diff --git a/Packs/CommonScripts/Scripts/IsIntegrationAvailable/README.md b/Packs/CommonScripts/Scripts/IsIntegrationAvailable/README.md index ee132bb08d24..3a6a70787f22 100644 --- a/Packs/CommonScripts/Scripts/IsIntegrationAvailable/README.md +++ b/Packs/CommonScripts/Scripts/IsIntegrationAvailable/README.md @@ -59,7 +59,7 @@ MITRE ATT&CK CoA - T1135 - Network Share Discovery MITRE ATT&CK CoA - T1083 - File and Directory Discovery Cyren Inbox Security Default Get User Devices by Email Address - Generic -FireEye ETP - Indicators Hunting +Trellix Email Security Cloud - Indicators Hunting FireEye HX - Isolate Endpoint MITRE ATT&CK CoA - T1133 - External Remote Services FireEye HX - Execution Flow Indicators Hunting diff --git a/Packs/CommonScripts/Scripts/ProvidesCommand/TestData/integration_search.json b/Packs/CommonScripts/Scripts/ProvidesCommand/TestData/integration_search.json index c3a44b1fcef1..faa966122dcd 100644 --- a/Packs/CommonScripts/Scripts/ProvidesCommand/TestData/integration_search.json +++ b/Packs/CommonScripts/Scripts/ProvidesCommand/TestData/integration_search.json @@ -1 +1 @@ -[{"ModuleName": "Demisto REST API_instance_1", "Brand": "Demisto REST API", "Category": "Utilities", "ID": "", "Version": 0, "Type": 1, "Contents": {"response": {"configurations": [{"brand": "", "canGetSamples": true, "category": "Data Enrichment & Threat Intelligence", "commitMessage": "", "configuration": [{"defaultValue": "https://api.abuseipdb.com/api/v2/", "display": "AbuseIP server URL", "hidden": false, "name": "server", "options": null, "required": true, "type": 0}, {"defaultValue": "", "display": "API Key (v2)", "hidden": false, "name": "apikey", "options": null, "required": true, "type": 4}, {"defaultValue": "80", "display": "Minimum score threshold", "hidden": false, "name": "threshold", "options": null, "required": false, "type": 0}, {"defaultValue": "30", "display": "Maximum reports age (in days)", "hidden": false, "name": "days", "options": null, "required": false, "type": 0}, {"defaultValue": "", "display": "Disregard quota errors", "hidden": false, "name": "disregard_quota", "options": null, "required": false, "type": 8}, {"defaultValue": "", "display": "Trust any certificate (not secure)", "hidden": false, "name": "insecure", "options": null, "required": false, "type": 8}, {"defaultValue": "", "display": "Use system proxy settings", "hidden": false, "name": "proxy", "options": null, "required": false, "type": 8}], "description": "Central repository to report and identify IP addresses that have been associated with malicious activity online. Check the Detailed Information section for more information on how to configure the integration.", "detailedDescription": "- Minimum score threshold \n- Minimum score from AbuseIPDB for the IP to be considered malicious. (Must be above 20). \n \n \n - Searching an entire subnet far back in time (e.g. Days=365) will likely cause a server timeout,\n and you will receive a 500 Internal Server Error response from AbuseIPDB server.\n \n \n \n - [Report Categories](https://www.abuseipdb.com/categories).\n", "display": "AbuseIPDB", "hidden": false, "icon": "", "id": "AbuseIPDB", "image": "", "integrationScript": {"commands": [{"arguments": [{"default": true, "deprecated": false, "description": "The IP address to check.", "name": "ip", "required": false, "secret": false}, {"default": false, "defaultValue": "30", "deprecated": false, "description": "The time range to return reports (in days). Default is 30.", "name": "days", "required": false, "secret": false}, {"auto": "PREDEFINED", "default": false, "defaultValue": "true", "deprecated": false, "description": "The length of the report. \"true\" returns the full report, \"false\" does not return reported categories. Default is \"true\".", "name": "verbose", "predefined": ["true", "false"], "required": false, "secret": false}, {"default": false, "defaultValue": "80", "deprecated": false, "description": "The minimum score from AbuseIPDB to consider whether the IP address is malicious (must be greater than 20). Default is 80.", "name": "threshold", "required": false, "secret": false}], "cartesian": false, "deprecated": false, "description": "Checks the specified IP address against the AbuseIP database.", "execution": false, "hidden": false, "important": null, "name": "ip", "outputs": [{"contentPath": "", "contextPath": "IP.Address", "description": "The address of the IP.", "type": "String"}, {"contentPath": "", "contextPath": "IP.Geo.Country", "description": "The country in which the IP address is located.", "type": "String"}, {"contentPath": "", "contextPath": "IP.Malicious.Vendor", "description": "The vendor reporting the IP address as malicious.", "type": "String"}, {"contentPath": "", "contextPath": "IP.Malicious.Detections", "description": "The Detections that led to the verdict.", "type": "String"}, {"contentPath": "", "contextPath": "AbuseIPDB.IP.Address", "description": "The IP address fetched from AbuseIPDB.", "type": "String"}, {"contentPath": "", "contextPath": "AbuseIPDB.IP.AbuseConfidenceScore", "description": "The confidence score fetched from AbuseIPDB.", "type": "String"}, {"contentPath": "", "contextPath": "AbuseIPDB.IP.TotalReports", "description": "The number of times the address has been reported.", "type": "Number"}, {"contentPath": "", "contextPath": "AbuseIPDB.IP.Geo.Country", "description": "The country associated with the IP Address.", "type": "String"}, {"contentPath": "", "contextPath": "AbuseIPDB.IP.Reports", "description": "The reports summary (for \"verbose\" reports).", "type": "String"}, {"contentPath": "", "contextPath": "DBotScore.Score", "description": "The actual score.", "type": "Number"}, {"contentPath": "", "contextPath": "DBotScore.Vendor", "description": "The vendor used to calculate the score.", "type": "String"}, {"contentPath": "", "contextPath": "DBotScore.Indicator", "description": "The indicator that was tested.", "type": "String"}, {"contentPath": "", "contextPath": "DBotScore.Type", "description": "The indicator type.", "type": "String"}, {"contentPath": "", "contextPath": "AbuseIPDB.IP.Malicious.Vendor", "description": "The vendor that determined this IP address to be malicious.", "type": "String"}, {"contentPath": "", "contextPath": "AbuseIPDB.IP.Malicious.Detections", "description": "The Detections that led to the verdict.", "type": "String"}, {"contentPath": "", "contextPath": "IP.Malicious.Description", "description": "A description explaining why the IP address was reported as malicious.", "type": "String"}, {"contentPath": "", "contextPath": "AbuseIPDB.IP.Malicious.Description", "description": "A description explaining why the IP address was reported as malicious.", "type": "String"}], "permitted": false, "sensitive": false, "timeout": 0}, {"arguments": [{"default": false, "deprecated": false, "description": "IPv4 Address Block in CIDR notation.", "name": "network", "required": true, "secret": false}, {"default": false, "defaultValue": "30", "deprecated": false, "description": "The time range to return reports (in days). Default is 30.", "name": "days", "required": false, "secret": false}, {"default": false, "defaultValue": "40", "deprecated": false, "description": "The maximum number of IPs to check. Default is 40.", "name": "limit", "required": false, "secret": false}, {"default": false, "defaultValue": "80", "deprecated": false, "description": "The minimum score from AbuseIPDB to consider whether the IP address is malicious (must be greater than 20). Default is 80.", "name": "threshold", "required": false, "secret": false}], "cartesian": false, "deprecated": false, "description": "Queries a block of IP addresses to check against the database.", "execution": false, "hidden": false, "important": null, "name": "abuseipdb-check-cidr-block", "outputs": [{"contentPath": "", "contextPath": "IP.Address", "description": "The IP address.", "type": "String"}, {"contentPath": "", "contextPath": "IP.Geo.Country", "description": "The country in which the IP address is located.", "type": "String"}, {"contentPath": "", "contextPath": "IP.Malicious.Vendor", "description": "The vendor reporting the IP address as malicious.", "type": "String"}, {"contentPath": "", "contextPath": "IP.Malicious.Detections", "description": "The Detections that led to the verdict.", "type": "String"}, {"contentPath": "", "contextPath": "AbuseIPDB.IP.Address", "description": "The IP address fetched from AbuseIPDB.", "type": "String"}, {"contentPath": "", "contextPath": "AbuseIPDB.IP.AbuseConfidenceScore", "description": "The confidence score fetched from AbuseIPDB.", "type": "Unknown"}, {"contentPath": "", "contextPath": "AbuseIPDB.IP.TotalReports", "description": "The number of times this address has been reported.", "type": "Unknown"}, {"contentPath": "", "contextPath": "AbuseIPDB.IP.Geo.Country", "description": "The country associated with this IP Address.", "type": "Unknown"}, {"contentPath": "", "contextPath": "AbuseIPDB.IP.Reports", "description": "Reports summary (for \"verbose\" reports).", "type": "Unknown"}, {"contentPath": "", "contextPath": "DBotScore.Score", "description": "The actual score.", "type": "Number"}, {"contentPath": "", "contextPath": "DBotScore.Vendor", "description": "The vendor used to calculate the score.", "type": "String"}, {"contentPath": "", "contextPath": "DBotScore.Indicator", "description": "The indicator that was tested.", "type": "String"}, {"contentPath": "", "contextPath": "DBotScore.Type", "description": "The indicator type.", "type": "String"}, {"contentPath": "", "contextPath": "AbuseIPDB.IP.Malicious.Vendor", "description": "The vendor used to calculate the score.", "type": "String"}, {"contentPath": "", "contextPath": "AbuseIPDB.IP.Malicious.Detections", "description": "The Detections that led to the verdict.", "type": "String"}, {"contentPath": "", "contextPath": "IP.Malicious.Description", "description": "A description explaining why the IP address was reported as malicious.", "type": "String"}], "permitted": false, "sensitive": false, "timeout": 0}, {"arguments": [{"default": false, "deprecated": false, "description": "The IP address to report.", "name": "ip", "required": true, "secret": false}, {"default": false, "deprecated": false, "description": "A CSV list of category IDs. For more information, see https://www.abuseipdb.com/categories.", "name": "categories", "required": true, "secret": false}], "cartesian": false, "deprecated": false, "description": "Reports an IP address to AbuseIPDB.", "execution": false, "hidden": false, "important": null, "name": "abuseipdb-report-ip", "outputs": null, "permitted": false, "sensitive": false, "timeout": 0}, {"arguments": [{"default": false, "defaultValue": "30", "deprecated": false, "description": "The time range to return reports (in days). Default is 30.", "name": "days", "required": false, "secret": false}, {"default": false, "defaultValue": "50", "deprecated": false, "description": "The maximum number of IPs to retrieve. Default is 50. ", "name": "limit", "required": false, "secret": false}, {"auto": "PREDEFINED", "default": false, "defaultValue": "false", "deprecated": false, "description": "Whether to save a list of blacklisted IPs in the Context Data in Demisto. Default is false.", "name": "saveToContext", "predefined": ["true", "false"], "required": false, "secret": false}], "cartesian": false, "deprecated": false, "description": "Returns a list of the most reported IP addresses.", "execution": false, "hidden": false, "important": null, "name": "abuseipdb-get-blacklist", "outputs": [{"contentPath": "", "contextPath": "AbuseIPDB.Blacklist", "description": "A list of blacklisted IP addresses.", "type": "Unknown"}], "permitted": false, "sensitive": false, "timeout": 0}, {"arguments": [], "cartesian": false, "deprecated": false, "description": "Returns a list of report categories from AbuseIPDB.", "execution": false, "hidden": false, "important": null, "name": "abuseipdb-get-categories", "outputs": [{"contentPath": "", "contextPath": "AbuseIPDB.Categories", "description": "The list of AbuseIPDB categories.", "type": "string"}], "permitted": false, "sensitive": false, "timeout": 0}], "isFetch": false, "isFetchCredentials": false, "longRunning": false, "longRunningPortMapping": false, "runOnce": false, "script": "\n\n''' IMPORTS '''\nimport requests\nimport os\nimport csv\n\n# disable insecure warnings\nrequests.packages.urllib3.disable_warnings()\n\n''' GLOBALS '''\nVERBOSE = True\nSERVER = demisto.params().get('server')\nif not SERVER.endswith('/'):\n SERVER += '/'\nAPI_KEY = demisto.params().get('apikey')\nMAX_AGE = demisto.params().get('days')\nTHRESHOLD = demisto.params().get('threshold')\nINSECURE = demisto.params().get('insecure')\nTEST_IP = \"127.0.0.2\"\nBLACKLIST_SCORE = 3\nCHECK_CMD = \"check\"\nCHECK_BLOCK_CMD = \"check-block\"\nREPORT_CMD = \"report\"\nBLACKLIST_CMD = 'blacklist'\nANALYSIS_TITLE = \"AbuseIPDB Analysis\"\nBLACKLIST_TITLE = \"AbuseIPDB Blacklist\"\nREPORT_SUCCESS = \"IP address reported successfully.\"\n\nHEADERS = {\n 'Key': API_KEY,\n 'Accept': 'application/json'\n}\n\nPROXY = demisto.params().get('proxy')\nif not demisto.params().get('proxy', False):\n del os.environ['HTTP_PROXY']\n del os.environ['HTTPS_PROXY']\n del os.environ['http_proxy']\n del os.environ['https_proxy']\n\nCATEGORIES_NAME = {\n 3: 'Frad_Orders',\n 4: 'DDoS_Attack',\n 5: 'FTP_Brute-Force',\n 6: 'Ping of Death',\n 7: 'Phishing',\n 8: 'Fraud VoIP',\n 9: 'Open_Proxy',\n 10: 'Web_Spam',\n 11: 'Email_Spam',\n 12: 'Blog_Spam',\n 13: 'VPN IP',\n 14: 'Port_Scan',\n 15: 'Hacking',\n 16: 'SQL Injection',\n 17: 'Spoofing',\n 18: 'Brute_Force',\n 19: 'Bad_Web_Bot',\n 20: 'Exploited_Host',\n 21: 'Web_App_Attack',\n 22: 'SSH',\n 23: 'IoT_Targeted'\n}\n\nCATEGORIES_ID = {\n \"Frad_Orders\": \"3\",\n \"DDoS_Attack\": \"4\",\n \"FTP_Brute\": \"5\",\n \"Ping of Death\": \"6\",\n \"Phishing\": \"7\",\n \"Fraud VoIP\": \"8\",\n \"Open_Proxy\": \"9\",\n \"Web_Spam\": \"10\",\n \"Email_Spam\": \"11\",\n \"Blog_Spam\": \"12\",\n \"VPN IP\": \"13\",\n \"Port_Scan\": \"14\",\n \"Hacking\": \"15\",\n \"SQL Injection\": \"16\",\n \"Spoofing\": \"17\",\n \"Brute_Force\": \"18\",\n \"Bad_Web_Bot\": \"19\",\n \"Exploited_Host\": \"20\",\n \"Web_App_Attack\": \"21\",\n \"SSH\": \"22\",\n \"IoT_Targeted\": \"23\"\n}\n\nsession = requests.session()\n\n\n''' HELPER FUNCTIONS '''\n\n\ndef http_request(method, url_suffix, params=None, headers=HEADERS, threshold=THRESHOLD):\n LOG('running request with url=%s' % (SERVER + url_suffix))\n try:\n analysis = session.request(method, SERVER + url_suffix, headers=headers, params=params, verify=not INSECURE)\n\n if analysis.status_code not in {200, 204, 429}:\n return_error('Bad connection attempet. Status code: ' + str(analysis.status_code))\n if analysis.status_code == 429:\n if demisto.params().get('disregard_quota'):\n return 'Too many requests (possibly bad API key). Status code: ' + str(analysis.status_code)\n else:\n return_error('Too many requests (possibly bad API key). Status code: ' + str(analysis.status_code))\n\n return REPORT_SUCCESS if url_suffix == REPORT_CMD else analysis.json()\n except Exception as e:\n LOG(e)\n return_error(e.message)\n\n\ndef analysis_to_entry(info, threshold=THRESHOLD, verbose=VERBOSE):\n if not isinstance(info, list):\n info = [info]\n\n context_ip_generic, context_ip, human_readable, dbot_scores = [], [], [], []\n for analysis in info:\n ip_ec = {\n \"Address\": analysis.get(\"ipAddress\"),\n \"Geo\": {\"Country\": analysis.get(\"countryName\") or analysis.get(\"countryCode\")}\n }\n abuse_ec = {\n \"IP\": {\n \"Address\": analysis.get(\"ipAddress\"),\n \"Geo\": {\"Country\": analysis.get(\"countryName\") or analysis.get(\"countryCode\")},\n 'AbuseConfidenceScore': analysis.get('abuseConfidenceScore'),\n \"TotalReports\": analysis.get(\"totalReports\") or analysis.get(\"numReports\") or \"0\"\n }\n }\n\n if verbose:\n reports = sum([report_dict.get(\"categories\") for report_dict in analysis.get(\"reports\")], []) # type: list\n categories = set(reports)\n abuse_ec[\"IP\"][\"Reports\"] = {CATEGORIES_NAME[c]: reports.count(c) for c in categories}\n\n human_readable.append(abuse_ec['IP'])\n\n dbot_score = getDBotScore(analysis, threshold)\n if dbot_score == 3:\n ip_ec[\"Malicious\"] = abuse_ec[\"IP\"][\"Malicious\"] = {\n 'Vendor': \"AbuseIPDB\",\n 'Detections': 'The address was reported as Malicious by AbuseIPDB.',\n 'Description': 'The address was reported as Malicious by AbuseIPDB.'\n\n }\n dbot_scores.append({\n \"Score\": dbot_score,\n \"Vendor\": \"AbuseIPDB\",\n \"Indicator\": analysis.get(\"ipAddress\"),\n \"Type\": \"ip\"\n })\n context_ip.append(abuse_ec)\n context_ip_generic.append(ip_ec)\n\n return createEntry(context_ip, context_ip_generic, human_readable, dbot_scores, title=ANALYSIS_TITLE)\n\n\ndef blacklist_to_entry(data, saveToContext):\n if not isinstance(data, list):\n data = [data]\n\n ips = [d.get(\"ipAddress\") for d in data]\n context = {\"Blacklist\": ips}\n temp = demisto.uniqueFile()\n with open(demisto.investigation()['id'] + '_' + temp, 'wb') as f:\n wr = csv.writer(f, quoting=csv.QUOTE_ALL)\n for ip in ips:\n wr.writerow([ip])\n entry = {\n 'HumanReadable': '',\n 'Contents': ips,\n 'ContentsFormat': formats['json'],\n 'Type': entryTypes['file'],\n 'File': \"Blacklist.csv\",\n 'FileID': temp,\n 'EntryContext': {'AbuseIPDB': createContext(context if saveToContext else None, removeNull=True)}\n }\n return entry\n\n\ndef getDBotScore(analysis, threshold=THRESHOLD):\n total_reports = analysis.get(\"totalReports\") or analysis.get(\"numReports\") or 0\n abuse_score = int(analysis.get(\"abuseConfidenceScore\"))\n dbot_score = 0 if total_reports == 0 else 1 if abuse_score < 20 else 2 if abuse_score < int(threshold) else 3\n return dbot_score\n\n\ndef createEntry(context_ip, context_ip_generic, human_readable, dbot_scores, title):\n entry = {\n 'ContentsFormat': formats['json'],\n 'Type': entryTypes['note'],\n 'Contents': context_ip,\n 'ReadableContentsFormat': formats['markdown'],\n 'HumanReadable': tableToMarkdown(title, human_readable, removeNull=True),\n 'EntryContext': {\n 'IP(val.Address && val.Address == obj.Address)': createContext(context_ip_generic, removeNull=True),\n 'AbuseIPDB(val.IP.Address && val.IP.Address == obj.IP.Address)': createContext(context_ip, removeNull=True),\n 'DBotScore': createContext(dbot_scores, removeNull=True)\n }\n }\n return entry\n\n\n''' FUNCTIONS '''\n\n\ndef check_ip_command(ip, days=MAX_AGE, verbose=VERBOSE, threshold=THRESHOLD):\n params = {\n \"ipAddress\": ip,\n \"maxAgeInDays\": days\n }\n if verbose:\n params['verbose'] = \"verbose\"\n analysis = http_request(\"GET\", url_suffix=CHECK_CMD, params=params).get(\"data\")\n return analysis_to_entry(analysis, verbose=verbose, threshold=threshold)\n\n\ndef check_block_command(network, limit, days=MAX_AGE, threshold=THRESHOLD):\n params = {\n \"network\": network,\n \"maxAgeInDays\": days\n }\n analysis = http_request(\"GET\", url_suffix=CHECK_BLOCK_CMD, params=params).get(\"data\").get(\"reportedAddress\")\n return analysis_to_entry(analysis[:int(limit) if limit.isdigit() else 40], verbose=False, threshold=threshold)\n\n\ndef report_ip_command(ip, categories):\n params = {\n \"ip\": ip,\n \"categories\": \",\".join([CATEGORIES_ID[c] if c in CATEGORIES_ID else c for c in categories.split()])\n }\n analysis = http_request(\"POST\", url_suffix=REPORT_CMD, params=params)\n return analysis\n\n\ndef get_blacklist_command(limit, days, saveToContext):\n params = {\n 'maxAgeInDays': days,\n \"limit\": limit\n }\n analysis = http_request(\"GET\", url_suffix=BLACKLIST_CMD, params=params)\n return analysis if type(analysis) is str else blacklist_to_entry(analysis.get(\"data\"), saveToContext)\n\n\ndef test_module():\n try:\n check_ip_command(ip=TEST_IP, verbose=False)\n except Exception as e:\n LOG(e)\n return_error(e.message)\n demisto.results('ok')\n\n\ndef get_categories_command():\n categories = {str(key): value for key, value in CATEGORIES_NAME.items()}\n entry = {\n 'ContentsFormat': formats['json'],\n 'Type': entryTypes['note'],\n 'Contents': categories,\n 'ReadableContentsFormat': formats['markdown'],\n 'HumanReadable': tableToMarkdown(\"AbuseIPDB report categories\", categories, removeNull=True),\n 'EntryContext': {'AbuseIPDB.Categories(val && val == obj)': createContext(categories, removeNull=True),\n }\n }\n return entry\n\n\ntry:\n if demisto.command() == 'test-module':\n # Tests connectivity and credentails on login\n test_module()\n elif demisto.command() == 'ip':\n demisto.results(check_ip_command(**demisto.args()))\n elif demisto.command() == 'abuseipdb-check-cidr-block':\n demisto.results(check_block_command(**demisto.args()))\n elif demisto.command() == 'abuseipdb-report-ip':\n demisto.results(report_ip_command(**demisto.args()))\n elif demisto.command() == 'abuseipdb-get-blacklist':\n demisto.results(get_blacklist_command(**demisto.args()))\n elif demisto.command() == 'abuseipdb-get-categories':\n demisto.results(get_categories_command(**demisto.args())) # type:ignore\n\nexcept Exception as e:\n LOG.print_log()\n return_error(e.message)", "subtype": "python2", "type": "python"}, "modified": "2019-10-18T15:51:41.34248721-04:00", "name": "AbuseIPDB", "prevName": "AbuseIPDB", "shouldCommit": false, "sortValues": null, "system": true, "vcShouldIgnore": false, "version": 6}, {"brand": "", "canGetSamples": false, "category": "Authentication", "commitMessage": "", "configuration": [{"defaultValue": "", "display": "Server IP (e.g. 192.168.0.1)", "hidden": false, "name": "LDAPServer", "options": null, "required": true, "type": 0}, {"defaultValue": "389", "display": "Port", "hidden": false, "name": "LDAPServerPort", "options": null, "required": false, "type": 0}, {"defaultValue": "", "display": "", "hidden": false, "name": "Credentials", "options": null, "required": true, "type": 9}, {"defaultValue": "", "display": "Base DN (e.g. DC=domain,DC=com)", "hidden": false, "name": "LDAPBaseDN", "options": null, "required": true, "type": 0}, {"defaultValue": "", "display": "Default Domain", "hidden": false, "name": "LDAPDefaultDomain", "options": null, "required": false, "type": 0}, {"defaultValue": "none", "display": "Security Type (none/ssl/tls)", "hidden": false, "name": "LDAPSecurity", "options": null, "required": true, "type": 0}, {"defaultValue": "", "display": "Do not validate server certificate (insecure)", "hidden": false, "name": "LDAPSInsecure", "options": null, "required": false, "type": 8}, {"defaultValue": "true", "display": "Auto populate groups", "hidden": false, "name": "LDAPSFetchGroups", "options": null, "required": false, "type": 8}, {"defaultValue": "", "display": "Full path for CA certificate of Active Directory server (type pem)", "hidden": false, "name": "LDAPCaCertFile", "options": null, "required": false, "type": 0}], "description": "Authenticate using Active Directory", "display": "Active Directory Authentication", "hidden": false, "icon": "activedirectory.png", "id": "activedir-login", "image": "", "integrationScript": null, "modified": "2019-10-28T16:18:38.469114945-04:00", "name": "activedir-login", "prevName": "activedir-login", "shouldCommit": false, "sortValues": null, "vcShouldIgnore": false, "version": 57}, {"brand": "", "canGetSamples": false, "category": "Data Enrichment & Threat Intelligence", "commitMessage": "", "configuration": [{"defaultValue": "", "display": "Server IP (e.g. 192.168.0.1)", "hidden": false, "name": "LDAPServer", "options": null, "required": true, "type": 0}, {"defaultValue": "389", "display": "Port", "hidden": false, "name": "LDAPServerPort", "options": null, "required": false, "type": 0}, {"defaultValue": "", "display": "", "hidden": false, "name": "Credentials", "options": null, "required": true, "type": 9}, {"defaultValue": "", "display": "Base DN (e.g. DC=domain,DC=com)", "hidden": false, "name": "LDAPBaseDN", "options": null, "required": true, "type": 0}, {"defaultValue": "none", "display": "Security Type (none/ssl/tls)", "hidden": false, "name": "LDAPSecurity", "options": null, "required": true, "type": 0}, {"defaultValue": "500", "display": "Page Size", "hidden": false, "name": "LDAPPageSize", "options": null, "required": false, "type": 0}, {"defaultValue": "", "display": "Do not validate server certificate (insecure)", "hidden": false, "name": "LDAPSInsecure", "options": null, "required": false, "type": 8}, {"defaultValue": "", "display": "Full path for CA certificate of Active Directory server (type pem)", "hidden": false, "name": "LDAPCaCertFile", "options": null, "required": false, "type": 0}], "deprecated": true, "description": "Query LDAP directory servers", "display": "Active Directory Query (Deprecated)", "hidden": false, "icon": "activedirectory.png", "id": "activedir", "image": "", "integrationScript": null, "isPasswordProtected": true, "modified": "2019-10-28T16:18:38.46569023-04:00", "name": "activedir", "prevName": "activedir", "shouldCommit": false, "sortValues": null, "vcShouldIgnore": false, "version": 57}, {"brand": "", "canGetSamples": true, "category": "Data Enrichment & Threat Intelligence", "commitMessage": "", "configuration": [{"defaultValue": "", "display": "Server IP address (e.g., 192.168.0.1)", "hidden": false, "name": "server_ip", "options": null, "required": true, "type": 0}, {"defaultValue": "", "display": "Port. If not specified, default the port is 389, or 636 for LDAPS.", "hidden": false, "name": "port", "options": null, "required": false, "type": 0}, {"defaultValue": "", "display": "Credentials", "hidden": false, "name": "credentials", "options": null, "required": true, "type": 9}, {"defaultValue": "", "display": "NTLM authentication", "hidden": false, "name": "ntlm", "options": null, "required": false, "type": 8}, {"defaultValue": "", "display": "Base DN (for example \"dc=company,dc=com\")", "hidden": false, "name": "base_dn", "options": null, "required": true, "type": 0}, {"defaultValue": "500", "display": "Page size", "hidden": false, "name": "page_size", "options": null, "required": true, "type": 0}, {"defaultValue": "SSL", "display": "Secure Connection", "hidden": false, "name": "secure_connection", "options": ["None", "SSL"], "required": true, "type": 15}, {"defaultValue": "", "display": "Trust any certificate (not secure)", "hidden": false, "name": "unsecure", "options": null, "required": false, "type": 8}], "description": "Active Directory Query integration enables you to access and manage Active Directory objects (users, contacts, and computers).", "detailedDescription": "Active Directory search uses paging. You set the page size by specifying the 'page size' parameter in the instance settings.\n", "display": "Active Directory Query v2", "hidden": false, "icon": "", "id": "Active Directory Query v2", "image": "", "integrationScript": {"commands": [{"arguments": [{"default": false, "deprecated": false, "description": "The username (samAccountName) of the user to modify", "name": "username", "required": true, "secret": false}, {"default": false, "deprecated": false, "description": "Root (e.g., DC=domain,DC=com)", "name": "base-dn", "required": false, "secret": false}], "cartesian": false, "deprecated": false, "description": "Expires the password of an Active Directory user.", "execution": false, "hidden": false, "important": null, "name": "ad-expire-password", "outputs": null, "permitted": false, "sensitive": false, "timeout": 0}, {"arguments": [{"default": false, "deprecated": false, "description": "The username (samAccountName) of the user to be modified", "name": "username", "required": true, "secret": false}, {"default": false, "deprecated": false, "description": "The initial password to set for the user. The user will be asked to change the password after login.", "name": "password", "required": true, "secret": false}, {"default": false, "deprecated": false, "description": "The user's DN", "name": "user-dn", "required": true, "secret": false}, {"default": false, "deprecated": false, "description": "The user's display name", "name": "display-name", "required": false, "secret": false}, {"default": false, "deprecated": false, "description": "Short description of the user", "name": "description", "required": false, "secret": false}, {"default": false, "deprecated": false, "description": "User email.", "name": "email", "required": false, "secret": false}, {"default": false, "deprecated": false, "description": "The user's telephone number", "name": "telephone-number", "required": false, "secret": false}, {"default": false, "deprecated": false, "description": "The user's job title", "name": "title", "required": false, "secret": false}, {"default": false, "deprecated": false, "description": "Sets basic or custom attributes of the user object. For example, custom-attributes=\"{\\\"notes\\\":\\\"a note about the contact\\\",\\\"company\\\":\\\"company name\\\"}\"", "name": "custom-attributes", "required": false, "secret": false}], "cartesian": false, "deprecated": false, "description": "Creates an Active Directory user. This command requires a secure connection (SSL,TLS). in order to use this command.", "execution": false, "hidden": false, "important": null, "name": "ad-create-user", "outputs": null, "permitted": false, "sensitive": false, "timeout": 0}, {"arguments": [{"default": false, "deprecated": false, "description": "Enables you to define search criteria in the Query Active Directory using Active Directory syntax. For example, the following query searches for all user objects, except Andy: \"(&(objectCategory=person)(objectClass=user)(!(cn=andy)))\". NOTE if you have special characters such as \"*\",\"(\",or \"\\\" the character must be preceded by two backslashes \"\\\\\". For example, to use \"*\", type \"\\\\*\". For more information about search filters, see syntax: https://docs.microsoft.com/en-us/windows/win32/adsi/search-filter-syntax", "name": "filter", "required": true, "secret": false}, {"default": false, "deprecated": false, "description": "Root (e.g. DC=domain,DC=com). By default, the Base DN configured for the instance will be used.", "name": "base-dn", "required": false, "secret": false}, {"default": false, "deprecated": false, "description": "CSV list of the object attributes to return, e.g., \"dn,memberOf\". To get all objects atributes, specify 'ALL'.", "name": "attributes", "required": false, "secret": false}, {"default": false, "defaultValue": "50", "deprecated": false, "description": "Maximum number of records to return", "name": "size-limit", "required": false, "secret": false}, {"default": false, "deprecated": false, "description": "Maximum time to pull records (in seconds)", "name": "time-limit", "required": false, "secret": false}, {"auto": "PREDEFINED", "default": false, "defaultValue": "yes", "deprecated": false, "description": "If set to 'no' will not output the the results of the search to the context.", "name": "context-output", "predefined": ["yes", "no"], "required": false, "secret": false}], "cartesian": false, "deprecated": false, "description": "Runs Active Directory queries.", "execution": false, "hidden": false, "important": null, "name": "ad-search", "outputs": [{"contentPath": "", "contextPath": "ActiveDirectory.Search.dn", "description": "The distinguished names that match the query.", "type": "string"}, {"contentPath": "", "contextPath": "ActiveDirectory.Search", "description": "Result of the search.", "type": "unknown"}], "permitted": false, "sensitive": false, "timeout": 0}, {"arguments": [{"default": false, "deprecated": false, "description": "The username of the user to add to the group. If this argument is not specified, the computer name argument must be specified.\t", "name": "username", "required": false, "secret": false}, {"default": false, "deprecated": false, "description": "The name of the computer to add to the group. If this argument is not specified, the username argument must be specified.", "name": "computer-name", "required": false, "secret": false}, {"default": false, "deprecated": false, "description": "The name of the group to add the user to", "name": "group-cn", "required": true, "secret": false}, {"default": false, "deprecated": false, "description": "Root (e.g., DC=domain,DC=com). By default, the Base DN configured for the instance will be used.", "name": "base-dn", "required": false, "secret": false}], "cartesian": false, "deprecated": false, "description": "Adds an Active Directory user or computer to a group.", "execution": false, "hidden": false, "important": null, "name": "ad-add-to-group", "outputs": null, "permitted": false, "sensitive": false, "timeout": 0}, {"arguments": [{"default": false, "deprecated": false, "description": "The name of the user to remove from the group. If this argument is not specified, the computer name argument must be specified.\t", "name": "username", "required": false, "secret": false}, {"default": false, "deprecated": false, "description": "The name of the computer to remove from the group. If this argument is not specified, the username argument must be specified.", "name": "computer-name", "required": false, "secret": false}, {"default": false, "deprecated": false, "description": "The name of the group to remove the user from\t", "name": "group-cn", "required": true, "secret": false}, {"default": false, "deprecated": false, "description": "Root (e.g., DC=domain,DC=com). By default, the Base DN configured for the instance will be used.", "name": "base-dn", "required": false, "secret": false}], "cartesian": false, "deprecated": false, "description": "Removes an Active Directory user or computer from a group.", "execution": false, "hidden": false, "important": null, "name": "ad-remove-from-group", "outputs": null, "permitted": false, "sensitive": false, "timeout": 0}, {"arguments": [{"default": false, "deprecated": false, "description": "The username of the account to update (sAMAccountName)\t", "name": "username", "required": true, "secret": false}, {"default": false, "deprecated": false, "description": "The name of the attribute to modify (e.g., sn, displayName, mail, etc.)", "name": "attribute-name", "required": true, "secret": false}, {"default": false, "deprecated": false, "description": "The value the attribute should be changed to\t", "name": "attribute-value", "required": true, "secret": false}, {"default": false, "deprecated": false, "description": "Root (e.g. DC=domain,DC=com). By default, the Base DN configured for the instance will be used.", "name": "base-dn", "required": false, "secret": false}], "cartesian": false, "deprecated": false, "description": "Updates attributes of an existing Active Directory user.", "execution": false, "hidden": false, "important": null, "name": "ad-update-user", "outputs": null, "permitted": false, "sensitive": false, "timeout": 0}, {"arguments": [{"default": false, "deprecated": false, "description": "The DN of the user to delete", "name": "user-dn", "required": true, "secret": false}], "cartesian": false, "deprecated": false, "description": "Deletes an Active Directory user.", "execution": false, "hidden": false, "important": null, "name": "ad-delete-user", "outputs": null, "permitted": false, "sensitive": false, "timeout": 0}, {"arguments": [{"default": false, "deprecated": false, "description": "The contact's DN\t", "name": "contact-dn", "required": true, "secret": false}, {"default": false, "deprecated": false, "description": "The contact's display name\t", "name": "display-name", "required": false, "secret": false}, {"default": false, "deprecated": false, "description": "Short description of the contact", "name": "description", "required": false, "secret": false}, {"default": false, "deprecated": false, "description": "The contact's email address", "name": "email", "required": false, "secret": false}, {"default": false, "deprecated": false, "description": "The contact's telephone number", "name": "telephone-number", "required": false, "secret": false}, {"default": false, "deprecated": false, "description": "Sets basic or custom attributes of the contact object. For example, custom-attributes=\"{\\\"notes\\\":\\\"some note about the contact\\\",\\\"company\\\":\\\"some company\\\"}\"", "name": "custom-attributes", "required": false, "secret": false}, {"default": false, "deprecated": false, "description": "The contact's job title", "name": "title", "required": false, "secret": false}], "cartesian": false, "deprecated": false, "description": "Creates an Active Directory contact.", "execution": false, "hidden": false, "important": null, "name": "ad-create-contact", "outputs": null, "permitted": false, "sensitive": false, "timeout": 0}, {"arguments": [{"default": false, "deprecated": false, "description": "The contact's DN\t", "name": "contact-dn", "required": true, "secret": false}, {"default": false, "deprecated": false, "description": "The attribute name to update\t", "name": "attribute-name", "required": true, "secret": false}, {"default": false, "deprecated": false, "description": "The attribute value to be updated\t", "name": "attribute-value", "required": true, "secret": false}], "cartesian": false, "deprecated": false, "description": "Updates attributes of an existing Active Directory contact.", "execution": false, "hidden": false, "important": null, "name": "ad-update-contact", "outputs": null, "permitted": false, "sensitive": false, "timeout": 0}, {"arguments": [{"default": false, "deprecated": false, "description": "The username of the account to disable (sAMAccountName)\t", "name": "username", "required": true, "secret": false}, {"default": false, "deprecated": false, "description": "Root (e.g., DC=domain,DC=com). By default, the Base DN configured for the instance will be used.", "name": "base-dn", "required": false, "secret": false}], "cartesian": false, "deprecated": false, "description": "Disables an Active Directory user account.", "execution": false, "hidden": false, "important": null, "name": "ad-disable-account", "outputs": null, "permitted": false, "sensitive": false, "timeout": 0}, {"arguments": [{"default": false, "deprecated": false, "description": "The username of the account to enable (sAMAccountName)\t", "name": "username", "required": true, "secret": false}, {"default": false, "deprecated": false, "description": "Root (e.g., DC=domain,DC=com). By default, the Base DN configured for the instance will be used.", "name": "base-dn", "required": false, "secret": false}], "cartesian": false, "deprecated": false, "description": "Enables a previously disabled Active Directory account.", "execution": false, "hidden": false, "important": null, "name": "ad-enable-account", "outputs": null, "permitted": false, "sensitive": false, "timeout": 0}, {"arguments": [{"default": false, "deprecated": false, "description": "The username of the account to unlock (sAMAccountName)\t", "name": "username", "required": true, "secret": false}, {"default": false, "deprecated": false, "description": "Root (e.g., DC=domain,DC=com). By default, the Base DN configured for the instance will be used.", "name": "base-dn", "required": false, "secret": false}], "cartesian": false, "deprecated": false, "description": "Unlocks a previously locked Active Directory user account.", "execution": false, "hidden": false, "important": null, "name": "ad-unlock-account", "outputs": null, "permitted": false, "sensitive": false, "timeout": 0}, {"arguments": [{"default": false, "deprecated": false, "description": "The username of the account to be disabled (sAMAccountName)\t", "name": "username", "required": true, "secret": false}, {"default": false, "deprecated": false, "description": "The password to set for the user\t", "name": "password", "required": true, "secret": false}, {"default": false, "deprecated": false, "description": "Root (e.g. DC=domain,DC=com). Base DN configured for the instance will be used as default.", "name": "base-dn", "required": false, "secret": false}], "cartesian": false, "deprecated": false, "description": "Sets a new password for an Active Directory user. This command requires a secure connection (SSL,TLS).", "execution": false, "hidden": false, "important": null, "name": "ad-set-new-password", "outputs": null, "permitted": false, "sensitive": false, "timeout": 0}, {"arguments": [{"default": false, "deprecated": false, "description": "The computer name\t", "name": "computer-name", "required": true, "secret": false}, {"default": false, "deprecated": false, "description": "Superior DN, e.g., OU=computers,DC=domain,DC=com (The specified domain must be the same as the current computer domain)", "name": "full-superior-dn", "required": false, "secret": false}], "cartesian": false, "deprecated": false, "description": "Modifies the computer organizational unit within a domain.", "execution": false, "hidden": false, "important": null, "name": "ad-modify-computer-ou", "outputs": null, "permitted": false, "sensitive": false, "timeout": 0}, {"arguments": [{"default": true, "deprecated": false, "description": "Query by the user's Active Directory Distinguished Name", "name": "dn", "required": false, "secret": false}, {"default": false, "deprecated": false, "description": "Query by the user's name", "name": "name", "required": false, "secret": false}, {"default": false, "deprecated": false, "description": "Include these AD attributes of the resulting objects in addition to the default attributes", "name": "attributes", "required": false, "secret": false}, {"default": false, "deprecated": false, "description": "Query users by this custom field type", "name": "custom-field-type", "required": false, "secret": false}, {"default": false, "deprecated": false, "description": "Query users by this custom field data (relevant only if the `custom-field-type` argument is provided)", "name": "custom-field-data", "required": false, "secret": false}, {"default": false, "deprecated": false, "description": "Query users by the samAccountName attribute", "name": "username", "required": false, "secret": false}, {"default": false, "deprecated": false, "description": "Maximum number of objects to return (default is 20)", "name": "limit", "required": false, "secret": false}, {"default": false, "deprecated": false, "description": "Query by the user's email address", "name": "email", "required": false, "secret": false}, {"auto": "PREDEFINED", "default": false, "defaultValue": "false", "deprecated": false, "description": "Include verbose translation for UserAccountControl flags", "name": "user-account-control-out", "predefined": ["true", "false"], "required": false, "secret": false}], "cartesian": false, "deprecated": false, "description": "Retrieves detailed information about a user account. The user can be specified by name, email address, or as an Active Directory Distinguished Name (DN). If no filter is specified, all users are returned.", "execution": false, "hidden": false, "important": null, "name": "ad-get-user", "outputs": [{"contentPath": "", "contextPath": "ActiveDirectory.Users.dn", "description": "The user's distinguished name", "type": "string"}, {"contentPath": "", "contextPath": "ActiveDirectory.Users.displayName", "description": "The user's display name", "type": "string"}, {"contentPath": "", "contextPath": "ActiveDirectory.Users.name", "description": "The user's common name", "type": "string"}, {"contentPath": "", "contextPath": "ActiveDirectory.Users.sAMAccountName", "description": "The user's sAMAccountName", "type": "string"}, {"contentPath": "", "contextPath": "ActiveDirectory.Users.userAccountControl", "description": "The user's account control flag", "type": "number"}, {"contentPath": "", "contextPath": "ActiveDirectory.Users.mail", "description": "The user's email address", "type": "string"}, {"contentPath": "", "contextPath": "ActiveDirectory.Users.manager", "description": "The user's manager", "type": "string"}, {"contentPath": "", "contextPath": "ActiveDirectory.Users.memberOf", "description": "Groups the user is member of", "type": "string"}, {"contentPath": "", "contextPath": "Account.DisplayName", "description": "The user's display name", "type": "string"}, {"contentPath": "", "contextPath": "Account.Groups", "description": "Groups the user is member of", "type": "string"}, {"contentPath": "", "contextPath": "Account.Manager", "description": "The user's manager", "type": "string"}, {"contentPath": "", "contextPath": "Account.ID", "description": "The user's distinguished name", "type": "string"}, {"contentPath": "", "contextPath": "Account.Username", "description": "The user's samAccountName", "type": "string"}, {"contentPath": "", "contextPath": "Account.Email", "description": "The user's email address", "type": "string"}], "permitted": false, "sensitive": false, "timeout": 0}, {"arguments": [{"default": false, "deprecated": false, "description": "The computer's DN", "name": "dn", "required": false, "secret": false}, {"default": false, "deprecated": false, "description": "Name of the computer to get information for", "name": "name", "required": false, "secret": false}, {"default": false, "deprecated": false, "description": "Include these AD attributes of the resulting objects in addition to the default attributes", "name": "attributes", "required": false, "secret": false}, {"default": false, "deprecated": false, "description": "Search computers by this custom field data (relevant only if the `customFieldType` argument is provided)", "name": "custom-field-data", "required": false, "secret": false}, {"default": false, "deprecated": false, "description": "Search computer by this custom field type", "name": "custom-field-type", "required": false, "secret": false}], "cartesian": false, "deprecated": false, "description": "Retrieves detailed information about a computer account. The computer can be specified by name, email address, or as an Active Directory Distinguished Name (DN). If no filters are provided, all computers are returned. ", "execution": false, "hidden": false, "important": null, "name": "ad-get-computer", "outputs": [{"contentPath": "", "contextPath": "ActiveDirectory.Computers.dn", "description": "The computer distinguished name", "type": ""}, {"contentPath": "", "contextPath": "ActiveDirectory.Computers.memberOf", "description": "Groups the computer is listed as a member", "type": ""}, {"contentPath": "", "contextPath": "ActiveDirectory.Computers.name", "description": "The computer name", "type": ""}, {"contentPath": "", "contextPath": "Endpoint.ID", "description": "The computer DN", "type": ""}, {"contentPath": "", "contextPath": "Endpoint.Hostname", "description": "The computer name", "type": ""}, {"contentPath": "", "contextPath": "Endpoint.Groups", "description": "Groups the computer is listed as a member of", "type": ""}], "permitted": false, "sensitive": false, "timeout": 0}, {"arguments": [{"default": false, "deprecated": false, "description": "Group's Active Directory Distinguished Name", "name": "group-dn", "required": true, "secret": false}, {"auto": "PREDEFINED", "default": false, "defaultValue": "person", "deprecated": false, "description": "Which members type to query", "name": "member-type", "predefined": ["person", "computer"], "required": true, "secret": false}, {"default": false, "deprecated": false, "description": "CSV list of attributes to include in the results, in addition to the default attributes", "name": "attributes", "required": false, "secret": false}], "cartesian": false, "deprecated": false, "description": "Retrieves the list of users or computers that are members of the specified group", "execution": false, "hidden": false, "important": null, "name": "ad-get-group-members", "outputs": [{"contentPath": "", "contextPath": "ActiveDirectory.Groups.dn", "description": "The group DN", "type": "string"}, {"contentPath": "", "contextPath": "ActiveDirectory.Groups.members.dn", "description": "The group member DN", "type": "string"}, {"contentPath": "", "contextPath": "ActiveDirectory.Groups.members.category", "description": "Person/computer", "type": "string"}], "permitted": false, "sensitive": false, "timeout": 0}], "dockerImage": "demisto/ldap:1.0.0.75", "isFetch": false, "isFetchCredentials": false, "longRunning": false, "longRunningPortMapping": false, "runOnce": false, "script": "\n\nfrom typing import *\nfrom ldap3 import Server, Connection, NTLM, SUBTREE, ALL_ATTRIBUTES, Tls, Entry\nfrom ldap3.extend import microsoft\nimport ssl\nfrom datetime import datetime\nimport traceback\nimport os\nfrom ldap3.utils.log import (set_library_log_detail_level, get_library_log_detail_level,\n set_library_log_hide_sensitive_data, EXTENDED)\n\n# global connection\nconn: Optional[Connection] = None\n\n''' GLOBAL VARS '''\n\n# userAccountControl is a bitmask used to store a number of settings.\n# find more at:\n# https://support.microsoft.com/en-gb/help/305144/how-to-use-the-useraccountcontrol-flags-to-manipulate-user-account-pro\n\nCOOMON_ACCOUNT_CONTROL_FLAGS = {\n 512: \"Enabled Account\",\n 514: \"Disabled account\",\n 544: \"Account Enabled - Require user to change password at first logon\",\n 4096: \"Workstation/server\",\n 66048: \"Enabled, password never expires\",\n 66050: \"Disabled, password never expires\",\n 66080: \"Enables, password never expires, password not required.\",\n 532480: \"Domain controller\"\n}\nNORMAL_ACCOUNT = 512\nDISABLED_ACCOUNT = 514\n\n# common attributes for specific AD objects\nDEFAULT_PERSON_ATTRIBUTES = [\n 'name',\n 'displayName',\n 'memberOf',\n 'mail',\n 'samAccountName',\n 'manager',\n 'userAccountControl'\n]\nDEFAULT_COMPUTER_ATTRIBUTES = [\n 'name',\n 'memberOf'\n]\n\n''' HELPER FUNCTIONS '''\n\n\ndef initialize_server(host, port, secure_connection, unsecure):\n \"\"\"\n uses the instance configuration to initialize the LDAP server\n\n :param host: host or ip\n :type host: string\n :param port: port or None\n :type port: number\n :param secure_connection: SSL or None\n :type secure_connection: string\n :param unsecure: trust any cert\n :type unsecure: boolean\n :return: ldap3 Server\n :rtype: Server\n \"\"\"\n\n if secure_connection == \"SSL\":\n # intialize server with ssl\n # port is configured by default as 389 or as 636 for LDAPS if not specified in configuration\n demisto.debug(\"initializing sever with ssl (unsecure: {}). port: {}\". format(unsecure, port or 'default(636)'))\n if not unsecure:\n demisto.debug(\"will require server certificate.\")\n tls = Tls(validate=ssl.CERT_REQUIRED, ca_certs_file=os.environ.get('SSL_CERT_FILE'))\n if port:\n return Server(host, port=port, use_ssl=True, tls=tls)\n return Server(host, use_ssl=True, tls=tls)\n if port:\n return Server(host, port=port, use_ssl=True)\n return Server(host, use_ssl=True)\n demisto.debug(\"initializing server without secure connection. port: {}\". format(port or 'default(389)'))\n if port:\n return Server(host, port=port)\n return Server(host)\n\n\ndef account_entry(person_object, custome_attributes):\n # create an account entry from a person objects\n account = {\n 'Type': 'AD',\n 'ID': person_object.get('dn'),\n 'Email': person_object.get('mail'),\n 'Username': person_object.get('samAccountName'),\n 'DisplayName': person_object.get('displayName'),\n 'Managr': person_object.get('manager'),\n 'Manager': person_object.get('manager'),\n 'Groups': person_object.get('memberOf')\n }\n\n for attr in custome_attributes:\n account[attr] = person_object[attr]\n\n return account\n\n\ndef endpoint_entry(computer_object, custome_attributes):\n # create an endpoint entry from a computer object\n endpoint = {\n 'Type': 'AD',\n 'ID': computer_object.get('dn'),\n 'Hostname': computer_object.get('name'),\n 'Groups': computer_object.get('memberOf')\n }\n\n for attr in custome_attributes:\n endpoint[attr] = computer_object[attr]\n\n return endpoint\n\n\ndef base_dn_verified(base_dn):\n # serch AD with a simple query to test base DN is configured correctly\n try:\n search(\n \"(objectClass=user)\",\n base_dn,\n size_limit=1\n )\n except Exception as e:\n demisto.info(str(e))\n return False\n return True\n\n\n''' COMMANDS '''\n\n''' SEARCH '''\n\n\ndef search(search_filter, search_base, attributes=None, size_limit=0, time_limit=0):\n \"\"\"\n find entries in the DIT\n\n Args:\n search_base: the location in the DIT where the search will start\n search_filte: LDAP query string\n attributes: the attributes to specify for each entry found in the DIT\n\n \"\"\"\n assert conn is not None\n success = conn.search(\n search_base=search_base,\n search_filter=search_filter,\n attributes=attributes,\n size_limit=size_limit,\n time_limit=time_limit\n )\n\n if not success:\n raise Exception(\"Search failed\")\n return conn.entries\n\n\ndef search_with_paging(search_filter, search_base, attributes=None, page_size=100, size_limit=0, time_limit=0):\n \"\"\"\n find entries in the DIT\n\n Args:\n search_base: the location in the DIT where the search will start\n search_filte: LDAP query string\n attributes: the attributes to specify for each entrxy found in the DIT\n\n \"\"\"\n assert conn is not None\n total_entries = 0\n cookie = None\n start = datetime.now()\n\n entries: List[Entry] = []\n entries_left_to_fetch = size_limit\n while True:\n if 0 < entries_left_to_fetch < page_size:\n page_size = entries_left_to_fetch\n\n conn.search(\n search_base,\n search_filter,\n search_scope=SUBTREE,\n attributes=attributes,\n paged_size=page_size,\n paged_cookie=cookie\n )\n\n entries_left_to_fetch -= len(conn.entries)\n total_entries += len(conn.entries)\n cookie = conn.result['controls']['1.2.840.113556.1.4.319']['value']['cookie']\n time_diff = (start - datetime.now()).seconds\n\n entries.extend(conn.entries)\n\n # stop when: 1.reached size limit 2.reached time limit 3. no cookie\n if (size_limit and size_limit <= total_entries) or (time_limit and time_diff >= time_limit) or (not cookie):\n break\n\n # keep the raw entry for raw content (backward compatability)\n raw = []\n # flaten the entries\n flat = []\n\n for entry in entries:\n entry = json.loads(entry.entry_to_json())\n\n flat_entry = {\n 'dn': entry['dn']\n }\n\n for attr in entry.get('attributes', {}):\n flat_entry[attr] = entry['attributes'][attr]\n\n raw.append(entry)\n flat.append(flat_entry)\n\n return {\n \"raw\": raw,\n \"flat\": flat\n }\n\n\ndef user_dn(sam_account_name, search_base):\n search_filter = '(&(objectClass=user)(sAMAccountName={}))'.format(sam_account_name)\n entries = search(\n search_filter,\n search_base\n )\n if not entries:\n raise Exception(\"Could not get full DN for user with sAMAccountName '{}'\".format(sam_account_name))\n entry = json.loads(entries[0].entry_to_json())\n return entry['dn']\n\n\ndef computer_dn(compuer_name, search_base):\n search_filter = '(&(objectClass=user)(objectCategory=computer)(name={}))'.format(compuer_name)\n entries = search(\n search_filter,\n search_base\n )\n if not entries:\n raise Exception(\"Could not get full DN for computer with name '{}'\".format(compuer_name))\n entry = json.loads(entries[0].entry_to_json())\n return entry['dn']\n\n\ndef group_dn(group_name, search_base):\n search_filter = '(&(objectClass=group)(cn={}))'.format(group_name)\n entries = search(\n search_filter,\n search_base\n )\n if not entries:\n raise Exception(\"Could not get full DN for group with name '{}'\".format(group_name))\n entry = json.loads(entries[0].entry_to_json())\n return entry['dn']\n\n\ndef convert_special_chars_to_unicode(search_filter):\n # We allow users to use special chars without explicitly typing their unicode values\n chars_to_replace = {\n '\\\\(': '\\\\28',\n '\\\\)': '\\\\29',\n '\\\\*': '\\\\2a',\n '\\\\/': '\\\\2f',\n '\\\\\\\\': '\\\\5c'\n }\n for i, j in chars_to_replace.items():\n search_filter = search_filter.replace(i, j)\n\n return search_filter\n\n\ndef free_search(default_base_dn, page_size):\n\n args = demisto.args()\n\n search_filter = args.get('filter')\n size_limit = int(args.get('size-limit', '0'))\n time_limit = int(args.get('time-limit', '0'))\n search_base = args.get('base-dn') or default_base_dn\n attributes = args.get('attributes')\n context_output = args.get('context-output')\n\n search_filter = convert_special_chars_to_unicode(search_filter)\n\n # if ALL was specified - get all the object's attributes, else expect a string of comma separated values\n if attributes:\n attributes = ALL_ATTRIBUTES if attributes == 'ALL' else attributes.split(',')\n\n entries = search_with_paging(\n search_filter,\n search_base,\n attributes=attributes,\n size_limit=size_limit,\n time_limit=time_limit,\n page_size=page_size\n )\n\n ec = {} if context_output == 'no' else {'ActiveDirectory.Search(obj.dn == val.dn)': entries['flat']}\n demisto_entry = {\n 'ContentsFormat': formats['json'],\n 'Type': entryTypes['note'],\n 'Contents': entries['raw'],\n 'ReadableContentsFormat': formats['markdown'],\n 'HumanReadable': tableToMarkdown(\"Active Directory Search\", entries['flat']),\n 'EntryContext': ec\n }\n demisto.results(demisto_entry)\n\n\ndef search_users(default_base_dn, page_size):\n # this command is equivalant to script ADGetUser\n # will preform a custom search to find users by a specific (one) attribute specified by the user\n\n args = demisto.args()\n\n attributes: List[str] = []\n custom_attributes: List[str] = []\n\n # zero is actually no limitation\n limit = int(args.get('limit', '0'))\n\n # default query - list all users\n query = \"(&(objectClass=User)(objectCategory=person))\"\n\n # query by user DN\n if args.get('dn'):\n query = \"(&(objectClass=User)(objectCategory=person)(distinguishedName={}))\".format(args['dn'])\n\n # query by name\n if args.get('name'):\n query = \"(&(objectClass=User)(objectCategory=person)(cn={}))\".format(args['name'])\n\n # query by email\n if args.get('email'):\n query = \"(&(objectClass=User)(objectCategory=person)(mail={}))\".format(args['email'])\n\n # query by sAMAccountName\n if args.get('username'):\n query = \"(&(objectClass=User)(objectCategory=person)(sAMAccountName={}))\".format(args['username'])\n\n # query by custom object attribute\n if args.get('custom-field-type'):\n if not args.get('custom-field-data'):\n raise Exception('Please specify \"custom-field-data\" as well when quering by \"custom-field-type\"')\n query = \"(&(objectClass=User)(objectCategory=person)({}={}))\".format(\n args['custom-field-type'], args['custom-field-data'])\n\n if args.get('attributes'):\n custom_attributes = args['attributes'].split(\",\")\n\n attributes = list(set(custom_attributes + DEFAULT_PERSON_ATTRIBUTES))\n\n entries = search_with_paging(\n query,\n default_base_dn,\n attributes=attributes,\n size_limit=limit,\n page_size=page_size\n )\n\n accounts = [account_entry(entry, custom_attributes) for entry in entries['flat']]\n\n if args.get('user-account-control-out', '') == 'true':\n # display a literal translation of the numeric account control flag\n for i, user in enumerate(entries['flat']):\n flag_no = user.get('userAccountControl')[0]\n entries['flat'][i]['userAccountControl'] = COOMON_ACCOUNT_CONTROL_FLAGS.get(flag_no) or flag_no\n\n demisto_entry = {\n 'ContentsFormat': formats['json'],\n 'Type': entryTypes['note'],\n 'Contents': entries['raw'],\n 'ReadableContentsFormat': formats['markdown'],\n 'HumanReadable': tableToMarkdown(\"Active Directory - Get Users\", entries['flat']),\n 'EntryContext': {\n 'ActiveDirectory.Users(obj.dn == val.dn)': entries['flat'],\n # 'backward compatability' with ADGetUser script\n 'Account(obj.ID == val.ID)': accounts\n }\n }\n demisto.results(demisto_entry)\n\n\ndef search_computers(default_base_dn, page_size):\n # this command is equivalent to ADGetComputer script\n\n args = demisto.args()\n\n attributes: List[str] = []\n custome_attributes: List[str] = []\n\n # default query - list all users (computer category)\n query = \"(&(objectClass=user)(objectCategory=computer))\"\n\n # query by user DN\n if args.get('dn'):\n query = \"(&(objectClass=user)(objectCategory=computer)(dn={}))\".format(args['dn'])\n\n # query by name\n if args.get('name'):\n query = \"(&(objectClass=user)(objectCategory=computer)(name={}))\".format(args['name'])\n\n # query by custom object attribute\n if args.get('custom-field-type'):\n if not args.get('custom-field-data'):\n raise Exception('Please specify \"custom-field-data\" as well when quering by \"custom-field-type\"')\n query = \"(&(objectClass=user)(objectCategory=computer)({}={}))\".format(\n args['custom-field-type'], args['ustom-field-data'])\n\n if args.get('attributes'):\n custome_attributes = args['attributes'].split(\",\")\n\n attributes = list(set(custome_attributes + DEFAULT_COMPUTER_ATTRIBUTES))\n\n entries = search_with_paging(\n query,\n default_base_dn,\n attributes=attributes,\n page_size=page_size\n )\n\n endpoints = [endpoint_entry(entry, custome_attributes) for entry in entries['flat']]\n\n demisto_entry = {\n 'ContentsFormat': formats['json'],\n 'Type': entryTypes['note'],\n 'Contents': entries['raw'],\n 'ReadableContentsFormat': formats['markdown'],\n 'HumanReadable': tableToMarkdown(\"Active Directory - Get Computers\", entries['flat']),\n 'EntryContext': {\n 'ActiveDirectory.Computers(obj.dn == val.dn)': entries['flat'],\n # 'backward compatability' with ADGetComputer script\n 'Endpoint(obj.ID == val.ID)': endpoints\n }\n }\n demisto.results(demisto_entry)\n\n\ndef search_group_members(default_base_dn, page_size):\n # this command is equivalent to ADGetGroupMembers script\n\n args = demisto.args()\n member_type = args.get('member-type')\n group_dn = args.get('group-dn')\n\n custome_attributes: List[str] = []\n default_attributes = DEFAULT_PERSON_ATTRIBUTES if member_type == 'person' else DEFAULT_COMPUTER_ATTRIBUTES\n\n if args.get('attributes'):\n custome_attributes = args['attributes'].split(\",\")\n\n attributes = list(set(custome_attributes + default_attributes))\n\n # neasted search\n query = \"(&(objectCategory={})(objectClass=user)(memberOf:1.2.840.113556.1.4.1941:={}))\".format(member_type,\n group_dn)\n\n entries = search_with_paging(\n query,\n default_base_dn,\n attributes=attributes,\n page_size=page_size\n )\n\n members = [{'dn': entry['dn'], 'category': member_type} for entry in entries['flat']]\n\n demisto_entry = {\n 'ContentsFormat': formats['json'],\n 'Type': entryTypes['note'],\n 'Contents': entries['raw'],\n 'ReadableContentsFormat': formats['markdown'],\n 'HumanReadable': tableToMarkdown(\"Active Directory - Get Group Members\", entries['flat']),\n 'EntryContext': {\n 'ActiveDirectory.Groups(obj.dn ==' + group_dn + ')': {\n 'dn': group_dn,\n 'members': members\n }\n }\n }\n\n if member_type == 'person':\n demisto_entry['EntryContext']['ActiveDirectory.Users(obj.dn == val.dn)'] = entries['flat']\n demisto_entry['EntryContext']['Account'] = [account_entry(\n entry, custome_attributes) for entry in entries['flat']]\n else:\n demisto_entry['EntryContext']['ActiveDirectory.Computers(obj.dn == val.dn)'] = entries['flat']\n demisto_entry['EntryContext']['Endpoint'] = [endpoint_entry(\n entry, custome_attributes) for entry in entries['flat']]\n\n demisto.results(demisto_entry)\n\n\n''' DATABASE OPERATIONS '''\n\n''' CREATE OBJECT'''\n\n\ndef create_user():\n assert conn is not None\n args = demisto.args()\n\n object_classes = [\"top\", \"person\", \"organizationalPerson\", \"user\"]\n user_dn = args.get('user-dn')\n username = args.get(\"username\")\n password = args.get(\"password\")\n custome_attributes = args.get('custom-attributes')\n attributes = {\n \"samAccountName\": username\n }\n\n # set common user attributes\n if args.get('display-name'):\n attributes['displayName'] = args['display-name']\n if args.get('description'):\n attributes['description'] = args['description']\n if args.get('email'):\n attributes['mail'] = args['email']\n if args.get('telephone-number'):\n attributes['telephoneNumber'] = args['telephone-number']\n if args.get('title'):\n attributes['title'] = args['title']\n\n # set user custome attributes\n if custome_attributes:\n try:\n custome_attributes = json.loads(custome_attributes)\n except Exception as e:\n demisto.info(str(e))\n raise Exception(\n \"Failed to parse custom attributes argument. Please see an example of this argument in the description.\"\n )\n for attribute_name, attribute_value in custome_attributes.items():\n # can run default attribute stting\n attributes[attribute_name] = attribute_value\n\n # add user\n success = conn.add(user_dn, object_classes, attributes)\n if not success:\n raise Exception(\"Failed to create user\")\n\n # set user password\n success = conn.extend.microsoft.modify_password(user_dn, password)\n if not success:\n raise Exception(\"Failed to reset user password\")\n\n # enable user and expire password\n modification = {\n # enable user\n 'userAccountControl': [('MODIFY_REPLACE', NORMAL_ACCOUNT)],\n # set to 0, to force password change on next login\n \"pwdLastSet\": [('MODIFY_REPLACE', \"0\")]\n }\n modify_object(user_dn, modification)\n\n demisto_entry = {\n 'ContentsFormat': formats['text'],\n 'Type': entryTypes['note'],\n 'Contents': \"Created user with DN: {}\".format(user_dn)\n }\n demisto.results(demisto_entry)\n\n\ndef create_contact():\n assert conn is not None\n args = demisto.args()\n\n object_classes = [\"top\", \"person\", \"organizationalPerson\", \"contact\"]\n contact_dn = args.get('contact-dn')\n\n # set contact attributes\n attributes: Dict = {}\n if args.get('custom-attributes'):\n try:\n attributes = json.loads(args['custom-attributes'])\n except Exception as e:\n demisto.info(str(e))\n raise Exception(\n 'Failed to parse custom attributes argument. Please see an example of this argument in the argument.'\n )\n\n # set common user attributes\n if args.get('display-name'):\n attributes['displayName'] = args['display-name']\n if args.get('description'):\n attributes['description'] = args['description']\n if args.get('email'):\n attributes['mail'] = args['email']\n if args.get('telephone-number'):\n attributes['telephoneNumber'] = args['telephone-number']\n if args.get('title'):\n attributes['title'] = args['title']\n\n # add contact\n\n success = conn.add(contact_dn, object_classes, attributes)\n if not success:\n raise Exception(\"Failed to create contact\")\n\n demisto_entry = {\n 'ContentsFormat': formats['text'],\n 'Type': entryTypes['note'],\n 'Contents': \"Created contact with DN: {}\".format(contact_dn)\n }\n demisto.results(demisto_entry)\n\n\n''' UPDATE OBJECT '''\n\n\ndef modify_object(dn, modification):\n \"\"\"\n modifys object in the DIT\n \"\"\"\n assert conn is not None\n success = conn.modify(dn, modification)\n if not success:\n raise Exception(\"Failed to update object {} with the following modofication: {}\".format(\n dn, json.dumps(modification)))\n\n\ndef update_user(default_base_dn):\n args = demisto.args()\n\n # get user DN\n sam_account_name = args.get('username')\n attribute_name = args.get('attribute-name')\n attribute_value = args.get('attribute-value')\n search_base = args.get('base-dn') or default_base_dn\n dn = user_dn(sam_account_name, search_base)\n\n modification = {}\n modification[attribute_name] = [('MODIFY_REPLACE', attribute_value)]\n\n # modify user\n modify_object(dn, modification)\n\n demisto_entry = {\n 'ContentsFormat': formats['text'],\n 'Type': entryTypes['note'],\n 'Contents': \"Updated user's {} to {} \".format(attribute_name, attribute_value)\n }\n demisto.results(demisto_entry)\n\n\ndef update_contact():\n args = demisto.args()\n\n contact_dn = args.get('contact-dn')\n modification = {}\n modification[args.get('attribute-name')] = [('MODIFY_REPLACE', args.get('attribute-value'))]\n\n # modify\n modify_object(contact_dn, modification)\n\n demisto_entry = {\n 'ContentsFormat': formats['text'],\n 'Type': entryTypes['note'],\n 'Contents': \"Updated contact's {} to: {} \".format(args.get('attribute-name'), args.get('attribute-value'))\n }\n demisto.results(demisto_entry)\n\n\ndef modify_computer_ou(default_base_dn):\n assert conn is not None\n args = demisto.args()\n\n computer_name = args.get('computer-name')\n dn = computer_dn(computer_name, args.get('base-dn') or default_base_dn)\n\n success = conn.modify_dn(dn, \"CN={}\".format(computer_name), new_superior=args.get('full-superior-dn'))\n if not success:\n raise Exception(\"Failed to modify computer OU\")\n\n demisto_entry = {\n 'ContentsFormat': formats['text'],\n 'Type': entryTypes['note'],\n 'Contents': \"Moved computer {} to {}\".format(computer_name, args.get('full-superior-dn'))\n }\n demisto.results(demisto_entry)\n\n\ndef expire_user_password(default_base_dn):\n args = demisto.args()\n\n # get user DN\n sam_account_name = args.get('username')\n search_base = args.get('base-dn') or default_base_dn\n dn = user_dn(sam_account_name, search_base)\n\n modification = {\n # set to 0, to force password change on next login\n \"pwdLastSet\": [('MODIFY_REPLACE', \"0\")]\n }\n\n # modify user\n modify_object(dn, modification)\n\n demisto_entry = {\n 'ContentsFormat': formats['text'],\n 'Type': entryTypes['note'],\n 'Contents': \"Expired password successfully\"\n }\n demisto.results(demisto_entry)\n\n\ndef set_user_password(default_base_dn):\n assert conn is not None\n args = demisto.args()\n\n # get user DN\n sam_account_name = args.get('username')\n password = args.get('password')\n search_base = args.get('base-dn') or default_base_dn\n dn = user_dn(sam_account_name, search_base)\n\n # set user password\n success = conn.extend.microsoft.modify_password(dn, password)\n if not success:\n raise Exception(\"Failed to reset user password\")\n\n demisto_entry = {\n 'ContentsFormat': formats['text'],\n 'Type': entryTypes['note'],\n 'Contents': \"User password successfully set\"\n }\n demisto.results(demisto_entry)\n\n\ndef enable_user(default_base_dn):\n args = demisto.args()\n\n # get user DN\n sam_account_name = args.get('username')\n search_base = args.get('base-dn') or default_base_dn\n dn = user_dn(sam_account_name, search_base)\n\n # modify user\n modification = {\n 'userAccountControl': [('MODIFY_REPLACE', NORMAL_ACCOUNT)]\n }\n modify_object(dn, modification)\n\n demisto_entry = {\n 'ContentsFormat': formats['text'],\n 'Type': entryTypes['note'],\n 'Contents': \"User {} was enabled\".format(sam_account_name)\n }\n demisto.results(demisto_entry)\n\n\ndef disable_user(default_base_dn):\n args = demisto.args()\n\n # get user DN\n sam_account_name = args.get('username')\n search_base = args.get('base-dn') or default_base_dn\n dn = user_dn(sam_account_name, search_base)\n\n # modify user\n modification = {\n 'userAccountControl': [('MODIFY_REPLACE', DISABLED_ACCOUNT)]\n }\n modify_object(dn, modification)\n\n demisto_entry = {\n 'ContentsFormat': formats['text'],\n 'Type': entryTypes['note'],\n 'Contents': \"User {} was disabled\".format(sam_account_name)\n }\n demisto.results(demisto_entry)\n\n\ndef add_member_to_group(default_base_dn):\n\n args = demisto.args()\n\n search_base = args.get('base-dn') or default_base_dn\n\n # get the dn of the member - either user or computer\n args_err = \"Pleade provide either username or computer-name\"\n member_dn = ''\n\n if args.get('username') and args.get('computer-name'):\n # both arguments passed\n raise Exception(args_err)\n if args.get('username'):\n member_dn = user_dn(args['username'], search_base)\n elif args.get('computer-name'):\n member_dn = computer_dn(args['computer-name'], search_base)\n else:\n # none of the arguments passed\n raise Exception(args_err)\n\n grp_dn = group_dn(args.get('group-cn'), search_base)\n\n success = microsoft.addMembersToGroups.ad_add_members_to_groups(conn, [member_dn], [grp_dn])\n if not success:\n raise Exception(\"Failed to add {} to group {]}\".format(\n args.get('username') or args.get('computer-name'),\n args.get('group_name')\n ))\n\n demisto_entry = {\n 'ContentsFormat': formats['text'],\n 'Type': entryTypes['note'],\n 'Contents': \"Object with dn {} was added to group {}\".format(member_dn, args.get('group-cn'))\n }\n demisto.results(demisto_entry)\n\n\ndef remove_member_from_group(default_base_dn):\n\n args = demisto.args()\n\n search_base = args.get('base-dn') or default_base_dn\n\n # get the dn of the member - either user or computer\n args_err = \"Pleade provide either username or computer-name\"\n member_dn = ''\n\n if args.get('username') and args.get('computer-name'):\n # both arguments passed\n raise Exception(args_err)\n if args.get('username'):\n member_dn = user_dn(args['username'], search_base)\n elif args.get('computer-name'):\n member_dn = computer_dn(args['computer-name'], search_base)\n else:\n # none of the arguments passed\n raise Exception(args_err)\n\n grp_dn = group_dn(args.get('group-cn'), search_base)\n\n success = microsoft.removeMembersFromGroups.ad_remove_members_from_groups(conn, [member_dn], [grp_dn], True)\n if not success:\n raise Exception(\"Failed to remove {member} from group {group_name}\".format({\n \"member\": args.get('username') or args.get('computer-name'),\n \"group_name\": args.get('group_name')\n }))\n\n demisto_entry = {\n 'ContentsFormat': formats['text'],\n 'Type': entryTypes['note'],\n 'Contents': \"Object with dn {} removed from group {}\".format(member_dn, args.get('group-cn'))\n }\n demisto.results(demisto_entry)\n\n\ndef unlock_account(default_base_dn):\n args = demisto.args()\n\n # get user DN\n sam_account_name = args.get('username')\n search_base = args.get('base-dn') or default_base_dn\n dn = user_dn(sam_account_name, search_base)\n\n success = microsoft.unlockAccount.ad_unlock_account(conn, dn)\n if not success:\n raise Exception(\"Failed to unlock user {}\".format(sam_account_name))\n\n demisto_entry = {\n 'ContentsFormat': formats['text'],\n 'Type': entryTypes['note'],\n 'Contents': \"Unlocked user {}\".format(sam_account_name)\n }\n demisto.results(demisto_entry)\n\n\n''' DELETE OBJECT '''\n\n\ndef delete_user():\n # can acually delete any object...\n assert conn is not None\n success = conn.delete(demisto.args().get('user-dn'))\n if not success:\n raise Exception('Failed to delete user')\n\n demisto_entry = {\n 'ContentsFormat': formats['text'],\n 'Type': entryTypes['note'],\n 'Contents': \"Deleted object with dn {}\".format(demisto.args().get('user-dn'))\n }\n demisto.results(demisto_entry)\n\n\n'''\n TEST CONFIGURATION\n authenticate user credentials while initializing connection wiith AD server\n verify base DN is configured correctly\n'''\n\n\ndef main():\n ''' INSTANCE CONFIGURATION '''\n SERVER_IP = demisto.params().get('server_ip')\n USERNAME = demisto.params().get('credentials')['identifier']\n PASSWORD = demisto.params().get('credentials')['password']\n DEFAULT_BASE_DN = demisto.params().get('base_dn')\n SECURE_CONNECTION = demisto.params().get('secure_connection')\n DEFAULT_PAGE_SIZE = int(demisto.params().get('page_size'))\n NTLM_AUTH = demisto.params().get('ntlm')\n UNSECURE = demisto.params().get('unsecure', False)\n PORT = demisto.params().get('port')\n\n if PORT:\n # port was configured, cast to int\n PORT = int(PORT)\n last_log_detail_level = None\n try:\n try:\n set_library_log_hide_sensitive_data(True)\n if is_debug_mode():\n demisto.info('debug-mode: setting library log detail to EXTENDED')\n last_log_detail_level = get_library_log_detail_level()\n set_library_log_detail_level(EXTENDED)\n server = initialize_server(SERVER_IP, PORT, SECURE_CONNECTION, UNSECURE)\n except Exception as e:\n return_error(str(e))\n return\n global conn\n if NTLM_AUTH:\n # intialize connection to LDAP server with NTLM authentication\n # user example: domain\\user\n domain_user = SERVER_IP + '\\\\' + USERNAME if '\\\\' not in USERNAME else USERNAME\n conn = Connection(server, user=domain_user, password=PASSWORD, authentication=NTLM)\n else:\n # here username should be the user dn\n conn = Connection(server, user=USERNAME, password=PASSWORD)\n\n # bind operation is the \u201cauthenticate\u201d operation.\n try:\n # open socket and bind to server\n if not conn.bind():\n message = \"Failed to bind to server. Please validate the credentials configured correctly.\\n{}\".format(\n json.dumps(conn.result))\n return_error(message)\n return\n except Exception as e:\n exc_msg = str(e)\n demisto.info(\"Failed bind to: {}:{}. {}: {}\".format(SERVER_IP, PORT, type(e), exc_msg\n + \"\\nTrace:\\n{}\".format(traceback.format_exc())))\n message = \"Failed to access LDAP server. Please validate the server host and port are configured correctly\"\n if 'ssl wrapping error' in exc_msg:\n message = \"Failed to access LDAP server. SSL error.\"\n if not UNSECURE:\n message += ' Try using: \"Trust any certificate\" option.'\n return_error(message)\n return\n\n demisto.info('Established connection with AD LDAP server')\n\n if not base_dn_verified(DEFAULT_BASE_DN):\n message = \"Failed to verify the base DN configured for the instance.\\n\" \\\n \"Last connection result: {}\\n\" \\\n \"Last error from LDAP server: {}\".format(json.dumps(conn.result), json.dumps(conn.last_error))\n return_error(message)\n return\n\n demisto.info('Verfied base DN \"{}\"'.format(DEFAULT_BASE_DN))\n\n ''' COMMAND EXECUTION '''\n\n if demisto.command() == 'test-module':\n if conn.user == '':\n # Empty response means you have no authentication status on the server, so you are an anonymous user.\n raise Exception(\"Failed to authenticate user\")\n demisto.results('ok')\n\n if demisto.command() == 'ad-search':\n free_search(DEFAULT_BASE_DN, DEFAULT_PAGE_SIZE)\n\n if demisto.command() == 'ad-expire-password':\n expire_user_password(DEFAULT_BASE_DN)\n\n if demisto.command() == 'ad-set-new-password':\n set_user_password(DEFAULT_BASE_DN)\n\n if demisto.command() == 'ad-unlock-account':\n unlock_account(DEFAULT_BASE_DN)\n\n if demisto.command() == 'ad-disable-account':\n disable_user(DEFAULT_BASE_DN)\n\n if demisto.command() == 'ad-enable-account':\n enable_user(DEFAULT_BASE_DN)\n\n if demisto.command() == 'ad-remove-from-group':\n remove_member_from_group(DEFAULT_BASE_DN)\n\n if demisto.command() == 'ad-add-to-group':\n add_member_to_group(DEFAULT_BASE_DN)\n\n if demisto.command() == 'ad-create-user':\n create_user()\n\n if demisto.command() == 'ad-delete-user':\n delete_user()\n\n if demisto.command() == 'ad-update-user':\n update_user(DEFAULT_BASE_DN)\n\n if demisto.command() == 'ad-modify-computer-ou':\n modify_computer_ou(DEFAULT_BASE_DN)\n\n if demisto.command() == 'ad-create-contact':\n create_contact()\n\n if demisto.command() == 'ad-update-contact':\n update_contact()\n\n if demisto.command() == 'ad-get-user':\n search_users(DEFAULT_BASE_DN, DEFAULT_PAGE_SIZE)\n\n if demisto.command() == 'ad-get-computer':\n search_computers(DEFAULT_BASE_DN, DEFAULT_PAGE_SIZE)\n\n if demisto.command() == 'ad-get-group-members':\n search_group_members(DEFAULT_BASE_DN, DEFAULT_PAGE_SIZE)\n\n except Exception as e:\n message = str(e)\n if conn:\n message += \"\\nLast connection result: {}\\nLast error from LDAP server: {}\".format(\n json.dumps(conn.result), conn.last_error)\n return_error(message)\n return\n finally:\n # disconnect and close the connection\n if conn:\n conn.unbind()\n if last_log_detail_level:\n set_library_log_detail_level(last_log_detail_level)\n\n\n# python2 uses __builtin__ python3 uses builtins\nif __name__ == \"__builtin__\" or __name__ == \"builtins\":\n main()", "subtype": "python3", "type": "python"}, "modified": "2019-10-18T15:51:33.995147038-04:00", "name": "Active Directory Query v2", "prevName": "Active Directory Query v2", "shouldCommit": false, "sortValues": null, "system": true, "vcShouldIgnore": false, "version": 8}, {"brand": "", "canGetSamples": true, "category": "Messaging", "commitMessage": "", "configuration": [{"defaultValue": "", "display": "Server IP (e.g. 192.168.0.1)", "hidden": false, "name": "hostname", "options": null, "required": true, "type": 0}, {"defaultValue": "61613", "display": "Port", "hidden": false, "name": "port", "options": null, "required": false, "type": 0}, {"defaultValue": "", "display": "Username", "hidden": false, "name": "credentials", "options": null, "required": false, "type": 9}, {"defaultValue": "Demisto", "display": "Client ID", "hidden": false, "name": "client-id", "options": null, "required": false, "type": 0}, {"defaultValue": "1", "display": "Subscription ID", "hidden": false, "name": "subscription-id", "options": null, "required": false, "type": 0}, {"defaultValue": "", "display": "Topic Name (for subscription)", "hidden": false, "name": "topic-name", "options": null, "required": false, "type": 0}, {"defaultValue": "", "display": "Fetch incidents", "hidden": false, "name": "isFetch", "options": null, "required": false, "type": 8}, {"defaultValue": "", "display": "Incident type", "hidden": false, "name": "incidentType", "options": null, "required": false, "type": 13}, {"defaultValue": "false", "display": "Use system proxy settings", "hidden": false, "name": "proxy", "options": null, "required": false, "type": 8}], "description": "Integration with ActiveMQ queue", "detailedDescription": "This integration uses ActiveMQ STOMP protocol, that must be enabled (usually port 61613 by default) in order to work.\nFetch incidents is based on using Durable Topic Subscribers, in order to fetch messages, and convert to Demisto incidents.", "display": "ActiveMQ", "hidden": false, "icon": "", "id": "ActiveMQ", "image": "", "integrationScript": {"commands": [{"arguments": [{"default": false, "deprecated": false, "description": "The message destination (such as a message queue - for example \u2018/queue/test\u2019 - or a message topic)", "name": "destination", "required": true, "secret": false}, {"default": false, "deprecated": false, "description": "The content of the message to be sent", "name": "body", "required": true, "secret": false}], "cartesian": false, "deprecated": false, "description": "Send messag to topic or queue", "execution": false, "hidden": false, "important": null, "name": "activemq-send", "outputs": null, "permitted": false, "sensitive": false, "timeout": 0}, {"arguments": [{"default": false, "defaultValue": "1", "deprecated": false, "description": "The identifier to uniquely identify the subscription", "name": "subscription-id", "required": true, "secret": false}, {"default": false, "deprecated": false, "description": "The topic to subscribe to", "name": "topic-name", "required": true, "secret": false}], "cartesian": false, "deprecated": false, "description": "Subscribe and read messages from topic", "execution": false, "hidden": false, "important": null, "name": "activemq-subscribe", "outputs": null, "permitted": false, "sensitive": false, "timeout": 0}], "dockerImage": "demisto/stomp", "isFetch": true, "isFetchCredentials": false, "longRunning": false, "longRunningPortMapping": false, "runOnce": false, "script": "''' IMPORTS '''\n\nimport stomp\nimport os\n\nif not demisto.params()['proxy']:\n del os.environ['HTTP_PROXY']\n del os.environ['HTTPS_PROXY']\n del os.environ['http_proxy']\n del os.environ['https_proxy']\n\n''' GLOBAL VARS '''\n\nHOSTNAME = demisto.params()['hostname']\nPORT = int(demisto.params()['port'])\nUSERNAME = demisto.params()['credentials']['identifier']\nPASSWORD = demisto.params()['credentials']['password']\n\nclass MsgListener(stomp.ConnectionListener):\n def __init__(self):\n self.result_arr = []\n self.msg_ids = []\n\n def on_error(self, headers, message):\n demisto.results('received an error \"%s\"' % message)\n def on_message(self, headers, message):\n self.result_arr.append(message)\n self.msg_ids.append(headers['message-id'])\n\n''' HELPER FUNCTIONS '''\n\ndef create_connection():\n conn = stomp.Connection([(HOSTNAME, PORT)])\n return conn\n\ndef connect(conn, client_id=None):\n conn.start()\n if client_id and len(client_id) > 0:\n conn.connect(USERNAME, PASSWORD, wait=True, headers = {'client-id': client_id })\n else:\n conn.connect(USERNAME, PASSWORD, wait=True)\n return conn\n\n''' FUNCTIONS '''\n\ndef send_message():\n\n txid = conn.begin()\n dest = demisto.args()['destination']\n body = demisto.args()['body']\n conn.send(dest, body, transaction=txid)\n conn.commit(txid)\n demisto.results('Message sent to ActiveMQ destination: ' + dest + ' with transaction ID: ' + txid)\n\ndef subscribe():\n\n listener = MsgListener()\n if client and len(client) > 0:\n conn.set_listener('Demisto', listener)\n subscription_id = demisto.args()['subscription-id']\n topic_name = demisto.args()['topic-name']\n conn.subscribe('/topic/'+topic_name, subscription_id, ack='client-individual', headers={'activemq.subscriptionName': client})\n time.sleep(1)\n for msg in listener.result_arr:\n demisto.results(msg)\n for msg_id in listener.msg_ids:\n conn.ack(msg_id, subscription_id)\n\ndef fetch_incidents():\n\n subscription_id = demisto.params()['subscription-id']\n listener = MsgListener()\n if client and len(client) > 0:\n conn.set_listener('Demisto', listener)\n topic_name = demisto.params()['topic-name']\n conn.subscribe(\n '/topic/' + topic_name,\n subscription_id,\n ack='client-individual',\n headers={'activemq.subscriptionName': client}\n )\n incidents = []\n time.sleep(3)\n for i in range(len(listener.result_arr)):\n msg = listener.result_arr[i]\n msg_id = listener.msg_ids[i]\n incidents.append({\n 'Name': 'ActiveMQ incident:' + msg_id,\n 'rawJSON': msg,\n 'details': msg\n })\n demisto.incidents(incidents)\n last_msg_id = \"\"\n for msg_id in listener.msg_ids:\n conn.ack(msg_id, subscription_id)\n last_msg_id = msg_id\n\n''' EXECUTION CODE '''\nclient = 'Demisto'\nif demisto.get(demisto.params(), 'client-id'):\n client = demisto.params()['client-id']\n\nconn = create_connection()\n\nLOG('command is %s' % (demisto.command(), ))\n\ntry:\n if demisto.command() == 'test-module':\n # Test connectivity\n connect(conn)\n demisto.results('ok')\n\n elif demisto.command() == 'activemq-send':\n connect(conn)\n send_message()\n\n elif demisto.command() == 'activemq-subscribe':\n connect(conn, client)\n subscribe()\n\n elif demisto.command() == 'fetch-incidents':\n connect(conn, client)\n fetch_incidents()\n\nexcept Exception, e:\n LOG(e.message)\n LOG.print_log()\n raise\n\nfinally:\n conn.disconnect()", "subtype": "", "type": "python"}, "modified": "2019-10-15T04:54:09.842836044-04:00", "name": "ActiveMQ", "prevName": "ActiveMQ", "shouldCommit": false, "sortValues": null, "system": true, "vcShouldIgnore": false, "version": 3}, {"brand": "", "canGetSamples": true, "category": "Data Enrichment & Threat Intelligence", "commitMessage": "", "configuration": [{"defaultValue": "", "display": "Server URL (e.g. https://starlight.companyname.com:8889)", "hidden": false, "name": "url", "options": null, "required": true, "type": 0}, {"defaultValue": "", "display": "User name", "hidden": false, "name": "credentials", "options": null, "required": true, "type": 9}, {"defaultValue": "", "display": "Fetch incidents", "hidden": false, "name": "isFetch", "options": null, "required": false, "type": 8}, {"defaultValue": "", "display": "Incident type", "hidden": false, "name": "incidentType", "options": null, "required": false, "type": 13}, {"defaultValue": "15", "display": "Fetching interval in minutes (default is 15, minimum is 15 )", "hidden": false, "name": "fetch_interval", "options": null, "required": false, "type": 0}, {"defaultValue": "", "display": "The specific security event to look for. Default is all events", "hidden": false, "name": "event_name", "options": null, "required": false, "type": 0}, {"defaultValue": "50", "display": "Security event severity threshold, between 0-100", "hidden": false, "name": "severity", "options": null, "required": false, "type": 0}, {"defaultValue": "true", "display": "Trust any certificate (unsecure)", "hidden": false, "name": "insecure", "options": null, "required": false, "type": 8}, {"defaultValue": "false", "display": "Use system proxy settings", "hidden": false, "name": "proxy", "options": null, "required": false, "type": 8}], "description": "Aella Star Light Integration", "display": "Aella Star Light", "hidden": false, "icon": "", "id": "Aella Star Light", "image": "", "integrationScript": {"commands": [{"arguments": [{"default": false, "deprecated": false, "description": "event id from the Star Light incident", "name": "event_id", "required": true, "secret": false}], "cartesian": false, "deprecated": false, "description": "Query the details for a specific Start Light event", "execution": false, "hidden": false, "important": null, "name": "aella-get-event", "outputs": [{"contentPath": "", "contextPath": "Aella.Event.event_name", "description": "The event name", "type": "string"}, {"contentPath": "", "contextPath": "Aella.Event.severity", "description": "The severity score", "type": "string"}, {"contentPath": "", "contextPath": "Aella.Event.dstip", "description": "The Destination IP", "type": "string"}, {"contentPath": "", "contextPath": "Aella.Event.srcip", "description": "The source IP", "type": "string"}, {"contentPath": "", "contextPath": "Aella.Event.tenantid", "description": "The tenant ID", "type": "string"}, {"contentPath": "", "contextPath": "Aella.Event.srcip_reputation", "description": "The source IP reputation", "type": "string"}, {"contentPath": "", "contextPath": "Aella.Event.dstip_reputation", "description": "The destination IP reputation", "type": "string"}, {"contentPath": "", "contextPath": "Aella.Event.dstip_geo", "description": "The destination IP geo location", "type": ""}, {"contentPath": "", "contextPath": "Aella.Event.srcip_geo", "description": "The source IP geo location", "type": ""}], "permitted": false, "sensitive": false, "timeout": 0}], "isFetch": true, "isFetchCredentials": false, "longRunning": false, "longRunningPortMapping": false, "runOnce": false, "script": "import requests\nimport json\nimport time\nimport datetime\nimport os\n\nimport urllib3\nurllib3.disable_warnings(urllib3.exceptions.InsecureRequestWarning)\n\nfrom requests.packages.urllib3.exceptions import InsecureRequestWarning\nrequests.packages.urllib3.disable_warnings(InsecureRequestWarning)\n\nif not demisto.params()['proxy']:\n del os.environ['HTTP_PROXY']\n del os.environ['HTTPS_PROXY']\n del os.environ['http_proxy']\n del os.environ['https_proxy']\n\n''' GLOBAL VARS '''\nURL = demisto.getParam('url') + '/aellaelastic'\nUSERNAME = demisto.getParam('credentials')['identifier']\nPASSWORD = demisto.getParam('credentials')['password']\nFETCH_INTERVAL = demisto.getParam('fetch_interval')\nVALIDATE_CERT = not demisto.params().get('insecure', True)\n\n''' HELPER FUNCTIONS '''\ndef make_rest_call(end_point, username, password, action_result,\n headers={}, params=None,\n data=None, method='get'):\n headers.update({'Accept': 'application/json'})\n headers.update({'Content-Type': 'application/json'})\n\n resp_json = None\n request_func = getattr(requests, method)\n if (not request_func):\n action_result['status']='Unsupported method {}'.format(method)\n return\n try:\n r = request_func(end_point, auth=(username, password),\n data=json.dumps(data) if data else None,\n headers=headers,\n verify=VALIDATE_CERT,params=params)\n except Exception as e:\n action_result['status']='Server REST API exception {}'.format(e)\n return\n\n if r is not None:\n action_result['r_text']=r.text\n action_result['headers']=r.headers\n action_result['r_status_code']=r.status_code\n\n try:\n resp_json = r.json()\n except Exception as e:\n action_result['status']='Json parse error {}'.format(\n r.text.replace('{', ' ').replace('}', ' '))\n return\n\n if (200 <= r.status_code <= 399):\n action_result['status'] = 'Success'\n else:\n action_result['status'] = 'Failed'\n\n action_result['data'] = resp_json\n return\n\n''' FUNCTIONS '''\ndef fetch_incidents_command():\n fetch_interval = demisto.getParam('fetch_interval')\n if fetch_interval is None:\n fetch_interval = 15 * 60 # 15 minutes\n else:\n try:\n fetch_interval = int(fetch_interval) * 60\n if fetch_interval < 15 * 60:\n # Min is 15 minutes\n fetch_interval = 15 * 60\n except:\n fetch_interval = 15 * 60\n\n cur_t = time.time()\n\n checkTime = cur_t - fetch_interval\n\n event = demisto.getParam('event_name')\n if event is None:\n event = '*'\n\n score = demisto.getParam('severity')\n if score:\n try:\n score = int(score)\n if score < 0 or score > 100:\n score = 50\n except:\n score = 50\n else:\n # Default score\n score = 50\n\n index_str = 'aella-ser*'\n\n query_str = 'event_name:{} AND severity:>{}'.format(event, score)\n\n ts_str = str(long(checkTime * 1000))\n\n query_json = { 'query':\n { 'bool':\n { 'must':\n [ { 'query_string':\n {\n 'query': query_str,\n 'analyze_wildcard': True\n }\n },\n { 'range':\n { 'timestamp':\n { 'gt': ts_str }\n }\n }\n ]\n }\n }\n }\n end_point = URL + '/{0}/{1}/_search'.format(index_str, 'amsg')\n\n action_result = {}\n make_rest_call(end_point,\n USERNAME, PASSWORD, action_result, data=query_json\n )\n\n if action_result['status'] == 'Success':\n demisto.info('Poll incidents ok')\n data = action_result.get('data')\n if not isinstance(data, dict):\n demisto.error('Data returned in wrong format {}'.format(data))\n demisto.incidents([])\n return\n hits = data.get('hits', {}).get('hits', [])\n incidents = []\n\n try:\n cached_event = demisto.getLastRun().get(\"cached_event\", {})\n except:\n cached_event = {}\n\n new_cached_event = {}\n\n for hit in hits:\n source = hit.get('_source', None)\n if not source:\n continue\n event_name = source.get('event_name', None)\n try:\n event_severity = int(source.get('severity', None))\n if event_severity > 75:\n severity = 3\n elif event_severity > 50:\n severity = 2\n else:\n severity = 1\n except:\n severity = 0\n\n if not event_name:\n continue\n eid = hit['_id']\n\n new_cached_event[eid] = True\n if cached_event.get(eid, False):\n continue\n\n sdi = '{}_{}'.format(event_name, eid)\n incident = {\n 'name' : sdi,\n 'severity' : severity,\n 'rawJSON':json.dumps({ 'name':sdi,\n 'label':'Starlight event',\n 'aella_eid': eid,\n 'aella_event':event_name,\n 'event_severity': event_severity\n })\n }\n incidents.append(incident)\n demisto.info('Incidents is {}'.format(incidents))\n demisto.setLastRun({'cached_event': new_cached_event})\n demisto.incidents(incidents)\n\n else:\n demisto.info('Poll incidents failed {}'.format(action_result))\n demisto.incidents([])\n\ndef aella_get_event_command():\n demisto.info('Aella started get-event with {}'.format(demisto.args()['event_id']))\n event_id = demisto.args()['event_id']\n query_json = { 'query': { 'match': { '_id': event_id } } }\n end_point = URL + '/{0}/{1}/_search'.format('aella-ser*', 'amsg')\n\n action_result = {}\n make_rest_call(end_point,\n USERNAME, PASSWORD, action_result, data=query_json\n )\n if action_result['status'] == 'Success':\n demisto.info('Run Query is successful')\n response = action_result.get('data')\n timed_out = response.get('timed_out', False)\n hits = response.get('hits', {}).get('hits', [])\n source = {}\n dbot_scores = []\n if len(hits) == 0:\n demisto.info('Get event got empty result')\n for item in hits:\n index = item.get('_index', '')\n source = item.get('_source', {})\n if index:\n source['_index'] = index\n source['timed_out'] = timed_out\n demisto.debug('This is my run_query result aellaEvent {}'.format(source))\n\n # Check url reputation\n url_str = source.get('url', '')\n if url_str:\n url_reputation = source.get('url_reputation', '')\n if url_reputation and url_reputation != 'Good':\n dbot_score = {\n 'Vendor' : 'Aella Data',\n 'Indicator' : url_str,\n 'Type' : 'url',\n 'Score' : 3,\n 'Malicious' : {\n 'Vendor' : 'Aella Data',\n 'Detections' : 'URL reputation {0}'.format(url_reputation),\n 'URL' : url_str\n }\n }\n else:\n dbot_score = {\n 'Vendor' : 'Aella Data',\n 'Indicator' : url_str,\n 'Type' : 'url',\n 'Malicious' : None\n }\n if url_reputation is None:\n # Unknonw\n dbot_score['Score'] = 0\n else:\n # Good\n dbot_score['Score'] = 1\n dbot_scores.append(dbot_score)\n\n # Check src ip reputation\n srcip_str = source.get('srcip', '')\n if srcip_str:\n srcip_reputation = source.get('srcip_reputation', '')\n if srcip_reputation and srcip_reputation != 'Good':\n dbot_score = {\n 'Vendor' : 'Aella Data',\n 'Indicator' : srcip_str,\n 'Type' : 'ip',\n 'Score' : 3,\n 'Malicious' : {\n 'Vendor' : 'Aella Data',\n 'Detections' : 'Source IP reputation {0}'.format(srcip_reputation),\n 'IP' : srcip_str\n }\n }\n else:\n dbot_score = {\n 'Vendor' : 'Aella Data',\n 'Indicator' : srcip_str,\n 'Type' : 'ip',\n 'Malicious' : None\n }\n if srcip_reputation is None:\n # Unknonw\n dbot_score['Score'] = 0\n else:\n # Good\n dbot_score['Score'] = 1\n dbot_scores.append(dbot_score)\n\n # Check dst ip reputation\n dstip_str = source.get('dstip', '')\n if dstip_str:\n dstip_reputation = source.get('dstip_reputation', '')\n if dstip_reputation and dstip_reputation != 'Good':\n dbot_score = {\n 'Vendor' : 'Aella Data',\n 'Indicator' : dstip_str,\n 'Type' : 'ip',\n 'Score' : 3,\n 'Malicious' : {\n 'Vendor' : 'Aella Data',\n 'Detections' : 'Destination IP reputation {0}'.format(dstip_reputation),\n 'IP' : dstip_str\n }\n }\n else:\n dbot_score = {\n 'Vendor' : 'Aella Data',\n 'Indicator' : dstip_str,\n 'Type' : 'ip',\n 'Malicious' : None\n }\n if dstip_reputation is None:\n # Unknonw\n dbot_score['Score'] = 0\n else:\n # Good\n dbot_score['Score'] = 1\n dbot_scores.append(dbot_score)\n\n break\n demisto.results({\n 'Type': entryTypes['note'],\n 'ContentsFormat': formats['json'],\n 'Contents': source,\n 'HumanReadable': tableToMarkdown('Aella Star Light Event <{0}>'.format(event_id), source),\n 'EntryContext': {\n 'Aella.Event(val._id==obj._id)': source,\n 'DBotScore' : createContext(dbot_scores, removeNull=True),\n }\n })\n else:\n demisto.info('Get event failed {}'.format(action_result))\n demisto.results(return_error('Failed to get event'))\n\n''' EXECUTION CODE '''\ndemisto.info('Command is {}'.format(demisto.command()))\n\nif demisto.command() == 'test-module':\n # This is the call made when pressing the integration test button.\n action_result = {}\n make_rest_call(URL + '/_cluster/health',\n USERNAME, PASSWORD, action_result\n )\n\n if action_result['status'] == 'Success':\n demisto.results('ok')\n else:\n demisto.results('failed')\n\nif demisto.command() == 'fetch-incidents':\n fetch_incidents_command()\n\nif demisto.command() == 'aella-get-event':\n aella_get_event_command()", "subtype": "", "type": "python"}, "modified": "2019-10-15T04:54:02.577442423-04:00", "name": "Aella Star Light", "prevName": "Aella Star Light", "shouldCommit": false, "sortValues": null, "system": true, "vcShouldIgnore": false, "version": 3}, {"brand": "", "canGetSamples": true, "category": "Data Enrichment & Threat Intelligence", "commitMessage": "", "configuration": [{"defaultValue": "2000000", "display": "Sensitivity threshold for configuring which domains are suspicious versus trusted.", "hidden": false, "name": "threshold", "options": null, "required": true, "type": 0}, {"defaultValue": "", "display": "Use System Proxy", "hidden": false, "name": "proxy", "options": null, "required": false, "type": 8}, {"defaultValue": "", "display": "Trust any certificate (Not Secure)", "hidden": false, "name": "insecure", "options": null, "required": false, "type": 8}], "description": "Alexa provides website ranking information that can be useful in determining if the domain in question has a strong web presence.", "detailedDescription": "To use this integration, simply call ```!domain```. !domain accepts\n \"domain\" as a parameter to query. Threshold is the point where a domain will be\n considered unknown versus suspicious.\n", "display": "Alexa Rank Indicator", "hidden": false, "icon": "", "id": "Alexa Rank Indicator", "image": "", "integrationScript": {"commands": [{"arguments": [{"default": false, "deprecated": false, "description": "domain to search", "name": "domain", "required": true, "secret": false}], "cartesian": false, "deprecated": false, "description": "Provides an Alexa ranking of the Domain in question.", "execution": false, "hidden": false, "important": null, "name": "domain", "outputs": [{"contentPath": "", "contextPath": "Domain.Name", "description": "The Domain being checked", "type": "string"}, {"contentPath": "", "contextPath": "DBotScore.Score", "description": "DBot score returned", "type": "number"}, {"contentPath": "", "contextPath": "DBotScore.Vendor", "description": "Vendor reporting the score", "type": "string"}, {"contentPath": "", "contextPath": "DBotScore.Domain", "description": "Domain being reported", "type": "string"}, {"contentPath": "", "contextPath": "Alexa.Domain.Data", "description": "The Domain being checked", "type": "string"}, {"contentPath": "", "contextPath": "Alexa.Domain.Rank", "description": "Alexa rank as determined by Amazon", "type": "string"}], "permitted": false, "sensitive": false, "timeout": 0}], "isFetch": false, "isFetchCredentials": false, "longRunning": false, "longRunningPortMapping": false, "runOnce": false, "script": "\n\n\n\nimport xml.etree.ElementTree as ET # type: ignore\nimport requests\n\n# Disable insecure warnings\nrequests.packages.urllib3.disable_warnings()\n\n\n\"\"\"GLOBAL VARIABLES/CONSTANTS\"\"\"\nTHRESHOLD = int(demisto.params().get('threshold'))\nUSE_SSL = not demisto.params().get('insecure', False)\n\n\n\"\"\"COMMAND FUNCTIONS\"\"\"\n\n\ndef alexa_domain_command():\n domain = demisto.args().get('domain')\n resp = requests.request('GET', 'https://data.alexa.com/data?cli=10&dat=s&url={}'.format(domain), verify=USE_SSL)\n root = ET.fromstring(str(resp.content))\n try:\n rank = root.find(\"SD[0]/POPULARITY\").attrib['TEXT'] # type: ignore\n if int(rank) > THRESHOLD:\n dbot_score = 2\n dbot_score_text = 'suspicious'\n else:\n dbot_score = 0\n dbot_score_text = 'unknown'\n except AttributeError:\n rank = 'Unknown'\n dbot_score = 2\n dbot_score_text = 'suspicious'\n dom_ec = {'Name': domain}\n dbot_ec = {\n 'Score': dbot_score,\n 'Vendor': 'Alexa Rank Indicator',\n 'Domain': domain,\n 'Type': 'domain'\n }\n ec = {\n 'Domain(val.Name && val.Name == obj.Name)': dom_ec,\n 'DBotScore': dbot_ec,\n 'Alexa.Domain(val.Name && val.Name == obj.Domain.Name)': {\n 'Name': domain,\n 'Rank': rank\n }\n }\n hr_string = ('The Alexa rank of {} is {} and has been marked as {}'\n ' while the threshold is {}'.format(domain, rank, dbot_score_text, THRESHOLD))\n demisto.results({\n 'Type': entryTypes['note'],\n 'ContentsFormat': formats['markdown'],\n 'Contents': xml2json(resp.content),\n 'HumanReadable': hr_string,\n 'EntryContext': ec\n })\n\n\ndef test_module_command():\n domain = 'google.com'\n resp = requests.request('GET', 'https://data.alexa.com/data?cli=10&dat=s&url={}'.format(domain), verify=USE_SSL)\n root = ET.fromstring(str(resp.content))\n rank = root.find(\"SD[0]/POPULARITY\").attrib['TEXT'] # type: ignore\n if rank == '1':\n result = 'ok'\n else:\n result = 'An error has occurred'\n return result\n\n\n\"\"\"EXECUTION BLOCK\"\"\"\ntry:\n handle_proxy()\n if demisto.command() == 'test-module':\n test_result = test_module_command()\n demisto.results(test_result)\n if demisto.command() == 'domain':\n alexa_domain_command()\nexcept Exception as e:\n LOG(e)\n LOG.print_log(False)\n return_error(e.message)", "subtype": "", "type": "python"}, "modified": "2019-10-15T04:54:10.330318-04:00", "name": "Alexa Rank Indicator", "prevName": "Alexa Rank Indicator", "shouldCommit": false, "sortValues": null, "system": true, "vcShouldIgnore": false, "version": 5}, {"brand": "", "canGetSamples": true, "category": "Network Security", "commitMessage": "", "configuration": [{"defaultValue": "", "display": "Server URL (e.g. https://192.168.0.1)", "hidden": false, "name": "server", "options": null, "required": true, "type": 0}, {"defaultValue": "", "display": "Credentials", "hidden": false, "name": "credentials", "options": null, "required": true, "type": 9}, {"defaultValue": "", "display": "Trust any certificate (unsecure)", "hidden": false, "name": "insecure", "options": null, "required": false, "type": 8}, {"defaultValue": "", "display": "Use system proxy settings", "hidden": false, "name": "proxy", "options": null, "required": false, "type": 8}], "description": "Algosec BusinessFlow(ABF), Firewall Analyzer (AFA) and FireFlow(AFF).", "display": "AlgoSec", "hidden": false, "icon": "", "id": "AlgoSec", "image": "", "integrationScript": {"commands": [{"arguments": [{"default": true, "deprecated": false, "description": "ID of requested change request", "name": "ticketId", "required": true, "secret": false}], "cartesian": false, "deprecated": false, "description": "Retrieves a FireFlow change request by its ID", "execution": false, "hidden": false, "important": null, "name": "algosec-get-ticket", "outputs": null, "permitted": false, "sensitive": false, "timeout": 0}, {"arguments": [{"default": false, "deprecated": false, "description": "A free text description of the issue", "name": "description", "required": false, "secret": false}, {"default": false, "deprecated": false, "description": "A list of device names, on which the change should be made", "name": "devices", "required": false, "secret": false}, {"default": false, "deprecated": false, "description": "The device action to perform for the traffic. This can be either\nof the following: \\U0010FC00 1 - Allow the traffic \\U0010FC00 0 - Block the\ntraffic\n", "name": "action", "predefined": ["0", "1"], "required": true, "secret": false}, {"default": false, "deprecated": false, "description": "The destination address to perform the action on", "name": "destAddress", "required": true, "secret": false}, {"default": false, "deprecated": false, "description": "The source address to perform the action on", "name": "sourceAddress", "required": true, "secret": false}, {"default": false, "deprecated": false, "description": "The email address of the requestor", "name": "requestor", "required": true, "secret": false}, {"default": false, "deprecated": false, "description": "The change request's title", "name": "subject", "required": true, "secret": false}, {"default": false, "deprecated": false, "description": "The device service or port for the connection, for example, \"http\" or \ufffc\ufffc\ufffc\ufffc\ufffc\ufffc\ufffc\ufffcMandatory \"tcp/123\"", "name": "service", "required": true, "secret": false}, {"default": false, "deprecated": false, "description": "The user for the connection", "name": "user", "required": true, "secret": false}, {"default": false, "deprecated": false, "description": "The application for the connection", "name": "application", "required": true, "secret": false}], "cartesian": false, "deprecated": false, "description": "Creates a new FireFlow change request", "execution": false, "hidden": false, "important": null, "name": "algosec-create-ticket", "outputs": null, "permitted": false, "sensitive": false, "timeout": 0}, {"arguments": [{"default": true, "deprecated": false, "description": "The IP/Subnet to search", "name": "address", "required": true, "secret": false}, {"auto": "PREDEFINED", "default": false, "deprecated": false, "description": "The search method for the address", "name": "type", "predefined": ["INTERSECT", "CONTAINED", "CONTAINING", "EXACT"], "required": false, "secret": false}], "cartesian": false, "deprecated": false, "description": "Find applications containing network objects related to IP address using BusinessFlow", "execution": false, "hidden": false, "important": null, "name": "algosec-get-applications", "outputs": null, "permitted": false, "sensitive": false, "timeout": 0}, {"arguments": [{"default": true, "deprecated": false, "description": "The IP/Subnet to search", "name": "address", "required": true, "secret": false}, {"auto": "PREDEFINED", "default": false, "deprecated": false, "description": "The search method for the address (default is INTERSECT)", "name": "type", "predefined": ["INTERSECT", "CONTAINED", "CONTAINING", "EXACT"], "required": false, "secret": false}], "cartesian": false, "deprecated": false, "description": "Find network objects related to IP address", "execution": false, "hidden": false, "important": null, "name": "algosec-get-network-object", "outputs": null, "permitted": false, "sensitive": false, "timeout": 0}, {"arguments": [{"default": true, "deprecated": false, "description": "source(s) for the query. Multiple values are separated by commas (,)", "name": "source", "required": true, "secret": false}, {"default": false, "deprecated": false, "description": "destination(s) for the query. Multiple values are separated by commas (,)", "name": "destination", "required": true, "secret": false}, {"default": false, "deprecated": false, "description": "service(s) for the query. Multiple values are separated by commas (,)", "name": "service", "required": true, "secret": false}, {"default": false, "deprecated": false, "description": "user for the query", "name": "user", "required": false, "secret": false}, {"default": false, "deprecated": false, "description": "application for the query", "name": "application", "required": false, "secret": false}], "cartesian": false, "deprecated": false, "description": "Performs a batch traffic simulation query using Firewall Analyzer", "execution": false, "hidden": false, "important": null, "name": "algosec-query", "outputs": null, "permitted": false, "sensitive": false, "timeout": 0}], "isFetch": false, "isFetchCredentials": false, "longRunning": false, "longRunningPortMapping": false, "runOnce": false, "script": "var username = params.credentials.identifier;\nvar password = params.credentials.password;\nvar server = params.server.replace(/[\\/]+$/, '');\nvar insecure = params.insecure;\nvar proxy = params.proxy;\nvar baseAFF = server + '/WebServices/WSDispatcher.pl';\nvar baseAFA = server + '/afa/php/ws.php';\nvar baseBF = server + '/BusinessFlow/rest/v1/network_objects/find';\n\nfunction fillInSoapContent(content, service) {\n var request = service === 'AFF' ? '%content%' : '%content%';\n return replaceInTemplates(request, {content: content});\n}\n\nfunction fillInSOAPRequestTemplate(method, content, service) {\n var template = service === 'AFF' ? '1%content%' : '%content%';\n return fillInSoapContent(replaceInTemplates(template, {content: content, method: method}), service);\n}\n\nvar responseDict = {\n 'authenticate': /(.*)<\\/sessionId/,\n 'getTicket': /((.|\\n)*)<\\/soap:Body/,\n 'createTicket': /((.|\\n)*)<\\/soap:Body/,\n 'ConnectRequest': /(.*)<\\/SessionID/,\n 'DisconnectRequest': /(.*)<\\/ns1:DisconnectResponse/,\n 'QueryRequest': /((.|\\n)*)<\\/SOAP-ENV:Body/\n};\n\nvar commandToMethod = {\n 'algosec-get-ticket': 'getTicket',\n 'algosec-create-ticket': 'createTicket',\n 'algosec-query': 'QueryRequest'\n};\n\nvar commandToURL = {\n 'algosec-get-applications': '/applications',\n 'algosec-get-network-object': ''\n};\n\nvar sessionData = '%sessionId%';\nvar methodDict = {\n 'authenticate': '%username%%password%',\n 'getTicket': sessionData + '%ticketId%',\n 'trafficLines': '%action%
%destAddress%
ftp
%sourceAddress%
%user%%application%
',\n 'createTicket': sessionData + '%requestor%%subject%%trafficLines%',\n 'ConnectRequest': '%username%%password%',\n 'DisconnectRequest': '%sessionId%',\n 'QueryRequest': '%sessionId%%query%',\n 'QueryInput': '%source%%destination%%service%'\n};\n\nvar affCommands = ['algosec-create-ticket', 'algosec-get-ticket'];\nvar bfCommands = ['algosec-get-applications', 'algosec-get-network-object'];\n\nfunction createTemplate(method, args) {\n switch (method) {\n case 'createTicket':\n var trafficLines =\n (args.description ? '\"%description%\"' : '') +\n methodDict['trafficLines'];\n return replaceInTemplates(methodDict[method], {trafficLines: trafficLines});\n case 'QueryRequest':\n var query = methodDict['QueryInput'] +\n (args.user ? '%user%' : '') +\n (args.application ? '%application%' : '');\n return replaceInTemplates(methodDict[method], {query: query});\n default:\n return methodDict[method];\n }\n}\n\nfunction sendSOAPRequest(method, args, service) {\n var req = fillInSOAPRequestTemplate(method, replaceInTemplates(createTemplate(method, args), args), service);\n var res = http(\n service === 'AFF' ? baseAFF : baseAFA,\n {\n Method: 'POST',\n Body: req\n },\n insecure,\n proxy\n );\n if (res.StatusCode < 200 || res.StatusCode >= 300) {\n throw 'Failed to ' + method + ', request status code: ' + res.StatusCode + ' and Body: ' + res.Body + '.';\n }\n return res.Body;\n}\n\nfunction sendRESTRequest(url, args) {\n var res = http(\n baseBF + url + encodeToURLQuery(args),\n {\n Method: 'GET',\n Username: username,\n Password: password\n },\n insecure,\n proxy\n );\n if (res.StatusCode < 200 || res.StatusCode >= 300) {\n throw 'Failed to ' + url + ', request status code: ' + res.StatusCode + ' and Body: ' + res.Body + '.';\n }\n return res.Body;\n}\n\nfunction sendAndParse(method, args, service) {\n var responseXML = sendSOAPRequest(method, args, service);\n var match = responseDict[method].exec(responseXML);\n if (match && match[1]) {\n return match[1];\n }\n throw method +' failed';\n}\n\nif (command === 'test-module') {\n if (sendAndParse('authenticate', {username: username, password: password}, 'AFF')) {\n return 'ok';\n }\n return 'something is wrong';\n}\n\nif (affCommands.indexOf(command) !== -1) {\n args.sessionId = sendAndParse('authenticate', {username: username, password: password}, 'AFF');\n return JSON.parse(x2j(sendAndParse(commandToMethod[command], args, 'AFF')));\n}\n\nif (bfCommands.indexOf(command) !== -1) {\n return JSON.parse(sendRESTRequest(commandToURL[command] ,args));\n}\n\nargs.sessionId = sendAndParse('ConnectRequest', {username: username, password: password}, 'AFA');\nres = JSON.parse(x2j(sendAndParse(commandToMethod[command], args)));\nsendAndParse('DisconnectRequest', {sessionId: args.sessionId}, 'AFA');\nreturn res;", "subtype": "", "type": "javascript"}, "modified": "2019-02-06T17:46:15.191691973-05:00", "name": "AlgoSec", "prevName": "AlgoSec", "shouldCommit": false, "sortValues": null, "system": true, "vcShouldIgnore": false, "version": 1}, {"brand": "", "canGetSamples": true, "category": "Data Enrichment & Threat Intelligence", "commitMessage": "", "configuration": [{"defaultValue": "https://otx.alienvault.com", "display": "Server URL (e.g. https://192.168.0.1)", "hidden": false, "name": "server", "options": null, "required": true, "type": 0}, {"defaultValue": "443", "display": "Port", "hidden": false, "name": "port", "options": null, "required": true, "type": 0}, {"defaultValue": "", "display": "API key", "hidden": false, "name": "apikey", "options": null, "required": true, "type": 4}, {"defaultValue": "", "display": "Trust any certificate (unsecure)", "hidden": false, "name": "insecure", "options": null, "required": false, "type": 8}, {"defaultValue": "", "display": "Use system proxy settings", "hidden": false, "name": "proxy", "options": null, "required": false, "type": 8}], "description": "Query IOCs in AlienVault", "detailedDescription": "To get Your API Key, go to [AlienVault OTX](https://otx.alienvault.com).\nAfter you login (or sign in), under Your profile settings You will find a OTX Key section.\nThis section contain Your key.\n", "display": "AlienVault OTX", "hidden": false, "icon": "", "id": "AlienVault OTX", "image": "", "integrationScript": {"commands": [{"arguments": [{"default": true, "deprecated": false, "description": "ip to query", "name": "ip", "required": true, "secret": false}], "cartesian": false, "deprecated": false, "description": "Query IP in AlienVault OTX", "execution": false, "hidden": false, "important": null, "name": "ip", "outputs": [{"contentPath": "", "contextPath": "IP.Address", "description": "IP Address", "type": ""}, {"contentPath": "", "contextPath": "IP.ASN", "description": "IP ASN", "type": ""}, {"contentPath": "", "contextPath": "IP.Geo.Country", "description": "IP country", "type": ""}, {"contentPath": "", "contextPath": "IP.Geo.Latitude", "description": "IP latitude", "type": ""}, {"contentPath": "", "contextPath": "IP.Geo.Longitude", "description": "IP longitude", "type": ""}, {"contentPath": "", "contextPath": "AlienVaultOTX.Reputation", "description": "IP reputation", "type": ""}], "permitted": false, "sensitive": false, "timeout": 0}, {"arguments": [{"default": true, "deprecated": false, "description": "domain to query", "name": "domain", "required": true, "secret": false}], "cartesian": false, "deprecated": false, "description": "Query domain in AlienVault OTX", "execution": false, "hidden": false, "important": null, "name": "domain", "outputs": [{"contentPath": "", "contextPath": "Domain.Name", "description": "Domain name", "type": ""}, {"contentPath": "", "contextPath": "Domain.AlienVaultOTX.Alexa", "description": "Alexa URL for domain data", "type": ""}, {"contentPath": "", "contextPath": "Domain.AlienVaultOTX.Whois", "description": "Whois URL for domain data", "type": ""}], "permitted": false, "sensitive": false, "timeout": 0}, {"arguments": [{"default": true, "deprecated": false, "description": "ip to query", "name": "ip", "required": true, "secret": false}], "cartesian": false, "deprecated": false, "description": "Query IP in AlienVault OTX", "execution": false, "hidden": false, "important": null, "name": "ipv6", "outputs": [{"contentPath": "", "contextPath": "IP.Address", "description": "IP Address", "type": ""}, {"contentPath": "", "contextPath": "IP.ASN", "description": "IP ASN", "type": ""}, {"contentPath": "", "contextPath": "IP.AlienVaultOTX.Reputation", "description": "AlienVault IP reputation", "type": ""}], "permitted": false, "sensitive": false, "timeout": 0}, {"arguments": [{"default": true, "deprecated": false, "description": "hostname to query", "name": "hostname", "required": true, "secret": false}], "cartesian": false, "deprecated": false, "description": "Query hostname in AlienVault OTX", "execution": false, "hidden": false, "important": null, "name": "hostname", "outputs": [{"contentPath": "", "contextPath": "Endpoint.Hostname", "description": "Endpoint hostname", "type": ""}, {"contentPath": "", "contextPath": "Endpoint.AlienVaultOTX.Alexa", "description": "Endpoint Alexa URL", "type": ""}, {"contentPath": "", "contextPath": "Endpoint.AlienVaultOTX.Whois", "description": "Endpoint Whois URL", "type": ""}], "permitted": false, "sensitive": false, "timeout": 0}, {"arguments": [{"default": true, "deprecated": false, "description": "md5 to query", "name": "file", "required": true, "secret": false}], "cartesian": false, "deprecated": false, "description": "Query file in AlienVault OTX", "execution": false, "hidden": false, "important": null, "name": "file", "outputs": [{"contentPath": "", "contextPath": "File.MD5", "description": "File md5", "type": ""}], "permitted": false, "sensitive": false, "timeout": 0}, {"arguments": [{"default": false, "deprecated": false, "description": "File md5", "name": "md5", "required": false, "secret": false}, {"default": false, "deprecated": false, "description": "File sha1", "name": "sha1", "required": false, "secret": false}, {"default": false, "deprecated": false, "description": "File sha256", "name": "sha256", "required": false, "secret": false}], "cartesian": false, "deprecated": false, "description": "Query file in AlienVault OTX", "execution": false, "hidden": false, "important": null, "name": "alienvault-query-file", "outputs": [{"contentPath": "", "contextPath": "File.MD5", "description": "File md5", "type": ""}, {"contentPath": "", "contextPath": "File.SHA1", "description": "File sha1", "type": ""}, {"contentPath": "", "contextPath": "File.SHA256", "description": "File sha256", "type": ""}], "permitted": false, "sensitive": false, "timeout": 0}, {"arguments": [{"default": false, "deprecated": false, "description": "The page of pulses to get", "name": "page", "required": false, "secret": false}], "cartesian": false, "deprecated": false, "description": "Search for pulses in AlienVault OTX", "execution": false, "hidden": false, "important": null, "name": "alienvault-search-pulses", "outputs": [{"contentPath": "", "contextPath": "AlienVault.Pulses.ID", "description": "Pulse ID", "type": ""}, {"contentPath": "", "contextPath": "AlienVault.Pulses.Author.ID", "description": "Author ID", "type": ""}, {"contentPath": "", "contextPath": "AlienVault.Pulses.Author.Username", "description": "Author username", "type": ""}, {"contentPath": "", "contextPath": "AlienVault.Pulses.Count", "description": "Pulse count", "type": ""}, {"contentPath": "", "contextPath": "AlienVault.Pulses.Modified", "description": "Pulse modification date", "type": ""}, {"contentPath": "", "contextPath": "AlienVault.Pulses.Name", "description": "Pulse name", "type": ""}, {"contentPath": "", "contextPath": "AlienVault.Pulses.Source", "description": "Pulse source", "type": ""}, {"contentPath": "", "contextPath": "AlienVault.Pulses.SubscriberCount", "description": "Pulse subscriber count", "type": ""}, {"contentPath": "", "contextPath": "AlienVault.Pulses.Tags", "description": "Pulse tags", "type": ""}, {"contentPath": "", "contextPath": "AlienVault.Pulses.Description", "description": "Pulse description", "type": ""}], "permitted": false, "sensitive": false, "timeout": 0}, {"arguments": [{"default": false, "deprecated": false, "description": "ID of desired pules\"", "name": "pulse-id", "required": true, "secret": false}], "cartesian": false, "deprecated": false, "description": "Get pulse details (pulse are alian vault's entities)", "execution": false, "hidden": false, "important": null, "name": "alienvault-get-pulse-details", "outputs": [{"contentPath": "", "contextPath": "AlienVault.Pulses.Description", "description": "Pulse description", "type": ""}, {"contentPath": "", "contextPath": "AlienVault.Pulses.Created", "description": "Pulse created", "type": ""}, {"contentPath": "", "contextPath": "AlienVault.Pulses.Author.Username", "description": "Pulse author username", "type": ""}, {"contentPath": "", "contextPath": "AlienVault.Pulses.ID", "description": "Pulse ID", "type": ""}, {"contentPath": "", "contextPath": "AlienVault.Pulses.Name", "description": "Pulse name", "type": ""}, {"contentPath": "", "contextPath": "AlienVault.Pulses.Tags", "description": "Pulse tags", "type": ""}, {"contentPath": "", "contextPath": "AlienVault.Pulses.TargetedCountries", "description": "Pulse targeted countries", "type": ""}, {"contentPath": "", "contextPath": "AlienVault.Pulses.Description", "description": "Pulse description", "type": ""}], "permitted": false, "sensitive": false, "timeout": 0}, {"arguments": [{"default": true, "deprecated": false, "description": "url to query", "name": "url", "required": true, "secret": false}], "cartesian": false, "deprecated": false, "description": "Query url in AlienVault OTX", "execution": false, "hidden": false, "important": null, "name": "url", "outputs": [{"contentPath": "", "contextPath": "URL.Data", "description": "URL ", "type": ""}], "permitted": false, "sensitive": false, "timeout": 0}], "isFetch": false, "isFetchCredentials": false, "longRunning": false, "longRunningPortMapping": false, "runOnce": false, "script": "function getArgName(args) {\n options = ['md5', 'sha1', 'sha256'];\n for (var k in options) {\n if (args[options[k]]) {\n return options[k];\n }\n }\n throw 'You must provide either sha1, sha256 or md5';\n}\n\nindicatorsCommands = {\n 'ip': {\n type: 'IPv4',\n argName: 'ip',\n contextKey: outputPaths.ip,\n translator: [\n {to: 'Address', from: 'indicator'},\n {to: 'ASN', from: 'asn'},\n {to: 'Geo-Country', from: 'country_name'},\n {to: 'Geo-Latitude', from: 'latitude'},\n {to: 'Geo-Longitude', from: 'longitude'},\n {to: 'AlienVaultOTX-Reputation', from: 'reputation'},\n ],\n tableTitle: 'AlienVault ip response %Address%',\n },\n 'url': {\n type: 'url',\n argName: 'url',\n contextKey: outputPaths.url,\n translator: [\n {to: 'Data', from: 'indicator'},\n ],\n tableTitle: 'AlienVault url response %Data%',\n },\n 'ipv6': {\n type: 'IPv6',\n argName: 'ip',\n contextKey: outputPaths.ip,\n translator: [\n {to: 'Address', from: 'indicator'},\n {to: 'ASN', from: 'asn'},\n {to: 'AlienVaultOTX-Reputation', from: 'reputation'},\n ],\n tableTitle: 'AlienVault ip response %Address%',\n },\n 'domain': {\n type: 'domain',\n argName: 'domain',\n contextKey: outputPaths.domain,\n translator: [\n {to: 'Name', from: 'indicator'},\n {to: 'AlienVaultOTX-Alexa', from: 'alexa'},\n {to: 'AlienVaultOTX-Whois', from: 'whois'},\n ],\n tableTitle: 'AlienVault domain response %Name%',\n },\n 'hostname': {\n type: 'hostname',\n argName: 'hostname',\n contextKey: 'Endpoint(val.Hostname==obj.Hostname)',\n translator: [\n {to: 'Hostname', from: 'indicator'},\n {to: 'AlienVaultOTX-Alexa', from: 'alexa'},\n {to: 'AlienVaultOTX-Whois', from: 'whois'},\n ],\n tableTitle: 'AlienVault hostname response %Hostname%',\n },\n 'file': {\n type: 'file',\n argName: 'file',\n contextKey: outputPaths.file,\n translator: [\n {to: 'MD5', from: 'indicator'},\n ],\n tableTitle: 'AlienVault file response %MD5%',\n },\n 'alienvault-query-file': {\n type: 'file',\n argName: getArgName,\n contextKey: outputPaths.file,\n translator: function(args) {\n return [\n {to: getArgName(args).toUpperCase(), from: 'indicator'},\n ];\n },\n tableTitle: function(args) {\n return 'AlienVault file response %' + getArgName(args).toUpperCase() + '%';\n },\n },\n};\n\nsearchCommands = {\n 'alienvault-search-pulses': {\n url: 'search/pulses',\n contextKey: 'AlienVault.Pulses(val.ID==obj.ID)',\n translator: [\n {to: 'ID', from: 'id'},\n {to: 'Author-ID', from: 'author.id'},\n {to: 'Author-Username', from: 'author.username'},\n {to: 'Count', from: 'indicator_count'},\n {to: 'Modified', from: 'modified'},\n {to: 'Name', from: 'name'},\n {to: 'Source', from: 'pulse_source'},\n {to: 'SubscriberCount', from: 'subscriber_count'},\n {to: 'Tags', from: 'tags'},\n {to: 'Description', from: 'description'},\n ],\n innerPath: 'results',\n tableTitle: 'AlienVault pulses',\n },\n 'alienvault-get-pulse-details': {\n url: 'pulses/%pulse-id%',\n contextKey: 'AlienVault.Pulses(val.ID==obj.ID)',\n translator: [\n {to: 'ID', from: 'id'},\n {to: 'Created', from: 'created'},\n {to: 'Author-Username', from: 'author_name'},\n {to: 'Description', from: 'description'},\n {to: 'Modified', from: 'modified'},\n {to: 'Name', from: 'name'},\n {to: 'Tags', from: 'tags'},\n {to: 'TargetedCountries', from: 'targeted_countries'},\n ],\n tableTitle: 'AlienVault pulse %ID%',\n },\n}\n\nvar mapObjFunction = function(mapFields) {\n var transformSingleObj= function(obj) {\n var res = {};\n mapFields.forEach(function(f) {\n res[f.to] = dq(obj, f.from);\n });\n return res;\n };\n return function(obj) {\n if (obj instanceof Array) {\n var res = [];\n for (var j=0; j < obj.length; j++) {\n res.push(transformSingleObj(obj[j]));\n }\n return res;\n }\n return transformSingleObj(obj);\n }\n}\n\nvar createContext = function(data) {\n var createContextSingle = function(obj) {\n var res = {};\n var keys = Object.keys(obj);\n keys.forEach(function(k) {\n var values = k.split('-');\n var current = res;\n for (var j = 0; j= 300) {\n throw 'Request to ' + url + ' failed, request status code: ' + res.StatusCode + ' and Body: ' + res.Body + '.';\n }\n // If we have a body, it should have application/json content type.\n if (res.Body &&\n (!res.Headers['Content-Type'] || !res.Headers['Content-Type'][0] || res.Headers['Content-Type'][0].indexOf('application/json') === -1) || res.Body.indexOf('Google Tag Manager') != -1) {\n log('Response body: ' + res.Body);\n throw 'Check your URL (excepting base URL) - got invalid content type, expected application/json.';\n }\n return res.Body ? JSON.parse(res.Body) : 'empty';\n}\n\nfunction queryIndicators(type, indicator, section) {\n return sendRequest('indicators/' + type + '/' + domain_from_url(indicator) + '/general');\n}\n\nif (command === 'test-module') {\n res = queryIndicators('IPv4', '8.8.8.8');\n return res !== 'Not found' ? 'ok' : 'Not found';\n}\n\ncurrentCommand = indicatorsCommands[command] || searchCommands[command];\nargName = currentCommand.argName instanceof Function ? currentCommand.argName(args) : currentCommand.argName;\nvar raw;\nif (indicatorsCommands[command]) {\n raw = queryIndicators(currentCommand.type, args[argName]);\n} else {\n raw = sendRequest(replaceInTemplatesAndRemove(currentCommand.url, args));\n}\nentry = {\n Type: entryTypes.note,\n Contents: raw,\n ContentsFormat: formats.json,\n ReadableContentsFormat: formats.markdown,\n};\n\ntranslator = currentCommand.translator instanceof Function ? currentCommand.translator(args) : currentCommand.translator;\ntableTitle = currentCommand.tableTitle instanceof Function ? currentCommand.tableTitle(args) : currentCommand.tableTitle;\ntranslated = mapObjFunction(translator)(currentCommand.innerPath ? raw[currentCommand.innerPath] : raw);\nentry.HumanReadable = tableToMarkdown(replaceInTemplates(tableTitle, translated), translated);\nif (currentCommand.contextKey) {\n ec = {};\n contextKey = currentCommand.getContextKey instanceof Function ? currentCommand.contextKey(args) : currentCommand.contextKey;\n outputPath = outputPaths.file;\n if (argName) {\n switch (argName) {\n case 'ip':\n outputPath = outputPaths.ip;\n break;\n case 'domain':\n outputPath = outputPaths.domain;\n break;\n case 'url':\n outputPath = outputPaths.url;\n break;\n }\n\n ec[contextKey] = createContext(translated);\n }\n entry.EntryContext = ec;\n if (ec && ec[contextKey] && ec[contextKey].Geo && ec[contextKey].Geo.Latitude && ec[contextKey].Geo.Longitude) {\n entry = [\n entry,\n {\n Type: entryTypes.map,\n Contents: {lat: parseFloat(ec[contextKey].Geo.Latitude),\n lng: parseFloat(ec[contextKey].Geo.Longitude)},\n ContentsFormat: formats.json\n }\n ];\n }\n}\n\nreturn entry;", "subtype": "", "type": "javascript"}, "modified": "2019-06-27T09:58:30.811824015-04:00", "name": "AlienVault OTX", "prevName": "AlienVault OTX", "shouldCommit": false, "sortValues": null, "system": true, "vcShouldIgnore": false, "version": 4}, {"brand": "", "canGetSamples": true, "category": "Data Enrichment & Threat Intelligence", "commitMessage": "", "configuration": [{"defaultValue": "https://www.example.com", "display": "Server URL (e.g., https://www.example.com)", "hidden": false, "name": "url", "options": null, "required": true, "type": 0}, {"defaultValue": "", "display": "Client ID", "hidden": false, "name": "client_id", "options": null, "required": true, "type": 0}, {"defaultValue": "", "display": "Client Secret", "hidden": false, "name": "client_secret", "options": null, "required": true, "type": 4}, {"defaultValue": "", "display": "Trust any certificate (not secure)", "hidden": false, "name": "insecure", "options": null, "required": false, "type": 8}, {"defaultValue": "", "display": "Use system proxy settings", "hidden": false, "name": "proxy", "options": null, "required": false, "type": 8}, {"defaultValue": "", "display": "Fetch incidents", "hidden": false, "name": "isFetch", "options": null, "required": false, "type": 8}, {"defaultValue": "", "display": "Incident type", "hidden": false, "name": "incidentType", "options": null, "required": false, "type": 13}, {"defaultValue": "3 days", "display": "First fetch timestamp (