Webhook
Overview
A webhook Action sends a notification to a URL of your choosing when one or more Rules with the Action flag an email message. The HTTP notification includes information about the message, the mailbox, the flagged Rules, and any triggered Actions, so you can set up an unlimited variety of integrations in response to flagged message events.
Here is some of what you can do with webhooks:
- Send Flagged Message events to your SIEM for analytics or to correlate with events from other tools, such as your EDR
- Send Flagged Message events to your SOAR, to trigger sophisticated DFIR workflows that are today only used on reported phish
- Trigger an AWS Lambda function
How to create a Webhook Action
- Set up the endpoint URL in the external system that you want to send Flagged Message events to (e.g. your SIEM, SOAR, or an intermediary like AWS Lambda or Zapier). Some systems call an integration that provides a URL for receiving webhook notifications a "webhook integration".
- Go to your Sublime Dashboard and click "Actions"
- Click "New Action" and select "Webhook"
- Give your webhook a name, provide the endpoint URL you generated externally in step 1, and optionally specify custom headers that will be included in the HTTP request for webhook notifications
- Click "Save"
How to add your new Webhook Action to Rules
- Click "Detection" or "Triage" in the left nav of your Dashboard
- Click on the Rule you wish to add the Action to, opening the detail view for this Rule
- Click "Edit" in the top right of this detail view
- Click the "Actions" dropdown and select your new webhook Action
- Click "Save Rule"
Custom headers
Any custom headers you configure for your webhook Actions will be be included in the HTTP requests for webhook notifications. You can include as many custom headers as you need.
The webhook payload
Once you activate a webhook Action and connect it to a Rule, whenever that Rule flags a message, the endpoint for the webhook will receive an HTTP POST
request with a body like the following:
{
"id": "5e02026c-55c1-4cbb-8a18-76eb2f3e06d3",
"api_version": "0",
"created_at": "2021-06-30T16:17:22.937642Z",
"type": "message.flagged",
"data": {
"message": {
"id": "dff3be1d-b348-4dab-bb60-582842909a88",
"canonical_id": "767f6519458092af9994a79ae73ca055e1365618",
"external_id": "17a5db5ea7fa760d",
"message_source_id": "a5843e7d-4ba5-40df-954e-73cb2f1e3e7e",
"mailbox": {
"id": "f0654b68-2686-491e-8fcb-e4ad61c61ad8",
"external_id": "109412126579921477721"
}
},
"flagged_rules": [
{
"id": "756b841c-4560-49b3-a8c0-dbacb2900aa5",
"name": "Brand impersonation: Chase Bank",
"severity": "high",
"tags": [
"brand-impersonation",
"suspicious-sender"
]
},
{
"id": "67cfc6d8-3cbb-4338-ae8e-fffb0adccd12",
"name": "Firebase storage link",
"severity": "medium",
"tags": [
"suspicious-content"
]
}
],
"triggered_actions": [
{
"id": "6f45c807-a9c7-47a8-8c3e-5391e4f477ca",
"name": "Auto-trash",
"type": "trash_message"
},
{
"id": "8edd8b97-7b85-40d2-9045-a619ae9b4e52",
"name": "Notify SIEM",
"type": "webhook"
}
]
}
}
Payload properties
The data.message
property contains information about the message that triggered the webhook notification, a subset of the data available through the retrieve message endpoint, including:
- the Sublime-provided
id
of the message, which can be used to fetch additional information through the retrieve message endpoint, including if and when the user read, forwarded, or replied to a message, as well as with other message endpoints like the endpoint to retrieve an image of a message - the
canonical_id
for the message group the message belongs to, which can be used with message group API endpoints like trash message group
Source IPs
If the domain receiving your webhook supports IPv6, you'll see frequently changing IPv6 addresses making the calls to your endpoint.
If only IPv4 is used, you'll see stable IPs making the calls to your endpoint. If you're using one of Sublime's Managed deployments you can check the table below. If you're using AWS CloudFormation, we recommending sending several test requests to establish to two IP addresses used by your deployment. And if you're using docker-compose, the IP address may vary or be stable based on the networking of the instance docker compose is running on.
Managed Instance | IPs |
---|---|
platform.sublime.security | 44.194.118.42 , 34.197.67.5 |
eu.platform.sublime.security | 54.247.166.54 , 52.214.238.48 |
au.platform.sublime.security | 13.238.69.11 , 54.252.178.93 |
ca.platform.sublime.security | 15.157.18.163 , 3.97.76.222 |
You may choose to create an IP allowlist based on these IPs, although we recommend checking the webhook signature instead.
Responding to webhook notifications
The endpoint URL you configure for a webhook should respond to the webhook's HTTP requests with an HTTP response code starting with 2
(for example, 200
) within 10 seconds. If your integration needs to do anything in response to webhook notifications that may take more than 10 seconds, it should send an HTTP response before doing that work.
Check the webhook signature
Webhook notifications include a signature in the X-Sublime-Signature
HTTP header. This allows you to verify that the requests were sent by Sublime, not some other party.
Before you can verify signatures, you need to retrieve your webhook Action's signing secret. Select a webhook Action from the "Actions" page, then click "Click to reveal", or click the "Copy" button to copy the secret directly to your clipboard.
The X-Sublime-Signature
header included in each request contains a timestamp and one or more signatures. The timestamp is prefixed by t=
, and each signature is prefixed by a scheme. Schemes start with v
, followed by an integer. Currently, the only valid signature scheme is v0
.
t=1626139320,v0=3d35d777510a7ccf73ded723f141b0ada2986aebe63c4e45a9c7e4bb1abaca06
Sublime generates signatures using a hash-based message authentication code (HMAC) with SHA-256. To prevent downgrade attacks, you should ignore all schemes that are not v0
.
Preventing replay attacks
A replay attack is when an attacker intercepts a valid payload and its signature, then re-transmits them. To mitigate such attacks, Sublime includes a timestamp in the X-Sublime-Signature
header. Because this timestamp is part of the signed payload, it is also verified by the signature, so an attacker cannot change the timestamp without invalidating the signature. If the signature is valid but the timestamp is too old, you can have your application reject the payload.
Sublime generates the timestamp and signature each time a webhook notification is sent to your endpoint. If Sublime retries a notification (for example, your endpoint previously replied with a non-2xx status code), then a new signature and timestamp is generated for the new delivery attempt.
Verifying signatures
Follow these steps to verify a request's signature:
Step 1: Extract the timestamp and signatures from the header
Split the header, using the ,
character as the separator, to get a list of elements. Then split each element, using the =
character as the separator, to get a prefix and value pair.
The value for the prefix t
corresponds to the timestamp, and v0
corresponds to the signature (or signatures). You can discard all other elements.
Step 2: Prepare the signed_payload
string
signed_payload
stringThe signed_payload
string is created by concatenating:
- The timestamp (as a string)
- The character
.
- The actual JSON payload (that is, the request body)
Step 3: Determine the expected signature
Compute an HMAC with the SHA256 hash function. Use the webhook's signing secret as the key, and use the signed_payload
string as the message.
Step 4: Compare the signatures
Compare the signature (or signatures) in the header to the expected signature. For an equality match, compute the difference between the current timestamp and the received timestamp, then decide if the difference is within your tolerance.
To protect against timing attacks, use a constant-time string comparison to compare the expected signature to each of the received signatures.
Updated 4 days ago