@@ -3,7 +3,8 @@ commonfields:
3
3
version : -1
4
4
name : Intezer
5
5
display : Intezer
6
- category : Data Enrichment & Threat Intelligence
6
+ releaseNotes : Add file upload command and update category
7
+ category : Forensics & Malware Analysis
7
8
image: 
8
9
description : Malware detection and analysis based on code reuse
9
10
detaileddescription : |
@@ -25,52 +26,82 @@ script:
25
26
var SERVER_API = 'https://analyze.intezer.com/api/';
26
27
var IS_AVAILABLE_URL = 'is-available';
27
28
var HASH_URL = 'v1-2/analyze-by-sha256';
29
+ var FILE_UPLOAD_URL = 'v1-2/analyze';
28
30
var ANALYSIS_RESULT_URL = 'v1-2/analyses/';
29
31
var SHA_256_REGEXP = new RegExp('[A-Fa-f0-9]{64}');
30
32
var SUCCESS_STATUS = 'succeeded';
31
33
var ERROR_PREFIX = 'Error from Intezer: ';
32
34
33
- var sendRequest = function(method, url, parameters ) {
35
+ var sendRequest = function(method, url, body, acceptedCodes ) {
34
36
// handle '/' at the end of the url
35
37
if (url[url.length - 1] === '/') {
36
38
url = url.substring(0, url.length - 1);
37
39
}
38
- if (!parameters ) {
39
- parameters = {};
40
+ if (!body ) {
41
+ body = {};
40
42
}
41
- parameters .api_key = params.APIKey;
43
+ body .api_key = params.APIKey;
42
44
// no need to handle url query encoding in GET (as no such api in Intezer)
43
45
var result = http(
44
46
SERVER_API + url,
45
47
{
46
48
Headers: {'Content-Type': ['application/json'], 'Accept': ['application/json']},
47
49
Method: method,
48
- Body: method == 'POST' ? JSON.stringify(parameters ) : ''
50
+ Body: method == 'POST' ? JSON.stringify(body ) : ''
49
51
},
50
52
false,
51
53
params.useproxy
52
54
);
53
55
56
+ return handleResponse(result, url, acceptedCodes);
57
+ }
58
+
59
+ var sendMultipartRequest = function(url, file, body) {
60
+ // handle '/' at the end of the url
61
+ if (url[url.length - 1] === '/') {
62
+ url = url.substring(0, url.length - 1);
63
+ }
64
+ if (!body) {
65
+ body = {};
66
+ }
67
+ body.api_key = params.APIKey;
68
+
69
+ var result = httpMultipart(
70
+ SERVER_API + url,
71
+ file,
72
+ {
73
+ Method: 'POST',
74
+ Headers: {'Content-Type': ['multipart/form-data']},
75
+ },
76
+ body
77
+ );
78
+
79
+ return handleResponse(result, url);
80
+ };
81
+
82
+ var handleResponse = function(result, url, acceptedCodes) {
83
+ var ignoreStatusCode = acceptedCodes && (acceptedCodes.indexOf(result.StatusCode) !== -1);
84
+
54
85
// validate response
55
- if (result.StatusCode < 200 || result.StatusCode > 299) {
86
+ if (!ignoreStatusCode && ( result.StatusCode < 200 || result.StatusCode > 299) ) {
56
87
switch (result.StatusCode) {
57
- case ' 400' :
88
+ case 400:
58
89
throw ERROR_PREFIX + '400 Bad Request - Wrong or invalid parameters\n';
59
- case ' 401' :
90
+ case 401:
60
91
throw ERROR_PREFIX + '401 Unauthorized - Wrong or invalid api key\n';
61
- case ' 403' :
92
+ case 403:
62
93
throw ERROR_PREFIX + '403 Forbidden - The API key is not valid\n';
63
- case ' 404' :
94
+ case 404:
64
95
throw ERROR_PREFIX + '404 Not Found - Access to expired analysis URL or sha256 not found for analysis by sha256\n';
65
- case ' 410' :
96
+ case 410:
66
97
throw ERROR_PREFIX + '410 Gone - Analysis no longer exists in the service\n';
67
- case ' 413' :
98
+ case 413:
68
99
throw ERROR_PREFIX + '413 Payload Too Large - Large request\n';
69
- case ' 415' :
100
+ case 415:
70
101
throw ERROR_PREFIX + '415 Unsupported Media Type - Wrong or missing content type\n';
71
- case ' 500' :
102
+ case 500:
72
103
throw ERROR_PREFIX + '500 Internal Server Error - Internal error\n';
73
- case ' 503' :
104
+ case 503:
74
105
throw ERROR_PREFIX + '503 Service Unavailable\n';
75
106
default:
76
107
throw ERROR_PREFIX + 'Failed to perform request ' + SERVER_API + url + ', request status code: ' + result.StatusCode + '\n';
@@ -93,12 +124,12 @@ script:
93
124
94
125
var getAnalysis = function(analysisId) {
95
126
var analysisRes = sendRequest('POST', ANALYSIS_RESULT_URL + analysisId);
127
+
96
128
return analysisRes.obj;
97
129
};
98
130
99
131
var analyzeSHA256 = function(hash, codeItemType, maxRetries, delay) {
100
132
// validate file is indeed sha256
101
- var entries = [];
102
133
if (!SHA_256_REGEXP.test(hash)) {
103
134
var warning = ERROR_PREFIX + 'File hash is not valid sha256.\nIntezer file hash reputation supports only sha256 hash.\n\n';
104
135
return {
@@ -112,15 +143,57 @@ script:
112
143
var res = sendRequest('POST', HASH_URL, {
113
144
sha256: hash,
114
145
code_item_type: codeItemType
115
- });
146
+ }, [404] );
116
147
117
148
var analysisId;
118
- if (res.statusCode === 201) {
149
+
150
+ if (res.statusCode === 404) {
151
+ var dBotScore = [{Indicator: hash, Type: 'hash', Vendor: 'Intezer', Score: 0}];
152
+ var file = {
153
+ SHA256: hash,
154
+ ExistsInIntezer: false,
155
+ Malicious: {
156
+ Vendor: 'Intezer'
157
+ }
158
+ };
159
+ var ec = {
160
+ 'File(val.SHA256==obj.SHA256)': file,
161
+ DBotScore: dBotScore
162
+ };
163
+
164
+ return [{
165
+ Type: entryTypes.note,
166
+ Contents: '',
167
+ ContentsFormat: formats.json,
168
+ HumanReadable: 'Hash ' + hash + ' does not exist on Intezer genome database',
169
+ EntryContext: ec
170
+ }];
171
+ } else if (res.statusCode === 201) {
119
172
analysisId = res.obj.analysis_id;
120
173
} else {
121
174
throw ERROR_PREFIX + 'Failed to create sha256 analysis for ' + hash + ', request status code: ' + res.statusCode + '\n';
122
175
}
123
176
177
+ return waitForResponse(analysisId, maxRetries, delay);
178
+ }
179
+
180
+ var analyzeUploadedFile = function(fileEntryId, codeItemType, maxRetries, delay) {
181
+ var res = sendMultipartRequest(FILE_UPLOAD_URL, fileEntryId, {code_item_type: codeItemType});
182
+
183
+ var analysisId;
184
+
185
+ if (res.statusCode === 201) {
186
+ analysisId = res.obj.analysis_id;
187
+ } else {
188
+ throw ERROR_PREFIX + 'Failed to create analysis, request status code: ' + res.statusCode + '\n';
189
+ }
190
+
191
+ return waitForResponse(analysisId, maxRetries, delay);
192
+ }
193
+
194
+ var waitForResponse = function(analysisId, maxRetries, delay) {
195
+ var entries = [];
196
+
124
197
// perform wait loop
125
198
res = getAnalysis(analysisId);
126
199
var tries = 0;
@@ -133,10 +206,11 @@ script:
133
206
134
207
// if polling did not retrieve response after {delay * maxRetries} seconds, return an error message
135
208
if (res.status !== SUCCESS_STATUS) {
136
- throw ERROR_PREFIX + 'Failed to pool sha256 analysis for ' + hash + ' , Try to change maxRetries or delay arguments\n';
209
+ throw ERROR_PREFIX + 'Failed to analyze , Try to change maxRetries or delay arguments\n';
137
210
} else {
138
211
// Create new indicator of sha256 according to verdict
139
212
var verdict = res.result.verdict;
213
+ var hash = res.result.sha256;
140
214
var dbotScore = 0;
141
215
var ec = {};
142
216
ec[outputPaths.file] = [];
@@ -153,7 +227,14 @@ script:
153
227
} else if (verdict === 'trusted' || verdict === 'neutral') {
154
228
dbotScore = 1;
155
229
}
156
- ec.DBotScore = [{Indicator: hash, Type: 'hash', Vendor: 'Intezer', Score: dbotScore}];
230
+ var dBotScore = [{Indicator: hash, Type: 'hash', Vendor: 'Intezer', Score: dbotScore}];
231
+ var file = {
232
+ SHA256: hash,
233
+ ExistsInIntezer: true,
234
+ Malicious: {
235
+ Vendor: 'Intezer'
236
+ }
237
+ };
157
238
var data = res.result;
158
239
var md = '## Intezer analysis result\n';
159
240
md += 'SHA256: ' + hash + '\n';
@@ -169,7 +250,10 @@ script:
169
250
Contents: res.body,
170
251
ContentsFormat: formats.json,
171
252
HumanReadable: md,
172
- EntryContext: ec
253
+ EntryContext: {
254
+ 'File(val.SHA256==obj.SHA256)': file,
255
+ DBotScore: dBotScore
256
+ }
173
257
});
174
258
}
175
259
@@ -181,7 +265,9 @@ script:
181
265
// send is available
182
266
return sendRequest('GET', IS_AVAILABLE_URL).obj.is_available;
183
267
case 'file':
184
- return analyzeSHA256(args.file, args.codeItemType, args.maxRetries, args.delay);
268
+ return analyzeSHA256(args.file, args.codeItuemType, args.maxRetries, args.delay);
269
+ case 'intezer-upload':
270
+ return analyzeUploadedFile(args.fileEntryId, args.codeItemType, args.maxRetries, args.delay);
185
271
default:
186
272
}
187
273
type : javascript
@@ -226,5 +312,53 @@ script:
226
312
- contextPath : DBotScore.Score
227
313
description : The actual score
228
314
type : number
315
+ - contextPath : Intezer.File
316
+ description : ' The file '
317
+ type : unknown
318
+ - contextPath : File.ExistsInIntezer
319
+ description : ' File exists in Intezer genome database (in case file does not
320
+ exist, consider upload the file) '
321
+ type : boolean
229
322
description : Checks file reputation of the given hash, supports SHA256
323
+ - name : intezer-upload
324
+ arguments :
325
+ - name : fileEntryId
326
+ required : true
327
+ default : true
328
+ description : The file entry id to upload
329
+ - name : codeItemType
330
+ auto : PREDEFINED
331
+ predefined :
332
+ - file
333
+ - memory_module
334
+ - fileless_code
335
+ description : Item type - File (file), Memory Module (memory_module) or Memory
336
+ Fileless Code (fileless_code)
337
+ defaultValue : file
338
+ - name : maxRetries
339
+ description : Number of retries for polling the analysis report
340
+ defaultValue : " 60"
341
+ - name : delay
342
+ description : Delay in seconds between polling of analysis results
343
+ defaultValue : " 5"
344
+ outputs :
345
+ - contextPath : File.SHA256
346
+ description : Bad hash SHA256
347
+ type : string
348
+ - contextPath : File.Malicious.Vendor
349
+ description : For malicious files, the vendor that made the decision
350
+ type : string
351
+ - contextPath : DBotScore.Indicator
352
+ description : The indicator we tested
353
+ type : string
354
+ - contextPath : DBotScore.Type
355
+ description : The type of the indicator
356
+ type : string
357
+ - contextPath : DBotScore.Vendor
358
+ description : Vendor used to calculate the score
359
+ type : string
360
+ - contextPath : DBotScore.Score
361
+ description : The actual score
362
+ type : number
363
+ description : Checks file reputation for uploaded file (up to 20MB)
230
364
runonce : false
0 commit comments