feat(triggers): add Intercom webhook triggers (#3990)

* feat(triggers): add Intercom webhook triggers

* fix(triggers): address PR review feedback for Intercom triggers
This commit is contained in:
Waleed
2026-04-06 11:49:28 -07:00
committed by GitHub
parent 62ea0f1d41
commit 590f37641c
13 changed files with 570 additions and 2 deletions

View File

@@ -6088,8 +6088,39 @@
}
],
"operationCount": 31,
"triggers": [],
"triggerCount": 0,
"triggers": [
{
"id": "intercom_conversation_created",
"name": "Intercom Conversation Created",
"description": "Trigger workflow when a new conversation is created in Intercom"
},
{
"id": "intercom_conversation_reply",
"name": "Intercom Conversation Reply",
"description": "Trigger workflow when someone replies to an Intercom conversation"
},
{
"id": "intercom_conversation_closed",
"name": "Intercom Conversation Closed",
"description": "Trigger workflow when a conversation is closed in Intercom"
},
{
"id": "intercom_contact_created",
"name": "Intercom Contact Created",
"description": "Trigger workflow when a new lead is created in Intercom"
},
{
"id": "intercom_user_created",
"name": "Intercom User Created",
"description": "Trigger workflow when a new user is created in Intercom"
},
{
"id": "intercom_webhook",
"name": "Intercom Webhook (All Events)",
"description": "Trigger workflow on any Intercom webhook event"
}
],
"triggerCount": 6,
"authType": "api-key",
"category": "tools",
"integrationType": "customer-support",

View File

@@ -2,6 +2,7 @@ import { IntercomIcon } from '@/components/icons'
import type { BlockConfig } from '@/blocks/types'
import { AuthMode, IntegrationType } from '@/blocks/types'
import { createVersionedToolSelector } from '@/blocks/utils'
import { getTrigger } from '@/triggers'
export const IntercomBlock: BlockConfig = {
type: 'intercom',
@@ -1409,6 +1410,26 @@ export const IntercomV2Block: BlockConfig = {
integrationType: IntegrationType.CustomerSupport,
tags: ['customer-support', 'messaging'],
hideFromToolbar: false,
subBlocks: [
...IntercomBlock.subBlocks,
...getTrigger('intercom_conversation_created').subBlocks,
...getTrigger('intercom_conversation_reply').subBlocks,
...getTrigger('intercom_conversation_closed').subBlocks,
...getTrigger('intercom_contact_created').subBlocks,
...getTrigger('intercom_user_created').subBlocks,
...getTrigger('intercom_webhook').subBlocks,
],
triggers: {
enabled: true,
available: [
'intercom_conversation_created',
'intercom_conversation_reply',
'intercom_conversation_closed',
'intercom_contact_created',
'intercom_user_created',
'intercom_webhook',
],
},
tools: {
...IntercomBlock.tools,
access: [

View File

@@ -0,0 +1,120 @@
import crypto from 'crypto'
import { createLogger } from '@sim/logger'
import { NextResponse } from 'next/server'
import { safeCompare } from '@/lib/core/security/encryption'
import type {
AuthContext,
EventMatchContext,
FormatInputContext,
FormatInputResult,
WebhookProviderHandler,
} from '@/lib/webhooks/providers/types'
const logger = createLogger('WebhookProvider:Intercom')
/**
* Validate Intercom webhook signature using HMAC-SHA1.
* Intercom signs payloads with the app's Client Secret and sends the
* signature in the X-Hub-Signature header as "sha1=<hex>".
*/
function validateIntercomSignature(secret: string, signature: string, body: string): boolean {
try {
if (!secret || !signature || !body) {
logger.warn('Intercom signature validation missing required fields', {
hasSecret: !!secret,
hasSignature: !!signature,
hasBody: !!body,
})
return false
}
if (!signature.startsWith('sha1=')) {
logger.warn('Intercom signature has invalid format', {
signature: `${signature.substring(0, 10)}...`,
})
return false
}
const providedSignature = signature.substring(5)
const computedHash = crypto.createHmac('sha1', secret).update(body, 'utf8').digest('hex')
return safeCompare(computedHash, providedSignature)
} catch (error) {
logger.error('Error validating Intercom signature:', error)
return false
}
}
export const intercomHandler: WebhookProviderHandler = {
verifyAuth({ request, rawBody, requestId, providerConfig }: AuthContext) {
const secret = providerConfig.webhookSecret as string | undefined
if (!secret) {
return null
}
const signature = request.headers.get('X-Hub-Signature')
if (!signature) {
logger.warn(`[${requestId}] Intercom webhook missing X-Hub-Signature header`)
return new NextResponse('Unauthorized - Missing Intercom signature', { status: 401 })
}
if (!validateIntercomSignature(secret, signature, rawBody)) {
logger.warn(`[${requestId}] Intercom signature verification failed`, {
signatureLength: signature.length,
secretLength: secret.length,
})
return new NextResponse('Unauthorized - Invalid Intercom signature', { status: 401 })
}
return null
},
handleReachabilityTest(body: unknown, requestId: string) {
const obj = body as Record<string, unknown> | null
if (obj?.topic === 'ping') {
logger.info(
`[${requestId}] Intercom ping event detected - returning 200 without triggering workflow`
)
return NextResponse.json({
status: 'ok',
message: 'Webhook endpoint verified',
})
}
return null
},
async formatInput({ body }: FormatInputContext): Promise<FormatInputResult> {
return { input: body }
},
async matchEvent({ webhook, body, requestId, providerConfig }: EventMatchContext) {
const triggerId = providerConfig.triggerId as string | undefined
const obj = body as Record<string, unknown>
const topic = obj?.topic as string | undefined
if (triggerId && triggerId !== 'intercom_webhook') {
const { isIntercomEventMatch } = await import('@/triggers/intercom/utils')
if (!isIntercomEventMatch(triggerId, topic || '')) {
logger.debug(
`[${requestId}] Intercom event mismatch for trigger ${triggerId}. Topic: ${topic}. Skipping execution.`,
{
webhookId: webhook.id,
triggerId,
receivedTopic: topic,
}
)
return false
}
}
return true
},
extractIdempotencyId(body: unknown) {
const obj = body as Record<string, unknown>
if (obj?.id && obj?.type === 'notification_event') {
return String(obj.id)
}
return null
},
}

View File

@@ -17,6 +17,7 @@ import { googleFormsHandler } from '@/lib/webhooks/providers/google-forms'
import { grainHandler } from '@/lib/webhooks/providers/grain'
import { hubspotHandler } from '@/lib/webhooks/providers/hubspot'
import { imapHandler } from '@/lib/webhooks/providers/imap'
import { intercomHandler } from '@/lib/webhooks/providers/intercom'
import { jiraHandler } from '@/lib/webhooks/providers/jira'
import { lemlistHandler } from '@/lib/webhooks/providers/lemlist'
import { linearHandler } from '@/lib/webhooks/providers/linear'
@@ -55,6 +56,7 @@ const PROVIDER_HANDLERS: Record<string, WebhookProviderHandler> = {
grain: grainHandler,
hubspot: hubspotHandler,
imap: imapHandler,
intercom: intercomHandler,
jira: jiraHandler,
lemlist: lemlistHandler,
linear: linearHandler,

View File

@@ -0,0 +1,41 @@
import { IntercomIcon } from '@/components/icons'
import { buildTriggerSubBlocks } from '@/triggers'
import {
buildIntercomContactOutputs,
buildIntercomExtraFields,
intercomSetupInstructions,
intercomTriggerOptions,
} from '@/triggers/intercom/utils'
import type { TriggerConfig } from '@/triggers/types'
/**
* Intercom Contact Created Trigger
*
* Fires when a new lead is created in Intercom.
* Note: In Intercom, contact.created fires for leads only.
* For identified users, use the User Created trigger (user.created topic).
*/
export const intercomContactCreatedTrigger: TriggerConfig = {
id: 'intercom_contact_created',
name: 'Intercom Contact Created',
provider: 'intercom',
description: 'Trigger workflow when a new lead is created in Intercom',
version: '1.0.0',
icon: IntercomIcon,
subBlocks: buildTriggerSubBlocks({
triggerId: 'intercom_contact_created',
triggerOptions: intercomTriggerOptions,
setupInstructions: intercomSetupInstructions('contact.created'),
extraFields: buildIntercomExtraFields('intercom_contact_created'),
}),
outputs: buildIntercomContactOutputs(),
webhook: {
method: 'POST',
headers: {
'Content-Type': 'application/json',
},
},
}

View File

@@ -0,0 +1,39 @@
import { IntercomIcon } from '@/components/icons'
import { buildTriggerSubBlocks } from '@/triggers'
import {
buildIntercomConversationOutputs,
buildIntercomExtraFields,
intercomSetupInstructions,
intercomTriggerOptions,
} from '@/triggers/intercom/utils'
import type { TriggerConfig } from '@/triggers/types'
/**
* Intercom Conversation Closed Trigger
*
* Fires when an admin closes a conversation.
*/
export const intercomConversationClosedTrigger: TriggerConfig = {
id: 'intercom_conversation_closed',
name: 'Intercom Conversation Closed',
provider: 'intercom',
description: 'Trigger workflow when a conversation is closed in Intercom',
version: '1.0.0',
icon: IntercomIcon,
subBlocks: buildTriggerSubBlocks({
triggerId: 'intercom_conversation_closed',
triggerOptions: intercomTriggerOptions,
setupInstructions: intercomSetupInstructions('conversation.admin.closed'),
extraFields: buildIntercomExtraFields('intercom_conversation_closed'),
}),
outputs: buildIntercomConversationOutputs(),
webhook: {
method: 'POST',
headers: {
'Content-Type': 'application/json',
},
},
}

View File

@@ -0,0 +1,43 @@
import { IntercomIcon } from '@/components/icons'
import { buildTriggerSubBlocks } from '@/triggers'
import {
buildIntercomConversationOutputs,
buildIntercomExtraFields,
intercomSetupInstructions,
intercomTriggerOptions,
} from '@/triggers/intercom/utils'
import type { TriggerConfig } from '@/triggers/types'
/**
* Intercom Conversation Created Trigger
*
* This is the PRIMARY trigger - it includes the dropdown for selecting trigger type.
* Fires when a user/lead starts a new conversation or an admin initiates a 1:1 conversation.
*/
export const intercomConversationCreatedTrigger: TriggerConfig = {
id: 'intercom_conversation_created',
name: 'Intercom Conversation Created',
provider: 'intercom',
description: 'Trigger workflow when a new conversation is created in Intercom',
version: '1.0.0',
icon: IntercomIcon,
subBlocks: buildTriggerSubBlocks({
triggerId: 'intercom_conversation_created',
triggerOptions: intercomTriggerOptions,
includeDropdown: true,
setupInstructions: intercomSetupInstructions(
'conversation.user.created and/or conversation.admin.single.created'
),
extraFields: buildIntercomExtraFields('intercom_conversation_created'),
}),
outputs: buildIntercomConversationOutputs(),
webhook: {
method: 'POST',
headers: {
'Content-Type': 'application/json',
},
},
}

View File

@@ -0,0 +1,41 @@
import { IntercomIcon } from '@/components/icons'
import { buildTriggerSubBlocks } from '@/triggers'
import {
buildIntercomConversationOutputs,
buildIntercomExtraFields,
intercomSetupInstructions,
intercomTriggerOptions,
} from '@/triggers/intercom/utils'
import type { TriggerConfig } from '@/triggers/types'
/**
* Intercom Conversation Reply Trigger
*
* Fires when a user, lead, or admin replies to a conversation.
*/
export const intercomConversationReplyTrigger: TriggerConfig = {
id: 'intercom_conversation_reply',
name: 'Intercom Conversation Reply',
provider: 'intercom',
description: 'Trigger workflow when someone replies to an Intercom conversation',
version: '1.0.0',
icon: IntercomIcon,
subBlocks: buildTriggerSubBlocks({
triggerId: 'intercom_conversation_reply',
triggerOptions: intercomTriggerOptions,
setupInstructions: intercomSetupInstructions(
'conversation.user.replied and/or conversation.admin.replied'
),
extraFields: buildIntercomExtraFields('intercom_conversation_reply'),
}),
outputs: buildIntercomConversationOutputs(),
webhook: {
method: 'POST',
headers: {
'Content-Type': 'application/json',
},
},
}

View File

@@ -0,0 +1,6 @@
export { intercomContactCreatedTrigger } from './contact_created'
export { intercomConversationClosedTrigger } from './conversation_closed'
export { intercomConversationCreatedTrigger } from './conversation_created'
export { intercomConversationReplyTrigger } from './conversation_reply'
export { intercomUserCreatedTrigger } from './user_created'
export { intercomWebhookTrigger } from './webhook'

View File

@@ -0,0 +1,41 @@
import { IntercomIcon } from '@/components/icons'
import { buildTriggerSubBlocks } from '@/triggers'
import {
buildIntercomContactOutputs,
buildIntercomExtraFields,
intercomSetupInstructions,
intercomTriggerOptions,
} from '@/triggers/intercom/utils'
import type { TriggerConfig } from '@/triggers/types'
/**
* Intercom User Created Trigger
*
* Fires when a new identified user is created in Intercom.
* Note: In Intercom, user.created fires for identified users only.
* For anonymous leads, use the Contact Created trigger (contact.created topic).
*/
export const intercomUserCreatedTrigger: TriggerConfig = {
id: 'intercom_user_created',
name: 'Intercom User Created',
provider: 'intercom',
description: 'Trigger workflow when a new user is created in Intercom',
version: '1.0.0',
icon: IntercomIcon,
subBlocks: buildTriggerSubBlocks({
triggerId: 'intercom_user_created',
triggerOptions: intercomTriggerOptions,
setupInstructions: intercomSetupInstructions('user.created'),
extraFields: buildIntercomExtraFields('intercom_user_created'),
}),
outputs: buildIntercomContactOutputs(),
webhook: {
method: 'POST',
headers: {
'Content-Type': 'application/json',
},
},
}

View File

@@ -0,0 +1,128 @@
import type { SubBlockConfig } from '@/blocks/types'
import type { TriggerOutput } from '@/triggers/types'
/**
* Dropdown options for the Intercom trigger type selector.
*/
export const intercomTriggerOptions = [
{ label: 'Conversation Created', id: 'intercom_conversation_created' },
{ label: 'Conversation Reply', id: 'intercom_conversation_reply' },
{ label: 'Conversation Closed', id: 'intercom_conversation_closed' },
{ label: 'Contact Created', id: 'intercom_contact_created' },
{ label: 'User Created', id: 'intercom_user_created' },
{ label: 'All Events', id: 'intercom_webhook' },
]
/**
* Generates HTML setup instructions for Intercom webhook triggers.
*/
export function intercomSetupInstructions(eventType: string): string {
const instructions = [
'Copy the <strong>Webhook URL</strong> above.',
'Go to your <a href="https://app.intercom.com/a/apps/_/developer-hub" target="_blank" rel="noopener noreferrer">Intercom Developer Hub</a>.',
'Select your app, then go to <strong>Webhooks</strong>.',
'Paste the webhook URL into the <strong>Endpoint URL</strong> field.',
`Select the <strong>${eventType}</strong> topic(s).`,
"Copy your app's <strong>Client Secret</strong> from the app's <strong>Basic Information</strong> page and paste it into the <strong>Webhook Secret</strong> field above (recommended for security).",
'Save the webhook configuration.',
'Deploy your workflow to activate the trigger.',
]
return instructions
.map(
(instruction, index) =>
`<div class="mb-3"><strong>${index + 1}.</strong> ${instruction}</div>`
)
.join('')
}
/**
* Extra fields for Intercom triggers (webhook secret for signature verification).
*/
export function buildIntercomExtraFields(triggerId: string): SubBlockConfig[] {
return [
{
id: 'webhookSecret',
title: 'Webhook Secret',
type: 'short-input',
placeholder: 'Enter your Intercom app Client Secret',
description:
"Your app's Client Secret from the Developer Hub. Used to verify webhook authenticity via X-Hub-Signature.",
password: true,
mode: 'trigger',
condition: { field: 'selectedTriggerId', value: triggerId },
},
]
}
/**
* Maps trigger IDs to the Intercom webhook topics they should match.
*/
export const INTERCOM_TRIGGER_TOPIC_MAP: Record<string, string[]> = {
intercom_conversation_created: ['conversation.user.created', 'conversation.admin.single.created'],
intercom_conversation_reply: ['conversation.user.replied', 'conversation.admin.replied'],
intercom_conversation_closed: ['conversation.admin.closed'],
intercom_contact_created: ['contact.created'],
intercom_user_created: ['user.created'],
intercom_webhook: [], // Empty = accept all events
}
/**
* Checks if an Intercom webhook event matches the configured trigger.
*/
export function isIntercomEventMatch(triggerId: string, topic: string): boolean {
const allowedTopics = INTERCOM_TRIGGER_TOPIC_MAP[triggerId]
if (allowedTopics === undefined) return false
if (allowedTopics.length === 0) {
return true
}
return allowedTopics.includes(topic)
}
/**
* Shared base outputs for all Intercom webhook triggers.
*/
function buildIntercomBaseOutputs(dataDescription: string): Record<string, TriggerOutput> {
return {
topic: { type: 'string', description: 'The webhook topic (e.g., conversation.user.created)' },
id: { type: 'string', description: 'Unique notification ID' },
app_id: { type: 'string', description: 'Your Intercom app ID' },
created_at: { type: 'number', description: 'Unix timestamp when the event occurred' },
delivery_attempts: {
type: 'number',
description: 'Number of delivery attempts for this notification',
},
first_sent_at: {
type: 'number',
description: 'Unix timestamp of first delivery attempt',
},
data: { type: 'json', description: dataDescription },
} as Record<string, TriggerOutput>
}
/**
* Build outputs for Intercom conversation triggers.
*/
export function buildIntercomConversationOutputs(): Record<string, TriggerOutput> {
return buildIntercomBaseOutputs(
'Event data containing the conversation object. Access via data.item for conversation details including id, state, open, assignee, contacts, conversation_parts, tags, and source'
)
}
/**
* Build outputs for Intercom contact triggers.
*/
export function buildIntercomContactOutputs(): Record<string, TriggerOutput> {
return buildIntercomBaseOutputs(
'Event data containing the contact object. Access via data.item for contact details including id, role, email, name, phone, external_id, custom_attributes, location, avatar, tags, companies, and timestamps'
)
}
/**
* Build outputs for the generic Intercom webhook trigger.
*/
export function buildIntercomGenericOutputs(): Record<string, TriggerOutput> {
return buildIntercomBaseOutputs(
'Event data containing the affected object. Access via data.item for the resource (conversation, contact, company, ticket, etc.)'
)
}

View File

@@ -0,0 +1,41 @@
import { IntercomIcon } from '@/components/icons'
import { buildTriggerSubBlocks } from '@/triggers'
import {
buildIntercomExtraFields,
buildIntercomGenericOutputs,
intercomSetupInstructions,
intercomTriggerOptions,
} from '@/triggers/intercom/utils'
import type { TriggerConfig } from '@/triggers/types'
/**
* Intercom Generic Webhook Trigger
*
* Accepts all Intercom webhook events.
*/
export const intercomWebhookTrigger: TriggerConfig = {
id: 'intercom_webhook',
name: 'Intercom Webhook (All Events)',
provider: 'intercom',
description: 'Trigger workflow on any Intercom webhook event',
version: '1.0.0',
icon: IntercomIcon,
subBlocks: buildTriggerSubBlocks({
triggerId: 'intercom_webhook',
triggerOptions: intercomTriggerOptions,
setupInstructions: intercomSetupInstructions(
'events you want to receive (conversation, contact, user, company, ticket, etc.)'
),
extraFields: buildIntercomExtraFields('intercom_webhook'),
}),
outputs: buildIntercomGenericOutputs(),
webhook: {
method: 'POST',
headers: {
'Content-Type': 'application/json',
},
},
}

View File

@@ -131,6 +131,14 @@ import {
hubspotWebhookTrigger,
} from '@/triggers/hubspot'
import { imapPollingTrigger } from '@/triggers/imap'
import {
intercomContactCreatedTrigger,
intercomConversationClosedTrigger,
intercomConversationCreatedTrigger,
intercomConversationReplyTrigger,
intercomUserCreatedTrigger,
intercomWebhookTrigger,
} from '@/triggers/intercom'
import {
jiraIssueCommentedTrigger,
jiraIssueCreatedTrigger,
@@ -381,4 +389,10 @@ export const TRIGGER_REGISTRY: TriggerRegistry = {
hubspot_ticket_restored: hubspotTicketRestoredTrigger,
hubspot_webhook: hubspotWebhookTrigger,
imap_poller: imapPollingTrigger,
intercom_conversation_created: intercomConversationCreatedTrigger,
intercom_conversation_reply: intercomConversationReplyTrigger,
intercom_conversation_closed: intercomConversationClosedTrigger,
intercom_contact_created: intercomContactCreatedTrigger,
intercom_user_created: intercomUserCreatedTrigger,
intercom_webhook: intercomWebhookTrigger,
}