forked from mitre-attack/attack-scripts
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathdiff_stix.py
706 lines (609 loc) · 30.5 KB
/
diff_stix.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
import argparse
from stix2 import MemoryStore, Filter, TAXIICollectionSource
from taxii2client.v20 import Collection
import os
import json
from tqdm import tqdm
import datetime
from string import Template
from itertools import chain
from dateutil import parser as dateparser
# helper maps
domainToDomainLabel = {
'enterprise-attack': 'Enterprise',
'mobile-attack': 'Mobile'
}
domainToTaxiiCollectionId = {
"enterprise-attack": "95ecc380-afe9-11e4-9b6c-751b66dd541e",
"mobile-attack": "2f669986-b40b-4423-b720-4396ca6a462b",
}
attackTypeToStixFilter = { # stix filters for querying for each type of data
'technique': [Filter('type', '=', 'attack-pattern')],
'software': [Filter('type', '=', 'malware'), Filter('type', '=', 'tool')],
'group': [Filter('type', '=', 'intrusion-set')],
'mitigation': [Filter('type', '=', 'course-of-action')]
}
attackTypeToPlural = { # because some of these pluralize differently
'technique': 'techniques',
'malware': 'malware',
'software': 'software',
'group': 'groups',
'mitigation': 'mitigations'
}
sectionNameToSectionHeaders = { # how we want to format headers for each section
"additions": "New {obj_type}",
"changes": "{obj_type} changes",
"minor_changes": "Minor {obj_type} changes",
"deprecations": "{obj_type} deprecations",
"revocations": "{obj_type} revocations",
"deletions": "{obj_type} deletions",
"unchanged": "Unchanged {obj_type}"
}
statusToColor = { # color key for layers
"additions": "#a1d99b",
"changes": "#fcf3a2",
"minor_changes": "#c7c4e0",
"deletions": "#ff00e1", # this will probably never show up but just in case
"revocations": "#ff9000",
"deprecations": "#ff6363",
"unchanged": "#ffffff"
}
statusDescriptions = { # explanation of modification types to data objects for legend in layer files
"additions": "objects which are present in the new data and not the old",
"changes": "objects which have a newer version number in the new data compared to the old",
"minor_changes": "objects which have a newer last edit date in the new data than in the old, but the same version number",
"revocations": "objects which are revoked in the new data but not in the old",
"deprecations": "objects which are deprecated in the new data but not in the old",
"deletions": "objects which are present in the old data but not the new",
"unchanged": "objects which did not change between the two versions"
}
class DiffStix(object):
"""
Utilities for detecting and summarizing differences between two versions of the ATT&CK content.
"""
def __init__(
self,
domains=['enterprise-attack', 'mobile-attack'],
layers=None,
markdown=None,
minor_changes=False,
unchanged=False,
new='new',
old='old',
show_key=False,
site_prefix='',
types=['technique', 'software', 'group', 'mitigation'],
use_taxii=False,
verbose=False
):
"""
Construct a new 'DiffStix' object.
params:
domains: list of domains to parse, e.g. enterprise-attack, mobile-attack
layers: array of output filenames for layer files, e.g. ['enterprise.json', 'mobile.json', 'pre.json']
markdown: output filename for markdown content to be written to
minor_changes: if true, also report minor changes section (changes which didn't increment version)
new: directory to load for new stix version
old: directory to load for old stix version
show_key: if true, output key to markdown file
site_prefix: prefix links in markdown output
types: which types of objects to report on, e.g technique, software
verbose: if true, print progress bar and status messages to stdout
"""
self.domains = domains
self.layers = layers
self.markdown = markdown
self.minor_changes = minor_changes
self.unchanged = unchanged
self.new = new
self.old = old
self.show_key = show_key
self.site_prefix = site_prefix
self.types = types
self.use_taxii = use_taxii
self.verbose = verbose
self.data = { # data gets load into here in the load() function. All other functionalities rely on this data structure
# technique {
# enterprise-attack {
# additions: [],
# deletions: [],
# changes: [],
# minor_changes: [],
# revocations: [],
# deprecations: [],
# unchanged: []
# }
# mobile-attack...
# }
# software...
}
self.stixIDToName = {} # stixID to object name
self.new_subtechnique_of_rels = [] # all subtechnique-of relationships in the new data
self.old_subtechnique_of_rels = [] # all subtechnique-of relationships in the old data
self.new_id_to_technique = {} # stixID => technique for every technique in the new data
self.old_id_to_technique = {} # stixID => technique for every technique in the old data
# build the bove data structures
self.load_data()
# remove duplicate relationships
self.new_subtechnique_of_rels = [i for n, i in enumerate(self.new_subtechnique_of_rels) if i not in self.new_subtechnique_of_rels[n+1:]]
self.old_subtechnique_of_rels = [i for n, i in enumerate(self.old_subtechnique_of_rels) if i not in self.old_subtechnique_of_rels[n+1:]]
def verboseprint(self, *args, **kwargs):
if self.verbose:
print(*args, **kwargs)
def getUrlFromStix(self, datum, is_subtechnique=False):
"""
Parse the website url from a stix object.
"""
url = datum['external_references'][0]['url']
split_url = url.split('/')
splitfrom = -3 if is_subtechnique else -2
link = '/'.join(split_url[splitfrom:])
return link
def deep_copy_stix(self, objects):
"""
Transform stix to dict and deep copy the dict.
"""
result = []
for obj in objects:
obj = dict(obj)
if 'external_references' in obj:
for i in range(len(obj['external_references'])):
obj['external_references'][i] = dict(
obj['external_references'][i])
if 'kill_chain_phases' in obj:
for i in range(len(obj['kill_chain_phases'])):
obj['kill_chain_phases'][i] = dict(obj['kill_chain_phases'][i])
if 'modified' in obj:
obj['modified'] = str(obj['modified'])
if 'definition' in obj:
obj['definition'] = dict(obj['definition'])
obj['created'] = str(obj['created'])
result.append(obj)
return result
# load data into data structure
def load_data(self):
"""
Load data from files into data dict.
"""
if self.verbose:
pbar = tqdm(total=len(self.types) * len(self.domains), desc="loading data", bar_format="{l_bar}{bar}| [{elapsed}<{remaining}, {rate_fmt}{postfix}]")
for obj_type in self.types:
for domain in self.domains:
# handle data loaded from either a directory or the TAXII server
def load_datastore(data_store):
raw_data = list(chain.from_iterable(
data_store.query(f) for f in attackTypeToStixFilter[obj_type]
))
raw_data = self.deep_copy_stix(raw_data)
id_to_obj = {item['id']: item for item in raw_data}
return {
"id_to_obj": id_to_obj,
"keys": set(id_to_obj.keys()),
"data_store": data_store
}
def parse_subtechniques(data_store, new=False):
# parse dataStore sub-technique-of relationships
if new:
for technique in list(data_store.query(attackTypeToStixFilter["technique"])):
self.new_id_to_technique[technique["id"]] = technique
self.new_subtechnique_of_rels += list(data_store.query([
Filter("type", "=", "relationship"),
Filter("relationship_type", "=", "subtechnique-of")
]))
else:
for technique in list(data_store.query(attackTypeToStixFilter["technique"])):
self.old_id_to_technique[technique["id"]] = technique
self.old_subtechnique_of_rels += list(data_store.query([
Filter("type", "=", "relationship"),
Filter("relationship_type", "=", "subtechnique-of")
]))
# load data from directory according to domain
def load_dir(dir, new=False):
data_store = MemoryStore()
datafile = os.path.join(dir, domain + ".json")
data_store.load_from_file(datafile)
parse_subtechniques(data_store, new)
return load_datastore(data_store)
# load data from TAXII server according to domain
def load_taxii(new=False):
collection = Collection("https://cti-taxii.mitre.org/stix/collections/" + domainToTaxiiCollectionId[domain])
data_store = TAXIICollectionSource(collection)
parse_subtechniques(data_store, new)
return load_datastore(data_store)
if self.use_taxii:
old = load_taxii(False)
else:
old = load_dir(self.old, False)
new = load_dir(self.new, True)
intersection = old["keys"] & new["keys"]
additions = new["keys"] - old["keys"]
deletions = old["keys"] - new["keys"]
# sets to store the ids of objects for each section
changes = set()
minor_changes = set()
revocations = set()
deprecations = set()
unchanged = set()
# find changes, revocations and deprecations
for key in intersection:
if "revoked" in new["id_to_obj"][key] and new["id_to_obj"][key]["revoked"]:
if not "revoked" in old["id_to_obj"][key] or not old["id_to_obj"][key]["revoked"]: # if it was previously revoked, it's not a change
# store the revoking object
revoked_by_key = new["data_store"].query([
Filter('type', '=', 'relationship'),
Filter('relationship_type', '=', 'revoked-by'),
Filter('source_ref', '=', key)
])
if (len(revoked_by_key) == 0):
print("WARNING: revoked object", key, "has no revoked-by relationship")
continue
else: revoked_by_key = revoked_by_key[0]["target_ref"]
new["id_to_obj"][key]["revoked_by"] = new["id_to_obj"][revoked_by_key]
revocations.add(key)
# else it was already revoked, and not a change; do nothing with it
elif "x_mitre_deprecated" in new["id_to_obj"][key] and new["id_to_obj"][key]["x_mitre_deprecated"]:
if not "x_mitre_deprecated" in old["id_to_obj"][key]: # if previously deprecated, not a change
deprecations.add(key)
else: # not revoked or deprecated
# try getting version numbers; should only lack version numbers if something has gone
# horribly wrong or a revoked object has slipped through
try:
old_version = float(old["id_to_obj"][key]["x_mitre_version"])
except:
print("ERROR: cannot get old version for object: " + key)
try:
new_version = float(new["id_to_obj"][key]["x_mitre_version"])
except:
print("ERROR: cannot get new version for object: " + key)
# check for changes
if new_version > old_version:
# an update has occurred to this object
changes.add(key)
else:
# check for minor change; modification date increased but not version
old_date = dateparser.parse(old["id_to_obj"][key]["modified"])
new_date = dateparser.parse(new["id_to_obj"][key]["modified"])
if new_date > old_date:
minor_changes.add(key)
else :
unchanged.add(key)
# set data
if obj_type not in self.data: self.data[obj_type] = {}
self.data[obj_type][domain] = {
"additions": [new["id_to_obj"][key] for key in additions],
"changes": [new["id_to_obj"][key] for key in changes]
}
# only create minor_changes data if we want to display it later
if self.minor_changes:
self.data[obj_type][domain]["minor_changes"] = [new["id_to_obj"][key] for key in minor_changes]
# ditto for unchanged
if self.unchanged:
self.data[obj_type][domain]["unchanged"] = [new["id_to_obj"][key] for key in unchanged]
self.data[obj_type][domain]["revocations"] = [new["id_to_obj"][key] for key in revocations]
self.data[obj_type][domain]["deprecations"] = [new["id_to_obj"][key] for key in deprecations]
# only show deletions if objects were deleted
if len(deletions) > 0:
self.data[obj_type][domain]["deletions"] = [old["id_to_obj"][key] for key in deletions]
if self.verbose:
pbar.update(1)
if self.verbose:
pbar.close()
def get_md_key(self):
"""
Create string describing each type of difference (change, addition, etc). Used in get_markdown_string.
Includes minor changes if the DiffStix instance was instantiated with the minor_changes argument.
Includes deletions if the changes include deletions.
"""
have_deletions = False
for types in self.data.keys():
for domain in self.data[types].keys():
if "deletions" in self.data[types][domain].keys():
have_deletions = True
key = "#### Key\n\n"
key += (
"* New objects: " + statusDescriptions['additions'] + "\n"
"* Object changes: " + statusDescriptions['changes'] + "\n"
)
if self.minor_changes:
key += "* Minor object changes: " + statusDescriptions['minor_changes'] + "\n"
if self.unchanged:
key += "* Unchanged objects: " + statusDescriptions['unchanged'] + "\n"
key += (
"* Object revocations: " + statusDescriptions['revocations'] + "\n"
"* Object deprecations: " + statusDescriptions['deprecations']
)
if have_deletions:
key += "\n" + "* Object deletions: " + statusDescriptions['deletions']
return f"{key}"
def has_subtechniques(self, sdo, new=False):
"""return true or false depending on whether the SDO has sub-techniques. new determines whether to parse from the new or old data"""
if new: return len(list(filter(lambda rel: rel["target_ref"] == sdo["id"], self.new_subtechnique_of_rels))) > 0
else: return len(list(filter(lambda rel: rel["target_ref"] == sdo["id"], self.old_subtechnique_of_rels))) > 0
def get_markdown_string(self):
"""
Return a markdown string summarizing detected differences.
"""
def getSectionList(items, obj_type, section):
"""
parse a list of items in a section and return a string for the items
"""
# get parents which have children
childless = list(filter(lambda item: not self.has_subtechniques(item, True) and not ("x_mitre_is_subtechnique" in item and item["x_mitre_is_subtechnique"]), items))
parents = list(filter(lambda item: self.has_subtechniques(item, True) and not ("x_mitre_is_subtechnique" in item and item["x_mitre_is_subtechnique"]), items))
children = { item["id"]: item for item in filter(lambda item: "x_mitre_is_subtechnique" in item and item["x_mitre_is_subtechnique"], items) }
subtechnique_of_rels = self.new_subtechnique_of_rels if section != "deletions" else self.old_subtechnique_of_rels
id_to_technique = self.new_id_to_technique if section != "deletions" else self.old_id_to_technique
parentToChildren = {} # stixID => [ children ]
for relationship in subtechnique_of_rels:
if relationship["target_ref"] in parentToChildren:
if relationship["source_ref"] in children:
parentToChildren[relationship["target_ref"]].append(children[relationship["source_ref"]])
else:
if relationship["source_ref"] in children:
parentToChildren[relationship["target_ref"]] = children[relationship["source_ref"]]
parentToChildren[relationship["target_ref"]] = [ children[relationship["source_ref"]] ]
# now group parents and children
groupings = []
for parent in childless + parents:
parent_children = parentToChildren.pop(parent["id"]) if parent["id"] in parentToChildren else []
groupings.append({
"parent": parent,
"parentInSection": True,
"children": parent_children
})
for parentID in parentToChildren:
groupings.append({
"parent": id_to_technique[parentID],
"parentInSection": False,
"children": parentToChildren[parentID]
})
groupings = sorted(groupings, key=lambda grouping: grouping["parent"]["name"])
def placard(item):
"""get a section list item for the given SDO according to section type"""
if section == "revocations":
revoker = item['revoked_by']
if "x_mitre_is_subtechnique" in revoker and revoker["x_mitre_is_subtechnique"]:
# get revoking technique's parent for display
parentID = list(filter(lambda rel: rel["source_ref"] == revoker["id"], subtechnique_of_rels))[0]["target_ref"]
parentName = id_to_technique[parentID]["name"] if parentID in id_to_technique else "ERROR NO PARENT"
return f"{item['name']} (revoked by { parentName}: [{revoker['name']}]({self.site_prefix}/{self.getUrlFromStix(revoker, True)}))"
else:
return f"{item['name']} (revoked by [{revoker['name']}]({self.site_prefix}/{self.getUrlFromStix(revoker)}))"
if section == "deletions":
return f"{item['name']}"
else:
is_subtechnique = item["type"] == "attack-pattern" and "x_mitre_is_subtechnique" in item and item["x_mitre_is_subtechnique"]
return f"[{item['name']}]({self.site_prefix}/{self.getUrlFromStix(item, is_subtechnique)})"
# build sectionList string
sectionString = ""
for grouping in groupings:
if grouping["parentInSection"]:
sectionString += f"* { placard(grouping['parent']) }\n"
# else:
# sectionString += f"* _{grouping['parent']['name']}_\n"
for child in sorted(grouping["children"], key=lambda child: child["name"]):
if grouping["parentInSection"]:
sectionString += f"\t* {placard(child) }\n"
else:
sectionString += f"* { grouping['parent']['name'] }: { placard(child) }\n"
return sectionString
self.verboseprint("generating markdown string... ", end="", flush="true")
content = ""
for obj_type in self.data.keys():
domains = ""
for domain in self.data[obj_type]:
domain_sections = ""
for section in self.data[obj_type][domain]:
if len(self.data[obj_type][domain][section]) > 0: # if there are items in the section
section_items = getSectionList(self.data[obj_type][domain][section], obj_type, section)
else: # no items in section
section_items = "No changes"
header = sectionNameToSectionHeaders[section] + ":"
if "{obj_type}" in header:
if section == "additions":
header = header.replace("{obj_type}", attackTypeToPlural[obj_type].capitalize())
else: header = header.replace("{obj_type}", obj_type.capitalize())
if section_items == "No changes":
domain_sections += f"{header}\n{section_items}\n\n" # e.g "added techniques:"
else: domain_sections += f"{header}\n\n{section_items}\n\n" # add empty line between header and section list
domains += f"**{domainToDomainLabel[domain]}**\n\n{domain_sections}" # e.g "enterprise"
content += f"### {attackTypeToPlural[obj_type].capitalize()}\n\n{domains}" # e.g "techniques"
if self.show_key:
key_content = self.get_md_key()
content = f"{key_content}\n\n{content}"
self.verboseprint("done")
return content
def get_layers_dict(self):
"""
Return ATT&CK Navigator layers in dict format summarizing detected differences. Returns a dict mapping domain to its layer dict.
"""
self.verboseprint("generating layers dict... ", end="", flush="true")
layers = {}
thedate = datetime.datetime.today().strftime('%B %Y')
# for each layer file in the domains mapping
for domain in self.domains:
# build techniques list
techniques = []
used_statuses = set()
for status in self.data["technique"][domain]:
if status == "revocations" or status == "deprecations": continue
for technique in self.data["technique"][domain][status]:
for phase in technique['kill_chain_phases']:
techniques.append({
"techniqueID": technique['external_references'][0]['external_id'],
"tactic": phase['phase_name'],
"enabled": True,
"color": statusToColor[status],
"comment": status[:-1] if status != "unchanged" else status # trim s off end of word
})
used_statuses.add(status)
# build legend based off used_statuses
legendItems = list(map(lambda status: {"color": statusToColor[status], "label": status + ": " + statusDescriptions[status]}, used_statuses))
# build layer structure
layer_json = {
"versions": {
"layer": "4.1",
"navigator": "4.1"
},
"name": f"{thedate} {domainToDomainLabel[domain]} Updates",
"description": f"{domainToDomainLabel[domain]} updates for the {thedate} release of ATT&CK",
"domain": domain,
"techniques": techniques,
"sorting": 0,
"hideDisabled": False,
"legendItems": legendItems,
"showTacticRowBackground": True,
"tacticRowBackground": "#205b8f",
"selectTechniquesAcrossTactics": True
}
layers[domain] = layer_json
self.verboseprint("done")
return layers
def markdown_string_to_file(outfile, content):
"""
Print the string passed in to the indicated output file.
"""
verboseprint("writing markdown string to file... ", end="", flush="true")
outfile = open(outfile, "w")
outfile.write(content)
outfile.close()
verboseprint("done")
def layers_dict_to_files(outfiles, layers):
"""
Print the layers dict passed in to layer files.
"""
verboseprint("writing layers dict to layer files... ", end="", flush="true")
# write each layer to separate files
if 'enterprise-attack' in layers: json.dump(layers['enterprise-attack'], open(outfiles[0], "w"), indent=4)
if 'mobile-attack' in layers: json.dump(layers['mobile-attack'], open(outfiles[1], "w"), indent=4)
verboseprint("done")
if __name__ == '__main__':
old_dir_default = "old"
date = datetime.datetime.today()
md_default = os.path.join("output", f"updates-{date.strftime('%B-%Y').lower()}.md")
layer_defaults = [
os.path.join("output", f"{date.strftime('%B_%Y')}_Updates_Enterprise.json"),
os.path.join("output", f"{date.strftime('%B_%Y')}_Updates_Mobile.json"),
os.path.join("output", f"{date.strftime('%B_%Y')}_Updates_Pre.json")
]
parser = argparse.ArgumentParser(
description="Create -markdown and/or -layers reporting on the changes between two versions of the ATT&CK content. Takes STIX bundles as input. For default operation, put enterprise-attack.json and mobile-attack.json bundles in 'old' and 'new' folders for the script to compare."
)
parser.add_argument("-old",
type=str,
metavar="OLD_DIR",
help=f"the directory of the old content. Default is '{old_dir_default}'"
)
parser.add_argument("-new",
type=str,
metavar="NEW_DIR",
default="new",
help="the directory of the new content. Default is '%(default)s'"
)
parser.add_argument("-types",
type=str,
nargs="+",
metavar=("OBJ_TYPE", "OBJ_TYPE"),
choices=[
"technique", "software", "group", "mitigation"
],
default=[
"technique", "software", "group", "mitigation"
],
help="which types of objects to report on. Choices (and defaults) are %(choices)s"
)
parser.add_argument("-domains",
type=str,
nargs="+",
metavar="DOMAIN",
choices=[
"enterprise-attack", "mobile-attack"
],
default=[
"enterprise-attack", "mobile-attack"
],
help="which domains to report on. Choices (and defaults) are %(choices)s"
)
parser.add_argument("-markdown",
type=str,
nargs="?",
const=md_default, # default if no value specified
help="create a markdown file reporting changes. If value is unspecified, defaults to %(const)s"
)
parser.add_argument("-layers",
type=str,
nargs="*",
# metavar=("ENTERPRISE", "MOBILE", "PRE"),
help=f'''
create layer files showing changes in each domain
expected order of filenames is 'enterprise', 'mobile', 'pre attack'.
If values are unspecified, defaults to {", ".join(layer_defaults)}
'''
)
parser.add_argument("-site_prefix",
type=str,
default="",
help="prefix links in markdown output, e.g. [prefix]/techniques/T1484"
)
parser.add_argument("-v", "--verbose",
action="store_true",
help="print progress bars and status messages"
)
parser.add_argument("--minor-changes",
action="store_true",
help="show changes to objects which didn't increment the version number"
)
parser.add_argument("--unchanged",
action="store_true",
help="show objects without changes in the markdown output"
)
parser.add_argument("--use-taxii",
action="store_true",
help="Use content from the ATT&CK TAXII server for the -old data"
)
parser.add_argument("--show-key",
action="store_true",
help="Add a key explaining the change types to the markdown"
)
args = parser.parse_args()
if args.use_taxii and args.old is not None:
parser.error('--use-taxii and -old cannot be used together')
if (not args.markdown and args.layers is None):
print("Script doesn't output anything unless -markdown and/or -layers are specified. Run 'python3 diff_stix.py -h' for usage instructions")
exit()
if args.old is None:
args.old = old_dir_default
diffStix = DiffStix(
domains=args.domains,
layers=args.layers,
markdown=args.markdown,
minor_changes=args.minor_changes,
unchanged=args.unchanged,
new=args.new,
old=args.old,
show_key=args.show_key,
site_prefix=args.site_prefix,
types=args.types,
use_taxii=args.use_taxii,
verbose=args.verbose
)
if args.verbose:
def verboseprint(*args, **kwargs):
print(*args, **kwargs)
else:
verboseprint = lambda *a, **k: None
if args.markdown:
md_string = diffStix.get_markdown_string()
markdown_string_to_file(args.markdown, md_string)
if args.layers is not None:
if len(args.layers) == 0:
# no files specified, e.g. '-layers', use defaults
diffStix.layers = layer_defaults
args.layers = layer_defaults
elif len(args.layers) == 3:
# files specified, e.g. '-layers file.json file2.json file3.json', use specified
diffStix.layers = args.layers # assumes order of files is enterprise, mobile, pre attack (same order as defaults)
else:
parser.error('-layers requires exactly three files to be specified or none at all')
layers_dict = diffStix.get_layers_dict()
layers_dict_to_files(args.layers, layers_dict)