Skip to content
Merged
Show file tree
Hide file tree
Changes from 4 commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
6 changes: 5 additions & 1 deletion src/appmixer/hubspot/bundle.json
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
{
"name": "appmixer.hubspot",
"version": "4.0.5",
"version": "4.1.0",
"engine": ">=6.0.0",
"changelog": {
"1.0.0": [
Expand Down Expand Up @@ -57,6 +57,10 @@
"4.0.5": [
"Removed webhook subscribing when using AuthHub.",
"Fixed an issue when HubSpot webhook was not automatically created when using own API key."
],
"4.1.0": [
"Added custom properties to the following components: `CreateContact`, `CreateDeal`, `CreateLineItem` and `Engagements - CreateEmail`.",
"Added caching of Contact and Deal properties, which can be further configured in the connector settings."
]
}
}
36 changes: 35 additions & 1 deletion src/appmixer/hubspot/commons.js
Original file line number Diff line number Diff line change
Expand Up @@ -36,7 +36,41 @@ module.exports = {
},

WATCHED_PROPERTIES_CONTACT: ['email', 'firstname', 'lastname', 'phone', 'website', 'company', 'address', 'city', 'state', 'zip'],
WATCHED_PROPERTIES_DEAL: ['dealname', 'dealstage', 'pipeline', 'hubSpotOwnerId', 'closedate', 'amount']
WATCHED_PROPERTIES_DEAL: ['dealname', 'dealstage', 'pipeline', 'hubSpotOwnerId', 'closedate', 'amount'],

async getObjectProperties(context, hubspot, objectType, output = 'all') {

// Default cache TTL set to 1 minute as property definitions rarely change
// Can be configured via context.config.objectPropertiesCacheTTL if needed
const objectPropertiesCacheTTL = context.config.objectPropertiesCacheTTL || (60 * 1000);
const cacheKeyPrefix = 'hubspot_properties_' + objectType;
let lock;
try {
lock = await context.lock(`hubspot_properties_${objectType}`);
const cached = await context.staticCache.get(cacheKeyPrefix + '_' + output);
if (cached) {
return cached;
}

// Get all properties from HubSpot.
const { data } = await hubspot.call('get', `crm/v3/properties/${objectType}`);
const properties = data.results.map(property => property.name);

// Save to cache both versions: triggers and actions.
await context.staticCache.set(cacheKeyPrefix + '_all', data.results, objectPropertiesCacheTTL);
await context.staticCache.set(cacheKeyPrefix + '_names', properties, objectPropertiesCacheTTL);

// For triggers return array of names: ['email', 'firstname', ...]
if (output === 'names') {
return properties;
}

// For actions return array of objects: [{ name: 'email', type: 'string', ... }, ...]
return data.results;
} finally {
await lock?.unlock();
}
}
};

/**
Expand Down
15 changes: 13 additions & 2 deletions src/appmixer/hubspot/crm/CreateContact/CreateContact.js
Original file line number Diff line number Diff line change
Expand Up @@ -21,6 +21,12 @@ module.exports = {
const { auth } = context;
const hs = new Hubspot(auth.accessToken, context.config);

const customFieldsArray = context.messages.in.content.customProperties?.AND || [];
const customProperties = customFieldsArray.reduce((acc, field) => {
acc[field.name] = field.value;
return acc;
}, {});

const payload = {
properties: {
email,
Expand All @@ -32,7 +38,8 @@ module.exports = {
address: address || '',
city: city || '' ,
state: state || '' ,
zip: zip || ''
zip: zip || '',
...customProperties
}
};
const { data } = await hs.call('post', 'crm/v3/objects/contacts', payload);
Expand All @@ -48,7 +55,11 @@ module.exports = {
phone: properties.phone ? properties.phone : '',
state: properties.state ? properties.state : '',
address: properties.address ? properties.address : '',
email: properties.email ? properties.email : ''
email: properties.email ? properties.email : '',
company: properties.company ? properties.company : '',
...Object.entries(properties)
.filter(([key]) => Object.keys(customProperties).includes(key))
.reduce((acc, [key, value]) => ({ ...acc, [key]: value || '' }), {})
}, 'contact');

}
Expand Down
33 changes: 31 additions & 2 deletions src/appmixer/hubspot/crm/CreateContact/component.json
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
{
"name": "appmixer.hubspot.crm.CreateContact",
"author": "Zulkafil Tabish <zulkafil@client.io>",
"author": "Appmixer <info@appmixer.com>",
"icon": "",
"description": "Create a new contact.",
"private": false,
Expand Down Expand Up @@ -29,7 +29,8 @@
"address": { "type": "string" },
"city": { "type": "string" },
"state": { "type": "string" },
"zip": { "type": "string" }
"zip": { "type": "string" },
"customProperties": { "type": "object" }
},
"required": [
"email"
Expand Down Expand Up @@ -96,6 +97,34 @@
"label": "Zip",
"tooltip": "The contact's zip.",
"index": 10
},
"customProperties": {
"type": "expression",
"label": "Custom Properties",
"tooltip": "Custom properties to add to the contact.",
"exclusiveFields": ["name"],
"index": 11,
"levels": ["AND"],
"fields": {
"name": {
"type": "select",
"label": "Property",
"tooltip": "The property to add to the contact.",
"index": 1,
"source": {
"url": "/component/appmixer/hubspot/crm/GetContactsProperties?outPort=out",
"data": {
"transform": "./GetContactsProperties#customFieldsToSelectArray"
}
}
},
"value": {
"type": "text",
"label": "Value",
"tooltip": "The value for the custom property.",
"index": 2
}
}
}
}
}
Expand Down
Loading