diff --git a/.github/labeler.yml b/.github/labeler.yml new file mode 100644 index 00000000..c53d2d66 --- /dev/null +++ b/.github/labeler.yml @@ -0,0 +1,2 @@ +App: + - '/.*/' diff --git a/.github/workflows/ci.yaml b/.github/workflows/ci.yaml index 53e0cbd7..aa536619 100644 --- a/.github/workflows/ci.yaml +++ b/.github/workflows/ci.yaml @@ -64,7 +64,7 @@ jobs: with: context: ${{ matrix.app }}/${{ matrix.version }} file: ${{ matrix.app }}/${{ matrix.version }}/Dockerfile - platforms: linux/amd64,linux/arm64,linux/386 + platforms: linux/amd64,linux/arm64 push: true tags: | ${{ secrets.DOCKERHUB_USERNAME }}/${{ matrix.app }}:${{ matrix.version }} diff --git a/.github/workflows/project_automation.yml b/.github/workflows/project_automation.yml index 0e3bf945..837957f6 100644 --- a/.github/workflows/project_automation.yml +++ b/.github/workflows/project_automation.yml @@ -6,6 +6,16 @@ on: - opened jobs: + add-label: + name: Add label to issue + runs-on: ubuntu-latest + steps: + - uses: github/issue-labeler@v3.3 #May not be the latest version + with: + configuration-path: .github/labeler.yml + repo-token: ${{ secrets.ADD_TO_PROJECT_PAT }} + enable-versioned-regex: 0 + add-to-project: name: Add issue to project runs-on: ubuntu-latest diff --git a/README.md b/README.md index 3ed3879d..8d19233d 100644 --- a/README.md +++ b/README.md @@ -1,7 +1,7 @@ # Shuffle Apps -This is a repository for apps to be used in [Shuffle](https://github.com/frikky/shuffle) +All public apps are available in the search, engine either in your local instance or on [https://shuffler.io/search?tab=apps](https://shuffler.io/search?tab=apps). This is a repository for apps to be used in [Shuffle](https://github.com/frikky/shuffle) -**PS:** These apps should be valid with WALKOFF, but the SDK is different, meaning you have to change the FIRST line in each Dockerfile (FROM frikky/shuffle:app_sdk). +**PS:** These apps should be valid with WALKOFF (from NSA), but the SDK is different, meaning you have to change the FIRST line in each Dockerfile (FROM frikky/shuffle:app_sdk) to make it compatible with Shuffle. ## App Creation App creation can be done with the Shuffle App Creator (exports as OpenAPI) or Python, which makes it possible to connect _literally_ any tool. Always prioritize using the App Creator when applicable. diff --git a/email/1.2.0/src/app.py b/email/1.2.0/src/app.py index 182d72d9..4a27c9b2 100644 --- a/email/1.2.0/src/app.py +++ b/email/1.2.0/src/app.py @@ -394,7 +394,7 @@ def parse_email_file(self, file_id, file_extension): print("File: %s" % file_path) if file_extension.lower() == 'eml': print('working with .eml file') - ep = eml_parser.EmlParser(include_attachment_data=True, include_raw_body=True) + ep = eml_parser.EmlParser(include_attachment_data=True, include_raw_body=True, parse_attachment=True) try: parsed_eml = ep.decode_email_bytes(file_path['data']) if str(parsed_eml["header"]["date"]) == "1970-01-01 00:00:00+00:00": diff --git a/servicenow/1.0.0/Dockerfile b/email/1.3.0/Dockerfile similarity index 99% rename from servicenow/1.0.0/Dockerfile rename to email/1.3.0/Dockerfile index 364e1531..bcc1273d 100644 --- a/servicenow/1.0.0/Dockerfile +++ b/email/1.3.0/Dockerfile @@ -24,3 +24,4 @@ COPY src /app # Finally, lets run our app! WORKDIR /app CMD python app.py --log-level DEBUG + diff --git a/email/1.3.0/api.yaml b/email/1.3.0/api.yaml new file mode 100644 index 00000000..40998adf --- /dev/null +++ b/email/1.3.0/api.yaml @@ -0,0 +1,280 @@ +walkoff_version: 1.3.0 +app_version: 1.3.0 +name: email +description: Email app +tags: + - email +categories: + - communication +contact_info: + name: "@frikkylikeme" + url: https://github.com/frikky + email: "frikky@shuffler.io" +actions: + - name: send_email_shuffle + description: Send an email from Shuffle + parameters: + - name: apikey + description: Your https://shuffler.io apikey + multiline: false + example: "https://shuffler.io apikey" + required: true + schema: + type: string + - name: recipients + description: The recipients of the email + multiline: false + example: "test@example.com,frikky@shuffler.io" + required: true + schema: + type: string + - name: subject + description: The subject to use + multiline: false + example: "SOS this is an alert :o" + required: true + schema: + type: string + - name: body + description: The body to add to the email + multiline: true + example: "This is an email alert from Shuffler.io :)" + required: true + schema: + type: string + returns: + schema: + type: string + - name: send_email_smtp + description: Send an email with SMTP + parameters: + - name: username + description: The SMTP login username + multiline: false + example: "frikky@shuffler.io" + required: false + schema: + type: string + - name: password + description: The password to log in with SMTP + multiline: false + example: "******************" + required: false + schema: + type: string + - name: smtp_host + description: The host of the SMTP + multiline: false + example: "smtp-mail.outlook.com" + required: true + schema: + type: string + - name: smtp_port + description: The port to use for SMTP + multiline: false + example: "587" + required: true + schema: + type: string + - name: recipient + description: The receiver(s) of the email + multiline: false + example: "frikky@shuffler.io,frikky@shuffler.io" + required: true + schema: + type: string + - name: subject + description: The subject of the email + multiline: false + example: "This is a subject, hello there :)" + required: true + schema: + type: string + - name: body + description: The body to add to the email + multiline: true + example: "This is an email alert from Shuffler.io :)" + required: true + schema: + type: string + - name: attachments + description: Send files from shuffle as part of the email + multiline: false + example: "file_id1,file_id2,file_id3" + required: false + schema: + type: string + - name: ssl_verify + description: Whether to use TLS or not + example: "true" + required: false + options: + - true + - false + schema: + type: string + - name: body_type + description: The type of body to send. HTML by default + example: "true" + required: false + options: + - "html" + - "plain" + schema: + type: string + returns: + schema: + type: string + - name: get_emails_imap + description: Get emails using IMAP (e.g. imap.gmail.com / Outlook.office365.com) + parameters: + - name: username + description: The SMTP login username + multiline: false + example: "frikky@shuffler.io" + required: true + schema: + type: string + - name: password + description: The password to log in with SMTP + multiline: false + example: "******************" + required: true + schema: + type: string + - name: imap_server + description: The imap server host + multiline: false + example: "Outlook.office365.com" + required: true + schema: + type: string + - name: foldername + description: The folder to use, e.g. "inbox" + multiline: false + example: "inbox" + required: true + schema: + type: string + - name: amount + description: Amount of emails to retrieve + multiline: false + example: "10" + required: true + schema: + type: string + - name: unread + description: Retrieve just unread emails + multiline: false + options: + - "false" + - "true" + required: true + schema: + type: bool + - name: fields + description: Comma separated list of fields to be exported + multiline: false + example: "body, header.subject, header.header.message-id" + required: false + schema: + type: string + - name: include_raw_body + description: Include raw body in email export + multiline: false + options: + - "true" + - "false" + required: true + schema: + type: bool + - name: include_attachment_data + description: Include raw attachments in email export + multiline: false + options: + - "false" + - "true" + required: true + schema: + type: bool + - name: upload_email_shuffle + description: Upload email in shuffle, return uid + multiline: false + options: + - "false" + - "true" + required: true + schema: + type: bool + - name: upload_attachments_shuffle + description: Upload attachments in shuffle, return uids + multiline: false + options: + - "false" + - "true" + required: true + schema: + type: bool + - name: ssl_verify + description: Whether to use TLS or not + example: "true" + required: false + options: + - true + - false + schema: + type: string + - name: mark_as_read + description: Mark email as read or not + multiline: false + options: + - "false" + - "true" + required: false + schema: + type: bool + - name: parse_email_file + description: Takes a file from shuffle and analyzes it if it's a valid .eml or .msg + parameters: + - name: file_id + description: file id + required: true + multiline: true + example: 'adf5e3d0fd85633be17004735a0a119e' + schema: + type: string + - name: extract_attachments + description: Whether to extract the attachments straight into files + required: true + options: + - true + - false + example: 'true' + schema: + type: string + - name: parse_email_headers + description: + parameters: + - name: email_headers + description: Email headers + required: true + multiline: true + example: 'Email Headers' + schema: + type: string + returns: + schema: + type: string + - name: analyze_headers + description: + parameters: + - name: headers + description: Email headers in any format + required: true + multiline: true + schema: + type: string + returns: + schema: + type: string +large_image:  diff --git a/email/1.3.0/requirements.txt b/email/1.3.0/requirements.txt new file mode 100644 index 00000000..926027e8 --- /dev/null +++ b/email/1.3.0/requirements.txt @@ -0,0 +1,8 @@ +requests==2.25.1 +glom==20.11.0 +eml-parser==1.17.0 +msg-parser==1.2.0 +mail-parser==3.15.0 +extract-msg==0.30.9 +jsonpickle==2.0.0 + diff --git a/email/1.3.0/src/app.py b/email/1.3.0/src/app.py new file mode 100644 index 00000000..62b3a26d --- /dev/null +++ b/email/1.3.0/src/app.py @@ -0,0 +1,617 @@ +import json +import uuid +import socket +import asyncio +import requests +import tempfile +import datetime +import base64 +import imaplib +import smtplib +import time +import random +import eml_parser +import mailparser +import extract_msg +import jsonpickle + +from glom import glom +from msg_parser import MsOxMessage +from email.mime.multipart import MIMEMultipart +from email.mime.text import MIMEText +from email.mime.application import MIMEApplication + +from walkoff_app_sdk.app_base import AppBase + +def json_serial(obj): + if isinstance(obj, datetime.datetime): + serial = obj.isoformat() + return serial + +def default(o): + """helpers to store item in json + arguments: + - o: field of the object to serialize + returns: + - valid serialized value for unserializable fields + """ + if isinstance(o, (datetime.date, datetime.datetime)): + return o.isoformat() + if isinstance(o, set): + return list(o) + if isinstance(o, bytes): + try: + return o.decode("utf-8") + except: + print("Failed parsing utf-8 string") + return o + + +class Email(AppBase): + __version__ = "1.3.0" + app_name = "email" + + 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) + + # This is an email function of Shuffle + def send_email_shuffle(self, apikey, recipients, subject, body): + targets = [recipients] + if ", " in recipients: + targets = recipients.split(", ") + elif "," in recipients: + targets = recipients.split(",") + + data = {"targets": targets, "body": body, "subject": subject, "type": "alert"} + + url = "https://shuffler.io/functions/sendmail" + headers = {"Authorization": "Bearer %s" % apikey} + return requests.post(url, headers=headers, json=data).text + + def send_email_smtp( + self, smtp_host, recipient, subject, body, smtp_port, attachments="", username="", password="", ssl_verify="True", body_type="html" + ): + if type(smtp_port) == str: + try: + smtp_port = int(smtp_port) + except ValueError: + return "SMTP port needs to be a number (Current: %s)" % smtp_port + + try: + s = smtplib.SMTP(host=smtp_host, port=smtp_port) + except socket.gaierror as e: + return f"Bad SMTP host or port: {e}" + + # This is not how it should work.. + # Port 465 & 587 = TLS. Sometimes 25. + if ssl_verify == "false" or ssl_verify == "False": + pass + else: + s.starttls() + + if len(username) > 1 or len(password) > 1: + try: + s.login(username, password) + except smtplib.SMTPAuthenticationError as e: + return { + "success": False, + "reason": f"Bad username or password: {e}" + } + + if body_type == "" or len(body_type) < 3: + body_type = "html" + + # setup the parameters of the message + msg = MIMEMultipart() + msg["From"] = username + msg["To"] = recipient + msg["Subject"] = subject + msg.attach(MIMEText(body, body_type)) + + # Read the attachments + attachment_count = 0 + try: + if attachments != None and len(attachments) > 0: + print("Got attachments: %s" % attachments) + attachmentsplit = attachments.split(",") + + #attachments = parse_list(attachments, splitter=",") + #print("Got attachments2: %s" % attachmentsplit) + print("Before loop") + files = [] + for file_id in attachmentsplit: + print(f"Looping {file_id}") + file_id = file_id.strip() + new_file = self.get_file(file_id) + print(f"New file: {new_file}") + try: + part = MIMEApplication( + new_file["data"], + Name=new_file["filename"], + ) + part["Content-Disposition"] = f"attachment; filename=\"{new_file['filename']}\"" + msg.attach(part) + attachment_count += 1 + except Exception as e: + print(f"[WARNING] Failed to attach {file_id}: {e}") + + + #files.append(new_file) + + #return files + #data["attachments"] = files + except Exception as e: + self.logger.info(f"Error in attachment parsing for email: {e}") + + + try: + s.send_message(msg) + except smtplib.SMTPDataError as e: + return { + "success": False, + "reason": f"Failed to send mail: {e}" + } + + self.logger.info("Successfully sent email with subject %s to %s" % (subject, recipient)) + return { + "success": True, + "reason": "Email sent to %s!" % recipient, + "attachments": attachment_count + } + + def get_emails_imap( + self, + username, + password, + imap_server, + foldername, + amount, + unread, + fields, + include_raw_body, + include_attachment_data, + upload_email_shuffle, + upload_attachments_shuffle, + ssl_verify="True", + mark_as_read="False", + ): + def path_to_dict(path, value=None): + def pack(parts): + return ( + {parts[0]: pack(parts[1:]) if len(parts) > 1 else value} + if len(parts) > 1 + else {parts[0]: value} + ) + + return pack(path.split(".")) + + def merge(d1, d2): + for k in d2: + if k in d1 and isinstance(d1[k], dict) and isinstance(d2[k], dict): + merge(d1[k], d2[k]) + else: + d1[k] = d2[k] + + #if isinstance(mark_as_read, str): + # if str(mark_as_read).lower() == "true": + # mark_as_read = True + # else: + # mark_as_read = False + + if type(amount) == str: + try: + amount = int(amount) + except ValueError: + return { + "success": False, + "reason": "Amount needs to be a number, not %s" % amount, + } + + try: + email = imaplib.IMAP4_SSL(imap_server) + except ConnectionRefusedError as error: + try: + email = imaplib.IMAP4(imap_server) + + if ssl_verify == "false" or ssl_verify == "False" or ssl_verify == False: + pass + else: + email.starttls() + except socket.gaierror as error: + return { + "success": False, + "reason": "Can't connect to IMAP server %s: %s" % (imap_server, error), + } + except socket.gaierror as error: + return { + "success": False, + "reason": "Can't connect to IMAP server %s: %s" % (imap_server, error), + } + + try: + email.login(username, password) + except imaplib.IMAP4.error as error: + return { + "success": False, + "reason": "Failed to log into %s: %s" % (username, error), + } + + email.select(foldername) + unread = True if unread.lower().strip() == "true" else False + + try: + # IMAP search queries, e.g. "seen" or "read" + # https://www.rebex.net/secure-mail.net/features/imap-search.aspx + mode = "(UNSEEN)" if unread else "ALL" + thistype, data = email.search(None, mode) + except imaplib.IMAP4.error as error: + return { + "success": False, + "reason": "Couldn't find folder %s." % (foldername), + } + + email_ids = data[0] + id_list = email_ids.split() + if id_list == None: + return { + "success": False, + "reason": f"Couldn't retrieve email. Data: {data}", + } + + #try: + # self.logger.info(f"LIST: {id_list}") + #except TypeError: + # return { + # "success": False, + # "reason": "Error getting email. Data: %s" % data, + # } + + mark_as_read = True if str(mark_as_read).lower().strip() == "true" else False + include_raw_body = True if str(include_raw_body).lower().strip() == "true" else False + include_attachment_data = ( + True if str(include_attachment_data).lower().strip() == "true" else False + ) + upload_email_shuffle = ( + True if str(upload_email_shuffle).lower().strip() == "true" else False + ) + upload_attachments_shuffle = ( + True if str(upload_attachments_shuffle).lower().strip() == "true" else False + ) + + # Convert of mails in json + emails = [] + ep = eml_parser.EmlParser( + include_attachment_data=include_attachment_data + or upload_attachments_shuffle, + include_raw_body=include_raw_body, + ) + + if len(id_list) == 0: + return { + "success": True, + "messages": [], + } + + try: + amount = len(id_list) if len(id_list)= 2: + parsed_headers[splitheader[0].strip()] = splititem.join(splitheader[1:]).strip() else: self.logger.info("Skipping header %s with split %s cus only one item" % (header, splititem)) continue diff --git a/http/1.4.0/Dockerfile b/http/1.4.0/Dockerfile new file mode 100644 index 00000000..9bbc5110 --- /dev/null +++ b/http/1.4.0/Dockerfile @@ -0,0 +1,27 @@ +# 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 +RUN apk add curl + +# 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/http/1.4.0/api.yaml b/http/1.4.0/api.yaml new file mode 100644 index 00000000..cad10c2c --- /dev/null +++ b/http/1.4.0/api.yaml @@ -0,0 +1,529 @@ +walkoff_version: 1.4.0 +app_version: 1.4.0 +name: http +description: HTTP app +tags: + - Testing + - HTTP +categories: + - Other + - HTTP +contact_info: + name: "@frikkylikeme" + url: https://github.com/frikky + email: "frikky@shuffler.io" +actions: + - name: GET + description: Runs a GET request towards the specified endpoint + parameters: + - name: url + description: The URL to get + multiline: false + example: "https://example.com" + required: true + schema: + type: string + - name: headers + description: Headers to use + multiline: true + required: false + example: "Content-Type: application/json" + schema: + type: string + - name: username + description: The username to use + multiline: false + required: false + example: "Username" + schema: + type: string + - name: password + description: The password to use + multiline: false + required: false + example: "*****" + schema: + type: string + - name: verify + description: Check certificate + multiline: false + options: + - false + - true + required: false + example: "false" + schema: + type: bool + - name: http_proxy + description: Add a HTTP proxy + multiline: false + required: false + example: "http://192.168.0.1:8080" + schema: + type: bool + - name: https_proxy + description: Add a HTTPS proxy + multiline: false + required: false + example: "http://192.168.0.1:8080" + schema: + type: bool + - name: timeout + description: Add a timeout for the request, in seconds + multiline: false + required: false + example: "10" + schema: + type: bool + - name: to_file + description: Makes the response into a file, and returns it as an ID + multiline: false + required: false + options: + - false + - true + example: "true" + schema: + type: bool + returns: + schema: + type: string + example: "404 NOT FOUND" + - name: POST + description: Runs a POST request towards the specified endpoint + parameters: + - name: url + description: The URL to post to + multiline: false + example: "https://example.com" + required: true + schema: + type: string + - name: body + description: The body to use + multiline: true + example: "{\n\t'json': 'blob'\n}" + required: false + schema: + type: string + - name: headers + description: Headers to use + multiline: true + required: false + example: "Content-Type: application/json" + schema: + type: string + - name: username + description: The username to use + multiline: false + required: false + example: "Username" + schema: + type: string + - name: password + description: The password to use + multiline: false + required: false + example: "*****" + schema: + type: string + - name: verify + description: Whether to check the certificate or not + multiline: false + required: false + options: + - false + - true + example: "false" + schema: + type: bool + - name: http_proxy + description: Add a HTTP proxy + multiline: false + required: false + example: "http://192.168.0.1:8080" + schema: + type: bool + - name: https_proxy + description: Add a HTTPS proxy + multiline: false + required: false + example: "http://192.168.0.1:8080" + schema: + type: bool + - name: timeout + description: Add a timeout for the request, in seconds + multiline: false + required: false + example: "10" + schema: + type: bool + returns: + schema: + type: string + example: "404 NOT FOUND" + - name: PATCH + description: Runs a PATCH request towards the specified endpoint + parameters: + - name: url + description: The URL to post to + multiline: false + example: "https://example.com" + required: true + schema: + type: string + - name: body + description: The body to use + multiline: true + example: "{\n\t'json': 'blob'\n}" + required: false + schema: + type: string + - name: headers + description: Headers to use + multiline: true + required: false + example: "Content-Type: application/json" + schema: + type: string + - name: username + description: The username to use + multiline: false + required: false + example: "Username" + schema: + type: string + - name: password + description: The password to use + multiline: false + required: false + example: "*****" + schema: + type: string + - name: verify + description: Whether to check the certificate or not + multiline: false + required: false + options: + - false + - true + example: "false" + schema: + type: bool + - name: http_proxy + description: Add a HTTP proxy + multiline: false + required: false + example: "http://192.168.0.1:8080" + schema: + type: bool + - name: https_proxy + description: Add a HTTPS proxy + multiline: false + required: false + example: "http://192.168.0.1:8080" + schema: + type: bool + - name: timeout + description: Add a timeout for the request, in seconds + multiline: false + required: false + example: "10" + schema: + type: bool + returns: + schema: + type: string + example: "404 NOT FOUND" + - name: PUT + description: Runs a PUT request towards the specified endpoint + parameters: + - name: url + description: The URL to PUT to + multiline: false + example: "https://example.com" + required: true + schema: + type: string + - name: body + description: The body to use + multiline: true + example: "{\n\t'json': 'blob'\n}" + required: false + schema: + type: string + - name: headers + description: Headers to use + multiline: true + required: false + example: "Content-Type: application/json" + schema: + type: string + - name: username + description: The username to use + multiline: false + required: false + example: "Username" + schema: + type: string + - name: password + description: The password to use + multiline: false + required: false + example: "*****" + schema: + type: string + - name: verify + description: Whether to check the certificate or not + multiline: false + required: false + options: + - false + - true + example: "false" + schema: + type: bool + - name: http_proxy + description: Add a HTTP proxy + multiline: false + required: false + example: "http://192.168.0.1:8080" + schema: + type: bool + - name: https_proxy + description: Add a HTTPS proxy + multiline: false + required: false + example: "http://192.168.0.1:8080" + schema: + type: bool + - name: timeout + description: Add a timeout for the request, in seconds + multiline: false + required: false + example: "10" + schema: + type: bool + returns: + schema: + type: string + example: "404 NOT FOUND" + - name: DELETE + description: Runs a DELETE request towards the specified endpoint + parameters: + - name: url + description: The URL to post to + multiline: false + example: "https://example.com" + required: true + schema: + type: string + - name: body + description: The body to use + multiline: true + example: "{\n\t'json': 'blob'\n}" + required: false + schema: + type: string + - name: headers + description: Headers to use + multiline: true + required: false + example: "Content-Type: application/json" + schema: + type: string + - name: username + description: The username to use + multiline: false + required: false + example: "Username" + schema: + type: string + - name: password + description: The password to use + multiline: false + required: false + example: "*****" + schema: + type: string + - name: verify + description: Whether to check the certificate or not + multiline: false + required: false + options: + - true + - false + example: "false" + schema: + type: bool + - name: http_proxy + description: Add a HTTP proxy + multiline: false + required: false + example: "http://192.168.0.1:8080" + schema: + type: bool + - name: https_proxy + description: Add a HTTPS proxy + multiline: false + required: false + example: "http://192.168.0.1:8080" + schema: + type: bool + - name: timeout + description: Add a timeout for the request, in seconds + multiline: false + required: false + example: "10" + schema: + type: bool + returns: + schema: + type: string + example: "404 NOT FOUND" + - name: HEAD + description: Runs a HEAD request towards the specified endpoint + parameters: + - name: url + description: The URL to HEAD to + multiline: false + example: "https://example.com" + required: true + schema: + type: string + - name: headers + description: Headers to use + multiline: true + required: false + example: "Content-Type: application/json" + schema: + type: string + - name: username + description: The username to use + multiline: false + required: false + example: "Username" + schema: + type: string + - name: password + description: The password to use + multiline: false + required: false + example: "*****" + schema: + type: string + - name: verify + description: Whether to check the certificate or not + multiline: false + required: false + options: + - false + - true + example: "false" + schema: + type: bool + - name: http_proxy + description: Add a HTTP proxy + multiline: false + required: false + example: "http://192.168.0.1:8080" + schema: + type: bool + - name: https_proxy + description: Add a HTTPS proxy + multiline: false + required: false + example: "http://192.168.0.1:8080" + schema: + type: bool + - name: timeout + description: Add a timeout for the request, in seconds + multiline: false + required: false + example: "10" + schema: + type: bool + returns: + schema: + type: string + example: "404 NOT FOUND" + - name: OPTIONS + description: Runs a OPTIONS request towards the specified endpoint + parameters: + - name: url + description: The URL to HEAD to + multiline: false + example: "https://example.com" + required: true + schema: + type: string + - name: headers + description: Headers to use + multiline: true + required: false + example: "Content-Type: application/json" + schema: + type: string + - name: username + description: The username to use + multiline: false + required: false + example: "Username" + schema: + type: string + - name: password + description: The password to use + multiline: false + required: false + example: "*****" + schema: + type: string + - name: verify + description: Whether to check the certificate or not + multiline: false + required: false + options: + - false + - true + example: "false" + schema: + type: bool + - name: http_proxy + description: Add a HTTP proxy + multiline: false + required: false + example: "http://192.168.0.1:8080" + schema: + type: bool + - name: https_proxy + description: Add a HTTPS proxy + multiline: false + required: false + example: "http://192.168.0.1:8080" + schema: + type: bool + - name: timeout + description: Add a timeout for the request, in seconds + multiline: false + required: false + example: "10" + schema: + type: bool + returns: + schema: + type: string + example: "404 NOT FOUND" + - name: curl + description: Run a curl command + parameters: + - name: statement + description: The curl command to run + multiline: true + example: "curl https://example.com" + required: true + schema: + type: string + returns: + schema: + type: string +large_image:  diff --git a/servicenow/1.0.0/requirements.txt b/http/1.4.0/requirements.txt similarity index 51% rename from servicenow/1.0.0/requirements.txt rename to http/1.4.0/requirements.txt index fd7d3e06..ae3e5391 100644 --- a/servicenow/1.0.0/requirements.txt +++ b/http/1.4.0/requirements.txt @@ -1 +1,2 @@ +uncurl==0.0.10 requests==2.25.1 \ No newline at end of file diff --git a/http/1.4.0/src/app.py b/http/1.4.0/src/app.py new file mode 100755 index 00000000..ff2ec91b --- /dev/null +++ b/http/1.4.0/src/app.py @@ -0,0 +1,450 @@ +import time +import json +import ast +import random +import socket +import uncurl +import asyncio +import requests +import subprocess + +from walkoff_app_sdk.app_base import AppBase + +class HTTP(AppBase): + __version__ = "1.3.0" + app_name = "http" + + def __init__(self, redis, logger, console_logger=None): + print("INIT") + """ + 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) + + # This is dangerously fun :) + # Do we care about arbitrary code execution here? + # Probably not huh + def curl(self, statement): + process = subprocess.Popen(statement, stdout=subprocess.PIPE, stderr=subprocess.PIPE, text=True, shell=True) + stdout = process.communicate() + item = "" + if len(stdout[0]) > 0: + print("Succesfully ran bash!") + item = stdout[0] + else: + print("FAILED to run bash!") + item = stdout[1] + + try: + ret = item.decode("utf-8") + return ret + except: + return item + + return item + + def splitheaders(self, headers): + parsed_headers = {} + if headers: + split_headers = headers.split("\n") + self.logger.info(split_headers) + for header in split_headers: + if ":" in header: + splititem = ":" + elif "=" in header: + splititem = "=" + else: + self.logger.info("Skipping header %s as its invalid" % header) + continue + + splitheader = header.split(splititem) + if len(splitheader) >= 2: + parsed_headers[splitheader[0].strip()] = splititem.join(splitheader[1:]).strip() + else: + self.logger.info("Skipping header %s with split %s cus only one item" % (header, splititem)) + continue + + return parsed_headers + + def checkverify(self, verify): + if str(verify).lower().strip() == "false": + return False + elif verify == None: + return False + elif verify: + return True + elif not verify: + return False + else: + return True + + def checkbody(self, body): + # Indicates json + if isinstance(body, str): + if body.strip().startswith("{"): + body = json.dumps(ast.literal_eval(body)) + + + # Not sure if loading is necessary + # Seemed to work with plain string into data=body too, and not parsed json=body + #try: + # body = json.loads(body) + #except json.decoder.JSONDecodeError as e: + # return body + + return body + else: + return body + + if isinstance(body, dict) or isinstance(body, list): + try: + body = json.dumps(body) + except: + return body + + return body + + def fix_url(self, url): + # Random bugs seen by users + if "hhttp" in url: + url = url.replace("hhttp", "http") + + if "http:/" in url and not "http://" in url: + url = url.replace("http:/", "http://", -1) + if "https:/" in url and not "https://" in url: + url = url.replace("https:/", "https://", -1) + if "http:///" in url: + url = url.replace("http:///", "http://", -1) + if "https:///" in url: + url = url.replace("https:///", "https://", -1) + if not "http://" in url and not "http" in url: + url = f"http://{url}" + + return url + + def return_file(self, requestdata): + filedata = { + "filename": "response.txt", + "data": requestdata, + } + fileret = self.set_files([filedata]) + if len(fileret) == 1: + return {"success": True, "file_id": fileret[0]} + + return fileret + + def prepare_response(self, request): + try: + parsedheaders = {} + for key, value in request.headers.items(): + parsedheaders[key] = value + + cookies = {} + if request.cookies: + for key, value in request.cookies.items(): + cookies[key] = value + + + jsondata = request.text + try: + jsondata = json.loads(jsondata) + except: + pass + + parseddata = { + "status": request.status_code, + "body": jsondata, + "url": request.url, + "headers": parsedheaders, + "cookies":cookies, + "success": True, + } + + return json.dumps(parseddata) + except Exception as e: + print(f"[WARNING] Failed in request: {e}") + return request.text + + def GET(self, url, headers="", username="", password="", verify=True, http_proxy="", https_proxy="", timeout=5, to_file=False): + url = self.fix_url(url) + + parsed_headers = self.splitheaders(headers) + parsed_headers["User-Agent"] = "Shuffle Automation" + verify = self.checkverify(verify) + proxies = None + if http_proxy: + proxies["http"] = http_proxy + if https_proxy: + proxies["https"] = https_proxy + + auth=None + if username or password: + # Shouldn't be used if authorization headers exist + if "Authorization" in parsed_headers: + #print("Found authorization - skipping username & pw") + pass + else: + auth = requests.auth.HTTPBasicAuth(username, password) + + if not timeout: + timeout = 5 + if timeout: + timeout = int(timeout) + + if to_file == "true": + to_file = True + else: + to_file = False + + request = requests.get(url, headers=parsed_headers, auth=auth, verify=verify, proxies=proxies, timeout=timeout) + if not to_file: + return self.prepare_response(request) + + return self.return_file(request.text) + + def POST(self, url, headers="", body="", username="", password="", verify=True, http_proxy="", https_proxy="", timeout=5, to_file=False): + url = self.fix_url(url) + + parsed_headers = self.splitheaders(headers) + parsed_headers["User-Agent"] = "Shuffle Automation" + verify = self.checkverify(verify) + body = self.checkbody(body) + proxies = None + if http_proxy: + proxies["http"] = http_proxy + if https_proxy: + proxies["https"] = https_proxy + + auth=None + if username or password: + # Shouldn't be used if authorization headers exist + if "Authorization" in parsed_headers: + #print("Found authorization - skipping username & pw") + pass + else: + auth = requests.auth.HTTPBasicAuth(username, password) + + if not timeout: + timeout = 5 + if timeout: + timeout = int(timeout) + + if to_file == "true": + to_file = True + else: + to_file = False + + request = requests.post(url, headers=parsed_headers, auth=auth, data=body, verify=verify, proxies=proxies, timeout=timeout) + if not to_file: + return self.prepare_response(request) + + return self.return_file(request.text) + + def PUT(self, url, headers="", body="", username="", password="", verify=True, http_proxy="", https_proxy="", timeout=5, to_file=False): + url = self.fix_url(url) + + parsed_headers = self.splitheaders(headers) + parsed_headers["User-Agent"] = "Shuffle Automation" + verify = self.checkverify(verify) + body = self.checkbody(body) + proxies = None + if http_proxy: + proxies["http"] = http_proxy + if https_proxy: + proxies["https"] = https_proxy + + + auth=None + if username or password: + # Shouldn't be used if authorization headers exist + if "Authorization" in parsed_headers: + #print("Found authorization - skipping username & pw") + pass + else: + auth = requests.auth.HTTPBasicAuth(username, password) + + if not timeout: + timeout = 5 + if timeout: + timeout = int(timeout) + + if to_file == "true": + to_file = True + else: + to_file = False + + request = requests.put(url, headers=parsed_headers, auth=auth, data=body, verify=verify, proxies=proxies, timeout=timeout) + if not to_file: + return self.prepare_response(request) + + return self.return_file(request.text) + + def PATCH(self, url, headers="", body="", username="", password="", verify=True, http_proxy="", https_proxy="", timeout=5, to_file=False): + url = self.fix_url(url) + + parsed_headers = self.splitheaders(headers) + parsed_headers["User-Agent"] = "Shuffle Automation" + verify = self.checkverify(verify) + body = self.checkbody(body) + proxies = None + if http_proxy: + proxies["http"] = http_proxy + if https_proxy: + proxies["https"] = https_proxy + + auth=None + if username or password: + # Shouldn't be used if authorization headers exist + if "Authorization" in parsed_headers: + #print("Found authorization - skipping username & pw") + pass + else: + auth = requests.auth.HTTPBasicAuth(username, password) + + if to_file == "true": + to_file = True + else: + to_file = False + + request = requests.patch(url, headers=parsed_headers, data=body, auth=auth, verify=verify, proxies=proxies, timeout=timeout) + if not to_file: + return self.prepare_response(request) + + return self.return_file(request.text) + + def DELETE(self, url, headers="", body="", username="", password="", verify=True, http_proxy="", https_proxy="", timeout=5, to_file=False): + url = self.fix_url(url) + + parsed_headers = self.splitheaders(headers) + parsed_headers["User-Agent"] = "Shuffle Automation" + verify = self.checkverify(verify) + body = self.checkbody(body) + proxies = None + if http_proxy: + proxies["http"] = http_proxy + if https_proxy: + proxies["https"] = https_proxy + + auth=None + if username or password: + # Shouldn't be used if authorization headers exist + if "Authorization" in parsed_headers: + #print("Found authorization - skipping username & pw") + pass + else: + auth = requests.auth.HTTPBasicAuth(username, password) + + if not timeout: + timeout = 5 + if timeout: + timeout = int(timeout) + + if to_file == "true": + to_file = True + else: + to_file = False + + request = requests.delete(url, headers=parsed_headers, data=body, auth=auth, verify=verify, proxies=proxies, timeout=timeout) + if not to_file: + return self.prepare_response(request) + + return self.return_file(request.text) + + def HEAD(self, url, headers="", body="", username="", password="", verify=True, http_proxy="", https_proxy="", timeout=5, to_file=False): + url = self.fix_url(url) + + parsed_headers = self.splitheaders(headers) + parsed_headers["User-Agent"] = "Shuffle Automation" + verify = self.checkverify(verify) + body = self.checkbody(body) + proxies = None + if http_proxy: + proxies["http"] = http_proxy + if https_proxy: + proxies["https"] = https_proxy + + auth=None + if username or password: + # Shouldn't be used if authorization headers exist + if "Authorization" in parsed_headers: + #print("Found authorization - skipping username & pw") + pass + else: + auth = requests.auth.HTTPBasicAuth(username, password) + + if not timeout: + timeout = 5 + if timeout: + timeout = int(timeout) + + if to_file == "true": + to_file = True + else: + to_file = False + + request = requests.head(url, headers=parsed_headers, auth=auth, verify=verify, proxies=proxies, timeout=timeout) + if not to_file: + return self.prepare_response(request) + + return self.return_file(request.text) + + def OPTIONS(self, url, headers="", body="", username="", password="", verify=True, http_proxy="", https_proxy="", timeout=5, to_file=False): + url = self.fix_url(url) + + parsed_headers = self.splitheaders(headers) + parsed_headers["User-Agent"] = "Shuffle Automation" + verify = self.checkverify(verify) + body = self.checkbody(body) + proxies = None + if http_proxy: + proxies["http"] = http_proxy + if https_proxy: + proxies["https"] = https_proxy + + auth=None + if username or password: + # Shouldn't be used if authorization headers exist + if "Authorization" in parsed_headers: + #print("Found authorization - skipping username & pw") + pass + else: + auth = requests.auth.HTTPBasicAuth(username, password) + + if not timeout: + timeout = 5 + + if timeout: + timeout = int(timeout) + + if to_file == "true": + to_file = True + else: + to_file = False + + request = requests.options(url, headers=parsed_headers, auth=auth, verify=verify, proxies=proxies, timeout=timeout) + if not to_file: + return self.prepare_response(request) + + return self.return_file(request.text) + + +# Run the actual thing after we've checked params +def run(request): + print("Starting cloud!") + action = request.get_json() + print(action) + print(type(action)) + authorization_key = action.get("authorization") + current_execution_id = action.get("execution_id") + + if action and "name" in action and "app_name" in action: + HTTP.run(action) + return f'Attempting to execute function {action["name"]} in app {action["app_name"]}' + else: + return f'Invalid action' + +if __name__ == "__main__": + HTTP.run() diff --git a/outlook-exchange/1.0.0/src/app.py b/outlook-exchange/1.0.0/src/app.py index 43673a10..38d8b2cd 100644 --- a/outlook-exchange/1.0.0/src/app.py +++ b/outlook-exchange/1.0.0/src/app.py @@ -198,14 +198,13 @@ def send_email( account=account, subject=subject, body=body, - to_recipients=[ - Mailbox(email_address=address) for address in recipient.split(", ") - ], - cc_recipients=[ - Mailbox(email_address=address) for address in ccrecipient.split(", ") - ], + to_recipients=[] ) + for address in recipient.split(", "): + address = address.strip() + m.to_recipients.append(Mailbox(email_address=address)) + file_uids = str(attachments).split() if len(file_uids) > 0: for file_uid in file_uids: diff --git a/servicenow/1.0.0/api.yaml b/servicenow/1.0.0/api.yaml deleted file mode 100644 index 4fa62bef..00000000 --- a/servicenow/1.0.0/api.yaml +++ /dev/null @@ -1,146 +0,0 @@ -walkoff_version: 1.0.0 -app_version: 1.0.0 -name: servicenow -description: servicenow app -tags: - - tickets -categories: - - tickets -contact_info: - name: "@frikkylikeme" - url: https://github.com/frikky - email: "frikky@shuffler.io" -authentication: - required: true - parameters: - - name: url - description: The url your instance is at - multiline: false - example: "test.service-now.com" - required: true - schema: - type: string - - name: username - description: The user to authenticate with - multiline: false - example: "username12345" - required: true - schema: - type: string - - name: password - description: The password for the user to authenticate with - multiline: false - example: "pw1234" - required: true - schema: - type: string -actions: - - name: get_ticket - description: Get ticket ids - parameters: - - name: table_name - description: The type to get. Empty as default - multiline: false - example: "incident" - required: true - schema: - type: string - - name: sys_id - description: The ID to get from the table - multiline: false - example: "INC123456" - required: true - schema: - type: string - - name: number - description: The number to get instead of record_id - multiline: false - example: "20" - required: false - schema: - type: string - returns: - schema: - type: string - - name: create_ticket - description: Create a ticket - parameters: - - name: table_name - description: The table to create the ticket in - multiline: false - example: "incident" - required: true - schema: - type: string - - name: body - description: The body of the ticket - multiline: true - example: "{'short_description':'Unable to connect to office wifi','assignment_group':'287ebd7da9fe198100f92cc8d1d2154e','urgency':'2','impact':'2'}" - required: true - schema: - type: string - - name: file_id - description: Optional file to attach - multiline: false - example: "ca0c88a6-626e-4235-896f-ca18c96fd48e" - required: false - schema: - type: string - returns: - schema: - type: string - - name: update_ticket - description: Update a ticket - parameters: - - name: table_name - description: The table to create the ticket in - multiline: false - example: "incident" - required: true - schema: - type: string - - name: sys_id - description: The ticket to edit - multiline: false - example: "incident" - required: true - schema: - type: string - - name: body - description: JSON data of the data to replace - multiline: true - example: "{'short_description':'Unable to connect to office wifi','assignment_group':'287ebd7da9fe198100f92cc8d1d2154e','urgency':'2','impact':'2'}" - required: true - schema: - type: string - - name: file_id - description: Optional file to attach - multiline: false - example: "ca0c88a6-626e-4235-896f-ca18c96fd48e" - required: false - schema: - type: string - returns: - schema: - type: string - - name: list_table - description: Get ticket ids - parameters: - - name: table_name - description: The type to get. Empty as default - multiline: false - example: "incident" - required: true - schema: - type: string - - name: limit - description: The limit of items to get - multiline: false - example: "1" - required: false - schema: - type: string - returns: - schema: - type: string -large_image:  diff --git a/servicenow/1.0.0/src/app.py b/servicenow/1.0.0/src/app.py deleted file mode 100755 index d85dc832..00000000 --- a/servicenow/1.0.0/src/app.py +++ /dev/null @@ -1,204 +0,0 @@ -import time -import json -import random -import socket -import asyncio -import requests - -from walkoff_app_sdk.app_base import AppBase - -class Servicenow(AppBase): - __version__ = "1.0.0" - app_name = "servicenow" - - 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) - - def send_request(self, url, username, password, path, method='get', body=None, params=None, headers=None, json=None, files=None): - body = body if body is not None else {} - params = params if params is not None else {} - - url = '{}{}'.format(url, path) - print("HEADERS: %s" % headers) - if not headers and files == None: - headers = { - 'Accept': 'application/json', - 'Content-Type': 'application/json' - } - - if files: - # Not supported in v2 - url = url.replace('v2', 'v1') - #{'file': ('report.csv', 'some,data,to,send\nanother,row,to,send\n')} - #file_entry = file['id'] - #file_name = file['name'] - try: - #shutil.copy(demisto.getFilePath(file_entry)['path'], file_name) - #with open(file_name, 'rb') as f: - #files = {'file': f} - - try: - res = requests.request(method, url, headers=headers, params=params, data=body, files=files, json=json, auth=(username, password)) - except requests.exceptions.ReadTimeout as e: - return "Readtimeout: %s" % e - except requests.exceptions.ConnectionError as e: - return "ConnectionError: %s" % e - - #shutil.rmtree(demisto.getFilePath(file_entry)['name'], ignore_errors=True) - except Exception as e: - return 'Failed to upload file - ' + str(e) - else: - try: - res = requests.request(method, url, headers=headers, data=json.dumps(body) if body else {}, json=json, params=params, auth=(username, password)) - except requests.exceptions.ReadTimeout as e: - return "Readtimeout: %s" % e - except requests.exceptions.ConnectionError as e: - return "ConnectionError: %s" % e - - try: - obj = res.json() - except Exception as e: - if not res.content: - return '' - return 'Error parsing reply - {} - {}'.format(res.content, str(e)) - - if 'error' in obj: - message = obj.get('error', {}).get('message') - details = obj.get('error', {}).get('detail') - if message == 'No Record found': - return { - # Return an empty results array - 'result': [] - } - return 'ServiceNow Error: {}, details: {}'.format(message, details) - - if res.status_code < 200 or res.status_code >= 300: - return 'Got status code {} with url {} with body {} with headers {}'.format(str(res.status_code), url, str(res.content), str(res.headers)) - - #print("RES: %s" % res) - #print("TEXT: %s" % res.text) - return res.text - - def get_ticket(self, url, username, password, table_name, sys_id, number=None): - path = None - query_params = {} # type: Dict - if sys_id: - path = "/api/now/v1/table/%s/%s" % (table_name, sys_id) - elif number: - path = '/api/now/v1/table/%s' % table_name - query_params = { - 'number': number - } - else: - # Only in cases where the table is of type ticket - return 'servicenow-get-ticket requires either ticket ID or ticket number' - - print("PATH: %s" % path) - return self.send_request(url, username, password, path, 'get', params=query_params) - - def list_table(self, url, username, password, table_name, limit=1): - query_params = { - "sysparm_limit": limit, - } - - #path = '/table/%s' % table_name - path = "/api/now/v1/table/%s" % table_name - - return self.send_request(url, username, password, path, 'get', params=query_params) - - def create_ticket(self, url, username, password, table_name, body, file_id=""): - if not isinstance(body, list) and not isinstance(body, object) and not isinstance(body, dict): - try: - data = json.loads(body) - except json.decoder.JSONDecodeError as e: - return {"success": False, "reason": e} - else: - data = body - - - path = "/api/now/v1/table/%s" % table_name - query_params = {} - base_request = self.send_request(url, username, password, path, 'post', params=query_params, json=data) - - if file_id: - tmp_file = self.get_file(file_id) - files = {'file': (tmp_file["filename"], tmp_file["data"])} - - try: - parsed_return = json.loads(base_request) - except: - print("[INFO] Failed parsed_return loading") - return base_request - - ticket_id = parsed_return["result"]["sys_id"] - params = { - "file_name": tmp_file["filename"], - "table_name": table_name, - "table_sys_id": ticket_id, - } - - filepath = "/api/now/v1/attachment/file" - file_request = self.send_request(url, username, password, filepath, 'post', params=params, files=files, headers={}) - print(file_request) - - return base_request - - def update_ticket(self, url, username, password, table_name, sys_id, body, file_id=""): - if not isinstance(body, list) and not isinstance(body, object) and not isinstance(body, dict): - try: - data = json.loads(body) - except json.decoder.JSONDecodeError as e: - return {"success": False, "reason": e} - else: - data = body - - - path = "/api/now/v1/table/%s/%s" % (table_name, sys_id) - query_params = {} - base_request = self.send_request(url, username, password, path, 'patch', params=query_params, json=data) - - if file_id: - tmp_file = self.get_file(file_id) - files = {'file': (tmp_file["filename"], tmp_file["data"])} - - try: - parsed_return = json.loads(base_request) - except: - print("[INFO] Failed parsed_return loading") - return base_request - - ticket_id = parsed_return["result"]["sys_id"] - params = { - "file_name": tmp_file["filename"], - "table_name": table_name, - "table_sys_id": ticket_id, - } - - filepath = "/api/now/v1/attachment/file" - file_request = self.send_request(url, username, password, filepath, '', params=params, files=files, headers={}) - print(file_request) - - return base_request - -# Run the actual thing after we've checked params -def run(request): - action = request.get_json() - print(action) - print(type(action)) - authorization_key = action.get("authorization") - current_execution_id = action.get("execution_id") - - if action and "name" in action and "app_name" in action: - Servicenow.run(action) - return f'Attempting to execute function {action["name"]} in app {action["app_name"]}' - else: - return f'Invalid action' - -if __name__ == "__main__": - Servicenow.run() diff --git a/shuffle-ai/1.0.0/api.yaml b/shuffle-ai/1.0.0/api.yaml index 40e0832c..ae0ae960 100644 --- a/shuffle-ai/1.0.0/api.yaml +++ b/shuffle-ai/1.0.0/api.yaml @@ -11,6 +11,67 @@ contact_info: url: https://shuffler.io email: support@shuffler.io actions: + - name: autoformat_text + description: Input ANY kind of data in the format you want, and the format you want it in. Default is a business-y email. Uses ShuffleGPT, which is based on OpenAI and our own model. + parameters: + - name: apikey + description: Your https://shuffler.io apikey + required: true + multiline: false + example: "" + schema: + type: string + - name: text + description: The text you want to be converted (ANY format) + required: true + multiline: true + example: "Bad IPs are 1.2.3.4 and there's no good way to format this. JSON works too!" + schema: + type: string + - name: formatting + description: The format to use. + required: false + multiline: true + example: "Make it work as a ticket we can put in service now that is human readable for security analysts" + schema: + type: string + returns: + schema: + type: string + - name: generate_report + description: Input ANY kind of data in the format you want, and it will make an HTML report for you. This can be downloaded from the File location. + parameters: + - name: apikey + description: Your https://shuffler.io apikey + required: true + multiline: false + example: "" + schema: + type: string + - name: input_data + description: The text you want to be converted (ANY format) + required: true + multiline: true + example: "Bad IPs are 1.2.3.4 and there's no good way to format this. JSON works too!" + schema: + type: string + - name: report_title + description: The report title to be used in the report + required: true + multiline: true + example: "Statistics for October" + schema: + type: string + - name: report_name + description: The name of the HTML file + required: false + multiline: true + example: "statistics.html" + schema: + type: string + returns: + schema: + type: string - name: extract_text_from_pdf description: Returns text from a pdf parameters: diff --git a/shuffle-ai/1.0.0/src/app.py b/shuffle-ai/1.0.0/src/app.py index 655f84bd..4a76c673 100644 --- a/shuffle-ai/1.0.0/src/app.py +++ b/shuffle-ai/1.0.0/src/app.py @@ -13,6 +13,89 @@ class Tools(AppBase): def __init__(self, redis, logger, console_logger=None): super().__init__(redis, logger, console_logger) + def autoformat_text(self, apikey, text, formatting="auto"): + headers = { + "Authorization": "Bearer %s" % apikey, + } + + if not formatting: + formatting = "auto" + + output_formatting= "Format the following data to be a good email that can be sent to customers. Don't make it too business sounding." + if formatting != "auto": + output_formatting = formatting + + ret = requests.post( + "https://shuffler.io/api/v1/conversation", + json={ + "query": text, + "formatting": output_formatting, + "output_format": "formatting" + }, + headers=headers, + ) + + if ret.status_code != 200: + print(ret.text) + return { + "success": False, + "reason": "Status code for auto-formatter is not 200" + } + + return ret.text + + def generate_report(self, apikey, input_data, report_title, report_name="generated_report.html"): + headers = { + "Authorization": "Bearer %s" % apikey, + } + + if not report_name: + report_name = "generated_report.html" + + if "." in report_name and not ".html" in report_name: + report_name = report_name.split(".")[0] + + if not "html" in report_name: + report_name = report_name + ".html" + + report_name = report_name.replace(" ", "_", -1) + + if not formatting: + formatting = "auto" + + output_formatting= "Format the following text into an HTML report with relevant graphs and tables. Title of the report should be {report_title}." + ret = requests.post( + "https://shuffler.io/api/v1/conversation", + json={ + "query": text, + "formatting": output_formatting, + "output_format": "formatting" + }, + headers=headers, + ) + + if ret.status_code != 200: + print(ret.text) + return { + "success": False, + "reason": "Status code for auto-formatter is not 200" + } + + # Make it into a shuffle file with self.set_files() + new_file = { + "name": report_name, + "data": ret.text, + } + + retdata = self.set_files([new_file]) + if retdata["success"]: + return retdata + + return { + "success": False, + "reason": "Failed to upload file" + } + def extract_text_from_pdf(self, file_id): def extract_pdf_text(pdf_path): @@ -105,7 +188,7 @@ def extract_text_from_image(self, file_id): image = Image.open(temp.name) image = image.resize((500,300)) custom_config = r'-l eng --oem 3 --psm 6' - text = pytesseract.image_to_string(image,config=custom_config + text = pytesseract.image_to_string(image,config=custom_config) data = { "success": True, diff --git a/shuffle-ai/1.0.0/upload.sh b/shuffle-ai/1.0.0/upload.sh new file mode 100755 index 00000000..33f84bac --- /dev/null +++ b/shuffle-ai/1.0.0/upload.sh @@ -0,0 +1,6 @@ + +gcloud run deploy shuffle-ai-1-0-0 \ + --region=europe-west2 \ + --max-instances=3 \ + --set-env-vars=SHUFFLE_APP_EXPOSED_PORT=8080,SHUFFLE_SWARM_CONFIG=run,SHUFFLE_LOGS_DISABLED=true --source=./ \ + --timeout=1800s diff --git a/shuffle-subflow/1.0.0/src/app.py b/shuffle-subflow/1.0.0/src/app.py index 2fc46527..82253179 100644 --- a/shuffle-subflow/1.0.0/src/app.py +++ b/shuffle-subflow/1.0.0/src/app.py @@ -172,6 +172,12 @@ def run_subflow(self, user_apikey, workflow, argument, source_workflow="", sourc else: print("No startnode") + if len(self.full_execution["execution_id"]) > 0 and self.full_execution["execution_id"] != source_execution: + params["source_execution"] = self.full_execution["execution_id"] + + if len(self.full_execution["authorization"]) > 0 and self.full_execution["authorization"] != source_auth: + params["source_auth"] = self.full_execution["authorization"] + if len(str(backend_url)) > 0: url = "%s/api/v1/workflows/%s/execute" % (backend_url, workflow) print("[INFO] Changed URL to %s for this execution" % url) diff --git a/shuffle-subflow/1.1.0/api.yaml b/shuffle-subflow/1.1.0/api.yaml index 4148c19a..0c64bda1 100644 --- a/shuffle-subflow/1.1.0/api.yaml +++ b/shuffle-subflow/1.1.0/api.yaml @@ -1,4 +1,4 @@ -app_version: 1.0.0 +app_version: 1.1.0 name: Shuffle Subflow description: The Shuffle Subflow app tags: diff --git a/shuffle-subflow/1.1.0/src/app.py b/shuffle-subflow/1.1.0/src/app.py index f1fa452a..5c4b4d8a 100644 --- a/shuffle-subflow/1.1.0/src/app.py +++ b/shuffle-subflow/1.1.0/src/app.py @@ -51,11 +51,11 @@ def run_userinput(self, user_apikey, sms="", email="", subflow="", information=" url = backend_url print("Found backend url: %s" % url) - print("AUTH: %s" % self.full_execution["authorization"]) + #print("AUTH: %s" % self.full_execution["authorization"]) #if len(information): # print("Should run arg: %s", information) - if len(subflow): + if len(subflow) > 0: #print("Should run subflow: %s", subflow) # Missing startnode (user input trigger) @@ -67,7 +67,8 @@ def run_userinput(self, user_apikey, sms="", email="", subflow="", information=" print("Should change port to 3001.") if "appspot.com" in frontend_url: frontend_url = "https://shuffler.io" - + if "run.app" in frontend_url: + frontend_url = "https://shuffler.io" for item in subflows: # In case of URL being passed, and not just ID @@ -79,10 +80,10 @@ def run_userinput(self, user_apikey, sms="", email="", subflow="", information=" argument = json.dumps({ "information": information, "parent_workflow": self.full_execution["workflow"]["id"], - "frontend_continue": "%s/workflows/%s/run?authorization=%s&reference_execution=%s&answer=true" % (frontend_url, self.full_execution["workflow"]["id"], self.full_execution["authorization"], self.full_execution["execution_id"]), - "frontend_abort": "%s/workflows/%s/run?authorization=%s&reference_execution=%s&answer=false" % (frontend_url, self.full_execution["workflow"]["id"], self.full_execution["authorization"], self.full_execution["execution_id"]), - "api_continue": "%s/api/v1/workflows/%s/execute?authorization=%s&reference_execution=%s&answer=true" % (frontend_url, self.full_execution["workflow"]["id"], self.full_execution["authorization"], self.full_execution["execution_id"]), - "api_abort": "%s/api/v1/workflows/%s/execute?authorization=%s&reference_execution=%s&answer=false" % (frontend_url, self.full_execution["workflow"]["id"], self.full_execution["authorization"], self.full_execution["execution_id"]), + "frontend_continue": "%s/workflows/%s/run?authorization=%s&reference_execution=%s&answer=true&source_node=%s" % (frontend_url, self.full_execution["workflow"]["id"], self.full_execution["authorization"], self.full_execution["execution_id"], source_node), + "frontend_abort": "%s/workflows/%s/run?authorization=%s&reference_execution=%s&answer=false&source_node=%s" % (frontend_url, self.full_execution["workflow"]["id"], self.full_execution["authorization"], self.full_execution["execution_id"], source_node), + "api_continue": "%s/api/v1/workflows/%s/execute?authorization=%s&reference_execution=%s&answer=true&source_node=%s" % (frontend_url, self.full_execution["workflow"]["id"], self.full_execution["authorization"], self.full_execution["execution_id"], source_node), + "api_abort": "%s/api/v1/workflows/%s/execute?authorization=%s&reference_execution=%s&answer=false&source_node=%s" % (frontend_url, self.full_execution["workflow"]["id"], self.full_execution["authorization"], self.full_execution["execution_id"], source_node), }) ret = self.run_subflow(user_apikey, item, argument, source_workflow=self.full_execution["workflow"]["id"], source_execution=self.full_execution["execution_id"], source_auth=self.full_execution["authorization"], startnode=startnode, backend_url=backend_url, source_node=source_node) @@ -106,7 +107,7 @@ def run_userinput(self, user_apikey, sms="", email="", subflow="", information=" print("Should run email with targets: %s", jsondata["targets"]) - ret = requests.post("%s/api/v1/functions/sendmail" % url, json=jsondata, headers=headers) + ret = requests.post("%s/api/v1/functions/sendmail" % url, json=jsondata, headers=headers, verify=False, proxies=self.proxy_config) if ret.status_code != 200: print("Failed sending email. Data: %s" % ret.text) result["email"] = False @@ -131,7 +132,7 @@ def run_userinput(self, user_apikey, sms="", email="", subflow="", information=" print("Should send sms with targets: %s", jsondata["numbers"]) - ret = requests.post("%s/api/v1/functions/sendsms" % url, json=jsondata, headers=headers) + ret = requests.post("%s/api/v1/functions/sendsms" % url, json=jsondata, headers=headers, verify=False, proxies=self.proxy_config) if ret.status_code != 200: print("Failed sending email. Data: %s" % ret.text) result["sms"] = False @@ -142,7 +143,7 @@ def run_userinput(self, user_apikey, sms="", email="", subflow="", information=" return json.dumps(result) - def run_subflow(self, user_apikey, workflow, argument, source_workflow="", source_execution="", source_node="", source_auth="", startnode="", backend_url=""): + def run_subflow(self, user_apikey, workflow, argument, source_workflow="", source_execution="", source_node="", source_auth="", startnode="", backend_url="", check_result=""): #print("STARTNODE: %s" % startnode) url = "%s/api/v1/workflows/%s/execute" % (self.url, workflow) if len(self.base_url) > 0: @@ -174,6 +175,12 @@ def run_subflow(self, user_apikey, workflow, argument, source_workflow="", sourc else: print("No startnode") + if len(self.full_execution["execution_id"]) > 0 and self.full_execution["execution_id"] != source_execution: + params["source_execution"] = self.full_execution["execution_id"] + + if len(self.full_execution["authorization"]) > 0 and self.full_execution["authorization"] != source_auth: + params["source_auth"] = self.full_execution["authorization"] + if len(str(backend_url)) > 0: url = "%s/api/v1/workflows/%s/execute" % (backend_url, workflow) print("[INFO] Changed URL to %s for this execution" % url) @@ -184,7 +191,7 @@ def run_subflow(self, user_apikey, workflow, argument, source_workflow="", sourc } if len(str(argument)) == 0: - ret = requests.post(url, headers=headers, params=params) + ret = requests.post(url, headers=headers, params=params, verify=False, proxies=self.proxy_config) else: if not isinstance(argument, list) and not isinstance(argument, dict): try: @@ -194,14 +201,14 @@ def run_subflow(self, user_apikey, workflow, argument, source_workflow="", sourc #print(f"ARG: {argument}") try: - ret = requests.post(url, headers=headers, params=params, json=argument) + ret = requests.post(url, headers=headers, params=params, json=argument, verify=False, proxies=self.proxy_config) print(f"Successfully sent argument of length {len(str(argument))} as JSON") except: try: - ret = requests.post(url, headers=headers, json=argument, params=params) + ret = requests.post(url, headers=headers, json=argument, params=params, verify=False, proxies=self.proxy_config) print("Successfully sent as JSON (2)") except: - ret = requests.post(url, headers=headers, data=argument, params=params) + ret = requests.post(url, headers=headers, data=argument, params=params, verify=False, proxies=self.proxy_config) print("Successfully sent as data (3)") print("Status: %d" % ret.status_code) diff --git a/shuffle-tools/1.2.0/api.yaml b/shuffle-tools/1.2.0/api.yaml index 50f19a25..cfa1a55b 100644 --- a/shuffle-tools/1.2.0/api.yaml +++ b/shuffle-tools/1.2.0/api.yaml @@ -1,7 +1,7 @@ --- app_version: 1.2.0 name: Shuffle Tools -description: A tool app for Shuffle. Gives access to most missing features along with Liquid. +description: A tool app for Shuffle. Gives access to most missing features along with Liquid. tags: - Testing - Shuffle @@ -25,6 +25,16 @@ actions: returns: schema: type: string + - name: execute_python + description: Runs python with the data input. Any prints will be returned. + parameters: + - name: code + description: The code to run. Can be a file ID from within Shuffle. + required: true + multiline: true + example: print("hello world") + schema: + type: string - name: check_cache_contains description: Checks Shuffle cache whether a user-provided key contains a value. Returns ALL the values previously appended. parameters: @@ -85,6 +95,19 @@ actions: returns: schema: type: string + - name: delete_cache_value + description: Delete a value saved to your organization in Shuffle + parameters: + - name: key + description: The key to delete + required: true + multiline: false + example: "timestamp" + schema: + type: string + returns: + schema: + type: string - name: send_sms_shuffle description: Send an SMS from Shuffle parameters: @@ -158,22 +181,22 @@ actions: skip_multicheck: true parameters: - name: input_list - description: The list to check + description: The list to filter from. Don't use .# into this. required: true multiline: false example: '[{"data": "1.2.3.4"}, {"data": "1.2.3.5"}]' schema: type: string - name: field - description: The field to check - required: false + description: The field to check in the input list + required: true multiline: false example: "data" schema: type: string - name: check description: Type of check - required: true + required: false example: "equals" options: - equals @@ -191,15 +214,15 @@ actions: schema: type: string - name: value - description: The value to check with - required: false + description: The value to compare with + required: false multiline: false example: "1.2.3.4" schema: type: string - name: opposite description: Whether to add or to NOT add - required: true + required: false options: - False - True @@ -245,7 +268,7 @@ actions: # schema: # type: string - name: parse_ioc - description: Parse IOC's based on https://github.com/fhightower/ioc-finder + description: "Parse IOC's based on https://github.com/fhightower/ioc-finder. Specify input type to optimize speed: domains, urls, email_addresses, ipv4s, ipv4_cidrs, md5s, sha256s, sha1s, cves and more.." parameters: - name: input_string description: The string to check @@ -258,7 +281,7 @@ actions: description: The string to check required: false multiline: false - example: "md5s" + example: "domains,urls,email_addresses,ipv4s,ipv4_cidrs,ipv6s,md5s,sha256s,sha1s,cves" schema: type: string returns: @@ -901,6 +924,7 @@ actions: options: - encode - decode + - "to image" schema: type: string - name: get_timestamp @@ -968,16 +992,7 @@ actions: example: '{"data": "Hello world"}' schema: type: string - - name: execute_python - description: Runs python with the data input. Any prints will be returned. - parameters: - - name: code - description: The code to run. Can be a file ID from within Shuffle. - required: true - multiline: true - example: print("hello world") - schema: - type: string + - name: run_math_operation description: Takes a math input and gives you the result parameters: @@ -1111,6 +1126,26 @@ actions: example: 'ls -la' schema: type: string + #- name: parse_ioc_new + # description: Parse IOC's based on https://github.com/fhightower/ioc-finder + # parameters: + # - name: input_string + # description: The string to check + # required: true + # multiline: true + # example: "123ijq192.168.3.6kljqwiejs8 https://shuffler.io" + # schema: + # type: string + # - name: input_type + # description: The string to check + # required: false + # multiline: false + # example: "md5s" + # schema: + # type: string + # returns: + # schema: + # type: string large_image:  # yamllint disable-line rule:line-length diff --git a/shuffle-tools/1.2.0/src/app.py b/shuffle-tools/1.2.0/src/app.py index 13cb7b4b..03019ed7 100644 --- a/shuffle-tools/1.2.0/src/app.py +++ b/shuffle-tools/1.2.0/src/app.py @@ -1,4 +1,4 @@ -import asyncio +import hmac import datetime import json import time @@ -13,8 +13,6 @@ import hashlib from io import StringIO from contextlib import redirect_stdout -from liquid import Liquid -import liquid import random import string @@ -36,6 +34,8 @@ import struct import paramiko +import concurrent.futures +import multiprocessing from walkoff_app_sdk.app_base import AppBase @@ -69,6 +69,30 @@ def base64_conversion(self, string, operation): encoded_string = str(encoded_bytes, "utf-8") return encoded_string + elif operation == "to image": + # Decode the base64 into an image and upload it as a file + decoded_bytes = base64.b64decode(string) + + # Make the bytes into unicode escaped bytes + # UnicodeDecodeError - 'utf-8' codec can't decode byte 0x89 in position 0: invalid start byte + try: + decoded_bytes = str(decoded_bytes, "utf-8") + except: + pass + + filename = "base64_image.png" + file = { + "filename": filename, + "data": decoded_bytes, + } + + fileret = self.set_files([file]) + value = {"success": True, "filename": filename, "file_id": fileret} + if len(fileret) == 1: + value = {"success": True, "filename": filename, "file_id": fileret[0]} + + return value + elif operation == "decode": try: decoded_bytes = base64.b64decode(string) @@ -145,7 +169,7 @@ def send_sms_shuffle(self, apikey, phone_numbers, body): url = "https://shuffler.io/api/v1/functions/sendsms" headers = {"Authorization": "Bearer %s" % apikey} - return requests.post(url, headers=headers, json=data).text + return requests.post(url, headers=headers, json=data, verify=False).text # This is an email function of Shuffle def send_email_shuffle(self, apikey, recipients, subject, body, attachments=""): @@ -181,7 +205,7 @@ def send_email_shuffle(self, apikey, recipients, subject, body, attachments=""): url = "https://shuffler.io/api/v1/functions/sendmail" headers = {"Authorization": "Bearer %s" % apikey} - return requests.post(url, headers=headers, json=data).text + return requests.post(url, headers=headers, json=data, verify=False).text def repeat_back_to_me(self, call): return call @@ -251,69 +275,16 @@ def parse(data): return "Invalid input" return return_value - # https://github.com/fhightower/ioc-finder - def parse_ioc(self, input_string, input_type="all"): - #if len(input_string) > 2500000 and (input_type == "" or input_type == "all"): - # return { - # "success": False, - # "reason": "Data too large (%d). Please reduce it below 2.5 Megabytes to use this action or specify the input type" % len(input_string) - # } - - # https://github.com/fhightower/ioc-finder/blob/6ff92a73a60e9233bf09b530ccafae4b4415b08a/ioc_finder/ioc_finder.py#L433 - ioc_types = ["domains", "urls", "email_addresses", "ipv6s", "ipv4s", "ipv4_cidrs", "md5s", "sha256s", "sha1s", "cves"] - input_string = str(input_string) - if input_type == "": - input_type = "all" - else: - input_type = input_type.split(",") - for item in input_type: - item = item.strip() - - ioc_types = input_type - - iocs = find_iocs(input_string, included_ioc_types=ioc_types) - newarray = [] - for key, value in iocs.items(): - if input_type != "all": - if key not in input_type: - continue - - if len(value) > 0: - for item in value: - # If in here: attack techniques. Shouldn't be 3 levels so no - # recursion necessary - if isinstance(value, dict): - for subkey, subvalue in value.items(): - if len(subvalue) > 0: - for subitem in subvalue: - data = { - "data": subitem, - "data_type": "%s_%s" % (key[:-1], subkey), - } - if data not in newarray: - newarray.append(data) - else: - data = {"data": item, "data_type": key[:-1]} - if data not in newarray: - newarray.append(data) - - # Reformatting IP - for item in newarray: - if "ip" in item["data_type"]: - item["data_type"] = "ip" - try: - item["is_private_ip"] = ipaddress.ip_address(item["data"]).is_private - except: - self.logger.info("Error parsing %s" % item["data"]) - + def parse_list(self, items, splitter="\n"): + # Check if it's already a list first try: - newarray = json.dumps(newarray) - except json.decoder.JSONDecodeError as e: - return "Failed to parse IOC's: %s" % e + newlist = json.loads(items) + if isinstance(newlist, list): + return newlist - return newarray + except Exception as e: + self.logger.info("[WARNING] Parse error - fallback: %s" % e) - def parse_list(self, items, splitter="\n"): if splitter == "": splitter = "\n" @@ -507,7 +478,7 @@ def regex_replace( return re.sub(regex, replace_string, input_data) def execute_python(self, code): - self.logger.info(f"Python code {len(code)} {code}. If uuid, we'll try to download and use the file.") + self.logger.info(f"Python code {len(code)}. If uuid, we'll try to download and use the file.") if len(code) == 36 and "-" in code: filedata = self.get_file(code) @@ -536,6 +507,14 @@ def custom_print(*args, **kwargs): # Add globals in it too globals_copy = globals().copy() globals_copy["print"] = custom_print + + # Add self to globals_copy + for key, value in locals().copy().items(): + if key not in globals_copy: + globals_copy[key] = value + + globals_copy["self"] = self + exec(code, globals_copy) s = f.getvalue() @@ -549,13 +528,19 @@ def custom_print(*args, **kwargs): try: return { "success": True, - "message": s.strip(), + "message": json.loads(s.strip()), } except Exception as e: - return { - "success": True, - "message": s, - } + try: + return { + "success": True, + "message": s.strip(), + } + except Exception as e: + return { + "success": True, + "message": s, + } except Exception as e: return { @@ -679,7 +664,7 @@ def filter_list(self, input_list, field, check, value, opposite): self.logger.info("Checklist and tmp: %s - %s" % (checklist, tmp)) found = False for subcheck in checklist: - subcheck = subcheck.strip() + subcheck = str(subcheck).strip() #ext.lower().strip() == value.lower().strip() if type(tmp) == list and subcheck in tmp: @@ -690,8 +675,17 @@ def filter_list(self, input_list, field, check, value, opposite): new_list.append(item) found = True break + elif type(tmp) == int and str(tmp) == subcheck: + new_list.append(item) + found = True + break else: - print("Nothing matching") + if str(tmp) == str(subcheck): + new_list.append(item) + found = True + break + else: + print("Nothing matching") if not found: failed_list.append(item) @@ -762,14 +756,62 @@ def filter_list(self, input_list, field, check, value, opposite): # CONTAINS FIND FOR LIST AND IN FOR STR elif check == "larger than": - if int(tmp) > int(value): - new_list.append(item) - else: + list_set = False + try: + if str(tmp).isdigit() and str(value).isdigit(): + if int(tmp) > int(value): + new_list.append(item) + list_set = True + except AttributeError as e: + self.logger.info("FAILED CHECKING LARGER THAN: %s" % e) + pass + + try: + value = len(json.loads(value)) + except Exception as e: + self.logger.info(f"[WARNING] Failed to convert destination to list: {e}") + + try: + # Check if it's a list in autocast and if so, check the length + if len(json.loads(tmp)) > int(value): + new_list.append(item) + list_set = True + except Exception as e: + self.logger.info(f"[WARNING] Failed to check if larger than as list: {e}") + + if not list_set: failed_list.append(item) elif check == "less than": - if int(tmp) < int(value): - new_list.append(item) - else: + # Old + #if int(tmp) < int(value): + # new_list.append(item) + #else: + # failed_list.append(item) + + list_set = False + try: + if str(tmp).isdigit() and str(value).isdigit(): + if int(tmp) < int(value): + new_list.append(item) + list_set = True + except AttributeError as e: + self.logger.info("FAILED CHECKING LARGER THAN: %s" % e) + pass + + try: + value = len(json.loads(value)) + except Exception as e: + self.logger.info(f"[WARNING] Failed to convert destination to list: {e}") + + try: + # Check if it's a list in autocast and if so, check the length + if len(json.loads(tmp)) < int(value): + new_list.append(item) + list_set = True + except Exception as e: + self.logger.info(f"[WARNING] Failed to check if larger than as list: {e}") + + if not list_set: failed_list.append(item) elif check == "in cache key": @@ -912,6 +954,7 @@ def get_file_meta(self, file_id): "%s/api/v1/files/%s?execution_id=%s" % (self.url, file_id, self.current_execution_id), headers=headers, + verify=False, ) self.logger.info(f"RET: {ret}") @@ -928,6 +971,7 @@ def delete_file(self, file_id): "%s/api/v1/files/%s?execution_id=%s" % (self.url, file_id, self.current_execution_id), headers=headers, + verify=False, ) return ret.text @@ -1456,7 +1500,7 @@ def merge_lists(self, list_one, list_two, set_field="", sort_key_list_one="", so return list_one def merge_json_objects(self, list_one, list_two, set_field="", sort_key_list_one="", sort_key_list_two=""): - self.merge_lists(self, list_one, list_two, set_field="", sort_key_list_one="", sort_key_list_two="") + return self.merge_lists(list_one, list_two, set_field=set_field, sort_key_list_one=sort_key_list_one, sort_key_list_two=sort_key_list_two) def fix_json(self, json_data): try: @@ -1705,7 +1749,7 @@ def check_cache_contains(self, key, value, append): else: append = False - get_response = requests.post(url, json=data) + get_response = requests.post(url, json=data, verify=False) try: allvalues = get_response.json() try: @@ -1720,7 +1764,7 @@ def check_cache_contains(self, key, value, append): data["value"] = json.dumps(new_value) set_url = "%s/api/v1/orgs/%s/set_cache" % (self.url, org_id) - set_response = requests.post(set_url, json=data) + set_response = requests.post(set_url, json=data, verify=False) try: allvalues = set_response.json() #allvalues["key"] = key @@ -1753,37 +1797,47 @@ def check_cache_contains(self, key, value, append): if allvalues["value"] == None or allvalues["value"] == "null": allvalues["value"] = "[]" + allvalues["value"] = str(allvalues["value"]) + try: parsedvalue = json.loads(allvalues["value"]) except json.decoder.JSONDecodeError as e: - parsedvalue = [] + parsedvalue = [str(allvalues["value"])] + except Exception as e: + print("Error parsing JSON - overriding: %s" % e) + parsedvalue = [str(allvalues["value"])] - #return parsedvalue + print("In ELSE2: '%s'" % parsedvalue) - for item in parsedvalue: - #return "%s %s" % (item, value) - if item == value: - if not append: - return { - "success": True, - "found": True, - "reason": "Found and not appending!", - "key": key, - "search": value, - "value": json.loads(allvalues["value"]), - } - else: - return { - "success": True, - "found": True, - "reason": "Found, was appending, but item already exists", - "key": key, - "search": value, - "value": json.loads(allvalues["value"]), - } - - # Lol - break + try: + for item in parsedvalue: + #return "%s %s" % (item, value) + if item == value: + if not append: + return { + "success": True, + "found": True, + "reason": "Found and not appending!", + "key": key, + "search": value, + "value": json.loads(allvalues["value"]), + } + else: + return { + "success": True, + "found": True, + "reason": "Found, was appending, but item already exists", + "key": key, + "search": value, + "value": json.loads(allvalues["value"]), + } + + # Lol + break + except Exception as e: + print("Error in check_cache_contains: %s" % e) + parsedvalue = [str(parsedvalue)] + append = True if not append: return { @@ -1795,24 +1849,15 @@ def check_cache_contains(self, key, value, append): "value": json.loads(allvalues["value"]), } - #parsedvalue = json.loads(allvalues["value"]) - #if parsedvalue == None: - # parsedvalue = [] - - #return parsedvalue new_value = parsedvalue if new_value == None: new_value = [value] new_value.append(value) - - #return new_value - data["value"] = json.dumps(new_value) - #return allvalues set_url = "%s/api/v1/orgs/%s/set_cache" % (self.url, org_id) - response = requests.post(set_url, json=data) + response = requests.post(set_url, json=data, verify=False) exception = "" try: allvalues = response.json() @@ -1843,10 +1888,11 @@ def check_cache_contains(self, key, value, append): #return allvalues except Exception as e: + print("[ERROR] Failed to handle cache contains: %s" % e) return { "success": False, "key": key, - "reason": f"Failed to get cache: {e}", + "reason": f"Failed to handle cache contains. Is the original value a list?: {e}", "search": value, "found": False, } @@ -1884,7 +1930,7 @@ def change_cache_subkey(self, key, subkey, value, overwrite): "value": value, } - response = requests.post(url, json=data) + response = requests.post(url, json=data, verify=False) try: allvalues = response.json() allvalues["key"] = key @@ -1904,6 +1950,9 @@ def change_cache_subkey(self, key, subkey, value, overwrite): self.logger.info("Value couldn't be parsed") return response.text + def delete_cache_value(self, key): + return self.delete_cache(key) + def get_cache_value(self, key): org_id = self.full_execution["workflow"]["execution_org"]["id"] url = "%s/api/v1/orgs/%s/get_cache" % (self.url, org_id) @@ -1915,14 +1964,14 @@ def get_cache_value(self, key): "key": key, } - value = requests.post(url, json=data) + value = requests.post(url, json=data, verify=False) try: allvalues = value.json() - self.logger.info("VAL1: ", allvalues) + #self.logger.info("VAL1: ", allvalues) allvalues["key"] = key - self.logger.info("VAL2: ", allvalues) + #self.logger.info("VAL2: ", allvalues) - if allvalues["success"] == True: + if allvalues["success"] == True and len(allvalues["value"]) > 0: allvalues["found"] = True else: allvalues["success"] = True @@ -1964,7 +2013,7 @@ def set_cache_value(self, key, value): "value": value, } - response = requests.post(url, json=data) + response = requests.post(url, json=data, verify=False) try: allvalues = response.json() allvalues["key"] = key @@ -2091,7 +2140,7 @@ def run_oauth_request(self, url, jwt): data = "grant_type=urn%3Aietf%3Aparams%3Aoauth%3Agrant-type%3Ajwt-bearer&assertion=%s" % jwt - return requests.post(url, data=data, headers=headers).text + return requests.post(url, data=data, headers=headers, verify=False).text # Based on https://google-auth.readthedocs.io/en/master/reference/google.auth.crypt.html def get_jwt_from_file(self, file_id, jwt_audience, scopes, complete_request=True): @@ -2386,7 +2435,7 @@ def run_ssh_command(self, host, port, user_name, private_key_file_id, password, except Exception as e: return {"success":"false","message":str(e)} else: - print("AUTH WITH PASSWORD") + #print("AUTH WITH PASSWORD") try: ssh_client.connect(hostname=host,username=user_name,port=port, password=str(password)) except Exception as e: @@ -2399,6 +2448,213 @@ def run_ssh_command(self, host, port, user_name, private_key_file_id, password, return {"success":"true","output": stdout.read().decode(errors='ignore')} + def parse_ioc(self, input_string, input_type="all"): + ioc_types = ["domains", "urls", "email_addresses", "ipv4s", "ipv4_cidrs", "md5s", "sha256s", "sha1s", "cves"] + + # Remember overriding ioc types we care about + if input_type == "" or input_type == "all": + input_type = "all" + else: + input_type = input_type.split(",") + for item in input_type: + item = item.strip() + + ioc_types = input_type + + iocs = find_iocs(str(input_string), included_ioc_types=ioc_types) + newarray = [] + for key, value in iocs.items(): + if input_type != "all": + if key not in input_type: + continue + + if len(value) > 0: + for item in value: + # If in here: attack techniques. Shouldn't be 3 levels so no + # recursion necessary + if isinstance(value, dict): + for subkey, subvalue in value.items(): + if len(subvalue) > 0: + for subitem in subvalue: + data = { + "data": subitem, + "data_type": "%s_%s" % (key[:-1], subkey), + } + if data not in newarray: + newarray.append(data) + else: + data = {"data": item, "data_type": key[:-1]} + if data not in newarray: + newarray.append(data) + + # Reformatting IP + for item in newarray: + if "ip" in item["data_type"]: + item["data_type"] = "ip" + try: + item["is_private_ip"] = ipaddress.ip_address(item["data"]).is_private + except: + self.logger.info("Error parsing %s" % item["data"]) + + try: + newarray = json.dumps(newarray) + except json.decoder.JSONDecodeError as e: + return "Failed to parse IOC's: %s" % e + + return newarray + + + def split_text(self, text): + # Split text into chunks of 10kb. Add each 10k to array + # In case e.g. 1.2.3.4 lands exactly on 20k boundary, it may be useful to overlap here. + # (just shitty code to reduce chance of issues) while still going fast + arr_one = [] + max_len = 5000 + current_string = "" + overlaps = 100 + + for i in range(0, len(text)): + current_string += text[i] + if len(current_string) > max_len: + # Appending just in case even with overlaps + if len(text) > i+overlaps: + current_string += text[i+1:i+overlaps] + else: + current_string += text[i+1:] + + arr_one.append(current_string) + current_string = "" + + if len(current_string) > 0: + arr_one.append(current_string) + + return arr_one + + def _format_result(self, result): + final_result = {} + + for res in result: + for key,val in res.items(): + if key in final_result: + if isinstance(val, list) and len(val) > 0: + for i in val: + final_result[key].append(i) + elif isinstance(val, dict): + #print(key,":::",val) + if key in final_result: + if isinstance(val, dict): + for k,v in val.items(): + #print("k:",k,"v:",v) + val[k].append(v) + #print(val) + #final_result[key].append([i for i in val if len(val) > 0]) + else: + final_result[key] = val + + return final_result + + # See function for how it works~: parse_ioc_new(..) + def _with_concurency(self, array_of_strings, ioc_types): + results = [] + #start = time.perf_counter() + + # Workers dont matter..? + # What can we use instead? + print("Strings:", len(array_of_strings)) + + workers = 4 + with concurrent.futures.ThreadPoolExecutor(max_workers=workers) as executor: + # Submit the find_iocs function for each string in the array + futures = [executor.submit( + find_iocs, + text=string, + included_ioc_types=ioc_types, + ) for string in array_of_strings] + + # Wait for all tasks to complete + concurrent.futures.wait(futures) + + # Retrieve the results if needed + results = [future.result() for future in futures] + + #print("Total time taken:", time.perf_counter()-start) + return self._format_result(results) + + # FIXME: Make this good and actually faster than normal + # For now: Concurrency doesn't make it faster due to GIL in python. + # May need to offload this to an executable or something + def parse_ioc_new(self, input_string, input_type="all"): + if input_type == "": + input_type = "all" + + ioc_types = ["domains", "urls", "email_addresses", "ipv4s", "ipv4_cidrs", "md5s", "sha256s", "sha1s", "cves"] + + if input_type == "" or input_type == "all": + ioc_types = ioc_types + else: + input_type = input_type.split(",") + for item in input_type: + item = item.strip() + + ioc_types = input_type + + input_string = str(input_string) + + if len(input_string) > 10000: + iocs = self._with_concurency(self.split_text(input_string), ioc_types=ioc_types) + else: + iocs = find_iocs(input_string, included_ioc_types=ioc_types) + + newarray = [] + for key, value in iocs.items(): + if input_type != "all": + if key not in input_type: + continue + + if len(value) == 0: + continue + + for item in value: + # If in here: attack techniques. Shouldn't be 3 levels so no + # recursion necessary + if isinstance(value, dict): + for subkey, subvalue in value.items(): + if len(subvalue) == 0: + continue + + for subitem in subvalue: + data = { + "data": subitem, + "data_type": "%s_%s" % (key[:-1], subkey), + } + + if data not in newarray: + newarray.append(data) + else: + data = {"data": item, "data_type": key[:-1]} + if data not in newarray: + newarray.append(data) + + # Reformatting IP + i = -1 + for item in newarray: + i += 1 + if "ip" not in item["data_type"]: + continue + + newarray[i]["data_type"] = "ip" + try: + newarray[i]["is_private_ip"] = ipaddress.ip_address(item["data"]).is_private + except Exception as e: + print("Error parsing %s: %s" % (item["data"], e)) + + try: + newarray = json.dumps(newarray) + except json.decoder.JSONDecodeError as e: + return "Failed to parse IOC's: %s" % e + + return newarray + if __name__ == "__main__": Tools.run() diff --git a/shuffle-tools/1.2.0/src/concurrency.py b/shuffle-tools/1.2.0/src/concurrency.py new file mode 100644 index 00000000..420d1686 --- /dev/null +++ b/shuffle-tools/1.2.0/src/concurrency.py @@ -0,0 +1,201 @@ +import time +import json +import ipaddress +import concurrent.futures +from functools import partial +from ioc_finder import find_iocs + +class Test(): + def split_text(self, text): + # Split text into chunks of 10kb. Add each 10k to array + # In case e.g. 1.2.3.4 lands exactly on 20k boundary, it may be useful to overlap here. + # (just shitty code to reduce chance of issues) while still going fast + + arr_one = [] + max_len = 2500 + current_string = "" + overlaps = 100 + + + for i in range(0, len(text)): + current_string += text[i] + if len(current_string) > max_len: + # Appending just in case even with overlaps + if len(text) > i+overlaps: + current_string += text[i+1:i+overlaps] + else: + current_string += text[i+1:] + + arr_one.append(current_string) + current_string = "" + + if len(current_string) > 0: + arr_one.append(current_string) + + #print("DATA:", arr_one) + print("Strings:", len(arr_one)) + #exit() + + return arr_one + + def _format_result(self, result): + final_result = {} + + for res in result: + for key, val in res.items(): + if key in final_result: + if isinstance(val, list) and len(val) > 0: + for i in val: + final_result[key].append(i) + elif isinstance(val, dict): + #print(key,":::",val) + if key in final_result: + if isinstance(val, dict): + for k,v in val.items(): + #print("k:",k,"v:",v) + val[k].append(v) + #print(val) + #final_result[key].append([i for i in val if len(val) > 0]) + else: + final_result[key] = val + + return final_result + + def worker_function(self, inputdata): + return find_iocs(inputdata["data"], included_ioc_types=inputdata["ioc_types"]) + + def _with_concurency(self, array_of_strings, ioc_types): + results = [] + #start = time.perf_counter() + + # Workers dont matter..? + # What can we use instead? + + results = [] + workers = 4 + with concurrent.futures.ThreadPoolExecutor(max_workers=workers) as executor: + # Submit the find_iocs function for each string in the array + futures = [executor.submit( + find_iocs, + text=string, + included_ioc_types=ioc_types, + ) for string in array_of_strings] + + # Wait for all tasks to complete + concurrent.futures.wait(futures) + + # Retrieve the results if needed + results = [future.result() for future in futures] + + return self._format_result(results) + + def parse_ioc_new(self, input_string, input_type="all"): + if input_type == "": + input_type = "all" + + #ioc_types = ["domains", "urls", "email_addresses", "ipv6s", "ipv4s", "ipv4_cidrs", "md5s", "sha256s", "sha1s", "cves"] + ioc_types = ["domains", "urls", "email_addresses", "ipv4s", "ipv4_cidrs", "md5s", "sha256s", "sha1s", "cves"] + + # urls = 10.4 -> 9.1 + # emails = 10.4 -> 9.48 + # ipv6s = 10.4 -> 7.37 + # ipv4 cidrs = 10.4 -> 10.44 + + if input_type == "" or input_type == "all": + ioc_types = ioc_types + else: + input_type = input_type.split(",") + for item in input_type: + item = item.strip() + + ioc_types = input_type + + input_string = str(input_string) + if len(input_string) > 10000: + iocs = self._with_concurency(self.split_text(input_string), ioc_types=ioc_types) + else: + iocs = find_iocs(input_string, included_ioc_types=ioc_types) + + newarray = [] + for key, value in iocs.items(): + if input_type != "all": + if key not in input_type: + continue + + if len(value) == 0: + continue + + for item in value: + # If in here: attack techniques. Shouldn't be 3 levels so no + # recursion necessary + if isinstance(value, dict): + for subkey, subvalue in value.items(): + if len(subvalue) == 0: + continue + + for subitem in subvalue: + data = { + "data": subitem, + "data_type": "%s_%s" % (key[:-1], subkey), + } + + if data not in newarray: + newarray.append(data) + else: + data = {"data": item, "data_type": key[:-1]} + if data not in newarray: + newarray.append(data) + + # Reformatting IP + i = -1 + for item in newarray: + i += 1 + if "ip" not in item["data_type"]: + continue + + newarray[i]["data_type"] = "ip" + try: + newarray[i]["is_private_ip"] = ipaddress.ip_address(item["data"]).is_private + except Exception as e: + print("Error parsing %s: %s" % (item["data"], e)) + + try: + newarray = json.dumps(newarray) + except json.decoder.JSONDecodeError as e: + return "Failed to parse IOC's: %s" % e + + return newarray + +# Make it not run this for multithreads +if __name__ == "__main__": + + input_string = "" + with open("testdata.txt", "r") as f: + input_string = f.read() + + try: + json_data = json.loads(input_string) + # If array, loop + if isinstance(json_data, list): + cnt = 0 + start = time.perf_counter() + for item in json_data: + cnt += 1 + classdata = Test() + + ret = classdata.parse_ioc_new(item) + #print("OUTPUT1: ", ret) + + #if cnt == 5: + # break + + print("Total time taken:", time.perf_counter()-start) + else: + classdata = Test() + ret = classdata.parse_ioc_new(input_string) + print("OUTPUT2: ", ret) + except Exception as e: + classdata = Test() + ret = classdata.parse_ioc_new(json_data) + print("OUTPUT3: ", ret) +