From b0da564ea0f72e32d9584f7f59c8f85900febf72 Mon Sep 17 00:00:00 2001 From: Lior Blobstein Date: Wed, 5 Dec 2018 21:15:46 +0200 Subject: [PATCH] Servicenow OOTB scripts (#2576) * comments * get table names * query script * add scripts, change snow logo * tests * fixes * fixes #2 * fixes --- Integrations/integration-ServiceNow.yml | 69 +++- Scripts/script-ServiceNowCreateIncident.yml | 223 ++++++++++++ Scripts/script-ServiceNowQueryIncident.yml | 192 +++++++++++ Scripts/script-ServiceNowUpdateIncident.yml | 257 ++++++++++++++ .../playbook-ServiceNow_TestScripts.yml | 322 ++++++++++++++++++ .../playbook-ServiceNow_Test_New.yml | 45 ++- 6 files changed, 1096 insertions(+), 12 deletions(-) create mode 100644 Scripts/script-ServiceNowCreateIncident.yml create mode 100644 Scripts/script-ServiceNowQueryIncident.yml create mode 100644 Scripts/script-ServiceNowUpdateIncident.yml create mode 100644 TestPlaybooks/NonCircleTests/playbook-ServiceNow_TestScripts.yml diff --git a/Integrations/integration-ServiceNow.yml b/Integrations/integration-ServiceNow.yml index 2cefa8349b8e..dfd220b756d0 100644 --- a/Integrations/integration-ServiceNow.yml +++ b/Integrations/integration-ServiceNow.yml @@ -4,7 +4,7 @@ commonfields: name: ServiceNow display: ServiceNow category: Case Management -image:  +image:  description: IT service management configuration: - display: ServiceNow URL, in the format https://company.service-now.com/ @@ -150,7 +150,7 @@ script: '2': 'On order', '3': 'On maintenance', '6': 'In stock/In transit', - '7': 'Retried', + '7': 'Retired', '100': 'Missing' } @@ -1205,6 +1205,44 @@ script: return res + def get_table_name_command(): + label = demisto.args()['label'] + offset = demisto.args().get('offset', DEFAULTS['offset']) + limit = demisto.args().get('limit', DEFAULTS['limit']) + + table_query = 'label=' + label + + res = query('sys_db_object', limit, offset, table_query) + + if not res or 'result' not in res: + return 'Cannot find table' + + tables = res['result'] + + if len(tables) == 0: + return 'Cannot find table' + + headers = ['ID', 'Name', 'SystemName'] + + mapped_tables = [{ + 'ID': table['sys_id'], + 'Name': table['name'], + 'SystemName': table['sys_name'] + } for table in tables] + + entry = { + 'Type': entryTypes['note'], + 'Contents': res, + 'ContentsFormat': formats['json'], + 'ReadableContentsFormat': formats['markdown'], + 'HumanReadable': tableToMarkdown('ServiceNow Tables for label - ' + label, mapped_tables, headers=headers), + 'EntryContext': { + 'ServiceNow.Table(val.ID===obj.ID)': createContext(mapped_tables), + } + } + + return entry + def fetch_incidents(): query_params = {} @@ -1321,6 +1359,8 @@ script: demisto.results(delete_record_command()) if demisto.command() == 'servicenow-list-table-fields': demisto.results(list_table_fields_command()) + if demisto.command() == 'servicenow-get-table-name': + demisto.results(get_table_name_command()) except Exception as e: LOG(e) LOG.print_log() @@ -3330,8 +3370,29 @@ script: description: User update time type: date description: Query the sys_user table in ServiceNow + - name: servicenow-get-table-name + arguments: + - name: label + required: true + description: The table label, e.g Asset, Incident, IP address etc. + - name: limit + description: Results limit + defaultValue: "10" + - name: offset + description: Starting record index to begin retrieving records from + outputs: + - contextPath: ServiceNow.Table.ID + description: Table system ID + type: string + - contextPath: ServiceNow.Table.Name + description: Table name to use in commands, e.g alm_asset + type: string + - contextPath: ServiceNow.Table.SystemName + description: Table system name, e.g Asset + type: string + description: Get table names by a label to use in commands isfetch: true runonce: false +releaseNotes: "Added command to retrieve table names, new ServiceNow logo" tests: - - No test - No instance -releaseNotes: "-" \ No newline at end of file + - No test - Hibernating instance \ No newline at end of file diff --git a/Scripts/script-ServiceNowCreateIncident.yml b/Scripts/script-ServiceNowCreateIncident.yml new file mode 100644 index 000000000000..96adc17be714 --- /dev/null +++ b/Scripts/script-ServiceNowCreateIncident.yml @@ -0,0 +1,223 @@ +commonfields: + id: ServiceNowCreateIncident + version: 38 +name: ServiceNowCreateIncident +script: | + """ + This script is used to wrap the generic create-record command in ServiceNow. + You can add fields that you want to create the record with as script arguments or in the + code and work with the records easily. + + + Mandatory fields in your ServiceNow table settings should be changed to be mandatory arguments in this script. + You can identify such fields by trying to get a record and receiving a response + stating that a required field is missing. + """ + + """ + Mapping of severity display names to their corresponding values in the API + """ + TICKET_SEVERITY = { + '1 - High': '1' , + '2 - Medium': '2', + '3 - Low': '3' + } + + + """ + Function to use the query command to retrieve records from the users table. + """ + def get_user(query): + user_args = { + 'table_name': 'sys_user', + 'query': query + } + + user_result = demisto.executeCommand('servicenow-query-table', user_args)[0] + user_data = demisto.get(user_result, 'Contents') + if not user_data: + return_error('Could not get the contents from the command result: ' + json.dumps(user_result)) + if not isinstance(user_data, dict): + # In case of string result, e.g "No incidents found" + demisto.results('User not found') + sys.exit(0) + user = user_data['result'] + + if not user or len(user) == 0: + demisto.results('User not found') + sys.exit(0) + + return user + + def get_user_id(user_name): + user_name = user_name.split(' ') + query = 'first_name={}^last_name={}'.format(user_name[0], user_name[1]) + + user = get_user(query) + + return user[0]['sys_id'] + + + """ + Function to use the query command to retrieve records from the groups table. + """ + def get_group(query): + group_args = { + 'table_name': 'sys_user_group', + 'query': query + } + + group_result = demisto.executeCommand('servicenow-query-table', group_args)[0] + group_data = demisto.get(group_result, 'Contents') + if not group_data: + return_error('Could not get the contents from the command result: ' + json.dumps(group_result)) + if not isinstance(group_data, dict): + # In case of string result, e.g "No incidents found" + demisto.results('Group not found') + sys.exit(0) + group = group_data['result'] + + if not group or len(group) == 0: + demisto.results('Group not found') + sys.exit(0) + + return group + + def get_group_id(group_name): + query = 'name=' + group_name + + group = get_group(query) + + return group[0]['sys_id'] + + + """ + The table name is required by the API. To acquire the table name, use the servicenow-get-table-name command. + """ + command_args = { + 'table_name': 'incident' + } + + """ + These record fields(columns) are mapped from their names in ServiceNow to your choice of field names. + To view all fields for a given table, use the servicenow-list-fields command. + The ID field must be included to manage unique context entries. + """ + fields_to_map = { + 'sys_id': 'ID', + 'number': 'Number' + } + + + """ + For each field in the arguments, you need to check if it was provided and apply any operations required (e.g, get a user id from a user name) to send them to the API. + """ + incident_severity = demisto.args().get('severity') + group_name = demisto.args().get('assigned_group') + user_name = demisto.args().get('assignee') + description = demisto.args().get('description') + user_id = None + group_id = None + + if user_name: + # Query the user table to get the system ID of the assignee + user_id = get_user_id(user_name) + if group_name: + # Query the group table to get the system ID of the assigned group + group_id = get_group_id(group_name) + + + """ + Every field that was provided needs to be formatted to the following syntax: 'field1=a;field2=b;...' to create the incident according to the arguments and execute the command. + In order to do that, to each field you need to concatenate the field's corresponding name in the ServiceNow API along with an '=' and the value. In the end each of those fields are joined by a ';'. + To view all the API fields for a record use the servicenow-list-fields-command. + """ + fields = [] + + if incident_severity: + fields.append('severity' + '=' + TICKET_SEVERITY[incident_severity]) + if user_id: + fields.append('assigned_to' + '=' + user_id) + if description: + fields.append('short_description' + '=' + description) + if group_id: + fields.append('assignment_group' + '=' + group_id) + + command_args['fields'] = ';'.join(fields) + + command_res = demisto.executeCommand('servicenow-create-record', command_args) + result = {} + try: + entry = command_res[0] + if isError(entry): + return_error(entry['Contents']) + else: + record_data = demisto.get(entry, 'Contents') + if not record_data: + return_error('Could not get the contents from the command result: ' + json.dumps(entry)) + if not isinstance(record_data, dict): + # In case of string result, e.g "No incidents found" + result = record_data + else: + # Get the actual record + record = record_data['result'] + # Map fields according to fields_to_map that were defined earlier + mapped_record = dict((fields_to_map[key], value) for (key, value) in list(filter(lambda (k,v): k in list(fields_to_map.keys()), record.items()))) + + display_headers = ['ID','Number'] + + # Output entry + result = { + 'Type': entryTypes['note'], + 'Contents': record_data, + 'ContentsFormat': formats['json'], + 'ReadableContentsFormat': formats['markdown'], + 'HumanReadable': tableToMarkdown('Incident successfully created', mapped_record, headers=display_headers, removeNull=True), + 'EntryContext': { + 'ServiceNow.Incident(val.ID===obj.ID)': createContext(mapped_record) + } + } + + except Exception as ex: + return_error(ex.message) + + demisto.results(result) +type: python +tags: +- servicenow +comment: |- + This script is used to wrap the generic create-record command in ServiceNow. + You can add fields that you want to create the record with as script arguments or in the + code and work with the records easily. +enabled: true +args: +- name: severity + auto: PREDEFINED + predefined: + - 1 - High + - 2 - Medium + - 3 - Low + description: Incident severity +- name: assignee + description: Incident assigne name (e.g John Smith) +- name: description + required: true + description: Incident description +- name: assigned_group + description: Incident assigned group name (e.g Incident Management Group) +outputs: +- contextPath: ServiceNow.Incident.ID + description: Incident ID + type: string +- contextPath: ServiceNow.Incident.Number + description: Incident number + type: string +scripttarget: 0 +dependson: + must: + - ServiceNow|||servicenow-create-record + - ServiceNow|||servicenow-query-table +runonce: false +releaseNotes: "ServiceNow create record example" +tests: + - No test - Hibernating instance \ No newline at end of file diff --git a/Scripts/script-ServiceNowQueryIncident.yml b/Scripts/script-ServiceNowQueryIncident.yml new file mode 100644 index 000000000000..d9f38442409c --- /dev/null +++ b/Scripts/script-ServiceNowQueryIncident.yml @@ -0,0 +1,192 @@ +commonfields: + id: ServiceNowQueryIncident + version: 100 +name: ServiceNowQueryIncident +script: | + """ + This script is used to wrap the generic query-table command in ServiceNow. + You can add fields that you want to use as inputs and outputs from the record as script arguments or in the + code and work with the records easily. + + + Mandatory fields in your ServiceNow table settings should be changed to be mandatory arguments in this script. + You can identify such fields by trying to get a record and receiving a response + stating that a required field is missing. + """ + + + """ + Mapping of priority values to their corresponding display in the UI + """ + INCIDENT_PRIORITY = { + '1': '1 - Critical', + '2': '2 - High', + '3': '3 - Moderate', + '4': '4 - Low', + '5': '5 - Planning' + } + + """ + Function to use the query command to retrieve records from the users table. + """ + def get_user(query): + user_args = { + 'table_name': 'sys_user', + 'query': query + } + + user_result = demisto.executeCommand('servicenow-query-table', user_args)[0] + user_data = demisto.get(user_result, 'Contents') + if not user_data: + return_error('Could not get the contents from the command result: ' + json.dumps(user_result)) + if not isinstance(user_data, dict): + # In case of string result, e.g "No incidents found" + demisto.results('User not found') + sys.exit(0) + user = user_data['result'] + + if not user or len(user) == 0: + demisto.results('User not found') + sys.exit(0) + + return user + + def get_user_id(user_name): + user_name = user_name.split(' ') + query = 'first_name={}^last_name={}'.format(user_name[0], user_name[1]) + + user = get_user(query) + + return user[0]['sys_id'] + + def get_user_name(user_id): + query = 'id=' + user_id + + user = get_user(query) + + return '{} {}'.format(user[0]['first_name'], user[0]['last_name']) + + """ + The table name is required by the API. To acquire the table name, use the servicenow-get-table-name command. + """ + command_args = { + 'table_name': 'incident' + } + + """ + These record fields(columns) are mapped from their names in ServiceNow to your choice of field names to be in the output. + To view all fields for a given table, use the servicenow-list-fields command. + The ID field must be included to manage unique context entries. + """ + fields_to_map = { + 'sys_id': 'ID', + 'priority': 'Priority', + 'opened_by': 'Caller', + 'number': 'Number', + 'short_description': 'Description' + } + + command_args['fields'] = list(fields_to_map.keys()) + + """ + For each field in the arguments, you need to check if it was provided and apply any operations required (e.g, get a user id from a user name) to send them to the API. + """ + incident_id = demisto.args().get('id') + incident_number = demisto.args().get('number') + user_name = demisto.args().get('assignee') + user_id = None + + if user_name: + # Query the user table to get the system ID of the assignee + user_id = get_user_id(user_name) + + + """ + Set up the query according to the arguments and execute the command + """ + if incident_id: + query = 'id=' + incident_id + elif incident_number: + query = 'number=' + incident_number + elif user_id: + query = 'assigned_to=' + user_id + + command_args['query'] = query + + command_res = demisto.executeCommand('servicenow-query-table', command_args) + result = {} + try: + entry = command_res[0] + if isError(entry): + return_error(entry['Contents']) + else: + record_data = demisto.get(entry, 'Contents') + if not record_data: + return_error('Could not get the contents from the command result: ' + json.dumps(entry)) + if not isinstance(record_data, dict): + # In case of string result, e.g "No incidents found" + result = record_data + else: + # Get the actual records + records = record_data['result'] + # Map fields according to fields_to_map that were defined earlier + mapped_records = [dict((fields_to_map[key], value) for (key, value) in list(filter(lambda (k,v): k in list(fields_to_map.keys()), r.items()))) for r in records] + for mr in mapped_records: + # Query the user table to get the name of the caller + mr['Caller'] = get_user_name(mr['Caller'].get('value')) + # Map the priority + mr['Priority'] = INCIDENT_PRIORITY.get(mr['Priority'], mr['Priority']) + display_headers = ['ID','Number','Priority', 'Description', 'Caller'] + + # Output entry + result = { + 'Type': entryTypes['note'], + 'Contents': record_data, + 'ContentsFormat': formats['json'], + 'ReadableContentsFormat': formats['markdown'], + 'HumanReadable': tableToMarkdown('ServiceNow Incidents', mapped_records, headers=display_headers, removeNull=True), + 'EntryContext': { + 'ServiceNow.Incident(val.ID===obj.ID)': createContext(mapped_records) + } + } + + except Exception as ex: + return_error(ex.message) + + demisto.results(result) +type: python +tags: +- servicenow +comment: |- + This script is used to wrap the generic query-table command in ServiceNow. + You can add fields that you want to use as inputs and outputs from the record as script arguments or in the + code and work with the records easily. +enabled: true +args: +- name: id + description: Incident System ID +- name: number + description: Incident number +- name: assignee + description: Incident assigne name (e.g John Smith) +outputs: +- contextPath: ServiceNow.Incident.ID + description: Incident ID + type: string +- contextPath: ServiceNow.Incident.Description + description: Incident description + type: string +- contextPath: ServiceNow.Incident.Number + description: Incident number + type: number +- contextPath: ServiceNow.Incident.Caller + description: Incident caller + type: string +scripttarget: 0 +dependson: + must: + - ServiceNow|||servicenow-query-table +runonce: false +releaseNotes: "ServiceNow query table example" +tests: + - No test - Hibernating instance \ No newline at end of file diff --git a/Scripts/script-ServiceNowUpdateIncident.yml b/Scripts/script-ServiceNowUpdateIncident.yml new file mode 100644 index 000000000000..89af4af6d6db --- /dev/null +++ b/Scripts/script-ServiceNowUpdateIncident.yml @@ -0,0 +1,257 @@ +commonfields: + id: ServiceNowUpdateIncident + version: 22 +name: ServiceNowUpdateIncident +script: | + """ + This script is used to wrap the generic update-record command in ServiceNow. + You can add fields that you want to update the record with as script arguments or in the + code and work with the records easily. + + + Mandatory fields in your ServiceNow table settings should be changed to be mandatory arguments in this script. + You can identify such fields by trying to get a record and receiving a response + stating that a required field is missing. + """ + + """ + Mapping of severity display names to their corresponding values in the API + """ + TICKET_SEVERITY = { + '1 - High': '1' , + '2 - Medium': '2', + '3 - Low': '3' + } + + """ + Function to use the query command to retrieve an incident by a query. + """ + def get_incident(query): + incident_args = { + 'table_name': 'incident', + 'query': query + } + + incident_result = demisto.executeCommand('servicenow-query-table', incident_args)[0] + incident_data = demisto.get(incident_result, 'Contents') + if not incident_data: + return_error('Could not get the contents from the command result: ' + json.dumps(incident_result)) + if not isinstance(incident_data, dict): + # In case of string result, e.g "No incidents found" + demisto.results('Incident not found') + sys.exit(0) + incident = incident_data['result'] + + if not incident or len(incident) == 0: + demisto.results('Incident not found') + sys.exit(0) + + return incident + + def get_incident_id(incident_number): + query = 'number=' + incident_number + + incident = get_incident(query) + + return incident[0]['sys_id'] + + """ + Function to use the query command to retrieve records from the users table. + """ + def get_user(query): + user_args = { + 'table_name': 'sys_user', + 'query': query + } + + user_result = demisto.executeCommand('servicenow-query-table', user_args)[0] + user_data = demisto.get(user_result, 'Contents') + if not user_data: + return_error('Could not get the contents from the command result: ' + json.dumps(user_result)) + if not isinstance(user_data, dict): + # In case of string result, e.g "No incidents found" + demisto.results('User not found') + sys.exit(0) + user = user_data['result'] + + if not user or len(user) == 0: + demisto.results('User not found') + sys.exit(0) + + return user + + def get_user_id(user_name): + user_name = user_name.split(' ') + query = 'first_name={}^last_name={}'.format(user_name[0], user_name[1]) + + user = get_user(query) + + return user[0]['sys_id'] + + + """ + Function to use the query command to retrieve records from the groups table. + """ + def get_group(query): + group_args = { + 'table_name': 'sys_user_group', + 'query': query + } + + group_result = demisto.executeCommand('servicenow-query-table', group_args)[0] + group_data = demisto.get(group_result, 'Contents') + if not group_data: + return_error('Could not get the contents from the command result: ' + json.dumps(group_result)) + if not isinstance(group_data, dict): + # In case of string result, e.g "No incidents found" + demisto.results('Group not found') + sys.exit(0) + group = group_data['result'] + + if not group or len(group) == 0: + demisto.results('Group not found') + sys.exit(0) + + return group + + def get_group_id(group_name): + query = 'name=' + group_name + + group = get_group(query) + + return group[0]['sys_id'] + + + """ + The table name is required by the API. To acquire the table name, use the servicenow-get-table-name command. + """ + command_args = { + 'table_name': 'incident' + } + + + """ + For each field in the arguments, you need to check if it was provided and apply any operations required (e.g, get a user id from a user name) to send them to the API. + """ + + incident_id = demisto.args().get('id') + incident_number = demisto.args().get('number') + incident_severity = demisto.args().get('severity') + description = demisto.args().get('description') + group_name = demisto.args().get('assigned_group') + user_name = demisto.args().get('assignee') + user_id = None + group_id = None + + if user_name: + # Query the user table to get the system ID of the assignee + user_id = get_user_id(user_name) + if group_name: + # Query the group table to get the system ID of the assigned group + group_id = get_group_id(group_name) + + + """ + Every field that was provided needs to be formatted to the following syntax: 'field1=a;field2=b;...' to update the incident according to the arguments and execute the command. + In order to do that, to each field you need to concatenate the field's corresponding name in the ServiceNow API along with an '=' and the value. In the end each of those fields are joined by a ';'. + To view all the API fields for a record use the servicenow-list-fields-command. + """ + fields = [] + + if incident_id: + command_args['id'] = incident_id + elif incident_number: + # Query the incident table to get the system ID of the incident + command_args['id'] = get_incident_id(incident_number) + else: + raise ValueError('Incident ID or number must be ') + if incident_severity: + fields.append('severity' + '=' + TICKET_SEVERITY[incident_severity]) + if user_id: + fields.append('assigned_to' + '=' + user_id) + if group_id: + fields.append('assignment_group' + '=' + group_id) + if description: + fields.append('short_description' + '=' + description) + + command_args['fields'] = ';'.join(fields) + + command_res = demisto.executeCommand("servicenow-update-record", command_args) + result = {} + try: + entry = command_res[0] + if isError(entry): + return_error(entry['Contents']) + else: + record_data = demisto.get(entry, 'Contents') + if not record_data: + return_error('Could not get the contents from the command result: ' + json.dumps(entry)) + if not isinstance(record_data, dict): + # In case of string result, e.g "No incidents found" + result = record_data + else: + # Get the actual record + record = record_data['result'] + # Map the ID + mapped_record = {'ID': record['sys_id']} + + # Output entry + result = { + 'Type': entryTypes['note'], + 'Contents': record_data, + 'ContentsFormat': formats['json'], + 'ReadableContentsFormat': formats['markdown'], + 'HumanReadable': 'Incident with ID ' + mapped_record['ID'] + ' successfully updated', + 'EntryContext': { + 'ServiceNow.Incident(val.ID===obj.ID)': createContext(mapped_record) + } + } + + except Exception as ex: + return_error(ex.message) + + demisto.results(result) +type: python +tags: +- servicenow +comment: |- + This script is used to wrap the generic update-record command in ServiceNow. + You can add fields that you want to update the record with as script arguments or in the + code and work with the records easily. +enabled: true +args: +- name: severity + auto: PREDEFINED + predefined: + - 1 - High + - 2 - Medium + - 3 - Low + description: Incident severity +- name: assignee + description: Incident assigne name (e.g John Smith) +- name: description + description: Incident description +- name: assigned_group + description: Incident assigned group name (e.g Incident Management Group) +- name: id + description: Incident ID to update +- name: number + description: Incident number to update +- name: query + description: 'Query to use ' +outputs: +- contextPath: ServiceNow.Incident.ID + description: Incident ID + type: string +- contextPath: ServiceNow.Incident.Number + description: Incident number + type: string +scripttarget: 0 +dependson: + must: + - ServiceNow|||servicenow-update-record + - ServiceNow|||servicenow-query-table +runonce: false +releaseNotes: "ServiceNow update record example" +tests: + - No test - Hibernating instance \ No newline at end of file diff --git a/TestPlaybooks/NonCircleTests/playbook-ServiceNow_TestScripts.yml b/TestPlaybooks/NonCircleTests/playbook-ServiceNow_TestScripts.yml new file mode 100644 index 000000000000..c07fc4c03e4f --- /dev/null +++ b/TestPlaybooks/NonCircleTests/playbook-ServiceNow_TestScripts.yml @@ -0,0 +1,322 @@ +id: test_servicenow_scripts +version: -1 +name: Test ServiceNow Scripts +starttaskid: "0" +tasks: + "0": + id: "0" + taskid: 827504d3-26ce-454d-813b-429d92397786 + type: start + task: + id: 827504d3-26ce-454d-813b-429d92397786 + version: -1 + name: "" + iscommand: false + brand: "" + nexttasks: + '#none#': + - "8" + separatecontext: false + view: |- + { + "position": { + "x": 50, + "y": 50 + } + } + note: false + timertriggers: [] + "1": + id: "1" + taskid: ee8f6a1c-f531-48f5-8351-01356f8d16f2 + type: regular + task: + id: ee8f6a1c-f531-48f5-8351-01356f8d16f2 + version: -1 + name: ServiceNowCreateIncident + description: |- + This script is used to wrap the generic create-record command in ServiceNow. + You can add fields that you want to create the record with as script arguments or in the + code and work with the records easily. + scriptName: ServiceNowCreateIncident + type: regular + iscommand: false + brand: "" + nexttasks: + '#none#': + - "2" + scriptarguments: + assigned_group: + simple: test1 + assignee: + simple: Joe Employee + description: + simple: Test incident + severity: + simple: 3 - Low + separatecontext: false + view: |- + { + "position": { + "x": 50, + "y": 370 + } + } + note: false + timertriggers: [] + "2": + id: "2" + taskid: e1b4195c-a3f7-4612-8ad2-a79bd5900767 + type: regular + task: + id: e1b4195c-a3f7-4612-8ad2-a79bd5900767 + version: -1 + name: ServiceNowQueryIncidents + description: |- + This script is used to wrap the generic query-table command in ServiceNow. + You can add fields that you want to use as inputs and outputs from the record as script arguments or in the + code and work with the records easily. + scriptName: ServiceNowQueryIncidents + type: regular + iscommand: false + brand: "" + nexttasks: + '#none#': + - "3" + scriptarguments: + assignee: {} + id: {} + number: + simple: ${ServiceNow.Incident.Number} + separatecontext: false + view: |- + { + "position": { + "x": 50, + "y": 545 + } + } + note: false + timertriggers: [] + "3": + id: "3" + taskid: e4cfaf94-65d2-442b-807d-1a60517e2846 + type: regular + task: + id: e4cfaf94-65d2-442b-807d-1a60517e2846 + version: -1 + name: Verify Incident + description: |- + Verifies path in context: + - Verifies path existence + - If matching object is an array: verify fields exists in each of the objects in the array + - If matching object is not an array: verify fields exists in matching object + - if 'expectedValue' is given: ensure that the given value is equal to the context path + scriptName: VerifyContext + type: regular + iscommand: false + brand: "" + nexttasks: + '#none#': + - "4" + scriptarguments: + expectedValue: + simple: Test incident + fields: {} + path: + simple: ServiceNow.Incident.Description + separatecontext: false + view: |- + { + "position": { + "x": 50, + "y": 720 + } + } + note: false + timertriggers: [] + "4": + id: "4" + taskid: 4eb2e521-d191-4669-8d36-c481a0079b27 + type: regular + task: + id: 4eb2e521-d191-4669-8d36-c481a0079b27 + version: -1 + name: ServiceNowUpdateIncident + description: |- + This script is used to wrap the generic update-record command in ServiceNow. + You can add fields that you want to update the record with as script arguments or in the + code and work with the records easily. + scriptName: ServiceNowUpdateIncident + type: regular + iscommand: false + brand: "" + nexttasks: + '#none#': + - "5" + scriptarguments: + assigned_group: {} + assignee: {} + description: + simple: Updated test incident + id: + simple: ${ServiceNow.Incident.ID} + number: {} + query: {} + severity: {} + separatecontext: false + view: |- + { + "position": { + "x": 50, + "y": 895 + } + } + note: false + timertriggers: [] + "5": + id: "5" + taskid: dc3eeca6-fb25-4bba-87a0-ef1552a9e0fa + type: regular + task: + id: dc3eeca6-fb25-4bba-87a0-ef1552a9e0fa + version: -1 + name: ServiceNowQueryIncidents + description: |- + This script is used to wrap the generic query-table command in ServiceNow. + You can add fields that you want to use as inputs and outputs from the record as script arguments or in the + code and work with the records easily. + scriptName: ServiceNowQueryIncidents + type: regular + iscommand: false + brand: "" + nexttasks: + '#none#': + - "6" + scriptarguments: + assignee: {} + id: {} + number: + simple: ${ServiceNow.Incident.Number} + separatecontext: false + view: |- + { + "position": { + "x": 50, + "y": 1070 + } + } + note: false + timertriggers: [] + "6": + id: "6" + taskid: fe013171-97ad-4475-8df3-fdc4ac16e268 + type: regular + task: + id: fe013171-97ad-4475-8df3-fdc4ac16e268 + version: -1 + name: Verify Incident + description: |- + Verifies path in context: + - Verifies path existence + - If matching object is an array: verify fields exists in each of the objects in the array + - If matching object is not an array: verify fields exists in matching object + - if 'expectedValue' is given: ensure that the given value is equal to the context path + scriptName: VerifyContext + type: regular + iscommand: false + brand: "" + nexttasks: + '#none#': + - "7" + scriptarguments: + expectedValue: + simple: Updated test incident + fields: {} + path: + simple: ServiceNow.Incident.Description + separatecontext: false + view: |- + { + "position": { + "x": 50, + "y": 1245 + } + } + note: false + timertriggers: [] + "7": + id: "7" + taskid: a163bb03-60b5-4df8-89ee-36c3754756c1 + type: regular + task: + id: a163bb03-60b5-4df8-89ee-36c3754756c1 + version: -1 + name: Delete the incident + description: Delete a record in a specified ServiceNow table + script: ServiceNow|||servicenow-delete-record + type: regular + iscommand: true + brand: ServiceNow + scriptarguments: + id: + simple: ${ServiceNow.Incident.ID} + table_name: + simple: incident + separatecontext: false + view: |- + { + "position": { + "x": 50, + "y": 1420 + } + } + note: false + timertriggers: [] + "8": + id: "8" + taskid: 2ab6f040-7fac-44d7-8ffb-45e1347ed9b7 + type: regular + task: + id: 2ab6f040-7fac-44d7-8ffb-45e1347ed9b7 + version: -1 + name: DeleteContext + description: Delete field from context + scriptName: DeleteContext + type: regular + iscommand: false + brand: "" + nexttasks: + '#none#': + - "1" + scriptarguments: + all: + simple: "yes" + index: {} + key: {} + keysToKeep: {} + subplaybook: {} + separatecontext: false + view: |- + { + "position": { + "x": 50, + "y": 195 + } + } + note: false + timertriggers: [] +view: |- + { + "linkLabelsPosition": {}, + "paper": { + "dimensions": { + "height": 1465, + "width": 380, + "x": 50, + "y": 50 + } + } + } +inputs: [] +outputs: [] diff --git a/TestPlaybooks/NonCircleTests/playbook-ServiceNow_Test_New.yml b/TestPlaybooks/NonCircleTests/playbook-ServiceNow_Test_New.yml index 13b8f9d88003..b87fe61019ea 100644 --- a/TestPlaybooks/NonCircleTests/playbook-ServiceNow_Test_New.yml +++ b/TestPlaybooks/NonCircleTests/playbook-ServiceNow_Test_New.yml @@ -1,5 +1,5 @@ -id: servicenow_test_new -version: -1 +id: ad055a1b-edec-4285-8bb6-fbfc558e22e9 +version: 15 name: ServiceNow Test New description: This playbook tests the ServiceNow integration commands. starttaskid: "0" @@ -1633,10 +1633,10 @@ tasks: timertriggers: [] "67": id: "67" - taskid: b5fb1279-37be-432c-8b23-46e33a2507c8 + taskid: f2de40ed-b11f-4d5c-8666-bf16eb5db67d type: regular task: - id: b5fb1279-37be-432c-8b23-46e33a2507c8 + id: f2de40ed-b11f-4d5c-8666-bf16eb5db67d version: -1 name: VerifyContext description: |- @@ -1652,7 +1652,7 @@ tasks: scriptarguments: expectedValue: {} fields: - simple: Field,Computer,Group,User + simple: Field,Computer,Group,User,Table path: simple: ServiceNow separatecontext: false @@ -1660,7 +1660,7 @@ tasks: { "position": { "x": 265, - "y": 7485 + "y": 7660 } } note: false @@ -1680,7 +1680,7 @@ tasks: brand: ServiceNow nexttasks: '#none#': - - "67" + - "70" scriptarguments: limit: simple: "1" @@ -1731,12 +1731,41 @@ tasks: } note: false timertriggers: [] + "70": + id: "70" + taskid: 563001d5-1061-4bb8-8609-648cda6953db + type: regular + task: + id: 563001d5-1061-4bb8-8609-648cda6953db + version: -1 + name: servicenow-get-table-name + description: Get table names by a label to use in commands + script: ServiceNow|||servicenow-get-table-name + type: regular + iscommand: true + brand: ServiceNow + nexttasks: + '#none#': + - "67" + scriptarguments: + label: + simple: Asset + separatecontext: false + view: |- + { + "position": { + "x": 265, + "y": 7485 + } + } + note: false + timertriggers: [] view: |- { "linkLabelsPosition": {}, "paper": { "dimensions": { - "height": 7530, + "height": 7705, "width": 810, "x": 50, "y": 50