2022-08-17 08:50:24 -07:00
|
|
|
import { createHmac } from 'crypto';
|
2023-01-27 03:22:44 -08:00
|
|
|
import type {
|
2023-03-09 09:13:15 -08:00
|
|
|
IHookFunctions,
|
|
|
|
IWebhookFunctions,
|
2023-01-27 03:22:44 -08:00
|
|
|
IDataObject,
|
|
|
|
INodeType,
|
|
|
|
INodeTypeDescription,
|
|
|
|
IWebhookResponseData,
|
|
|
|
} from 'n8n-workflow';
|
2024-08-29 06:55:53 -07:00
|
|
|
import { NodeConnectionType, randomString } from 'n8n-workflow';
|
2020-03-16 19:02:48 -07:00
|
|
|
|
2022-08-17 08:50:24 -07:00
|
|
|
import { helpscoutApiRequest, helpscoutApiRequestAllItems } from './GenericFunctions';
|
2020-03-16 19:02:48 -07:00
|
|
|
|
|
|
|
export class HelpScoutTrigger implements INodeType {
|
|
|
|
description: INodeTypeDescription = {
|
|
|
|
displayName: 'HelpScout Trigger',
|
|
|
|
name: 'helpScoutTrigger',
|
2021-03-01 14:38:22 -08:00
|
|
|
icon: 'file:helpScout.svg',
|
2020-03-16 19:02:48 -07:00
|
|
|
group: ['trigger'],
|
|
|
|
version: 1,
|
2021-07-03 05:40:16 -07:00
|
|
|
description: 'Starts the workflow when HelpScout events occur',
|
2020-03-16 19:02:48 -07:00
|
|
|
defaults: {
|
|
|
|
name: 'HelpScout Trigger',
|
|
|
|
},
|
|
|
|
inputs: [],
|
2024-08-29 06:55:53 -07:00
|
|
|
outputs: [NodeConnectionType.Main],
|
2020-03-16 19:02:48 -07:00
|
|
|
credentials: [
|
|
|
|
{
|
|
|
|
name: 'helpScoutOAuth2Api',
|
|
|
|
required: true,
|
|
|
|
},
|
|
|
|
],
|
|
|
|
webhooks: [
|
|
|
|
{
|
|
|
|
name: 'default',
|
|
|
|
httpMethod: 'POST',
|
|
|
|
responseMode: 'onReceived',
|
|
|
|
path: 'webhook',
|
|
|
|
},
|
|
|
|
],
|
|
|
|
properties: [
|
|
|
|
{
|
|
|
|
displayName: 'Events',
|
|
|
|
name: 'events',
|
|
|
|
type: 'multiOptions',
|
|
|
|
options: [
|
|
|
|
{
|
2020-03-29 10:10:54 -07:00
|
|
|
name: 'Conversation - Assigned',
|
2020-03-16 19:02:48 -07:00
|
|
|
value: 'convo.assigned',
|
|
|
|
},
|
|
|
|
{
|
2020-03-29 10:10:54 -07:00
|
|
|
name: 'Conversation - Created',
|
2020-03-16 19:02:48 -07:00
|
|
|
value: 'convo.created',
|
|
|
|
},
|
|
|
|
{
|
2020-03-29 10:10:54 -07:00
|
|
|
name: 'Conversation - Deleted',
|
2020-03-16 19:02:48 -07:00
|
|
|
value: 'convo.deleted',
|
|
|
|
},
|
|
|
|
{
|
2020-03-29 10:10:54 -07:00
|
|
|
name: 'Conversation - Merged',
|
2020-03-16 19:02:48 -07:00
|
|
|
value: 'convo.merged',
|
|
|
|
},
|
|
|
|
{
|
2020-03-29 10:10:54 -07:00
|
|
|
name: 'Conversation - Moved',
|
2020-03-16 19:02:48 -07:00
|
|
|
value: 'convo.moved',
|
|
|
|
},
|
|
|
|
{
|
2020-03-29 10:10:54 -07:00
|
|
|
name: 'Conversation - Status',
|
2020-03-16 19:02:48 -07:00
|
|
|
value: 'convo.status',
|
|
|
|
},
|
|
|
|
{
|
2020-03-29 10:10:54 -07:00
|
|
|
name: 'Conversation - Tags',
|
2020-03-16 19:02:48 -07:00
|
|
|
value: 'convo.tags',
|
|
|
|
},
|
|
|
|
{
|
2020-03-29 10:10:54 -07:00
|
|
|
name: 'Conversation Agent Reply - Created',
|
|
|
|
value: 'convo.agent.reply.created',
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: 'Conversation Customer Reply - Created',
|
|
|
|
value: 'convo.customer.reply.created',
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: 'Conversation Note - Created',
|
|
|
|
value: 'convo.note.created',
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: 'Customer - Created',
|
2020-03-16 19:02:48 -07:00
|
|
|
value: 'customer.created',
|
|
|
|
},
|
|
|
|
{
|
2020-03-29 10:10:54 -07:00
|
|
|
name: 'Rating - Received',
|
2020-03-16 19:02:48 -07:00
|
|
|
value: 'satisfaction.ratings',
|
|
|
|
},
|
|
|
|
],
|
|
|
|
default: [],
|
|
|
|
required: true,
|
|
|
|
},
|
|
|
|
],
|
|
|
|
};
|
|
|
|
|
|
|
|
webhookMethods = {
|
|
|
|
default: {
|
|
|
|
async checkExists(this: IHookFunctions): Promise<boolean> {
|
|
|
|
const webhookUrl = this.getNodeWebhookUrl('default');
|
|
|
|
const webhookData = this.getWorkflowStaticData('node');
|
|
|
|
const events = this.getNodeParameter('events') as string;
|
|
|
|
|
|
|
|
// Check all the webhooks which exist already if it is identical to the
|
|
|
|
// one that is supposed to get created.
|
|
|
|
const endpoint = '/v2/webhooks';
|
2022-08-17 08:50:24 -07:00
|
|
|
const data = await helpscoutApiRequestAllItems.call(
|
|
|
|
this,
|
|
|
|
'_embedded.webhooks',
|
|
|
|
'GET',
|
|
|
|
endpoint,
|
|
|
|
{},
|
|
|
|
);
|
2020-03-16 19:02:48 -07:00
|
|
|
|
|
|
|
for (const webhook of data) {
|
|
|
|
if (webhook.url === webhookUrl) {
|
|
|
|
for (const event of events) {
|
2022-08-17 08:50:24 -07:00
|
|
|
if (!webhook.events.includes(event) && webhook.state === 'enabled') {
|
2020-03-16 19:02:48 -07:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// Set webhook-id to be sure that it can be deleted
|
|
|
|
webhookData.webhookId = webhook.id as string;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
},
|
|
|
|
async create(this: IHookFunctions): Promise<boolean> {
|
|
|
|
const webhookData = this.getWorkflowStaticData('node');
|
|
|
|
const webhookUrl = this.getNodeWebhookUrl('default');
|
|
|
|
const events = this.getNodeParameter('events') as string;
|
|
|
|
|
|
|
|
const endpoint = '/v2/webhooks';
|
|
|
|
|
|
|
|
const body = {
|
|
|
|
url: webhookUrl,
|
|
|
|
events,
|
2024-06-19 04:33:57 -07:00
|
|
|
secret: randomString(10).toLowerCase(),
|
2020-03-16 19:02:48 -07:00
|
|
|
};
|
|
|
|
|
2022-08-17 08:50:24 -07:00
|
|
|
const responseData = await helpscoutApiRequest.call(
|
|
|
|
this,
|
|
|
|
'POST',
|
|
|
|
endpoint,
|
|
|
|
body,
|
|
|
|
{},
|
|
|
|
undefined,
|
|
|
|
{ resolveWithFullResponse: true },
|
|
|
|
);
|
2020-03-16 19:02:48 -07:00
|
|
|
|
|
|
|
if (responseData.headers['resource-id'] === undefined) {
|
|
|
|
// Required data is missing so was not successful
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
webhookData.webhookId = responseData.headers['resource-id'] as string;
|
|
|
|
webhookData.secret = body.secret;
|
|
|
|
return true;
|
|
|
|
},
|
|
|
|
async delete(this: IHookFunctions): Promise<boolean> {
|
|
|
|
const webhookData = this.getWorkflowStaticData('node');
|
|
|
|
if (webhookData.webhookId !== undefined) {
|
|
|
|
const endpoint = `/v2/webhooks/${webhookData.webhookId}`;
|
|
|
|
try {
|
|
|
|
await helpscoutApiRequest.call(this, 'DELETE', endpoint);
|
2021-04-16 09:33:36 -07:00
|
|
|
} catch (error) {
|
2020-03-16 19:02:48 -07:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Remove from the static workflow data so that it is clear
|
2023-03-03 09:49:19 -08:00
|
|
|
// that no webhooks are registered anymore
|
2020-03-16 19:02:48 -07:00
|
|
|
delete webhookData.webhookId;
|
|
|
|
delete webhookData.secret;
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
},
|
|
|
|
},
|
|
|
|
};
|
|
|
|
|
|
|
|
async webhook(this: IWebhookFunctions): Promise<IWebhookResponseData> {
|
|
|
|
const req = this.getRequestObject();
|
|
|
|
const bodyData = this.getBodyData();
|
|
|
|
const headerData = this.getHeaderData() as IDataObject;
|
|
|
|
const webhookData = this.getWorkflowStaticData('node');
|
2021-03-01 14:38:22 -08:00
|
|
|
if (headerData['x-helpscout-signature'] === undefined) {
|
2020-03-16 19:02:48 -07:00
|
|
|
return {};
|
|
|
|
}
|
2022-08-17 08:50:24 -07:00
|
|
|
|
|
|
|
const computedSignature = createHmac('sha1', webhookData.secret as string)
|
|
|
|
.update(req.rawBody)
|
|
|
|
.digest('base64');
|
2020-03-16 19:02:48 -07:00
|
|
|
if (headerData['x-helpscout-signature'] !== computedSignature) {
|
|
|
|
return {};
|
|
|
|
}
|
|
|
|
return {
|
2022-08-17 08:50:24 -07:00
|
|
|
workflowData: [this.helpers.returnJsonArray(bodyData)],
|
2020-03-16 19:02:48 -07:00
|
|
|
};
|
|
|
|
}
|
|
|
|
}
|