fleet/website/api/controllers/webhooks/receive-from-zapier.js
Eric c4e8f83a70
Website: update Zapier webhook (#26741)
ChangeS:
- updated the `receive-from-zapier` webhook's expected inputs when
receiving a `receive-new-customer-data` event
2025-02-28 19:44:32 -06:00

162 lines
7.7 KiB
JavaScript
Vendored

module.exports = {
friendlyName: 'Receive Zapier events',
description: 'Receive events from Zapier.',
inputs: {
eventName: {
type: 'string',
description: 'The unique identifier for this Zap.',
moreInfoUrl: 'https://zapier.com/app/assets/zaps/folders/2035513',
required: true,
},
data: {
type: {},
description: 'Data associated with this event.',
whereToGet: { description: 'Check out the Zap in question and see what it\'s sending via HTTP.' },
required: true,
},
webhookSecret: {
type: 'string',
description: 'Used to verify that requests are coming from where we think they are.',
required: true,
},
},
exits: {
success: { description: 'An event has successfully been received.' },
unrecognizedEventName: { description: 'I do not know how to handle that kind of event.', responseType: 'ok' },// TODO: how will zapier react to receiving a bad request response?
couldNotMatchLinkedinId: { description: 'A linkedIn company could not be found using the provided linkedIn url', responseType: 'ok' }
},
fn: async function ({eventName, data, webhookSecret}) {
let assert = require('assert');
if (!sails.config.custom.zapierWebhookSecret) {
throw new Error('No webhook secret configured! (Please set `sails.config.custom.zapierWebhookSecret`.)');
}
if (!sails.config.custom.iqSecret) {
throw new Error('No iqSecret configured! (Please set `sails.config.custom.iqSecret`.)');
}
if (sails.config.custom.zapierWebhookSecret !== webhookSecret) {
throw new Error('Received unexpected webhook request with webhookSecret set to: '+webhookSecret);
}
// Search for any campaigns that have a placeholder URN. If there are more than one records with a placeholder URN, throw an error.
let adCampaignsWithPlaceholderUrns = await AdCampaign.find({
isLatest: true,
linkedinCampaignUrn: {startsWith: 'PLACEHOLDER-'}
});
if(adCampaignsWithPlaceholderUrns.length > 1) {
throw new Error(`Consistency violation. When the receive-from-zapier webhook received an event from the ${eventName} zap. More than one adcampaigns with a placeholder campaign URN exist in the database.`);
}
// Zap: https://zapier.com/editor/280954803
if(eventName === 'update-placeholder-campaign-urn') {
assert(_.isObject(data));
assert(_.isString(data.placeholderUrn));
assert(_.isString(data.linkedinCampaignUrn));
let adCampaignWithThisPlaceholderUrn = await AdCampaign.findOne({linkedinCampaignUrn: data.placeholderUrn});
if(!adCampaignWithThisPlaceholderUrn) {
sails.log.warn(`when the receive-from-zapier webhook received an event to update an AdCampaign record with a non-placeholder linkedinCampaignUrn value (${data.linkedinCampaignUrn}), no record could be found with the specified placeholder (${data.placeholderUrn}).`);
}
await AdCampaign.updateOne({linkedinCampaignUrn: data.placeholderUrn}).set({
linkedinCampaignUrn: data.linkedinCampaignUrn
});
// Zap: https://zapier.com/editor/281086063
} else if (eventName === 'receive-new-customer-data') {
assert(_.isObject(data));
assert(_.isString(data.newMarketingStage));
assert(_.isString(data.persona) && AdCampaign.validate('persona', data.persona));
assert(data.linkedinCompanyId);
// (FUTURE: make field for this and have it already in CRM so this step isn't necessary)
let linkedinCompanyId = data.linkedinCompanyId;
// Check if we have enough space in our current active campaign.
// If so, then use it and update its inventory. Otherwise, prepare to create
// a new campaign, and use that instead, updating our set of active campaigns
// in the db, including marking the new one as the latest and greatest. Along the way,
// communicate with Campaign Manager to update or create the appropriate campaign.
let latestCampaign = await AdCampaign.findOne({ isLatest: true, persona: data.persona });
if (latestCampaign && latestCampaign.linkedinCompanyIds.length < 100) {
// Update ad campaign in Campaign Manager
// > For help w/ Linkedin API, see https://github.com/fleetdm/confidential/tree/main/ads
let filterCriteriaForLatestCampaign = latestCampaign.linkedinCompanyIds.map((id)=>{
return `urn:li:organization:${id}`;
});
// Append the new organization to the end of the the filterCriteria
filterCriteriaForLatestCampaign.push(`urn:li:organization:${linkedinCompanyId}`);
// Send a request to update this campaign.
await sails.helpers.http.sendHttpRequest.with({
method: 'POST',
url: `https://hooks.zapier.com/hooks/catch/3627242/2wdx23r?webhookSecret=${ encodeURIComponent(sails.config.custom.zapierWebhookSecret)}`,
body: {
campaignGroup: sails.config.custom.linkedinAbmCampaignGroupUrn,
name: latestCampaign.name,
linkedinCampaignUrn: latestCampaign.linkedinCampaignUrn,
targetingCriteria: filterCriteriaForLatestCampaign,
}
}).retry();
await AdCampaign.updateOne({ id: latestCampaign.id }).set({
linkedinCompanyIds: _.uniq(latestCampaign.linkedinCompanyIds.concat(linkedinCompanyId))
});
} else {
// First, mark the old campaign as no longer the latest.
// Note: Since we might not have done the first-time setup for
// this persona yet, it's possible there won't actually be a
// campaign record yet. (In that case, we'll create it momentarily.)
if (latestCampaign) {
await AdCampaign.updateOne({ id: latestCampaign.id }).set({
isLatest: false,
});
}//fi
// Create a placeholder linkedinCampaignUrn value to create the record with initially
// We'll use this value in a subsequent webhook run that will save update the record with the real linkedinCampaignUrn (once it has been created).
// Note: there is a possibility that a new campaign can't be created with only one linkedInCompanyID, (There is a minimum audience size of 300)
// In this case, we will treat this new campaign as the latest campaign in the website's database, and send updates for it as new company IDs are added.
// When the campaign actually exists in LinkedIn, Zapier will send another event to update the campaign urn in the website's database.
let placeholderUrn = 'PLACEHOLDER-'+sails.helpers.strings.random();
let nowAt = new Date();
let newCampaignName = `${data.persona} - ${nowAt.toISOString().split('T')[0]} @ ${nowAt.toLocaleString(undefined, {timezone: 'America/Chicago'}).split(', ')[1]} CT`;
// Now save an incomplete reference to the new LinkedIn campaign.
latestCampaign = await AdCampaign.create({
isLatest: true,
persona: data.persona,
name: newCampaignName,
linkedinCampaignUrn: placeholderUrn,
linkedinCompanyIds: [ linkedinCompanyId ],
}).fetch();
// Then create new ad campaign in Campaign Manager
// > For help w/ Linkedin API, see https://github.com/fleetdm/confidential/tree/main/ads
await sails.helpers.http.sendHttpRequest.with({
method: 'POST',
url: `https://hooks.zapier.com/hooks/catch/3627242/2wdx23r?webhookSecret=${ encodeURIComponent(sails.config.custom.zapierWebhookSecret)}`,
body: {
campaignGroup: sails.config.custom.linkedinAbmCampaignGroupUrn,
name: newCampaignName,
targetingCriteria: [`urn:li:organization:${linkedinCompanyId}`],
linkedinCampaignUrn: placeholderUrn,
},
}).retry();
}
} else {
throw 'unrecognizedEventName';
}
}
};