diff --git a/Integrations/integration-HybridAnalysis.yml b/Integrations/integration-HybridAnalysis.yml new file mode 100644 index 000000000000..e7fc07774466 --- /dev/null +++ b/Integrations/integration-HybridAnalysis.yml @@ -0,0 +1,484 @@ +commonfields: + id: Hybrid Analysis + version: -1 +name: Hybrid Analysis +display: Hybrid Analysis +category: Forensics & Malware Analysis +image:  +description: Fully automated malware analysis with unique Hybrid Analysis. +defaultEnabled: true +fromversion: 3.6.1 +configuration: +- display: Server URL (e.g. https://216.3.128.82) + name: serverUrl + defaultvalue: https://www.hybrid-analysis.com + type: 0 + required: true +- display: API Key + name: apiKey + defaultvalue: "" + type: 4 + required: false +- display: Do not validate server certificate (insecure) + name: insecure + defaultvalue: "" + type: 8 + required: false +- display: Use system proxy settings + name: proxy + defaultvalue: "" + type: 8 + required: false +script: + script: |+ + var serverUrl = params.serverUrl; + var insecure = params.insecure; + var proxy = params.proxy; + var version = params.version; + var apiKey; + var integrationContext = getIntegrationContext(); + var licenseID = getLicenseID(); + var HEADERS = { + 'User-Agent': ['Hybrid Analysis'], + 'accept': ['application/json'], + 'Content-Type': ['application/x-www-form-urlencoded'], + 'DemistoLicense': [licenseID] + }; + + if (params.apiKey) { + apiKey = params.apiKey; + } else if (integrationContext.apiKey) { + apiKey = integrationContext.apiKey; + } else { + apiKey = generateKey(); + } + + HEADERS['api-key'] = [apiKey]; + + // handle '/' at the end of serverUrl + if (serverUrl[serverUrl.length - 1] === '/') { + serverUrl = serverUrl.substring(0, serverUrl.length - 1); + } + + function generateKey() { + HEADERS['api-key'] = [integrationContext.masterApiKey]; + var cmdUrl = '/api/v2/key/create'; + var response = sendRequest('POST', cmdUrl, 'uid=DemistoLimitedEdition'); + key = response.api_key; + integrationContext.apiKey = key; + setIntegrationContext(integrationContext); + return key; + } + + function entryError(errorCode, text) { + var error = 'Hybrid Analysis returned an error (' + errorCode + ') - ' + text; + return {Type: entryTypes.error, ContentsFormat: formats.text, Contents: error}; + } + + // return a function that maps object keys by mapper (or capitlize keys if key is not exists in mapper) + function mapObject(mapper) { + return function(obj) { + var res = {}; + Object.keys(obj).forEach(function(key) { + // map key or capitalize if not exists + var newKey = mapper[key] || key; + res[newKey] = obj[key]; + }); + return res; + }; + } + + function createTableEntry(name, rawResponse, table, context, headers) { + return { + Type: entryTypes.note, + ContentsFormat: formats.json, + Contents: rawResponse, + ReadableContentsFormat: formats.markdown, + HumanReadable: tableToMarkdown(name, table, headers, undefined, headerTransform=underscoreToCamelCase, removeNull=true), + EntryContext: context + }; + } + + function sendRequest(method, endpoint, body) { + var requestUrl = serverUrl + endpoint; + var res = http( + requestUrl, + { + Method: method, + Headers: HEADERS, + Body: body + }, + insecure, + proxy + ); + + if (res.StatusCode < 200 || res.StatusCode >= 300) { + throw 'Request Failed.\nStatus code: ' + res.StatusCode + '.\nBody: ' + JSON.stringify(res) + '.'; + } + + var body; + try { + body = JSON.parse(res.Body); + } catch (ex) { + throw 'Error parsing response - ' + res.Body + ' - ' + ex; + } + + return body; + } + + function scan(hash) { + return sendRequest('POST', '/api/v2/search/hash', 'hash='+hash); + } + function scanToEntry(res) { + + var response = res; + + // create table from response + var tableMapper = { + threatlevel: 'threat level', + total_network_connections: 'total network connections', + targeturl: 'target url', + classification_tags: 'classification tags', + threatscore: 'threat score', + total_processes: 'total processes', + submitname: 'submit name', + environmentDescription: 'environment description', + isinteresting: 'interesting', + environmentId: 'environment id', + isurlanalysis: 'url analysis', + analysis_start_time: 'analysis start time', + total_signatures: 'total signatures' + }; + + var table = response.map(mapObject(tableMapper)); + + // create context from response + var context = {}; + + var contextMapper = { + sha1: 'SHA1', + sha256: 'SHA256', + md5: 'MD5', + }; + context[outputPaths.file] = response.map(mapObject(contextMapper)); + response.forEach(function(res) { + if(res.threatlevel && res.threatlevel > 1) { + addMalicious(context, outputPaths.file, { + MD5: res.md5, + SHA1: res.sha1, + SHA256: res.sha256, + Malicious: { Vendor: 'Hybrid Analysis', Description: 'Score above ' + res.threatscore } + }); + } + }); + return createTableEntry('Scan Results:', response, table, context); + } + + function submitFile (entryId, environmentId) { + var requestUrl = serverUrl + '/api/v2/submit/file'; + + // submit file + var res = httpMultipart( + requestUrl, // URL + entryId, // Optional - FilePath / EntryID + { + Method: 'POST', + Headers: HEADERS + }, + { // Multipart Contents + environment_id: environmentId // For API v2 + }, + insecure, + proxy + ); + + if (res.StatusCode < 200 || res.StatusCode >= 300) { + throw 'Multipart Request Failed.\nStatus code: ' + res.StatusCode + '.\nBody: ' + JSON.stringify(res) + '.'; + } + var body; + + try { + body = JSON.parse(res.Body); + } catch (ex) { + throw 'Error parsing response - ' + res.Body + ' - ' + ex; + } + + var response = body; + var resMessage = ''; + if (response.sha256) { + resMessage = resMessage + '\nsha256 - ' + response.sha256; + } + if (response['job_id']) { + resMessage = resMessage + '\nJob ID - ' + response['job_id']; + } + if (response['environment_id']) { + resMessage = resMessage + '\nEnvironment ID - ' + response['environment_id']; + } + + return resMessage; + } + + function submitFileMsg(hash) { + var resMessage = "File submitted successfully" + hash; + return resMessage; + } + + function searchQuery(query) { + body = ''; + if (args.query) { + args.query.split(',').forEach(function(keyValue){ + splittedObject = keyValue.split(/:(.+)/); // Split by first ':' only + key = splittedObject[0]; + value = splittedObject[1]; + body += key + '=' + value + '&' + }); + } else { + // Build Crowd Strike query syntax from arguments, i.e. key:value + for (var key in args) { + body += key + '=' + args[key] + '&'; + } + } + var res = sendRequest('POST', '/api/v2/search/terms', body); + + var result = res.result; + + // create table from search result + var tableMapper = { + environmentDescription: 'environment description', + start_time: 'start time', + submitname: 'submit name', + threatscore: 'threat score', + type_short: 'type short', + }; + + var table = result.map(mapObject(tableMapper)); + + // create context from search result + var contextMapper = { + job_id: 'JobID', + sha256: 'SHA256', + environment_id: 'EnvironmentID' + }; + + var context = { + 'HybridAnalysis.Search(val.JobID && val.JobID == obj.JobID || val.SHA256 && val.SHA256 == obj.SHA256)': result.map(mapObject(contextMapper)) + }; + + result.forEach(function(res) { + if(res.threatlevel && res.threatlevel > 1) { + addMalicious(context, outputPaths.file, { + MD5: res.md5, + SHA1: res.sha1, + SHA256: res.sha256, + Malicious: { Vendor: 'Hybrid Analysis', Description: 'Score above ' + res.threatscore } + }); + } + }); + + return createTableEntry('Search results:', result, table, context); + } + + function detonateFile(entryId, delay, timeout) { + var environmentId; + if (args.environmentID) { + environmentId = args.environmentID; + } else { + var environments = getEnvironments().backend.global_environment; + for (var i=0; i 0) { + return scanToEntry(res); + } else { + waitTime = waitTime + delayTime; + wait(delayTime); + } + } + throw ('Timeout due to no answer after ' + timeOut + ' seconds.'); + + } + + switch (command) { + case 'test-module': + var entry = searchQuery('url:google'); + if (entry && entry.Type === entryTypes.note) { + return 'ok'; + } + return entry && entry.Contents; + case 'hybrid-analysis-scan': + var res = scan(args.file); + return scanToEntry(res); + case 'hybrid-analysis-submit-sample': + var hash = submitFile(args.entryId, args.environmentID); + return submitFileMsg(hash); + case 'hybrid-analysis-search': + return searchQuery(args.query); + case 'hybrid-analysis-detonate-file': + return detonateFile(args.entryId, args.delay, args.timeout); + } + + type: javascript + commands: + - name: hybrid-analysis-scan + arguments: + - name: file + required: true + default: true + description: File hash (MD5, SHA1 or SHA256) + outputs: + - contextPath: File.SHA256 + description: SHA256 of the file + - contextPath: File.SHA1 + description: SHA1 of the file + - contextPath: File.MD5 + description: MD5 of the file + - contextPath: File.environmentId + description: 'Environment id of the file ' + - contextPath: File.analysis_start_time + description: Analysis start time of the file + - contextPath: File.submitname + description: Submission name of the file + - contextPath: File.classification_tags + description: List of classification tags of the file + - contextPath: File.vxfamily + description: Family classification of the file + - contextPath: File.total_network_connections + description: Total network connections of the file + - contextPath: File.total_processes + description: Total processes count of the file + - contextPath: File.total_signatures + description: Total signatures count if the file + - contextPath: File.hosts + description: List of file's hosts + - contextPath: File.isinteresting + description: If server found this file interesting + - contextPath: File.domains + description: List of file's related domains + - contextPath: File.isurlanalysis + description: If file analyzed by url + - contextPath: File.Malicious.Vendor + description: or malicious files, the vendor that made the decision + - contextPath: File.Malicious.Description + description: For malicious files, the reason for the vendor to make the decision + description: Get summary information for a given MD5, SHA1 or SHA256 and all the + reports generated for any environment ID + - name: hybrid-analysis-submit-sample + arguments: + - name: entryId + required: true + default: true + description: War-room entry ID of sample file + - name: environmentID + required: true + description: Environment ID to submit file to (get all IDs via vx-get-environments) + description: Submit a file from investigation to analysis server (NOTE - minimum + required authorization is `default`) + - name: hybrid-analysis-search + arguments: + - name: query + description: Hybrid Analysis query syntax (see `/faq#advanced-search-options` + for more details). examples - url:google, host:95.181.53.78 + - name: filename + description: Filename e.g. invoice.exe + - name: filetype + description: Filetype e.g. docx + - name: filetype_desc + description: Filetype description e.g. PE32 executable + - name: env_id + description: Environment Id + - name: country + description: Country (3 digit ISO) e.g. swe + - name: verdict + description: 'Verdict e.g. 1 (available: 1 ''whitelisted’, 2 ''no verdict’, + 3 ''no specific threat’, 4 ''suspicious’, 5 ‘malicious’)' + - name: av_detect + description: AV Multiscan range e.g. 50-70 (min 0, max 100) + - name: vx_family + description: AV Family Substring e.g. nemucod + - name: tag + description: Hashtag e.g. ransomware + - name: port + description: Port e.g. 8080 + - name: host + description: Host e.g. 192.168.0.1 + - name: domain + description: Domain e.g. checkip.dyndns.org + - name: url + description: HTTP Request Substring e.g. google + - name: similar_to + description: Similar Samples e.g. + - name: context + description: Sample Context e.g. + - name: imp_hash + description: Import Hash + - name: ssdeep + description: SSDeep + - name: authentihash + description: Authentication Hash + outputs: + - contextPath: HybridAnalysis.Search.SHA256 + description: SHA256 of search result + - contextPath: HybridAnalysis.Search.SHA1 + description: SHA1of search result + - contextPath: HybridAnalysis.Search.MD5 + description: MD5 of search result + - contextPath: HybridAnalysis.Search.environmentId + description: Environment Id of search result + - contextPath: HybridAnalysis.Search.start_time + description: Start time of search result + - contextPath: HybridAnalysis.Search.threatscore + description: Threat score of search result by server + - contextPath: HybridAnalysis.Search.verdict + description: Verdict of search result + - contextPath: HybridAnalysis.Search.environmentDescription + description: Environment description of search result + - contextPath: HybridAnalysis.Search.submitname + description: Submission name of search result + - contextPath: HybridAnalysis.Search.vxfamily + description: Family of search result + - contextPath: HybridAnalysis.Search.threatscore + description: Threat score of search result + - contextPath: HybridAnalysis.Search.type_short + description: type of search result (url, host, etc.) + - contextPath: HybridAnalysis.Search.size + description: Size of search result + - contextPath: File.Malicious.Vendor + description: or malicious files, the vendor that made the decision + - contextPath: File.Malicious.Description + description: For malicious files, the reason for the vendor to make the decision + description: Search the database using the Hybrid Analysis search syntax + - name: hybrid-analysis-detonate-file + arguments: + - name: entryId + required: true + description: War-room entry ID of sample file + - name: environmentID + description: environment ID to submit file to(get all via vx-get-environments). + Default is 100, or other WINDOWS ID + - name: delay + description: Delay wait time between calls (in seconds) + defaultValue: "3" + - name: timeout + description: Total wait time (in seconds) + defaultValue: "60" + description: Detonate file through Hybrid Analysis + runonce: false