commonfields: id: Binalyze AIR version: -1 vcShouldKeepItemLegacyProdMachine: false name: Binalyze AIR display: Binalyze AIR category: Endpoint image:  description: Collect your forensics data under 10 minutes. detaileddescription: "## Binalyze AIR Integration\n\n- This integration allows you to use the Binalyze AIR's powerful features easily.\n---\n\n## SETUP\n- Give a name to your instance,\n- Type AIR server's URL(do not put a forward-slash at the end),\n- Type your username and password that you created in AIR Server,\n- Click Save & Exit on the right bottom corner.\n---\n## USAGE \nYou can use the integration in Automation, Playboks or in Playground.\n In Playground: \n- !air-isolation endpoint= organization= isolation=enable or\n- !air-isolation endpoint= organization= isolation=disable or \n- !air-acquisition endpoint= profile= .\n---\n For more information, please refer to [View Integration Documentation](https://kb.binalyze.com/air/integrations/cortex-xsoar-integration).\n For support, please e-mail us at support@binalyze.com." configuration: - display: Server URL (e.g. https://air.local) name: server defaultvalue: https://air.local type: 0 required: true - display: Username name: creds type: 9 required: true - display: API Key name: api_key type: 4 required: false - display: Trust any certificate (not secure) name: insecure type: 8 required: false - display: Use system proxy settings name: proxy type: 8 required: false script: script: | import json import urllib3 import traceback urllib3.disable_warnings() # Enter your webhook urls QUICK_URL = 'http://air.local/api/webhook/cortex6/{endpoint}?token=58a5f249-6d6d-4bea-80f6-e326617cc3c4' EVENTLOG_URL = 'http://air.local/api/webhook/cortex2/{endpoint}?token=818cad88-0ce7-4840-9ead-2b71d0670e7a' FULL_URL = 'http://air.local/api/webhook/cortex3/{endpoint}?token=cbcf96d7-164e-4e0b-a871-e960422c7b03' MEMORY_URL = 'http://air.local/api/webhook/cortex4/{endpoint}?token=6b2a67b8-0f4a-44b6-8889-4d6d6e6ca143' BROWSINGHISTORY_URL = 'http://air.local/api/webhook/cortex5/{endpoint}?token=49e29f58-1453-4161-9a4d-f286d7181be5' PARAMS = demisto.params() ARGS = demisto.args() AIR_SERVER = demisto.params()['server'] USERNAME = demisto.params()['creds']['identifier'] PASSWORD = demisto.params()['creds']['password'] USE_SSL = not demisto.params()['insecure'] def set_proxies(): if demisto.params()['proxy']: http = os.environ['http_proxy'] or os.environ['HTTP_PROXY'] https = os.environ['https_proxy'] or os.environ['HTTPS_PROXY'] proxies = { 'http': http, 'https': https } return proxies return None PROXIES = set_proxies() QUICK_URI = '/' + '/'.join(QUICK_URL.split('/')[3:6]) QUICK_TOKEN = QUICK_URL.split('=')[-1] EVENTLOG_URI = '/' + '/'.join(EVENTLOG_URL.split('/')[3:6]) EVENTLOG_TOKEN = EVENTLOG_URL.split('=')[-1] FULL_URI = '/' + '/'.join(FULL_URL.split('/')[3:6]) FULL_TOKEN = FULL_URL.split('=')[-1] MEMORY_URI = '/' + '/'.join(MEMORY_URL.split('/')[3:6]) MEMORY_TOKEN = MEMORY_URL.split('=')[-1] BROWSINGHISTORY_URI = '/' + '/'.join(BROWSINGHISTORY_URL.split('/')[3:6]) BROWSINGHISTORY_TOKEN = BROWSINGHISTORY_URL.split('=')[-1] API_LIST = { 'air_login' : '{0}/api/auth/login'.format(AIR_SERVER), 'isolate_uri' : '{0}/api/endpoints/tasks/isolation'.format(AIR_SERVER), 'info' : '{0}/api/app/info'.format(AIR_SERVER), 'organisation_id' : '{0}/api/organizations'.format(AIR_SERVER) } def test_connection(): url = API_LIST['info'] response = requests.get(url) if response.status_code == 200: demisto.results('ok') else: demisto.results('test connection failed') def air_acquisition(endpoint: str, profile: str) -> Dict[str, str]: output = 'The acquisiton profile "' + profile + '" has been started by AIR in the endpoint named:' return {output: endpoint} def air_acquisition_command(args: Dict[str, Any]) -> CommandResults: endpoint = args.get('endpoint', 'example endpoint') if not endpoint: raise ValueError('endpoint not specified') # Call the standalone function and get the raw response profile = args.get('profile', 'example profile') if not profile: raise ValueError('endpoint not specified') # Generate the url and make the request. if profile == 'quick': url = '{0}{1}/{2}?token={3}'.format(AIR_SERVER, QUICK_URI, endpoint, QUICK_TOKEN) response = requests.get(url) if profile == 'event logs': url = '{0}{1}/{2}?token={3}'.format(AIR_SERVER, EVENTLOG_URI, endpoint, EVENTLOG_TOKEN) response = requests.get(url) if profile == 'full': url = '{0}{1}/{2}?token={3}'.format(AIR_SERVER, FULL_URI, endpoint, FULL_TOKEN) response = requests.get(url) if profile == 'memory': url = '{0}{1}/{2}?token={3}'.format(AIR_SERVER, MEMORY_URI, endpoint, MEMORY_TOKEN) response = requests.get(url) if profile == 'browsing history': url = '{0}{1}/{2}?token={3}'.format(AIR_SERVER, BROWSINGHISTORY_URI, endpoint, BROWSINGHISTORY_TOKEN) response = requests.get(url) result = air_acquisition(endpoint, profile) return CommandResults( outputs_prefix='Air acquisition by webhook started in', outputs_key_field='endpoint', outputs=result, ) def air_login(args: Dict[str, Any]) -> CommandResults: headers = {'User-Agent': 'Binalyze AIR', 'Content-type': 'application/json', 'Accept-Charset': 'UTF-8'} data = {'username': USERNAME, 'password': PASSWORD, 'rememberMe': True ,} data = json.dumps(data, indent=2) air_login = API_LIST['air_login'] login = requests.post(url=air_login, headers=headers, data=data) accessToken = login.json()['accessToken'] refreshToken = login.json()['refreshToken'] air_token = 'Authentication=' + accessToken + '; ' + 'RefreshToken=' + refreshToken global air_header air_header = {'Cookie': air_token, 'User-Agent': 'Binalyze AIR', 'Content-type': 'application/json', 'Accept-Charset': 'UTF-8'} def air_isolation(endpoint: str, isolation: str) -> Dict[str, str]: output = 'The isolation ' + isolation + 'd in:' return {output: endpoint} def air_isolation_command(args: Dict[str, Any]) -> CommandResults: endpoint = args.get('endpoint', 'Hostname of endpoint') if not endpoint: raise ValueError('endpoint not specified') isolation = args.get('isolation', 'enable') if not isolation: raise ValueError('endpoint not specified') organizations = args.get('organization', 'Organization name of the Endpoint') url = API_LIST['organisation_id'] response = requests.request("GET", url, headers=air_header) response_info = response.json() organizations_list = [organizations for organizations in response_info['entities']] for name in organizations_list: organizations_name = name['name'] global organizations_id if organizations_name in organizations: organizations_id = name['_id'] if isolation == 'enable': data = json.dumps({ "enabled": True, "filter": {"name": endpoint, "organizationIds": [organizations_id]} }) url = API_LIST['isolate_uri'] response = requests.request("POST", url, headers=air_header, data=data) result = air_isolation(endpoint, isolation) return CommandResults( outputs_prefix='Air Isolated', outputs_key_field='endpoint', outputs=result, ) if isolation == 'disable': data = json.dumps({ "enabled": False, "filter": {"name": endpoint, "organizationIds": [organizations_id]} }) url = API_LIST['isolate_uri'] response = requests.request("POST", url, headers=air_header, data=data) result = air_isolation(endpoint, isolation) return CommandResults( outputs_prefix='Air unisolated', outputs_key_field='endpoint', outputs=result, ) def main() -> None: try: demisto.debug(f'Command being called is {demisto.command()}') air_login(demisto.args()) if demisto.command() == 'test-module': test_connection() if demisto.command() == 'air-acquisition': return_results(air_acquisition_command(demisto.args())) elif demisto.command() == 'air-isolation': return_results(air_isolation_command(demisto.args())) except Exception as ex: demisto.error(traceback.format_exc()) # print the traceback return_error(f'Failed to execute BaseScript. Error: {str(ex)}') ''' ENTRY POINT ''' if __name__ in ('__main__', '__builtin__', 'builtins'): main() type: python commands: - name: air-isolation arguments: - name: endpoint required: true description: Hostname of endpoint - name: organization required: true auto: PREDEFINED predefined: - aaa - bbb description: Organization name of the endpoint defaultValue: aaa - name: isolation required: true auto: PREDEFINED predefined: - enable - disable defaultValue: enable description: Isolate an endpoint - name: air-acquisition arguments: - name: endpoint required: true description: Hostname of endpoint - name: profile required: true auto: PREDEFINED predefined: - quick - event logs - full - memory - browsing history description: Select the acquisiton profile description: Collect forensics data dockerimage: demisto/python3:3.9.6.22912 runonce: false subtype: python3