-
Notifications
You must be signed in to change notification settings - Fork 90
/
Copy pathstart-alert-scan.gmp.py
387 lines (323 loc) · 10.5 KB
/
start-alert-scan.gmp.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
# SPDX-FileCopyrightText: 2018 Henning Häcker
# SPDX-FileCopyrightText: 2019-2021 Greenbone AG
#
# SPDX-License-Identifier: GPL-3.0-or-later
from argparse import ArgumentParser, Namespace, RawTextHelpFormatter
from typing import List
from gvm.protocols.gmp import Gmp
HELP_TEXT = """
This script makes an E-Mail alert scan.
Usage examples:
$ gvm-script --gmp-username name --gmp-password pass ssh --hostname
... start-alert-scan.gmp.py +h
... start-alert-scan.gmp.py ++target-name ++hosts ++ports \
++port-list-name +C +R +S
... start-alert-scan.gmp.py ++target-name ++hosts ++port-list-id \
+C ++recipient ++sender
... start-alert-scan.gmp.py ++target-id +C ++recipient ++sender
"""
def get_scan_config(gmp: Gmp, config: int, debug: bool = False):
# get all configs of the openvas instance
# filter for all rows!
res = gmp.get_scan_configs(filter_string="rows=-1")
if config < 0 or config > 4:
raise ValueError("Wrong config identifier. Choose between [0,4].")
# match the config abbreviation to accepted config names
config_list = [
"Full and fast",
"Full and fast ultimate",
"Full and very deep",
"Full and very deep ultimate",
"System Discovery",
]
template_abbreviation_mapper = {
0: config_list[0],
1: config_list[1],
2: config_list[2],
3: config_list[3],
4: config_list[4],
}
for conf in res.xpath("config"):
cid = conf.xpath("@id")[0]
name = conf.xpath("name/text()")[0]
# get the config id of the desired template
if template_abbreviation_mapper.get(config) == name:
config_id = cid
if debug:
print(name + ": " + config_id)
break
return config_id
def get_target(
gmp,
target_name: str = None,
hosts: List[str] = None,
ports: str = None,
port_list_name: str = None,
port_list_id: str = None,
debug: bool = False,
):
if target_name is None:
target_name = "target"
targets = gmp.get_targets(filter_string=target_name)
existing_targets = [""]
for target in targets.findall("target"):
existing_targets.append(str(target.find("name").text))
counter = 0
# iterate over existing targets and find a vacant targetName
if target_name in existing_targets:
while True:
tmp_name = f"{target_name} ({str(counter)})"
if tmp_name in existing_targets:
counter += 1
else:
target_name = tmp_name
break
if debug:
print(f"target name: {target_name}")
if not port_list_id:
existing_port_lists = [""]
port_lists_tree = gmp.get_port_lists()
for plist in port_lists_tree.findall("port_list"):
existing_port_lists.append(str(plist.find("name").text))
if port_list_name is None:
port_list_name = "portlist"
if port_list_name in existing_port_lists:
counter = 0
while True:
tmp_name = f"{port_list_name} ({str(counter)})"
if tmp_name in existing_port_lists:
counter += 1
else:
port_list_name = tmp_name
break
port_list = gmp.create_port_list(name=port_list_name, port_range=ports)
# create port list
port_list_id = port_list.xpath("@id")[0]
if debug:
print(f"New Portlist-name:\t{port_list_name}")
print(f"New Portlist-id: \t{str(port_list_id)}")
# integrate port list id into create_target
res = gmp.create_target(target_name, hosts=hosts, port_list_id=port_list_id)
print(f"New target '{target_name}' created.")
return res.xpath("@id")[0]
def get_alert(
gmp: Gmp,
sender_email: str,
recipient_email: str,
alert_name: str = None,
debug: bool = False,
):
# create alert if necessary
alert_object = gmp.get_alerts(filter_string=f"name={alert_name}")
alert = alert_object.xpath("alert")
if len(alert) == 0:
print(f"creating new alert {alert_name}")
gmp.create_alert(
name=alert_name,
event=gmp.types.AlertEvent.TASK_RUN_STATUS_CHANGED,
event_data={"status": "Done"},
condition=gmp.types.AlertCondition.ALWAYS,
method=gmp.types.AlertMethod.EMAIL,
method_data={
"""Task '$n': $e
After the event $e,
the following condition was met: $c
This email escalation is configured to attach report format '$r'.
Full details and other report formats are available on the scan engine.
$t
Note:
This email was sent to you as a configured security scan escalation.
Please contact your local system administrator if you think you
should not have received it.
""": "message",
"2": "notice",
sender_email: "from_address",
"[OpenVAS-Manager] Task": "subject",
"c402cc3e-b531-11e1-9163-406186ea4fc5": "notice_attach_format",
recipient_email: "to_address",
},
)
alert_object = gmp.get_alerts(filter_string=f"name={recipient_email}")
alert = alert_object.xpath("alert")
alert_id = alert[0].get("id", "no id found")
if debug:
print(f"alert_id: {str(alert_id)}")
return alert_id
def get_scanner(gmp: Gmp):
res = gmp.get_scanners()
scanner_ids = res.xpath("scanner/@id")
return scanner_ids[1] # "default scanner"
def create_and_start_task(
gmp: Gmp,
config_id: str,
target_id: str,
scanner_id: str,
alert_id: str,
alert_name: str,
debug: bool = False,
) -> str:
# Create the task
task_name = f"Alert Scan for Alert {alert_name}"
tasks = gmp.get_tasks(filter_string=f'name="{task_name}"')
existing_tasks = tasks.findall("task")
if existing_tasks:
task_name = f"Alert Scan for Alert {alert_name} ({len(existing_tasks)})"
task_comment = "Alert Scan"
res = gmp.create_task(
name=task_name,
config_id=config_id,
target_id=target_id,
scanner_id=scanner_id,
alert_ids=[alert_id],
comment=task_comment,
)
# Start the task
task_id = res.xpath("@id")[0]
gmp.start_task(task_id)
if debug:
# Stop the task (for performance reasons)
gmp.stop_task(task_id=task_id)
print("Task stopped")
return task_name
def parse_args(args: Namespace): # pylint: disable=unused-argument
parser = ArgumentParser(
prefix_chars="+",
add_help=False,
formatter_class=RawTextHelpFormatter,
description=HELP_TEXT,
)
parser.add_argument(
"+h",
"++help",
action="help",
help="Show this help message and exit.",
)
target = parser.add_mutually_exclusive_group(required=True)
target.add_argument(
"++target-id",
type=str,
dest="target_id",
help="Use an existing target by target id",
)
target.add_argument(
"++target-name",
type=str,
dest="target_name",
help="Create a target by name",
)
parser.add_argument(
"++hosts",
nargs="+",
dest="hosts",
help="Host(s) for the new target",
)
ports = parser.add_mutually_exclusive_group()
ports.add_argument(
"++port-list-id",
type=str,
dest="port_list_id",
help="An existing portlist id for the new target",
)
ports.add_argument(
"++ports",
type=str,
dest="ports",
help="Ports in the new target: e.g. T:80-80,8080",
)
parser.add_argument(
"++port-list-name",
type=str,
dest="port_list_name",
help="Name for the new portlist in the new target",
)
config = parser.add_mutually_exclusive_group()
config.add_argument(
"+C",
"++scan-config",
default=0,
type=int,
dest="config",
help="Choose from existing scan config:"
"\n 0: Full and fast"
"\n 1: Full and fast ultimate"
"\n 2: Full and very deep"
"\n 3: Full and very deep ultimate"
"\n 4: System Discovery",
)
config.add_argument(
"++scan-config-id",
type=str,
dest="scan_config_id",
help="Use existing scan config by id",
)
parser.add_argument(
"++scanner-id",
type=str,
dest="scanner_id",
help="Use existing scanner by id",
)
parser.add_argument(
"+R",
"++recipient",
required=True,
dest="recipient_email",
type=str,
help="Alert recipient E-Mail address",
)
parser.add_argument(
"+S",
"++sender",
required=True,
dest="sender_email",
type=str,
help="Alert senders E-Mail address",
)
parser.add_argument(
"++alert-name",
dest="alert_name",
type=str,
help="Optional Alert name",
)
script_args, _ = parser.parse_known_args()
return script_args
def main(gmp: Gmp, args: Namespace) -> None:
# pylint: disable=undefined-variable, unused-argument
script_args = parse_args(args)
# set alert_name to recipient email if no other name
# is given
if script_args.alert_name is None:
script_args.alert_name = script_args.recipient_email
# use existing config from argument
if not script_args.scan_config_id:
config_id = get_scan_config(gmp, script_args.config)
else:
config_id = script_args.scan_config_id
# create new target or use existing one from id
if not script_args.target_id:
target_id = get_target(
gmp,
target_name=script_args.target_name,
hosts=script_args.hosts,
ports=script_args.ports,
port_list_name=script_args.port_list_name,
port_list_id=script_args.port_list_id,
)
else:
target_id = script_args.target_id
alert_id = get_alert(
gmp,
script_args.sender_email,
script_args.recipient_email,
script_args.alert_name,
)
if not script_args.scanner_id:
scanner_id = get_scanner(gmp)
else:
scanner_id = script_args.scanner_id
task_name = create_and_start_task(
gmp, config_id, target_id, scanner_id, alert_id, script_args.alert_name
)
print(f"Task started: {task_name}\n")
print("Script finished\n")
if __name__ == "__gmp__":
main(gmp, args)