Microsoft Sentinel¶
Table of Contents¶
Release Notes¶
Version |
Date |
Notes |
---|---|---|
2.1.3 |
12/2024 |
Bug fix for Sentinel URL entities. |
2.1.2 |
10/2024 |
Added retry logic to SOAR incident updates |
2.1.1 |
07/2024 |
Correction to documentation on template references |
2.1.0 |
03/2024 |
Add support for multi-tenancy. |
2.0.0 |
01/2024 |
|
1.2.0 |
07/2023 |
Add config setting to not close SOAR case was Sentinel incident is closed |
1.1.0 |
04/2023 |
Update function ‘Sentinel Update Incident’ |
1.0.4 |
06/2022 |
Several template fixes for tags (labels) and severity |
1.0.3 |
04/2022 |
Support for app.config verify and cert parameters |
1.0.2 |
02/2022 |
Bug fix in some situations updating Sentinel from SOAR |
1.0.1 |
11/2021 |
Minor bug fixes and datatable improvements |
1.0.0 |
08/2021 |
Initial Release |
v2.1.1¶
The default configuration file incorrectly specified the override Sentinel template references were:
update_sentinel_incident_template
close_sentinel_incident_template
The correct settings are:
sentinel_update_incident_template
sentinel_close_incident_template
v2.1.0¶
When upgrading from a previous version to v2.1.0, manually update your app.config to be able to use the multi-tenant capability. The app.config changes are documented here App Configuration. The previous version of the app.config from versions before 2.1.0 are compatible with version 2.1.0+.
Overview¶
SOAR integration for ‘fn_microsoft_sentinel’
This app allows bi-directional synchronization between IBM SOAR and Microsoft Sentinel. Sentinel entities are exposed as artifacts for further investigation.
Key Features¶
Escalate Microsoft Sentinel Incidents to IBM SOAR Cases
Automatically keep Incidents and Cases synchronized
Retrieve Sentinel Incident alert entities as artifacts and data table rows
Sync comments to and from Sentinel Incidents
Support editable templates for field mapping between the two systems
Supports configuring multiple Sentinel servers
Requirements¶
This app supports the IBM Security QRadar SOAR Platform and the IBM Security QRadar SOAR for IBM Cloud Pak for Security.
SOAR platform¶
The SOAR platform supports two app deployment mechanisms, Edge Gateway (also known as App Host) and integration server.
If deploying to a SOAR platform with an App Host, the requirements are:
SOAR platform >=
50.0.9097
.The app is in a container-based format (available from the AppExchange as a
zip
file).
If deploying to a SOAR platform with an integration server, the requirements are:
SOAR platform >=
50.0.9097
.The app is in the older integration format (available from the AppExchange as a
zip
file which contains atar.gz
file).Integration server is running
resilient_circuits>=50.0.0
.If using an API key account, make sure the account provides the following minimum permissions:
Name
Permissions
Org Data
Read
Function
Read
Incident
Read, Edit, Create, Owner, Status
Incident Notes
Edit
Layout
Read, Edit
The following SOAR platform guides provide additional information:
Edge Gateway Deployment Guide or App Host Deployment Guide: provides installation, configuration, and troubleshooting information, including proxy server settings.
Integration Server Guide: provides installation, configuration, and troubleshooting information, including proxy server settings.
System Administrator Guide: provides the procedure to install, configure and deploy apps.
The above guides are available on the IBM Documentation website at ibm.biz/soar-docs. On this web page, select your SOAR platform version. On the follow-on page, you can find the Edge Gateway Deployment Guide, App Host Deployment Guide, or Integration Server Guide by expanding Apps in the Table of Contents pane. The System Administrator Guide is available by expanding System Administrator.
Cloud Pak for Security¶
If you are deploying to IBM Cloud Pak for Security, the requirements are:
IBM Cloud Pak for Security >=
1.10.15
.Cloud Pak is configured with an Edge Gateway.
The app is in a container-based format (available from the AppExchange as a
zip
file).
The following Cloud Pak guides provide additional information:
Edge Gateway Deployment Guide or App Host Deployment Guide: provides installation, configuration, and troubleshooting information, including proxy server settings. From the Table of Contents, select Case Management and Orchestration & Automation > Orchestration and Automation Apps.
System Administrator Guide: provides information to install, configure, and deploy apps. From the IBM Cloud Pak for Security IBM Documentation table of contents, select Case Management and Orchestration & Automation > System administrator.
These guides are available on the IBM Documentation website at ibm.biz/cp4s-docs. From this web page, select your IBM Cloud Pak for Security version. From the version-specific IBM Documentation page, select Case Management and Orchestration & Automation.
Proxy Server¶
The app does support a proxy server.
Python Environment¶
Python 3.9 and Python 3.11 are supported. Additional package dependencies may exist for each of these packages:
jinja2~=3.1.0
resilient_circuits>=50.0.0
simplejson~=3.19.0
Sentinel Configuration¶
Several steps are necessary to enable API access to Sentinel. Below is a brief list of steps to complete:
Define an App under
App registration
, for Sentinel access. The Application (client) ID will be used as the app.configclient_id
.a. Specify a client secret. This will be used in the app.config file as
api_secret
.
Define a workspace and resource group for the subscription. These names will be defined in the app.config file as
workspace_name
andresource_groupname
.
For your resource group, specify the app with the Sentinel Responder role.
Add your tenant_id (Under
Tenant properties
) and subscription_id to the app.config file.
Installation¶
Install¶
To install or uninstall an App or Integration on the SOAR platform, see the documentation at ibm.biz/soar-docs.
To install or uninstall an App on IBM Cloud Pak for Security, see the documentation at ibm.biz/cp4s-docs and follow the instructions above to navigate to Orchestration and Automation.
App Configuration¶
The following table provides the settings you need to configure the app. These settings are made in the app.config file. See the documentation discussed in the Requirements section for the procedure.
Config |
Required |
Example |
Description |
---|---|---|---|
azure_url |
Yes |
|
URL for Azure access. This field is typically not changed |
ms_sentinel_labels |
No |
|
A comma separated list of label(s) to access |
polling_lookback |
Yes |
|
# of minutes to look back for incident changes. This is used only the first time the app starts |
polling_interval |
Yes |
|
# of Seconds to wait until checking for changes in Sentinel. Comment out to disable the poller |
api_version |
No |
|
API version to use when making API calls to Microsoft Sentinel. Sentinel API versions can be found https://learn.microsoft.com/en-us/rest/api/securityinsights/api-versions |
https_proxy |
No |
|
- |
http_proxy |
No |
|
- |
verify |
No |
|
verify= false or /path/to/client_certificate.pem |
For each labeled server:
Config |
Required |
Example |
Description |
---|---|---|---|
tenant_id |
Yes |
|
Your tenant_id |
client_id |
Yes |
|
Your App Id or client_id |
app_secret |
Yes |
|
Your app secret |
subscription_id |
Yes |
|
*subscription_id for incident access * |
workspace_name |
Yes |
`` |
workspace name for incident access. |
resource_groupname |
Yes |
`` |
resource group for incident access. |
new_incident_filters |
No |
|
*Set of filters to apply when escalating incidents to SOAR. Incidents not matching the criteria are not synchronized. In this example, both a match of status and severity would be required. * |
poller_filters_template |
No |
/var/rescircuits/poller_filters_template.jinja |
Customer supplied template for filters to apply to new Sentinel incidents found by the poller. |
max_alerts |
Yes |
10 |
limit the number of alerts per sentinel incident to the first n alerts or leave blank for all alerts |
close_soar_case |
No |
|
Whether the SOAR case should be closed when its linked Sentinel incident is closed. If set to False the linked SOAR case will be updated, but not closed. |
create_incident_template |
No |
/var/rescircuits/create_incident_template.jinja |
Customer supplied template for mapping Sentinel Incident fields to an SOAR incident. If not specified, a default template is used. |
update_incident_template |
No |
/var/rescircuits/update_incident_template.jinja |
Customer supplied template for mapping Sentinel Incident fields to an SOAR incident. If not specified, a default template is used. |
close_incident_template |
No |
/var/rescircuits/close_incident_template.jinja |
Customer supplied template for mapping Sentinel Incident fields to an SOAR incident. If not specified, a default template is used. This is useful when a customer customizes the fields used when closing an incident. |
sentinel_update_incident_template |
No |
/var/rescircuits/update_sentinel_incident_template.jinja |
Customer supplied template for updating a sentinel incident when the SOAR incident is updated |
sentinel_close_incident_template |
No |
/var/rescircuits/update_sentinel_incident_template.jinja |
Customer supplied template for closing a sentinel incident when the SOAR incident is closed |
See the section below for examples of the templates.
Function - Sentinel Add Incident Comment¶
Create a comment for a given Sentinel incident. This is used by a playbook to automatically sync SOAR Comments with Sentinel.
Inputs:
Name |
Type |
Required |
Example |
Tooltip |
---|---|---|---|---|
|
|
Yes |
|
Comment to add to the Sentinel Incident. |
|
|
No |
|
The ID of a Sentinel Incident. |
|
|
No |
|
Label give to the Sentinel server in the app.config. Example in app.config [fn_microsoft_sentinel:label1] the label here is label1. |
|
|
No |
|
The Sentinel profile name. |
Outputs:
NOTE: This example might be in JSON format, but
results
is a Python Dictionary on the SOAR platform.
results = {
"content": {
"etag": "\"1f017068-0000-0100-0000-6616d57f0000\"",
"id": "/subscriptions/xxxxxxxx-xxxx-xxxx-xxxx-xxxxxxxxxxxx/resourceGroups/demoassets/providers/Microsoft.OperationalInsights/workspaces/AzureSentinelDemo/providers/Microsoft.SecurityInsights/Incidents/b3763ac5-6319-496b-b404-6c5000418a81/Comments/42acd50c-f765-11ee-ab89-fa89897771d8",
"name": "42acd50c-f765-11ee-ab89-fa89897771d8",
"properties": {
"author": {
"email": null,
"name": "Comment created from external application - MS Sentinel",
"objectId": "fb5360be-0c6a-4260-ae69-9b07dd735441",
"userPrincipalName": null
},
"createdTimeUtc": "2024-04-10T18:07:59.058255Z",
"lastModifiedTimeUtc": "2024-04-10T18:07:59.058255Z",
"message": "From IBM SOAR:\nhello"
},
"type": "Microsoft.SecurityInsights/Incidents/Comments"
},
"inputs": {
"sentinel_incident_comment": "hello",
"sentinel_incident_id": "b3763ac5-6319-496b-b404-6c5000418a81",
"sentinel_label": "label1",
"sentinel_profile": null
},
"metrics": {
"execution_time_ms": 1497,
"host": "local",
"package": "fn-microsoft-sentinel",
"package_version": "2.1.0",
"timestamp": "2024-04-10 14:07:59",
"version": "1.0"
},
"raw": null,
"reason": null,
"success": true,
"version": 2.0
}
Example Function Input Script:
inputs.sentinel_incident_comment = note.text.content
inputs.sentinel_incident_id = incident.properties.sentinel_incident_number
if incident.properties.sentinel_profile:
inputs.sentinel_profile = incident.properties.sentinel_profile
if incident.properties.sentinel_label:
inputs.sentinel_label = incident.properties.sentinel_label
Example Function Post Process Script:
# Import datetime
from datetime import datetime
results = playbook.functions.results.comment_results
if results.get("success"):
# Get the current time
dt_now = datetime.now()
# note.text = f"<b>Sent to Sentinel at {dt_now}</b><br>{str(note.text.content)}"
else:
incident.addNote(f"Sentinel Comment Sync failed with reason: {results.get('reason')}")
Function - Sentinel Get Incident Alerts¶
Sentinel Get Incident Alerts
Inputs:
Name |
Type |
Required |
Example |
Tooltip |
---|---|---|---|---|
|
|
No |
|
The ID of a Sentinel Incident. |
|
|
No |
|
Label give to the Sentinel server in the app.config. Example in app.config [fn_microsoft_sentinel:label1] the label here is label1. |
|
|
No |
|
The Sentinel profile name. |
Outputs:
NOTE: This example might be in JSON format, but
results
is a Python Dictionary on the SOAR platform.
results = {
"content": {
"value": [
{
"id": "/subscriptions/xxxxxxxx-xxxx-xxxx-xxxx-xxxxxxxxxxxx/resourceGroups/demoassets/providers/Microsoft.OperationalInsights/workspaces/AzureSentinelDemo/providers/Microsoft.SecurityInsights/Entities/4c4831e9-2ed7-a44a-e530-c784cc7c2ddc",
"kind": "SecurityAlert",
"name": "4c4831e9-2ed7-a44a-e530-c784cc7c2ddc",
"properties": {
"additionalData": {
"AlertMessageEnqueueTime": "2024-04-10T15:29:49.911Z",
"MTP_Classification": "Unknown",
"OriginalProductComponentName": "Databases",
"OriginalProductName": "Azure Security Center",
"StoreAlertPublisher": "{\"Succeeded\":true,\"Reason\":null,\"PublishTime\":\"2024-04-10T15:29:41.6565662Z\"}",
"effectiveSubscriptionId": "xxxxxxxx-xxxx-xxxx-xxxx-xxxxxxxxxxxx"
},
"alertDisplayName": "Suspected brute-force attack attempt",
"alertLink": "https://portal.azure.com/#blade/Microsoft_Azure_Security_AzureDefenderForData/AlertBlade/alertId/2516895378499999999_c01aa88b-4520-4c62-a80d-c61f3dfaf47e/subscriptionId/xxxxxxxx-xxxx-xxxx-xxxx-xxxxxxxxxxxx/resourceGroup/DemoAssets/referencedFrom/alertDeepLink/location/centralus",
"alertType": "SQL.VM_BruteForce",
"compromisedEntity": "PAMSqlSv",
"confidenceLevel": "Unknown",
"description": "Brute-force attack is a common attack technique for finding valid credentials to the database. By submitting many users/passwords combinations, an attacker can guess a correct one.\r\nOnce obtained, an attacker can have full access to the database. While this specific alert doesn\u0027t indicate a successful brute-force, it is advised to take safety measures to protect your resource against this attack.\r\nTo investigate this suspected brute-force attempt, review it\u0027s origin (based on the application name and IP/Location), and try to find out whether it\u0027s recognized to you, or suspicious.\r\nIf you believe this to be an attack on your database, use firewall rules to limit the access to your resource, and make sure you use strong passwords and not well known user names.\r\nAlso, consider using only AAD authentication to further enhance your security posture.\r\n",
"endTimeUtc": "2024-04-10T15:29:10Z",
"friendlyName": "Suspected brute-force attack attempt",
"processingEndTime": "2024-04-10T15:29:40.734843Z",
"productComponentName": "Databases",
"productName": "Azure Security Center",
"providerAlertId": "2516895378499999999_c01aa88b-4520-4c62-a80d-c61f3dfaf47e",
"remediationSteps": [
"* Apply [workflow automation](https://go.microsoft.com/fwlink/?linkid=2174808) to block future attacks.",
"* Consider [blocking the IP address](https://go.microsoft.com/fwlink/?linkid=2099055) of the attacking client and hardening your firewall.",
"* When possible, use [Windows authentication](https://go.microsoft.com/fwlink/?linkid=2129121) and disable SQL Server authentication.",
"* Use [strong passwords](https://go.microsoft.com/fwlink/?linkid=2099068) and avoid reusing them across multiple databases.",
"* If applicable, disable default and well known application/database accounts such as SA."
],
"resourceIdentifiers": [
{
"resourceId": "/subscriptions/xxxxxxxx-xxxx-xxxx-xxxx-xxxxxxxxxxxx/resourceGroups/DemoAssets/providers/Microsoft.Compute/virtualMachines/PAMSqlSv",
"subscriptionId": "xxxxxxxx-xxxx-xxxx-xxxx-xxxxxxxxxxxx",
"type": "AzureResource"
},
{
"agentId": "7db452ea-1cae-4079-94a5-85dc76156e96",
"subscriptionId": "xxxxxxxx-xxxx-xxxx-xxxx-xxxxxxxxxxxx",
"type": "LogAnalytics",
"workspaceId": "c6c712b0-dc3d-4408-945a-4bde1db6579b"
}
],
"severity": "High",
"startTimeUtc": "2024-04-10T15:29:10Z",
"status": "New",
"systemAlertId": "4c4831e9-2ed7-a44a-e530-c784cc7c2ddc",
"tactics": [
"PreAttack"
],
"timeGenerated": "2024-04-10T15:29:49.9010714Z",
"timeGenerated_ms": 1712762989000,
"vendorName": "Microsoft"
},
"type": "Microsoft.SecurityInsights/Entities"
}
]
},
"inputs": {
"sentinel_incident_id": "b3763ac5-6319-496b-b404-6c5000418a81",
"sentinel_label": "label1",
"sentinel_profile": null
},
"metrics": {
"execution_time_ms": 883,
"host": "local",
"package": "fn-microsoft-sentinel",
"package_version": "2.1.0",
"timestamp": "2024-04-10 14:07:28",
"version": "1.0"
},
"raw": null,
"reason": null,
"success": true,
"version": 2.0
}
Example Function Input Script:
inputs.sentinel_incident_id = incident.properties.sentinel_incident_number
if incident.properties.sentinel_profile:
inputs.sentinel_profile = incident.properties.sentinel_profile
if incident.properties.sentinel_label:
inputs.sentinel_label = incident.properties.sentinel_label
Example Function Post Process Script:
from datetime import datetime
current_dt = datetime.now()
results = playbook.functions.results.alerts_results
if results.get('success'):
for alert in results.get('content', {}).get('value'):
properties = alert.get('properties', {})
row = incident.addRow("sentinel_incident_alerts")
row['report_date'] = current_dt
row['alert_date'] = properties.get('timeGenerated_ms')
row['alert_name'] = properties.get('alertDisplayName')
row['alert_description'] = properties.get('description')
row['alert_type'] = properties.get('alertType')
row['alert_status'] = properties.get('status')
row['alert_severity'] = properties.get('severity')
row['alert_confidence_level'] = properties.get('confidenceLevel')
row['alert_tactics'] = ",".join(properties.get('tactics', []))
row['alert_compromised_entity'] = properties.get('compromisedEntity')
row['alert_remediation_steps'] = helper.createPlainText('\n'.join(properties.get('remediationSteps', [])))
row['alert_id'] = properties.get('systemAlertId')
if properties.get('alertLink'):
row['alert_url'] = helper.createRichText(f"<a target='blank' href='{properties.get('alertLink')}'>Alert Link</a>")
else:
row['alert_url'] = helper.createRichText(f"<a target='blank' href='https://security.microsoft.com/alerts/{properties.get('systemAlertId')}'>Alert Link</a>")
else:
incident.addNote(f"Sentinel Incident Sync failed with reason: {results.get('reason')}")
Function - Sentinel Get Incident Comments¶
Get Comments from a Sentinel Incident.
Inputs:
Name |
Type |
Required |
Example |
Tooltip |
---|---|---|---|---|
|
|
No |
|
SOAR Incident ID |
|
|
No |
|
The ID of a Sentinel Incident. |
|
|
No |
|
Label give to the Sentinel server in the app.config. Example in app.config [fn_microsoft_sentinel:label1] the label here is label1. |
|
|
No |
|
The Sentinel profile name. |
Outputs:
NOTE: This example might be in JSON format, but
results
is a Python Dictionary on the SOAR platform.
results = {
"content": {
"value": [
{
"etag": "\"1f016d66-0000-0100-0000-6616d56e0000\"",
"id": "/subscriptions/xxxxxxxx-xxxx-xxxx-xxxx-xxxxxxxxxxxx/resourceGroups/demoassets/providers/Microsoft.OperationalInsights/workspaces/AzureSentinelDemo/providers/Microsoft.SecurityInsights/Incidents/b3763ac5-6319-496b-b404-6c5000418a81/Comments/aebab623-5e91-43a7-b7a6-541548fc738f",
"name": "aebab623-5e91-43a7-b7a6-541548fc738f",
"properties": {
"author": {
"email": "example@microsoft.com",
"name": "John Doe",
"objectId": "43dd7b73-6a70-475e-88c3-609a9f30b514",
"userPrincipalName": "example@microsoft.com"
},
"createdTimeUtc": "2024-04-10T18:07:42.5540333Z",
"lastModifiedTimeUtc": "2024-04-10T18:07:42.5540333Z",
"message": "hello"
},
"type": "Microsoft.SecurityInsights/Incidents/Comments"
}
]
},
"inputs": {
"incident_id": 2121,
"sentinel_incident_id": "b3763ac5-6319-496b-b404-6c5000418a81",
"sentinel_label": "label1",
"sentinel_profile": null
},
"metrics": {
"execution_time_ms": 1044,
"host": "local",
"package": "fn-microsoft-sentinel",
"package_version": "2.1.0",
"timestamp": "2024-04-10 14:07:55",
"version": "1.0"
},
"raw": null,
"reason": null,
"success": true,
"version": 2.0
}
Example Function Input Script:
inputs.sentinel_incident_id = incident.properties.sentinel_incident_number
inputs.incident_id = incident.id
if incident.properties.sentinel_profile:
inputs.sentinel_profile = incident.properties.sentinel_profile
if incident.properties.sentinel_label:
inputs.sentinel_label = incident.properties.sentinel_label
Example Function Post Process Script:
results = playbook.functions.results.comment_results
if results.get("success"):
for comment in results.get("content", {}).get('value'):
incident.addNote(helper.createRichText(comment.get('properties', {}).get('message')))
else:
incident.addNote(f"Sentinel Get Incident Comments failed with reason: {results.get('reason')}")
Function - Sentinel Get Incident Entities¶
Get the Entities associated with a Sentinel Incident.
Inputs:
Name |
Type |
Required |
Example |
Tooltip |
---|---|---|---|---|
|
|
No |
|
The ID of a Sentinel Incident. |
|
|
No |
|
Label give to the Sentinel server in the app.config. Example in app.config [fn_microsoft_sentinel:label1] the label here is label1. |
|
|
No |
|
The Sentinel profile name. |
Outputs:
NOTE: This example might be in JSON format, but
results
is a Python Dictionary on the SOAR platform.
results = {
"content": {
"b3763ac5-6319-496b-b404-6c5000418a81_4c4831e9-2ed7-a44a-e530-c784cc7c2ddc": [
{
"id": "/subscriptions/xxxxxxxx-xxxx-xxxx-xxxx-xxxxxxxxxxxx/resourceGroups/demoassets/providers/Microsoft.OperationalInsights/workspaces/AzureSentinelDemo/providers/Microsoft.SecurityInsights/entities/b9ce5c65-0b04-27bf-c4af-c5926008a5d3",
"kind": "Ip",
"name": "b9ce5c65-0b04-27bf-c4af-c5926008a5d3",
"properties": {
"address": "80.66.76.30",
"friendlyName": "80.66.76.30"
},
"resilient_artifact_type": "IP Address",
"resilient_artifact_value": "80.66.76.30",
"type": "Microsoft.SecurityInsights/entities"
},
{
"id": "/subscriptions/xxxxxxxx-xxxx-xxxx-xxxx-xxxxxxxxxxxx/resourceGroups/demoassets/providers/Microsoft.OperationalInsights/workspaces/AzureSentinelDemo/providers/Microsoft.SecurityInsights/entities/8e6ffdf6-05ab-b16e-aaa7-c3465af09c46",
"kind": "AzureResource",
"name": "8e6ffdf6-05ab-b16e-aaa7-c3465af09c46",
"properties": {
"additionalData": {
"ResourceName": "PAMSqlSv",
"ResourceType": "Virtual Machine"
},
"friendlyName": "PAMSqlSv",
"resourceId": "/subscriptions/xxxxxxxx-xxxx-xxxx-xxxx-xxxxxxxxxxxx/resourceGroups/DemoAssets/providers/Microsoft.Compute/virtualMachines/PAMSqlSv",
"subscriptionId": "xxxxxxxx-xxxx-xxxx-xxxx-xxxxxxxxxxxx"
},
"resilient_artifact_type": "String",
"resilient_artifact_value": "PAMSqlSv",
"type": "Microsoft.SecurityInsights/entities"
},
{
"id": "/subscriptions/xxxxxxxx-xxxx-xxxx-xxxx-xxxxxxxxxxxx/resourceGroups/demoassets/providers/Microsoft.OperationalInsights/workspaces/AzureSentinelDemo/providers/Microsoft.SecurityInsights/entities/eda45dbc-9011-cfbc-4e9b-4fd97d44f069",
"kind": "Host",
"name": "eda45dbc-9011-cfbc-4e9b-4fd97d44f069",
"properties": {
"azureID": "/subscriptions/xxxxxxxx-xxxx-xxxx-xxxx-xxxxxxxxxxxx/resourceGroups/DemoAssets/providers/Microsoft.Compute/virtualMachines/PAMSqlSv",
"friendlyName": "PAMSqlSv",
"hostName": "PAMSqlSv",
"omsAgentID": "7db452ea-1cae-4079-94a5-85dc76156e96"
},
"resilient_artifact_type": "System Name",
"resilient_artifact_value": "PAMSqlSv",
"type": "Microsoft.SecurityInsights/entities"
},
{
"id": "/subscriptions/xxxxxxxx-xxxx-xxxx-xxxx-xxxxxxxxxxxx/resourceGroups/demoassets/providers/Microsoft.OperationalInsights/workspaces/AzureSentinelDemo/providers/Microsoft.SecurityInsights/entities/7cc5b89b-7a07-55eb-cfb1-a8a8afd5c67e",
"kind": "Account",
"name": "7cc5b89b-7a07-55eb-cfb1-a8a8afd5c67e",
"properties": {
"accountName": "sa",
"friendlyName": "sa"
},
"resilient_artifact_type": "User Account",
"resilient_artifact_value": "sa",
"type": "Microsoft.SecurityInsights/entities"
}
]
},
"inputs": {
"sentinel_incident_id": "b3763ac5-6319-496b-b404-6c5000418a81",
"sentinel_label": "label1",
"sentinel_profile": null
},
"metrics": {
"execution_time_ms": 2788,
"host": "local",
"package": "fn-microsoft-sentinel",
"package_version": "2.1.0",
"timestamp": "2024-04-10 14:08:00",
"version": "1.0"
},
"raw": null,
"reason": null,
"success": true,
"version": 2.0
}
Example Function Input Script:
inputs.sentinel_incident_id = incident.properties.sentinel_incident_number
if incident.properties.sentinel_profile:
inputs.sentinel_profile = incident.properties.sentinel_profile
if incident.properties.sentinel_label:
inputs.sentinel_label = incident.properties.sentinel_label
Example Function Post Process Script:
from datetime import datetime
current_dt = datetime.now()
results = playbook.functions.results.entities_results
if results.get("success"):
for alert in results.get("content", {}).keys():
for entity in results.get("content", {}).get(alert):
row = incident.addRow("sentinel_incident_entities")
row['report_date'] = current_dt
row['alert_id'] = alert
row['entity_id'] = entity.get('name')
row['entity_type'] = entity.get('kind')
row['entity_value'] = entity.get('properties', {}).get('friendlyName')
row['entity_properties'] = "<br>".join(["<b>{}</b>: {}".format(k, v) for k, v in entity.get('properties', {}).items()])
# Create an artifact
desc = [f"created from Sentinel entity: {entity.get('name')}"]
if entity.get('properties', {}).get('azureID'):
desc.append(entity.get('properties', {}).get('azureID'))
if entity.get('resilient_artifact_type'):
incident.addArtifact(entity.get('resilient_artifact_type'), entity.get('resilient_artifact_value'), "\n".join(desc))
else:
incident.addNote(f"Sentinel Incident Sync failed with reason: {results.get('reason')}")
Function - Sentinel Update Incident¶
Update / Close a Sentinel incident based on Sentinel field changes in the SOAR Incident.
Inputs:
Name |
Type |
Required |
Example |
Tooltip |
---|---|---|---|---|
|
|
No |
|
SOAR Incident ID |
|
|
No |
|
The ID of a Sentinel Incident. |
|
|
No |
|
Label give to the Sentinel server in the app.config. Example in app.config [fn_microsoft_sentinel:label1] the label here is label1. |
|
|
No |
|
The Sentinel profile name. |
Outputs:
NOTE: This example might be in JSON format, but
results
is a Python Dictionary on the SOAR platform.
results = {
"content": {
"etag": "\"1f01df97-0000-0100-0000-6616d7020000\"",
"id": "/subscriptions/xxxxxxxx-xxxx-xxxx-xxxx-xxxxxxxxxxxx/resourceGroups/demoassets/providers/Microsoft.OperationalInsights/workspaces/AzureSentinelDemo/providers/Microsoft.SecurityInsights/Incidents/b3763ac5-6319-496b-b404-6c5000418a81",
"name": "b3763ac5-6319-496b-b404-6c5000418a81",
"properties": {
"additionalData": {
"alertProductNames": [
"Azure Security Center"
],
"alertsCount": 1,
"bookmarksCount": 0,
"commentsCount": 2,
"tactics": [
"PreAttack"
],
"techniques": []
},
"createdTimeUtc": "2024-04-10T15:29:50.0960914Z",
"description": "Brute-force attack is a common attack technique for finding valid credentials to the database. By submitting many users/passwords combinations, an attacker can guess a correct one.\u00a0 Once obtained, an attacker can have full access to the database. While this specific alert doesn\u0027t indicate a successful brute-force, it is advised to take safety measures to protect your resource against this attack.\u00a0 To investigate this suspected brute-force attempt, review it\u0027s origin (based on the application name and IP/Location), and try to find out whether it\u0027s recognized to you, or suspicious.\u00a0 If you believe this to be an attack on your database, use firewall rules to limit the access to your resource, and make sure you use strong passwords and not well known user names.\u00a0 Also, consider using only AAD authentication to further enhance your security posture.\u00a0",
"incidentNumber": 4609,
"incidentUrl": "https://portal.azure.com/#asset/Microsoft_Azure_Security_Insights/Incident/subscriptions/xxxxxxxx-xxxx-xxxx-xxxx-xxxxxxxxxxxx/resourceGroups/demoassets/providers/Microsoft.OperationalInsights/workspaces/AzureSentinelDemo/providers/Microsoft.SecurityInsights/Incidents/b3763ac5-6319-496b-b404-6c5000418a81",
"labels": [
{
"labelName": "test tag",
"labelType": "User"
},
{
"labelName": "another label",
"labelType": "User"
}
],
"lastModifiedTimeUtc": "2024-04-10T18:14:26.8445378Z",
"owner": {
"assignedTo": null,
"email": null,
"objectId": null,
"ownerType": null,
"userPrincipalName": null
},
"providerIncidentId": "4609",
"providerName": "Azure Sentinel",
"relatedAnalyticRuleIds": [
"/subscriptions/xxxxxxxx-xxxx-xxxx-xxxx-xxxxxxxxxxxx/resourceGroups/demoassets/providers/Microsoft.OperationalInsights/workspaces/AzureSentinelDemo/providers/Microsoft.SecurityInsights/alertRules/14fefa89-3cdd-429f-923b-b524aa8c65c3"
],
"severity": "High",
"status": "Active",
"title": "Suspected brute-force attack attempt"
},
"type": "Microsoft.SecurityInsights/Incidents"
},
"inputs": {
"incident_id": 2121,
"sentinel_incident_id": "b3763ac5-6319-496b-b404-6c5000418a81",
"sentinel_label": "label1",
"sentinel_profile": null
},
"metrics": {
"execution_time_ms": 2209,
"host": "local",
"package": "fn-microsoft-sentinel",
"package_version": "2.1.0",
"timestamp": "2024-04-10 14:14:26",
"version": "1.0"
},
"raw": null,
"reason": null,
"success": true,
"version": 2.0
}
Example Function Input Script:
inputs.incident_id = incident.id
inputs.sentinel_incident_id = incident.properties.sentinel_incident_number
if incident.properties.sentinel_profile:
inputs.sentinel_profile = incident.properties.sentinel_profile
if incident.properties.sentinel_label:
inputs.sentinel_label = incident.properties.sentinel_label
Example Function Post Process Script:
None
Playbooks¶
Playbook Name |
Description |
Activation Type |
Object |
Status |
Condition |
---|---|---|---|---|---|
Sentinel: Comment Sync - Example (PB) |
Sync an Incident note to a Sentinel Incident comment. |
Automatic |
note |
|
|
Sentinel: Get Incident Alerts - Example (PB) |
Get alerts for a Sentinel Incident. |
Manual |
incident |
|
|
Sentinel: Get Incident Comments - Example (PB) |
Get Sentinel incident comments. |
Manual |
incident |
|
|
Sentinel: Get Incident Entities - Example (PB) |
Get Entities for a Sentinel Incident. |
Manual |
incident |
|
|
Sentinel: Incident Sync - Example (PB) |
Sync SOAR incident with linked Sentinel incident. |
Automatic |
incident |
|
|
Sentinel: Update Incident - Example (PB) |
Update SOAR incident with new information from the Sentinel incident. |
Automatic |
incident |
|
|
Custom Layouts¶
Import the Data Tables and Custom Fields like the screenshot below:
The Sentinel fields are used to capture information specific on a Sentinel Incident, including close incident information (Classification). The datatable, Sentinel Entities, is used to capture specific entity information. Another datatable, Sentinel Comment IDs, is used for tracking purposes and need not be added to a layout.
Data Table - Sentinel Incident Alerts¶
API Name:¶
sentinel_incident_alerts
Columns:¶
Column Name |
API Access Name |
Type |
Tooltip |
---|---|---|---|
Alert Date |
|
|
Date and time of the Sentinel Alert. |
Alert Id |
|
|
Sentinel Alert ID. |
Alert Name |
|
|
Name of the Sentinel Alert. |
Alert Type |
|
|
The Type of Sentinel Alert. |
Compromised Entity |
|
|
Compromised Entities on the Sentinel Alert. |
Confidence Level |
|
|
Confidence level of the Sentinel Alert. |
Description |
|
|
Description of the Sentinel Alert. |
Link |
|
|
URL link to the Sentinel Alert. |
Remediation Steps |
|
|
Remediation steps from the Sentinel Alert. |
Report Date |
|
|
Report date of the Sentinel Incident. |
Severity |
|
|
Severity of the Sentinel Alert. |
Status |
|
|
Status of the Sentinel Alert. |
Tactics |
|
|
Tactics of the Sentinel Alert. |
Data Table - Sentinel Incident Entities¶
API Name:¶
sentinel_incident_entities
Columns:¶
Column Name |
API Access Name |
Type |
Tooltip |
---|---|---|---|
Alert ID |
|
|
The Sentinel Alert ID. |
Entity ID |
|
|
The Sentinel Entity ID. |
Entity Properties |
|
|
The properties of the Sentinel Entity. |
Entity Type |
|
|
The type of the Sentinel Entity. |
Entity Value |
|
|
The value of the Sentinel Entity. |
Report Date |
|
|
The reported date and time of the Sentinel Incident. |
Custom Fields¶
Label |
API Access Name |
Type |
Prefix |
Placeholder |
Tooltip |
---|---|---|---|---|---|
Sentinel Incident Assigned To |
|
|
|
- |
- |
Sentinel Incident Classification |
|
|
|
- |
- |
Sentinel Incident Classification Comment |
|
|
|
- |
- |
Sentinel Incident Classification Reason |
|
|
|
- |
- |
Sentinel Incident ID |
|
|
|
- |
- |
Sentinel Incident Labels |
|
|
|
- |
- |
Sentinel Incident Number |
|
|
|
- |
- |
Sentinel Incident Status |
|
|
|
- |
- |
Sentinel Incident Tactics |
|
|
|
- |
- |
Sentinel Incident URL |
|
|
|
- |
- |
Sentinel Server Label |
|
|
|
- |
Label give to the Sentinel server in the app.config. Example in app.config [fn_microsoft_sentinel:label1] the label here is label1. |
Sentinel Profile |
|
|
|
- |
- |
Custom Templates¶
Below are examples of templates for creating, updating, and closing SOAR incidents. Customize these templates and refer to them in our app.config file.
Each template should be reviewed for correctness in your enterprise.
For instance, closing a SOAR incident may include additional custom fields which the default template
does not include. Or, remove the owner_id
mapping entry entirely.
In addition, the incident_creation_template
and incident_update_template
includes the following
mapping:
{# if Sentinel users are different than SOAR users, consider using a mapping table using soar_substitute: #}
{# {{ properties.owner.assignedTo|soar_substitute('{"sentinel_user1@co.com": "soar_user1@ent.com", "sentinel_user2@co.com": "soar_user2@ent.com"}') }} #}
"owner_id": "{{ properties.owner.assignedTo }}",
If your Sentinel login users differ from SOAR users, modify the owner_id
mapping to use a substitution map as indicated in the jinja comment.
incident_create_template.jinja¶
{
{# JINJA template for creating a new SOAR incident from a Sentinel incident. #}
"name": "Sentinel Incident {{ properties.incidentNumber|e }} - {{ properties.title|replace('"', '\\"') }}",
"discovered_date": {{ properties.createdTimeUtc|soar_datetimeformat }},
"start_date": {% if properties.firstActivityTimeUtc %} {{ properties.firstActivityTimeUtc|soar_datetimeformat }} {% else %} {{ properties.createdTimeUtc|soar_datetimeformat }} {%endif %},
{% if properties.description %}
"description": {
"format": "text",
"content": "{{ properties.description|replace('"', '\\"') }}"
},
{% endif %}
{# if Sentinel users are different than SOAR users, consider using a mapping table using soar_substitute: #}
{# "owner_id": "{{ properties.owner.userPrincipalName|soar_substitute('{"sentinel_user1@co.com": "soar_user1@ent.com", "sentinel_user2@co.com": "soar_user2@ent.com", "DEFAULT": "default_user@ent.com" }') }}", #}
"plan_status": "{{ properties.status|soar_substitute('{"Closed": "C", "Active": "A", "New": "A"}') }}",
"severity_code": "{{ properties.severity|soar_substitute('{"Informational": "Low"}') }}",
"properties": {
"sentinel_incident_number": "{{ name|e }}",
"sentinel_incident_id": "{{ properties.incidentNumber }}",
"sentinel_incident_status": "{{ properties.status }}",
"sentinel_incident_url": "<a target='blank' href='{{ properties.incidentUrl }}'>Sentinel Incident</a>"
{% if properties.classification %}
,"sentinel_incident_classification": "{{ properties.classification }}"
{% endif %}
{% if properties.classificationReason %}
,"sentinel_incident_classification_reason": "{{ properties.classificationReason }}"
{% endif %}
{% if properties.classificationComment %}
,"sentinel_incident_classification_comment": "{{ properties.classificationComment|replace('"', '\\"') }}"
{% endif %}
{% if properties.owner.assignedTo %}
,"sentinel_incident_assigned_to": "{{ properties.owner.assignedTo }}"
{% endif %}
{% if properties.labels %}
,"sentinel_incident_labels": "{{ properties.labels | map(attribute='labelName') | join(',') | replace('"', '\\"') | replace('\\', '\\\\') }}"
{% endif %}
{% if properties.additionalData.tactics %}
,"sentinel_incident_tactics": "{{ properties.additionalData.tactics|join(' ') }}"
{% endif %}
{% if soar_profile %}
,"sentinel_profile": "{{ soar_profile }}"
{% endif %}
{% if resilient_profile %}
,"sentinel_profile": "{{ resilient_profile }}"
{% endif %}
{% if soar_label %}
,"sentinel_label": "{{ soar_label }}"
{% endif %}
}
{% if properties.status == "Closed" %}
,
"resolution_id": "Resolved",
"resolution_summary": "Closed by Sentinel"
{%endif %}
}
incident_update_template.jinja¶
{
{# JINJA template for updating a new SOAR incident from a Sentinel incident. #}
"description": {
"format": "text",
"content": "{{ properties.description|replace('"', '\\"') }}"
},
{# if Sentinel users are different than SOAR users, consider using a mapping table using soar_substitute: #}
{# "owner_id": "{{ properties.owner.userPrincipalName|soar_substitute('{"sentinel_user1@co.com": "soar_user1@ent.com", "sentinel_user2@co.com": "soar_user2@ent.com", "DEFAULT": "default_user@ent.com" }') }}", #}
"plan_status": "{{ properties.status|soar_substitute('{"Closed": "A", "Active": "A", "New": "A"}') }}",
"severity_code": "{{ properties.severity|soar_substitute('{"Informational": "Low"}') }}",
"properties": {
"sentinel_incident_status": "{{ properties.status }}",
"sentinel_incident_classification": "{{ properties.classification }}",
"sentinel_incident_classification_reason": "{{ properties.classificationReason }}",
"sentinel_incident_classification_comment": "{{ properties.classificationComment|replace('"', '\\"') }}",
"sentinel_incident_assigned_to": "{{ properties.owner.assignedTo }}",
"sentinel_incident_labels": "{{ properties.labels | map(attribute='labelName') | join(',') | replace('"', '\\"') | replace('\\', '\\\\') }}",
"sentinel_incident_tactics": "{{ properties.additionalData.tactics|join(' ') }}"
}
}
incident_close_template.jinja¶
{
{# JINJA template for closing a new SOAR incident from a Sentinel incident. #}
"plan_status": "C",
"resolution_id": "Resolved",
"resolution_summary": "Closed by Sentinel",
"properties": {
"sentinel_incident_status": "{{ properties.status }}",
"sentinel_incident_classification": "{{ properties.classification }}",
"sentinel_incident_classification_reason": "{{ properties.classificationReason }}",
"sentinel_incident_classification_comment": "{{ properties.classificationComment|safe }}",
"sentinel_incident_assigned_to": "{{ properties.owner.assignedTo }}",
"sentinel_incident_labels": "{{ properties.labels | map(attribute='labelName') | join(',') | replace('"', '\\"') | replace('\\', '\\\\') }}",
"sentinel_incident_tactics": "{{ properties.additionalData.tactics|join(' ') }}"
}
}
sentinel_close_incident_template.jinja¶
{
{# JINJA template for closing a new Sentinel incident from a SOAR incident. #}
"properties": {
"title": "{{ name|soar_splitpart(1)}}",
"severity": "{{ severity_code|string|soar_substitute('{"Low": "Low", "Medium": "Medium", "High": "High"}') }}",
"status": "Closed",
"classification": "{{ resolution_id|string|soar_substitute('{"Unresolved": "Undetermined", "Duplicate": "FalsePositive", "Not an Issue": "BenignPositive", "Resolved": "TruePositive"}') }}",
"classificationReason": "{{ resolution_id|string|soar_substitute('{"Unresolved": "", "Duplicate": "InaccurateData", "Not an Issue": "SuspiciousButExpected", "Resolved": "SuspiciousActivity"}') }}",
"classificationComment": "{{ resolution_summary|striptags|safe }}"
{# modify as necessary #}
{% if properties.sentinel_incident_labels %}
{% set label_list = properties.sentinel_incident_labels.split(',') -%}
,"labels": [
{% for label in label_list %}
{% if label %}
{
"labelName": {{ label | trim | json }},
"labelType": "User"
}
{% endif %}
{% if not loop.last %},{% endif %}
{% endfor %}
]
{% endif %}
}
}
sentinel_update_incident_template.jinja¶
{
{# JINJA template for updating a new Sentinel incident from a SOAR incident. #}
"properties": {
"title": "{{ name|soar_splitpart(1) }}",
"severity": "{{ severity_code|string|soar_substitute('{"Low": "Low", "Medium": "Medium", "High": "High"}') }}",
"status": "{{ properties.sentinel_incident_status }}"
{% if description %}
,"description": "{{ description|striptags|safe }}"
{% endif %}
{% if properties.sentinel_incident_labels %}
{% set label_list = properties.sentinel_incident_labels.split(',') -%}
,"labels": [
{% for label in label_list %}
{% if label %}
{
"labelName": {{ label | trim | json }},
"labelType": "User"
}
{% endif %}
{% if not loop.last %},{% endif %}
{% endfor %}
]
{% endif %}
}
}
Custom poller filter template¶
In version 2.0.0 the ability to use a jinja template for the poller filter was added. The following is the example template that is provided with this integration. This template also shows other filter examples that are commented out. The example template shows all Sentinel incident properties that can be filter for.
poller_filters_template.jinja¶
{
{# JINJA template for filtering Sentinel Incidents. #}
{# If the value equals true then it passed the filter. #}
{# To add a filter for a property add an if else to that property where it equals true if it passes the filter. #}
"title": true,
{# Example filter for title that will filter out Sentinel incidents that do not have the string "TEST-" in the title:
"title": {% if "TEST-" in title %} true {% else %} false {% endif %}
#}
"description": true,
{# If the severity of the Sentinel incident is High, or, Medium, or Low then the Sentinel incident will pass this filter. #}
"severity": {% if severity in ["High", "Medium", "Low"] %} true {% else %} false {% endif %},
{# If the status of the Sentinel incident is either New or Active than the Sentinel incident will pass this filter. #}
"status": {% if status in ["New", "Active"] %} true {% else %} false {% endif %},
"email": true,
"assignedTo": true,
{# Example filter for assignedTo that will test if the Sentinel incident is assigned to a user in the given list:
"assignedTo": {% if assignedTo in ["Test User", "sys admin"] %} true {% else %} false {% endif %}
#}
"userPrincipalName": true,
{# labels are equal to Tags on Sentinel #}
"labels": true,
{# Example of filtering labels for a specific labelName:
"labels": false,
{% set present = false %}
{% for tag in labels %}
{% if tag.labelName == "tech3" %}
"labels": true,
{% endif %}
{% endfor %}
#}
{# If the alertsCount of the Sentinel incident is 0 or greater then the Sentinel incident will pass this filter. #}
"alertsCount": {% if alertsCount >= 0 %} true {% else %} false {% endif %},
"bookmarksCount": true,
"commentsCount": true,
"alertProductNames": true,
{# If the tactics property of the Sentinel incident is equal to PreAttack, or InitialAccess or blank then the Sentinel incident will pass this filter. #}
"tactics": {% if "PreAttack" in tactics or "InitialAccess" in tactics or tactics == [] %} true {% else %} false {% endif %},
"relatedAnalyticRuleIds": true,
"providerName": true
}
Troubleshooting & Support¶
Refer to the documentation listed in the Requirements section for troubleshooting information.
For Support¶
This is a IBM supported App. For assistance, see https://ibm.com/mysupport.