C3-Client: refactored collectRelays

This commit is contained in:
Mariusz B. / mgeeky 2021-03-27 18:12:59 +01:00
parent 9a5d9d00e0
commit 6337d9418e

View File

@ -443,62 +443,34 @@ Gateway {num}:\t{g['name']}''')
if config['format'] == 'json': if config['format'] == 'json':
printJson(relays) printJson(relays)
def collectRelays(args): def collectRelays(args, nonFatal = False):
relays = [] relays = []
gateways = getRequest('/api/gateway') gateways = getRequest('/api/gateway')
gateway_id = ''
relay_id = ''
gateway_id = None if hasattr(args, 'gateway_id'):
if hasattr(args, 'gateway_id') and args.gateway_id != None:
gateway_id = args.gateway_id gateway_id = args.gateway_id
relay_id = None if hasattr(args, 'relay_id'):
if hasattr(args, 'relay_id') and args.relay_id != None:
relay_id = args.relay_id relay_id = args.relay_id
if gateway_id != None: for _gateway in gateways:
gatewayId = '' if len(gateway_id) > 0:
if _gateway["agentId"].lower() != gateway_id.lower() and _gateway["name"].lower() != gateway_id.lower():
continue
for g in gateways: gateway = getRequest(f'/api/gateway/{_gateway["agentId"]}')
if gateway_id.lower() == g['name'].lower():
gatewayId = g['agentId']
break
elif gateway_id.lower() == g['agentId'].lower():
gatewayId = g['agentId']
break
if gatewayId == '': for relay in gateway['relays']:
Logger.err('Gateway with given Name/ID could not be found.') if len(relay_id) > 0:
if config['format'] == 'json': print('{}') if relay["agentId"].lower() != relay_id.lower() and relay["name"].lower() != relay_id.lower():
sys.exit(1) continue
gateway = getRequest(f'/api/gateway/{gatewayId}') relays.append((gateway, relay))
if 'relays' not in gateway.keys(): if len(relays) == 0 and not nonFatal:
Logger.err('Specified Gateway did not have any Relay.') Logger.fatal('Could not find Relays matching filter criteria. Try changing gateway, relay criteria.')
if config['format'] == 'json': print('{}')
sys.exit(1)
if relay_id != None:
for r in gateway['relays']:
if relay_id.lower() == r['name'].lower():
relays.append((gateway, r))
elif relay_id.lower() == r['agentId'].lower():
relays.append((gateway, r))
else:
for r in gateway['relays']:
relays.append((gateway, r))
else:
for g in gateways:
gr = getRequest(f'/api/gateway/{g["agentId"]}')
if 'relays' in gr.keys():
for r in gr['relays']:
if relay_id != None:
if relay_id.lower() == r['name'].lower():
relays.append((g, r))
elif relay_id.lower() == r['agentId'].lower():
relays.append((g, r))
else:
relays.append((g, r))
return relays return relays
@ -991,7 +963,7 @@ def shell(cmd, alternative = False, stdErrToStdout = False, surpressStderr = Fal
return status return status
def onAlarmRelay(args): def onAlarmRelay(args):
origRelays = collectRelays(args) origRelays = collectRelays(args, nonFatal = True)
lastTimestamp = 0 lastTimestamp = 0
origRelayIds = set() origRelayIds = set()
@ -1005,26 +977,39 @@ def onAlarmRelay(args):
try: try:
while True: while True:
time.sleep(2) currRelays = collectRelays(args, nonFatal = True)
currRelays = collectRelays(args)
currRelayIds = set() currRelayIds = set()
currLastTimestamp = 0 currLastTimestamp = 0
newestRelay = None
for gateway, relay in currRelays: for gateway, relay in currRelays:
currRelayIds.add(relay['agentId']) currRelayIds.add(relay['agentId'])
if relay['timestamp'] > currLastTimestamp: if relay['timestamp'] > currLastTimestamp:
currLastTimestamp = relay['timestamp'] currLastTimestamp = relay['timestamp']
newestRelay = relay
if currLastTimestamp > lastTimestamp and currRelayIds != origRelayIds: relaysDiff = currRelayIds.difference(origRelayIds)
if currLastTimestamp > lastTimestamp and len(relaysDiff) > 0:
lastTimestamp = currLastTimestamp lastTimestamp = currLastTimestamp
origRelayIds = currRelayIds origRelayIds = currRelayIds
newestRelay = None
newestRelayGateway = None
newestRelayId = relaysDiff.pop()
for gateway, relay in currRelays:
if relay['agentId'] == newestRelayId:
newestRelay = relay
newestRelayGateway = gateway
break
if newestRelay == None:
continue
print('[+] New Relay checked-in!') print('[+] New Relay checked-in!')
printFullRelay(newestRelay, len(currRelays)) printFullRelay(newestRelay, len(currRelays))
time.sleep(2)
try: try:
if args.execute != None and len(args.execute) > 0: if args.execute != None and len(args.execute) > 0:
for command in args.execute: for command in args.execute:
@ -1039,7 +1024,8 @@ def onAlarmRelay(args):
cmd = cmd.replace("<relayId>", newestRelay['agentId']) cmd = cmd.replace("<relayId>", newestRelay['agentId'])
cmd = cmd.replace("<buildId>", newestRelay['buildId']) cmd = cmd.replace("<buildId>", newestRelay['buildId'])
cmd = cmd.replace("<timestamp>", str(datetime.fromtimestamp(newestRelay['timestamp']))) cmd = cmd.replace("<timestamp>", str(datetime.fromtimestamp(newestRelay['timestamp'])))
cmd = cmd.replace("<gatewayId>", newestRelay['name']) cmd = cmd.replace("<gatewayId>", newestRelayGateway['agentId'])
cmd = cmd.replace("<gatewayName>", newestRelayGateway['name'])
print(f'[.] Executing command: {cmd}') print(f'[.] Executing command: {cmd}')
print(shell(cmd)) print(shell(cmd))
@ -1059,11 +1045,16 @@ def onAlarmRelay(args):
"<relayId>", newestRelay['agentId'], "<relayId>", newestRelay['agentId'],
"<buildId>", newestRelay['buildId'], "<buildId>", newestRelay['buildId'],
"<timestamp>", datetime.fromtimestamp(newestRelay['timestamp']), "<timestamp>", datetime.fromtimestamp(newestRelay['timestamp']),
"<gatewayId>", newestRelay['name'], "<gatewayId>", newestRelayGateway['agentId'],
"<gatewayName>", newestRelayGateway['name'],
} }
print(f'[.] Triggering a webhook: {webhook}') print(f'[.] Triggering a webhook: {webhook}')
try:
requests.post(webhook, data = data, headers = headears) requests.post(webhook, data = data, headers = headears)
except Exception as e:
print(f'[-] Webhook failed: {e}')
print('[.] Webhooks triggered.') print('[.] Webhooks triggered.')
@ -1745,7 +1736,7 @@ def parseArgs(argv):
alarm_sub = alarm.add_subparsers(help = 'Alarm on what?', required = True) alarm_sub = alarm.add_subparsers(help = 'Alarm on what?', required = True)
alarm_relay = alarm_sub.add_parser('relay', help = 'Trigger an alarm whenever a new Relay checks-in.') alarm_relay = alarm_sub.add_parser('relay', help = 'Trigger an alarm whenever a new Relay checks-in.')
alarm_relay.add_argument('-e', '--execute', action='append', default=[], help = 'If new Relay checks in - execute this command. Use following placeholders in your command: <computerName>, <userName>, <domain>, <isElevated>, <osVersion>, <processId>, <relayName>, <relayId>, <buildId>, <timestamp> to customize executed command\'s parameters. Example: powershell -c "Add-Type -AssemblyName System.Speech; $synth = New-Object -TypeName System.Speech.Synthesis.SpeechSynthesizer; $synth.Speak(\'New Relay just checked-in <domain>/<userName>@<computerName>\')"') alarm_relay.add_argument('-e', '--execute', action='append', default=[], help = 'If new Relay checks in - execute this command. Use following placeholders in your command: <computerName>, <userName>, <domain>, <isElevated>, <osVersion>, <processId>, <relayName>, <relayId>, <buildId>, <gatewayId>, <gatewayName>, <timestamp> to customize executed command\'s parameters. Example: powershell -c "Add-Type -AssemblyName System.Speech; $synth = New-Object -TypeName System.Speech.Synthesis.SpeechSynthesizer; $synth.Speak(\'New Relay just checked-in <domain>/<userName>@<computerName>\')"')
alarm_relay.add_argument('-x', '--webhook', action='append', default=[], help = 'Trigger a Webhook (HTTP POST request) to this URL whenever a new Relay checks-in. The request will contain JSON message with all the fields available, mentioned in --execute option.') alarm_relay.add_argument('-x', '--webhook', action='append', default=[], help = 'Trigger a Webhook (HTTP POST request) to this URL whenever a new Relay checks-in. The request will contain JSON message with all the fields available, mentioned in --execute option.')
alarm_relay.add_argument('-g', '--gateway-id', metavar='gateway_id', default='', help = 'ID (or Name) of the Gateway which Relays should be returned. If not given, will result all relays from all gateways.') alarm_relay.add_argument('-g', '--gateway-id', metavar='gateway_id', default='', help = 'ID (or Name) of the Gateway which Relays should be returned. If not given, will result all relays from all gateways.')
alarm_relay.set_defaults(func = onAlarmRelay) alarm_relay.set_defaults(func = onAlarmRelay)