-
Notifications
You must be signed in to change notification settings - Fork 364
/
Copy pathattack_range.py
235 lines (192 loc) · 9.56 KB
/
attack_range.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
import os
import sys
import argparse
from modules.config_handler import ConfigHandler
from modules.aws_controller import AwsController
from modules.azure_controller import AzureController
from modules import configuration
# need to set this ENV var due to a OSX High Sierra forking bug
# see this discussion for more details: https://github.com/ansible/ansible/issues/34056#issuecomment-352862252
os.environ['OBJC_DISABLE_INITIALIZE_FORK_SAFETY'] = 'YES'
def init(args):
config_path = args.config
print("""
__
.d$$b
.' TO$;\\
/ : TP._;
/ _.; :Tb|
/ / ;j$j
_.-" d$$$$
.' .. d$$$$;
/ /P' d$$$$P. |\\
/ " .d$$$P' |\\^"l
.' `T$P^\"\"\"\"\" :
._.' _.' ;
`-.-".-'-' ._. _.-" .-"
`.-" _____ ._ .-"
-(.g$$$$$$$b. .'
""^^T$$$P^) .(:
_/ -" /.' /:/;
._.'-'`-' ")/ /;/;
`-.-"..--"" " / / ;
.-" ..--"" -' :
..--""--.-" (\\ .-(\\
..--"" `-\\(\\/;`
_. :
;`-
:\\
; bug
By: Splunk Threat Research Team [STRT] - research@splunk.com
""")
# parse config
config = ConfigHandler.read_config(config_path)
ConfigHandler.validate_config(config)
if config['general']['cloud_provider'] == 'aws':
config.pop('azure')
config.pop('local')
controller = AwsController(config)
elif config['general']['cloud_provider'] == 'azure':
config.pop('aws')
config.pop('local')
controller = AzureController(config)
elif config['general']['cloud_provider'] == 'local':
from modules.vagrant_controller import VagrantController
config.pop('azure')
config.pop('aws')
controller = VagrantController(config)
return controller
def simulate(args):
controller = init(args)
controller.simulate(args.engine, args.target, args.technique, args.playbook)
def dump(args):
controller = init(args)
controller.dump(args.file_name, args.search, args.earliest, args.latest)
def replay(args):
controller = init(args)
controller.replay(args.file_name, args.index, args.sourcetype, args.source)
def build(args):
controller = init(args)
controller.build()
def destroy(args):
controller = init(args)
controller.destroy()
def stop(args):
controller = init(args)
instance_ids = [id.strip() for id in args.instance_ids.split(',')] if args.instance_ids else None
controller.stop(instance_ids)
def resume(args):
controller = init(args)
instance_ids = [id.strip() for id in args.instance_ids.split(',')] if args.instance_ids else None
controller.resume(instance_ids)
def packer(args):
controller = init(args)
controller.packer(args.image_name)
def configure(args):
configuration.new(args.config)
def show(args):
controller = init(args)
controller.show()
def create_remote_backend(args):
controller = init(args)
controller.create_remote_backend(args.backend_name)
def delete_remote_backend(args):
controller = init(args)
controller.delete_remote_backend(args.backend_name)
def init_remote_backend(args):
controller = init(args)
controller.init_remote_backend(args.backend_name)
def main(args):
"""
main function parses the arguments passed to the script and calls the respctive method.
:param args: Arguments passed by the user on command line while calling the script.
:return: returns the output of the function called.
"""
# grab arguments
parser = argparse.ArgumentParser(
description="Use `attack_range.py action -h` to get help with any Attack Range action")
parser.add_argument("-c", "--config", required=False, default="attack_range.yml",
help="path to the configuration file of the attack range")
parser.set_defaults(func=lambda _: parser.print_help())
actions_parser = parser.add_subparsers(title="attack Range actions", dest="action")
configure_parser = actions_parser.add_parser("configure", help="configure a new attack range")
build_parser = actions_parser.add_parser("build", help="builds attack range instances")
simulate_parser = actions_parser.add_parser("simulate", help="simulates attack techniques")
destroy_parser = actions_parser.add_parser("destroy", help="destroy attack range instances")
stop_parser = actions_parser.add_parser("stop", help="stops attack range instances")
resume_parser = actions_parser.add_parser("resume", help="resumes previously stopped attack range instances")
packer_parser = actions_parser.add_parser("packer", help="create golden images")
show_parser = actions_parser.add_parser("show", help="list machines")
dump_parser = actions_parser.add_parser("dump", help="dump locally logs from attack range instances")
replay_parser = actions_parser.add_parser("replay", help="replay dumps into the splunk server")
create_remote_backend_parser = actions_parser.add_parser("create_remote_backend", help="Create a Remote Backend")
delete_remote_backend_parser = actions_parser.add_parser("delete_remote_backend", help="Delete a Remote Backend")
init_remote_backend_parser = actions_parser.add_parser("init_remote_backend", help="Init a Remote Backend")
# Build arguments
build_parser.set_defaults(func=build)
# Destroy arguments
destroy_parser.set_defaults(func=destroy)
# Stop arguments
stop_parser.set_defaults(func=stop)
stop_parser.add_argument("--instance_ids", required=False, type=str, help="comma-separated list of instance IDs to stop")
# Resume arguments
resume_parser.set_defaults(func=resume)
resume_parser.add_argument("--instance_ids", required=False, type=str, help="comma-separated list of instance IDs to resume")
# Packer agruments
packer_parser.add_argument("-in", "--image_name", required=True, type=str,
help="provide image name such as splunk, linux, windows-2016, windows-2019, nginx, windows-10, windows-11")
packer_parser.set_defaults(func=packer)
# Configure arguments
configure_parser.add_argument("-c", "--config", required=False, type=str, default='attack_range.yml',
help="provide path to write configuration to")
configure_parser.set_defaults(func=configure)
# Simulation arguments
simulate_parser.add_argument("-e", "--engine", required=False, default="ART",
help="simulation engine to use. Available options are: PurpleSharp and ART (default)")
simulate_parser.add_argument("-t", "--target", required=True,
help="target for attack simulation. Use the name of the aws EC2 name")
simulate_parser.add_argument("-te", "--technique", required=False, type=str, default="",
help="comma delimited list of MITRE ATT&CK technique ID to simulate in the "
"attack_range, example: T1117, T1118")
simulate_parser.add_argument("-p", "--playbook", required=False, type=str, default="",
help="file path for a simulation playbook")
simulate_parser.set_defaults(func=simulate)
# Dump Arguments
dump_parser.add_argument("-fn", "--file_name", required=True,
help="file name of the attack_data")
dump_parser.add_argument("--search", required=True,
help="splunk search to export")
dump_parser.add_argument("--earliest", required=True,
help="earliest time of the splunk search")
dump_parser.add_argument("--latest", required=False, default="now",
help="latest time of the splunk search")
dump_parser.set_defaults(func=dump)
# Replay Arguments
replay_parser.add_argument("-fn", "--file_name", required=True,
help="file name of the attack_data")
replay_parser.add_argument("--source", required=True,
help="source of replayed data")
replay_parser.add_argument("--sourcetype", required=True,
help="sourcetype of replayed data")
replay_parser.add_argument("--index", required=False, default="test",
help="index of replayed data")
replay_parser.set_defaults(func=replay)
# Show arguments
show_parser.set_defaults(func=show, machines=True)
# Create Remote Backend
create_remote_backend_parser.add_argument("-bn", "--backend_name", required=True,
help="name of the remote backend")
create_remote_backend_parser.set_defaults(func=create_remote_backend)
# Delete Remote Backend
delete_remote_backend_parser.add_argument("-bn", "--backend_name", required=True,
help="name of the remote backend")
delete_remote_backend_parser.set_defaults(func=delete_remote_backend)
# Init Remote Backend
init_remote_backend_parser.add_argument("-bn", "--backend_name", required=True,
help="name of the remote backend")
init_remote_backend_parser.set_defaults(func=init_remote_backend)
# # parse them
args = parser.parse_args()
return args.func(args)
if __name__ == "__main__":
main(sys.argv[1:])