diff --git a/Cylance/1.0.0/api.yaml b/Cylance/1.0.0/api.yaml index 48146a49..1b0abc44 100644 --- a/Cylance/1.0.0/api.yaml +++ b/Cylance/1.0.0/api.yaml @@ -1,6 +1,6 @@ app_version: 1.0.0 name: Cylance -description: An app to handle Cylance. Awful authentication. +description: An app to handle Cylance. contact_info: name: "@frikkylikeme" url: https://shuffler.io diff --git a/Docs/aws-iam.md b/Docs/aws-iam.md new file mode 100644 index 00000000..c071763a --- /dev/null +++ b/Docs/aws-iam.md @@ -0,0 +1,38 @@ +## AWS IAM App +Aws IAM (Identity and Access Management) app for managing IAM operations from the shuffle. + +![alt_text](https://github.com/dhaval055/Shuffle-apps/blob/master/Docs/iam.png?raw=true) + +## Actions + +| No. | Action | Description | Parameters | +|-----|--------|-------------|------------| +|1 | Get user | Retrieves information about the specified IAM user, including the user's creation date, path, unique ID, and ARN | access_key, secret_key, region, user_name +|2 | Change password | Change password of the specified user | access_key, secret_key, region, username, password +|3 | List users | Lists the IAM users | access_key, secret_key, region, path_prefix, marker, max_items +|4 | List user tags | Lists the tags that are attached to the specified IAM user | access_key, secret_key, region, user_name, marker, max_items +|5 | List attached user policies | Lists all managed policies that are attached to the specified IAM user | access_key, secret_key, region, user_name, marker, max_items +|6 | Attach user policy | Attach policy to the user | access_key, secret_key, region, username, policy_arn +|7 | Get instance profile | Retrieves information about the specified instance profile, including the instance profile's path, GUID, ARN, and role. | access_key, secret_key, region, instance_profile_name +|8 | List access keys | List all access keys | access_key, secret_key, region, username, marker, max_items +|9 | List ssh public keys | List SSH public keys | access_key, secret_key, region, username, marker, max_items + +__Note__: access_key, secret_key and region are used for authentication. + +## Requirements + +1. AWS account +2. Access key, Secret key and region of the user. + +- __How to find access key & secret key ?__ +1. Open https://console.aws.amazon.com/ +2. From navbar click on user dropwodown → My Security Credentials. +3. Open the Access keys tab, and then choose Create access key. +4. To see the new access key, choose Show. Your credentials resemble the following: + - Access key ID: AKIAIOSFODNN7EXAMPLE + - Secret access key: wJalrXUtnFEMI/K7MDENG/bPxRfiCYEXAMPLEKEY + + ## Note + Some actions have marker and max_items parameters (Both are used for paginating results). + - marker : Use this parameter only when paginating results and only after you receive a response indicating that the results are truncated. Set it to the value of the marker element in the response that you received to indicate where the next call should start. + - max_items : Use this only when paginating results to indicate the maximum number of items you want in the response. diff --git a/Docs/iam.png b/Docs/iam.png new file mode 100644 index 00000000..3ee433f6 Binary files /dev/null and b/Docs/iam.png differ diff --git a/Microsoft-Teams/1.0.0/api.yaml b/Microsoft-Teams/1.0.0/api.yaml index aa0cb634..39ea7cd8 100644 --- a/Microsoft-Teams/1.0.0/api.yaml +++ b/Microsoft-Teams/1.0.0/api.yaml @@ -104,6 +104,13 @@ actions: example: 'Alert...' schema: type: string + - name: added_information + description: Some extra information to be added to the callback. E.g. an alert + required: true + multiline: true + example: '$new_ticket.ticket_id' + schema: + type: string - name: callback_url description: webhook url of your workflow in shuffle required: true diff --git a/Microsoft-Teams/1.0.0/src/app.py b/Microsoft-Teams/1.0.0/src/app.py index 4e34e9de..1be9556e 100644 --- a/Microsoft-Teams/1.0.0/src/app.py +++ b/Microsoft-Teams/1.0.0/src/app.py @@ -43,15 +43,38 @@ async def send_rich_text(self, webhook_url, title, message, link_button_text, li return f'Message Sent' - async def send_actionable_msg(self, webhook_url, title, message, callback_url): + async def send_actionable_msg(self, webhook_url, title, message, added_information, callback_url): try: myTeamsMessage = teams.connectorcard(webhook_url) # You must create the connectorcard object with the Microsoft Webhook URL myTeamsMessage.title(title) # title for your card myTeamsMessage.text(message) # Add text to the message. myTeamsPotentialAction3 = teams.potentialaction(_name = "Select_Action") - myTeamsPotentialAction3.choices.addChoices("Accept","Accept") #option 1 - myTeamsPotentialAction3.choices.addChoices("Reject","Reject") #option 2 - myTeamsPotentialAction3.addInput("MultichoiceInput","list","Select Action",False) #Dropdown menu + + value = { + "choice": "ACCEPT", + "extra": added_information, + } + + #print(f"VALUE: {value}") + + try: + accept = json.dumps(value) + except: + print("FAILED ENCODING ACCEPT") + accept = "ACCEPT" + + myTeamsPotentialAction3.choices.addChoices("Accept", accept) #option 1 + + value["choice"] = "REJECT" + try: + deny = json.dumps(value) + except: + print("FAILED ENCODING REJECT") + deny = "REJECT" + + myTeamsPotentialAction3.choices.addChoices("Reject", deny) #option 2 + + myTeamsPotentialAction3.addInput("MultichoiceInput","list","Select Action", False) #Dropdown menu myTeamsPotentialAction3.addAction("HttpPost","Submit",callback_url) #post request to Shuffle myTeamsMessage.addPotentialAction(myTeamsPotentialAction3) myTeamsMessage.send()# send the message. @@ -74,5 +97,6 @@ async def get_user_input(self, webhook_url, title, message, callback_url): return f'{e.__class__} occured' return f'Message Sent' + if __name__ == "__main__": asyncio.run(MsTeams.run(), debug=True) diff --git a/aws-iam/1.0.0/Dockerfile b/aws-iam/1.0.0/Dockerfile new file mode 100644 index 00000000..364e1531 --- /dev/null +++ b/aws-iam/1.0.0/Dockerfile @@ -0,0 +1,26 @@ +# Base our app image off of the WALKOFF App SDK image +FROM frikky/shuffle:app_sdk as base + +# We're going to stage away all of the bloat from the build tools so lets create a builder stage +FROM base as builder + +# Install all alpine build tools needed for our pip installs +RUN apk --no-cache add --update alpine-sdk libffi libffi-dev musl-dev openssl-dev + +# Install all of our pip packages in a single directory that we can copy to our base image later +RUN mkdir /install +WORKDIR /install +COPY requirements.txt /requirements.txt +RUN pip install --prefix="/install" -r /requirements.txt + +# Switch back to our base image and copy in all of our built packages and source code +FROM base +COPY --from=builder /install /usr/local +COPY src /app + +# Install any binary dependencies needed in our final image +# RUN apk --no-cache add --update my_binary_dependency + +# Finally, lets run our app! +WORKDIR /app +CMD python app.py --log-level DEBUG diff --git a/aws-iam/1.0.0/api.yaml b/aws-iam/1.0.0/api.yaml new file mode 100644 index 00000000..ec2615f3 --- /dev/null +++ b/aws-iam/1.0.0/api.yaml @@ -0,0 +1,238 @@ +app_version: 1.0.0 +name: AWS IAM +description: An app to interact with Amazon IAM +contact_info: + name: "@dhaval055" + url: https://shuffler.io + email: dhavald@inforpercept.com +tags: + - Access + - Users +categories: + - Access + - Users +authentication: + required: true + parameters: + - name: access_key + description: The access key to use + example: "*****" + required: true + schema: + type: string + - name: secret_key + description: The secret key to use + example: "*****" + required: true + schema: + type: string + - name: region + description: The region to use + example: "ap-south-1" + required: true + schema: + type: string +actions: + - name: change_password + description: Change password of the specified user + parameters: + - name: username + description: Username to change password + required: true + multiline: false + example: 'johnwiliams' + schema: + type: string + - name: password + description: New password for user + required: true + multiline: false + example: '*****' + schema: + type: string + returns: + schema: + type: string + - name: attach_user_policy + description: Attach policy to the user. + parameters: + - name: username + description: Username you want to attach policy to. + required: true + multiline: false + example: 'johnwiliams' + schema: + type: string + - name: policy_arn + description: Amazon Resource Names (ARNs) uniquely identify AWS resources. + required: true + multiline: true + example: 'arn:aws:iam::aws:policy/AdministratorAccess' + schema: + type: string + returns: + schema: + type: string + - name: list_access_keys + description: List all access keys + parameters: + - name: username + description: List all access keys of this username + required: true + multiline: false + example: 'johnwiliams' + schema: + type: string + - name: marker + description: Use this parameter only when paginating results and only after you receive a response indicating that the results are truncated. + required: false + multiline: false + example: '10' + schema: + type: string + - name: max_items + description: Use this only when paginating results to indicate the maximum number of items you want in the response. + required: false + multiline: false + example: '123' + schema: + type: string + returns: + schema: + type: string + - name: list_ssh_public_keys + description: List SSH public keys + parameters: + - name: username + description: List SSH public keys of this username + required: true + multiline: false + example: 'johnwiliams' + schema: + type: string + - name: marker + description: Use this parameter only when paginating results and only after you receive a response indicating that the results are truncated. + required: false + multiline: false + example: '10' + schema: + type: string + - name: max_items + description: Use this only when paginating results to indicate the maximum number of items you want in the response. + required: false + multiline: false + example: '123' + schema: + type: string + returns: + schema: + type: string + - name: get_instance_profile + description: Retrieves information about the specified instance profile, including the instance profile's path, GUID, ARN, and role. + parameters: + - name: instance_profile_name + description: The name of the instance profile to get information about. + required: true + multiline: false + example: 'ExampleInstanceProfile' + schema: + type: string + returns: + schema: + type: string + - name: get_user + description: Retrieves information about the specified IAM user, including the user's creation date, path, unique ID, and ARN. + parameters: + - name: user_name + description: The name of the user to get information about. + required: true + multiline: false + example: 'Bob' + schema: + type: string + returns: + schema: + type: string + - name: list_attached_user_policies + description: Lists all managed policies that are attached to the specified IAM user. + parameters: + - name: user_name + description: The name (friendly name, not ARN) of the user to list attached policies for. + required: true + multiline: false + example: 'johnwiliams' + schema: + type: string + - name: marker + description: Use this parameter only when paginating results and only after you receive a response indicating that the results are truncated. + required: false + multiline: false + example: '10' + schema: + type: string + - name: max_items + description: Use this only when paginating results to indicate the maximum number of items you want in the response. + required: false + multiline: false + example: '123' + schema: + type: string + returns: + schema: + type: string + - name: list_users + description: Lists the IAM users + parameters: + - name: path_prefix + description: The path prefix for filtering the results. + required: true + multiline: false + example: '"/" for all users' + schema: + type: string + - name: marker + description: Use this parameter only when paginating results and only after you receive a response indicating that the results are truncated. + required: false + multiline: false + example: '10' + schema: + type: string + - name: max_items + description: Use this only when paginating results to indicate the maximum number of items you want in the response. + required: false + multiline: false + example: '123' + schema: + type: string + returns: + schema: + type: string + - name: list_user_tags + description: Lists the tags that are attached to the specified IAM user. + parameters: + - name: user_name + description: The path prefix for filtering the results. + required: true + multiline: false + example: 'JohnDoe' + schema: + type: string + - name: marker + description: Use this parameter only when paginating results and only after you receive a response indicating that the results are truncated. + required: false + multiline: false + example: '10' + schema: + type: string + - name: max_items + description: Use this only when paginating results to indicate the maximum number of items you want in the response. + required: false + multiline: false + example: '123' + schema: + type: string + returns: + schema: + type: string + +large_image:  diff --git a/aws-iam/1.0.0/requirements.txt b/aws-iam/1.0.0/requirements.txt new file mode 100644 index 00000000..06ef1c78 --- /dev/null +++ b/aws-iam/1.0.0/requirements.txt @@ -0,0 +1,2 @@ +boto3==1.16.59 +requests==2.25.1 \ No newline at end of file diff --git a/aws-iam/1.0.0/src/app.py b/aws-iam/1.0.0/src/app.py new file mode 100644 index 00000000..2088df88 --- /dev/null +++ b/aws-iam/1.0.0/src/app.py @@ -0,0 +1,248 @@ +import socket +import asyncio +import time +import random +import json +import boto3 +import botocore +from botocore.config import Config +import datetime + +from walkoff_app_sdk.app_base import AppBase + +def datetime_handler(x): + """ This function is used make datetime object json serilizable, + removing this function can cause error in some actions """ + + if isinstance(x, datetime.datetime): + return x.isoformat() + raise TypeError("Unknown type") + +class AWSIAM(AppBase): + __version__ = "1.0.0" + app_name = "AWS IAM" + + def __init__(self, redis, logger, console_logger=None): + """ + Each app should have this __init__ to set up Redis and logging. + :param redis: + :param logger: + :param console_logger: + """ + super().__init__(redis, logger, console_logger) + + async def auth_iam(self, access_key, secret_key, region): + my_config = Config( + region_name = region, + signature_version = 'v4', + retries = { + 'max_attempts': 10, + 'mode': 'standard' + }, + ) + + self.iam = boto3.resource( + 'iam', + config=my_config, + aws_access_key_id=access_key, + aws_secret_access_key=secret_key, + ) + + return self.iam + + async def change_password(self, access_key, secret_key, region, username, password): + self.iam = await self.auth_iam(access_key, secret_key, region) + client = self.iam.meta.client + + try: + return client.update_login_profile(UserName=username, Password=password, PasswordResetRequired=True) + except botocore.exceptions.ClientError as e: + print("Error: %s" % e) + return "%s" % e + + async def attach_user_policy(self, access_key, secret_key, region, username, policy_arn): + self.iam = await self.auth_iam(access_key, secret_key, region) + client = self.iam.meta.client + + try: + response = client.attach_user_policy( + PolicyArn=str(policy_arn), + UserName= str(username), + ) + return json.dumps(response) + except botocore.exceptions.ClientError as e: + print(f"Error: {e}") + return f'{e}' + + async def list_access_keys(self, access_key, secret_key, region, username, marker, max_items): + self.iam = await self.auth_iam(access_key, secret_key, region) + client = self.iam.meta.client + + try: + response = client.list_access_keys( + UserName= str(username) + ) + if marker: + response = client.list_access_keys( + UserName= str(username), + Marker = str(marker) + ) + if max_items: + response = client.list_access_keys( + UserName= str(username), + MaxItems = int(max_items) + ) + if marker and max_items: + response = client.list_access_keys( + UserName= str(username), + MaxItems = int(max_items), + Marker = str(marker) + ) + return json.dumps(response, default=datetime_handler) + except botocore.exceptions.ClientError as e: + return f'{e}' + + async def list_ssh_public_keys(self, access_key, secret_key, region, username, marker, max_items): + self.iam = await self.auth_iam(access_key, secret_key, region) + client = self.iam.meta.client + + try: + response = client.list_ssh_public_keys( + UserName= str(username) + ) + if marker: + response = client.list_ssh_public_keys( + UserName= str(username), + Marker = str(marker) + ) + if max_items: + response = client.list_ssh_public_keys( + UserName= str(username), + MaxItems = int(max_items) + ) + if marker and max_items: + response = client.list_ssh_public_keys( + UserName= str(username), + MaxItems = int(max_items), + Marker = str(marker) + ) + + return json.dumps(response, default=datetime_handler) + except botocore.exceptions.ClientError as e: + return f'{e}' + + async def get_instance_profile(self, access_key, secret_key, region, instance_profile_name): + self.iam = await self.auth_iam(access_key, secret_key, region) + client = self.iam.meta.client + + try: + response = client.get_instance_profile( + InstanceProfileName= str(instance_profile_name) + ) + return json.dumps(response, default=datetime_handler) + except botocore.exceptions.ClientError as e: + print(f"Error: {e}") + return f'{e}' + + async def get_user(self, access_key, secret_key, region, user_name): + self.iam = await self.auth_iam(access_key, secret_key, region) + client = self.iam.meta.client + + try: + response = client.get_user( + UserName= str(user_name) + ) + return json.dumps(response, default= datetime_handler) + except botocore.exceptions.ClientError as e: + print(f"Error: {e}") + return f'{e}' + + async def list_attached_user_policies(self, access_key, secret_key, region, user_name, marker, max_items): + self.iam = await self.auth_iam(access_key, secret_key, region) + client = self.iam.meta.client + + try: + response = client.list_attached_user_policies( + UserName= str(user_name) + ) + if marker: + response = client.list_attached_user_policies( + UserName= str(user_name), + Marker = str(marker) + ) + if max_items: + response = client.list_attached_user_policies( + UserName= str(user_name), + MaxItems = int(max_items) + ) + if marker and max_items: + response = client.list_attached_user_policies( + userName= str(user_name), + MaxItems = int(max_items), + Marker = str(marker) + ) + + return json.dumps(response, default=datetime_handler) + except botocore.exceptions.ClientError as e: + return f'{e}' + + async def list_users(self, access_key, secret_key, region, path_prefix, marker, max_items): + self.iam = await self.auth_iam(access_key, secret_key, region) + client = self.iam.meta.client + + try: + response = client.list_users( + PathPrefix = path_prefix + ) + if marker: + response = client.list_users( + PathPrefix = path_prefix, + Marker = str(marker) + ) + if max_items: + response = client.list_users( + PathPrefix = path_prefix, + MaxItems = int(max_items) + ) + if marker and max_items: + response = client.list_users( + PathPrefix = path_prefix, + MaxItems = int(max_items), + Marker = str(marker) + ) + + return json.dumps(response, default=datetime_handler) + except botocore.exceptions.ClientError as e: + return f'{e}' + + async def list_user_tags(self, access_key, secret_key, region, user_name, marker, max_items): + self.iam = await self.auth_iam(access_key, secret_key, region) + client = self.iam.meta.client + + try: + response = client.list_user_tags( + UserName = str(user_name) + ) + if marker: + response = client.list_user_tags( + UserName = str(user_name), + Marker = str(marker) + ) + if max_items: + response = client.list_user_tags( + UserName = str(user_name), + MaxItems = int(max_items) + ) + if marker and max_items: + response = client.list_user_tags( + UserName = str(user_name), + MaxItems = int(max_items), + Marker = str(marker) + ) + + return json.dumps(response, default=datetime_handler) + except botocore.exceptions.ClientError as e: + return f'{e}' + +if __name__ == "__main__": + asyncio.run(AWSIAM.run(), debug=True) diff --git a/shuffle-tools/1.0.0/api.yaml b/shuffle-tools/1.0.0/api.yaml index 735c832a..6d8c4b28 100644 --- a/shuffle-tools/1.0.0/api.yaml +++ b/shuffle-tools/1.0.0/api.yaml @@ -111,7 +111,7 @@ actions: options: - equals - 'larger than' - - 'less than' + - 'less than' - is empty - contains - starts with @@ -239,6 +239,31 @@ actions: returns: schema: type: string + - name: map_value + description: Takes a mapping dictionary and translates the input data + parameters: + - name: input_data + description: The input data to use + required: true + multiline: true + example: $exec.field1 + schema: + type: string + - name: mapping + description: The mapping dictionary + required: true + multiline: true + example: | + { + "Low": 1, + "Medium": 2, + "High": 3, + } + schema: + type: string + returns: + schema: + type: string - name: parse_list description: Parses a list and returns it as a json object parameters: @@ -285,25 +310,25 @@ actions: multiline: true example: "REPEATING: Hello world" schema: - type: file + type: file returns: schema: - type: string - - name: download_remote_file + type: string + - name: download_remote_file description: Downloads a file from a URL parameters: - - name: url - description: + - name: url + description: required: true - multiline: false + multiline: false example: "https://secure.eicar.org/eicar.com.txt" schema: - type: string + type: string returns: schema: - type: string - - name: get_file_meta - description: Gets the file meta + type: string + - name: get_file_meta + description: Gets the file meta parameters: - name: file_id description: diff --git a/shuffle-tools/1.0.0/src/app.py b/shuffle-tools/1.0.0/src/app.py index efcd0cd1..637d3b64 100644 --- a/shuffle-tools/1.0.0/src/app.py +++ b/shuffle-tools/1.0.0/src/app.py @@ -206,6 +206,17 @@ async def translate_value(self, input_data, translate_from, translate_to): return input_data + async def map_value(self, input_data, mapping): + + mapping = json.loads(mapping) + print(f"Got mapping {json.dumps(mapping, indent=2)}") + + # Get value if input_data in map, otherwise return original input_data + output_data = mapping.get(input_data, input_data) + print(f"Mapping {input_data} to {output_data}") + + return output_data + async def execute_python(self, code, shuffle_input): print("Run with shuffle_data %s" % shuffle_input) print("And python code %s" % code) @@ -261,15 +272,17 @@ async def filter_list(self, input_list, field, check, value, opposite): if not isinstance(input_list, list): return { "success": False, - "reason": "Error: input isnt a list. Remove # to use this app." % type(input_list) + "reason": "Error: input isnt a list. Remove # to use this app." % type(input_list), + "valid": [], + "invalid": [], } input_list = [input_list] new_list = [] failed_list = [] - try: - for item in input_list: + for item in input_list: + try: try: item = json.loads(item) except: @@ -407,7 +420,7 @@ async def filter_list(self, input_list, field, check, value, opposite): else: new_list = file_list #else: - # failed_list = file_list + # failed_list = file_list elif type(tmp) == str: filedata = self.get_file(tmp) @@ -419,8 +432,11 @@ async def filter_list(self, input_list, field, check, value, opposite): else: failed_list.append(item) - except Exception as e: - return "Error: %s" % e + except Exception as e: + #"Error: %s" % e + print("FAILED WITH EXCEPTION: %s" % e) + failed_list.append(item) + #return try: return json.dumps({ @@ -435,7 +451,7 @@ async def filter_list(self, input_list, field, check, value, opposite): "reason": "Failed parsing filter list output" + e, }) - return new_list + return new_list async def multi_list_filter(self, input_list, field, check, value): input_list = input_list.replace("'", '"', -1) @@ -525,7 +541,7 @@ async def delete_file(self, file_id): async def get_file_value(self, filedata): if filedata == None: return "File is empty?" - + print("INSIDE APP DATA: %s" % filedata) return "%s" % filedata["data"].decode() @@ -542,7 +558,7 @@ async def download_remote_file(self, url): else: value = {"success": False, "reason": "No files downloaded"} - return value + return value async def extract_archive(self, file_ids, fileformat="zip", password=None): try: @@ -589,7 +605,7 @@ async def extract_archive(self, file_ids, fileformat="zip", password=None): to_be_uploaded.append( {"filename": source.name, "data": source.read()} ) - return_data["success"] = True + return_data["success"] = True except (zipfile.BadZipFile, Exception): return_data["files"].append( { @@ -617,7 +633,7 @@ async def extract_archive(self, file_ids, fileformat="zip", password=None): to_be_uploaded.append( {"filename": source.name, "data": source.read()} ) - return_data["success"] = True + return_data["success"] = True except Exception: return_data["files"].append( { @@ -646,7 +662,7 @@ async def extract_archive(self, file_ids, fileformat="zip", password=None): "data": source.read(), } ) - return_data["success"] = True + return_data["success"] = True except Exception: return_data["files"].append( { diff --git a/thehive/1.0.0/api.yaml b/thehive/1.0.0/api.yaml index d854bee5..8faedceb 100644 --- a/thehive/1.0.0/api.yaml +++ b/thehive/1.0.0/api.yaml @@ -1,48 +1,48 @@ walkoff_version: 1.0.0 app_version: 1.0.0 name: thehive -description: TheHive implementation for Shuffle +description: TheHive implementation for Shuffle tags: - - Ticketing + - Ticketing - Search categories: - - Ticketing + - Ticketing - Search contact_info: - name: "@frikkylikeme" + name: "@frikkylikeme" url: https://github.com/frikky authentication: required: true parameters: - - name: apikey - description: The Apikey to use + - name: apikey + description: The Apikey to use example: "*****" required: true schema: type: string - - name: url - description: The URL to use + - name: url + description: The URL to use example: "http://localhost:9000" required: true schema: type: string actions: - - name: create_alert - description: Get an item from TheHive + - name: create_alert + description: Create an alert in TheHive parameters: - - name: type + - name: type description: The type to use for the alert example: "incident" required: true schema: type: string - - name: source + - name: source description: The source to use example: "SIEM" required: true schema: type: string - - name: sourceref + - name: sourceref description: The source reference to use example: "incident-1234" required: true @@ -54,26 +54,26 @@ actions: required: false schema: type: string - - name: description + - name: description description: The description to use example: "" required: false multiline: true schema: type: string - - name: tlp + - name: tlp description: The tlp to use example: "2" required: false schema: type: string - - name: severity + - name: severity description: The severity to use example: "2" required: false schema: type: string - - name: tags + - name: tags description: The tags to use example: "ioc,incident,this is a tag,what" required: false @@ -82,37 +82,122 @@ actions: returns: example: '{"data": "this is a test", "this_is_a_number": 1, "this_is_a_list": [{"item": [{"hello": "there", "how_is_this": {"sub_in_sub": [{"another": "list"}]}}]}, {"item": "2"}], "subobject": {"data": "subobject"}}' schema: - type: string + type: string + - name: create_alert_artifact + description: Create an alert artifact (TheHive 4 ONLY) + parameters: + - name: alert_id + description: Alert identifier + example: "~1234" + required: true + schema: + type: string + - name: dataType + description: "Observable's type, must be a valid type, one of the defined data types in TheHive" + example: "ip" + required: true + schema: + type: string + - name: data + description: Observable's data/value + example: "8.8.8.8" + required: true + schema: + type: string + - name: message + description: Observable's description + example: "Extracted IP entity from product X" + required: false + schema: + type: string + - name: tlp + description: "Case's TLP: 0, 1, 2, 3 for WHITE, GREEN, AMBER, RED. Default: 2" + example: "2" + required: false + schema: + type: string + - name: ioc + description: "Observable's ioc flag, True to mark an observable as IOC. Default: False" + example: "False" + required: false + multiline: false + schema: + type: string + - name: sighted + description: "Observable's sighted flag, True to mark the observable as sighted. Default: False" + example: "False" + required: false + multiline: false + schema: + type: string + - name: ignoreSimilarity + description: "Observable's similarity ignore flag. Trueto ignore the observable during similarity computing" + example: "False" + required: false + multiline: false + schema: + type: string + - name: tags + description: List of observable tags + example: "ioc,alienvault,abuse.ch" + required: false + schema: + type: string + returns: + example: | + [ + { + "_id": "~4321", + "id": "~4321", + "createdBy": "user.1@example.com", + "createdAt": 1616443009693, + "_type": "case_artifact", + "dataType": "ip", + "data": "8.8.8.8", + "startDate": 1616443009693, + "tlp": 2, + "tags": [ + "test1" + ], + "ioc": false, + "sighted": false, + "message": "Test IP entity", + "reports": {}, + "stats": {} + } + ] + schema: + type: string - name: create_case description: Get an item from TheHive parameters: - - name: title + - name: title description: The title to use example: "" required: false multiline: true schema: type: string - - name: description + - name: description description: The description to use example: "" required: false multiline: true schema: type: string - - name: tlp + - name: tlp description: The tlp to use example: "2" required: false schema: type: string - - name: severity + - name: severity description: The severity to use example: "2" required: false schema: type: string - - name: tags + - name: tags description: The tags to use example: "ioc,incident,this is a tag,what" required: false @@ -120,7 +205,7 @@ actions: type: string returns: schema: - type: string + type: string - name: create_case_from_alert description: Create a case from alert parameters: @@ -130,7 +215,7 @@ actions: required: true schema: type: string - - name: case_template + - name: case_template description: Case template name to apply when creating the case example: "" required: false @@ -148,7 +233,7 @@ actions: required: true schema: type: string - - name: case_id + - name: case_id description: The case to merge it to example: "" required: true @@ -157,28 +242,28 @@ actions: returns: schema: type: string - - name: add_observable + - name: add_observable description: Add an observable to TheHive parameters: - - name: case_id + - name: case_id description: The case to add it to example: "" required: true schema: - type: string - - name: data + type: string + - name: data description: The item to add itself example: "shuffler.io" required: true schema: type: string - - name: datatype + - name: datatype description: The type of the item to add example: "domain" required: true schema: type: string - - name: tags + - name: tags description: The tags to use example: "shuffle,is,cool" required: false @@ -186,11 +271,11 @@ actions: type: string returns: schema: - type: string - - name: get_item + type: string + - name: get_item description: Get an item from TheHive parameters: - - name: field_type + - name: field_type description: The type to get (alert, case..) example: "alert" options: @@ -199,13 +284,13 @@ actions: - case_observables - case_task - case_tasks - - linked_cases - - task_log + - linked_cases + - task_log - task_logs required: true schema: type: string - - name: cur_id + - name: cur_id description: The ID of the item to retrieve example: "" required: true @@ -213,11 +298,11 @@ actions: type: string returns: schema: - type: string + type: string - name: update_field description: Update an alert field parameters: - - name: field_type + - name: field_type description: The type to modify (alert, case..) options: - alert @@ -225,8 +310,8 @@ actions: - case_observables - case_task - case_tasks - - linked_cases - - task_log + - linked_cases + - task_log - task_logs required: true schema: @@ -236,12 +321,12 @@ actions: required: true schema: type: string - - name: field + - name: field description: The field to modify required: true schema: type: string - - name: data + - name: data description: The data to set the field to. If you want to append to what already exists, start with %s. required: true multiline: true @@ -250,10 +335,10 @@ actions: returns: schema: type: number - - name: search_cases + - name: search_cases description: Get an item from TheHive parameters: - - name: title_query + - name: title_query description: The title to search for example: "injection" required: true @@ -270,34 +355,34 @@ actions: required: true schema: type: string - - name: custom_query + - name: custom_query description: Custom query for search - example: "{\"_field\": \"title\", \"_value\": \"shuffle\"}" + example: "{\"_field\": \"title\", \"_value\": \"shuffle\"}" required: true schema: type: string - name: search_alerts description: Get an item from TheHive parameters: - - name: title_query + - name: title_query description: The title to search for example: "alert" required: true schema: type: string - - name: search_range + - name: search_range description: The amount of alerts to get. Defaults to 0-25 example: "0-50" - required: false + required: false schema: type: string returns: schema: - type: string - - name: close_alert - description: Close an alert in thehive + type: string + - name: close_alert + description: Close an alert in thehive parameters: - - name: alert_id + - name: alert_id description: The ID to close example: "adf5e3d0fd85633be17004735a0a119e" required: true @@ -305,21 +390,21 @@ actions: type: string returns: schema: - type: string - - name: reopen_alert + type: string + - name: reopen_alert description: Reopen an alert in TheHive parameters: - - name: alert_id + - name: alert_id description: The ID to close example: "adf5e3d0fd85633be17004735a0a119e" required: true schema: type: string - - name: run_analyzer + - name: run_analyzer description: Reopen an alert in TheHive parameters: - - name: cortex_id - description: The cortex ID + - name: cortex_id + description: The cortex ID example: "MISP_2_0" required: true schema: @@ -330,19 +415,19 @@ actions: required: true schema: type: string - - name: artifact_id - description: The artifact ID + - name: artifact_id + description: The artifact ID example: "adf5e3d0fd85633be17004735a0a119e" required: true schema: type: string returns: schema: - type: string - - name: create_task_log + type: string + - name: create_task_log description: Creates a task log in TheHive parameters: - - name: task_id + - name: task_id description: The task ID example: "AXX1SWs8Oc6KiwR-tT2f" required: true @@ -354,34 +439,34 @@ actions: required: true schema: type: string - - name: filedata + - name: filedata description: The file ID from Shuffle example: "adf5e3d0fd85633be17004735a0a119e" - required: false + required: false schema: - type: file - - name: create_case_file_observable + type: file + - name: create_case_file_observable description: Creates a task log in TheHive parameters: - - name: case_id + - name: case_id description: The case ID example: "AXX1SWs8Oc6KiwR-tT2f" required: true schema: type: string - - name: tags + - name: tags description: Tags for the case artifact example: "ioc,cool,artifact" required: true schema: type: string - - name: filedata + - name: filedata description: The file ID from Shuffle example: "adf5e3d0fd85633be17004735a0a119e" - required: true + required: true schema: - type: file + type: file returns: schema: - type: string + type: string large_image:  diff --git a/thehive/1.0.0/src/app.py b/thehive/1.0.0/src/app.py index d2bf8c76..95f087a9 100644 --- a/thehive/1.0.0/src/app.py +++ b/thehive/1.0.0/src/app.py @@ -34,13 +34,13 @@ def __init__(self, redis, logger, console_logger=None): super().__init__(redis, logger, console_logger) # async def run_analyzer(self, apikey, url, title_query): - # self.thehive = TheHiveApi(url, apikey) + # self.thehive = TheHiveApi(url, apikey, cert=False) # response = self.thehive.find_cases(query=String("title:'%s'" % title_query), range='all', sort=[]) # return response.text async def search_cases(self, apikey, url, title_query): - self.thehive = TheHiveApi(url, apikey) + self.thehive = TheHiveApi(url, apikey, cert=False) response = self.thehive.find_cases( query=ContainsString("title", title_query), range="all", sort=[] @@ -48,7 +48,7 @@ async def search_cases(self, apikey, url, title_query): return response.text async def search_query(self, apikey, url, search_for, custom_query): - self.thehive = TheHiveApi(url, apikey) + self.thehive = TheHiveApi(url, apikey, cert=False) try: query = json.loads(custom_query) @@ -63,10 +63,10 @@ async def search_query(self, apikey, url, search_for, custom_query): if response.status_code == 200: return response.text else: - raise IOError(response.text) + raise IOError(response.text) async def add_observable(self, apikey, url, case_id, data, datatype, tags): - self.thehive = TheHiveApi(url, apikey) + self.thehive = TheHiveApi(url, apikey, cert=False) if tags: if ", " in tags: @@ -91,7 +91,7 @@ async def add_observable(self, apikey, url, case_id, data, datatype, tags): return self.thehive.create_case_observable(case_id, item).text async def search_alerts(self, apikey, url, title_query, search_range="0-25"): - self.thehive = TheHiveApi(url, apikey) + self.thehive = TheHiveApi(url, apikey, cert=False) # Could be "all" too if search_range == "": @@ -105,7 +105,7 @@ async def search_alerts(self, apikey, url, title_query, search_range="0-25"): async def create_case( self, apikey, url, title, description="", tlp=1, severity=1, tags="" ): - self.thehive = TheHiveApi(url, apikey) + self.thehive = TheHiveApi(url, apikey, cert=False) if tags: if ", " in tags: tags = tags.split(", ") @@ -164,7 +164,7 @@ async def create_alert( severity=1, tags="", ): - self.thehive = TheHiveApi(url, apikey) + self.thehive = TheHiveApi(url, apikey, cert=False) if tags: if ", " in tags: tags = tags.split(", ") @@ -188,14 +188,14 @@ async def create_alert( tlp = int(tlp) if isinstance(severity, str): if not severity.isdigit(): - return "Severity needs to be a number from 1-3, not %s" % severity + return "Severity needs to be a number from 1-3, not %s" % severity severity = int(severity) if tlp > 3 or tlp < 0: return "TLP needs to be a number from 0-3, not %d" % tlp if severity > 3 or severity < 1: - return "Severity needs to be a number from 1-3, not %d" % severity + return "Severity needs to be a number from 1-3, not %d" % severity alert = thehive4py.models.Alert( title=title, @@ -214,9 +214,61 @@ async def create_alert( except requests.exceptions.ConnectionError as e: return "ConnectionError: %s" % e + async def create_alert_artifact( + self, + apikey, + url, + alert_id, + dataType, + data, + message=None, + tlp="2", + ioc="False", + sighted="False", + ignoreSimilarity="False", + tags=None + ): + self.thehive = TheHiveApi(url, apikey, cert=False, version=4) + + if tlp: + tlp = int(tlp) + else: + tlp = 2 + + ioc = ioc.lower().strip() == "true" + sighted = sighted.lower().strip() == "true" + ignoreSimilarity = ignoreSimilarity.lower().strip() == "true" + + if tags: + tags = [x.strip() for x in tags.split(",")] + else: + tags = [] + + + + alert_artifact = thehive4py.models.AlertArtifact( + dataType=dataType, + data=data, + message=message, + tlp=tlp, + ioc=ioc, + sighted=sighted, + ignoreSimilarity=ignoreSimilarity, + tags=tags + ) + + try: + ret = self.thehive.create_alert_artifact(alert_id, alert_artifact) + except requests.exceptions.ConnectionError as e: + return "ConnectionError: %s" % e + if ret.status_code > 299: + raise ConnectionError(ret.text) + + return ret.text + # Gets an item based on input. E.g. field_type = Alert async def get_item(self, apikey, url, field_type, cur_id): - self.thehive = TheHiveApi(url, apikey) + self.thehive = TheHiveApi(url, apikey, cert=False) newstr = "" ret = "" @@ -247,22 +299,22 @@ async def get_item(self, apikey, url, field_type, cur_id): return ret.text async def close_alert(self, apikey, url, alert_id): - self.thehive = TheHiveApi(url, apikey) + self.thehive = TheHiveApi(url, apikey, cert=False) return self.thehive.mark_alert_as_read(alert_id).text async def reopen_alert(self, apikey, url, alert_id): - self.thehive = TheHiveApi(url, apikey) + self.thehive = TheHiveApi(url, apikey, cert=False) return self.thehive.mark_alert_as_unread(alert_id).text async def create_case_from_alert(self, apikey, url, alert_id, case_template=None): - self.thehive = TheHiveApi(url, apikey) + self.thehive = TheHiveApi(url, apikey, cert=False) response = self.thehive.promote_alert_to_case( alert_id=alert_id, case_template=case_template ) return response.text async def merge_alert_into_case(self, apikey, url, alert_id, case_id): - self.thehive = TheHiveApi(url, apikey) + self.thehive = TheHiveApi(url, apikey, cert=False) req = url + f"/api/alert/{alert_id}/merge/{case_id}" ret = requests.post(req, auth=self.thehive.auth) return ret.text @@ -316,7 +368,7 @@ async def update_field(self, apikey, url, field_type, cur_id, field, data): # https://github.com/TheHive-Project/TheHiveDocs/tree/master/api/connectors/cortex async def run_analyzer(self, apikey, url, cortex_id, analyzer_id, artifact_id): - self.thehive = TheHiveApi(url, apikey) + self.thehive = TheHiveApi(url, apikey, cert=False) return self.thehive.run_analyzer(cortex_id, artifact_id, analyzer_id).text # Creates a task log in TheHive with file