diff --git a/src/appmixer/hubspot/bundle.json b/src/appmixer/hubspot/bundle.json index a5ab46ef5..b4d23e2a2 100644 --- a/src/appmixer/hubspot/bundle.json +++ b/src/appmixer/hubspot/bundle.json @@ -1,6 +1,6 @@ { "name": "appmixer.hubspot", - "version": "4.0.6", + "version": "4.1.0", "engine": ">=6.0.0", "changelog": { "1.0.0": [ @@ -60,6 +60,11 @@ ], "4.0.6": [ "Fixed an issue when the `CreateDeal` component was showing error `resource not found`." - ] + ], + "4.1.0": [ + "Added custom properties and additional properties to the following components: `CreateContact`, `CreateDeal`, `UpdateContact` and `UpdateDeal`. Custom properties are new properties that are not part of the default HubSpot properties. Additional properties default HubSpot properties but were not part of the default properties list before.", + "Added caching of Contact and Deal properties, which can be further configured in the connector settings.", + "Improved listing output variables for `GetContact` to make it consistent with the selected properties in the input." + ] } } diff --git a/src/appmixer/hubspot/commons.js b/src/appmixer/hubspot/commons.js index 1f5472389..fe048b449 100644 --- a/src/appmixer/hubspot/commons.js +++ b/src/appmixer/hubspot/commons.js @@ -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(); + } + } }; /** diff --git a/src/appmixer/hubspot/crm/CreateContact/CreateContact.js b/src/appmixer/hubspot/crm/CreateContact/CreateContact.js index 2505f86fa..fb3d6d4ce 100644 --- a/src/appmixer/hubspot/crm/CreateContact/CreateContact.js +++ b/src/appmixer/hubspot/crm/CreateContact/CreateContact.js @@ -21,6 +21,12 @@ module.exports = { const { auth } = context; const hs = new Hubspot(auth.accessToken, context.config); + const additionalPropertiesArray = context.messages.in.content.additionalProperties?.AND || []; + const additionalProperties = additionalPropertiesArray.reduce((acc, field) => { + acc[field.name] = field.value; + return acc; + }, {}); + const payload = { properties: { email, @@ -32,7 +38,8 @@ module.exports = { address: address || '', city: city || '' , state: state || '' , - zip: zip || '' + zip: zip || '', + ...additionalProperties } }; const { data } = await hs.call('post', 'crm/v3/objects/contacts', payload); @@ -48,7 +55,9 @@ 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 : '', + ...additionalProperties }, 'contact'); } diff --git a/src/appmixer/hubspot/crm/CreateContact/component.json b/src/appmixer/hubspot/crm/CreateContact/component.json index 8807c8a7c..e9011d6cb 100644 --- a/src/appmixer/hubspot/crm/CreateContact/component.json +++ b/src/appmixer/hubspot/crm/CreateContact/component.json @@ -1,6 +1,6 @@ { "name": "appmixer.hubspot.crm.CreateContact", - "author": "Zulkafil Tabish ", + "author": "Appmixer ", "icon": "", "description": "Create a new contact.", "private": false, @@ -29,7 +29,8 @@ "address": { "type": "string" }, "city": { "type": "string" }, "state": { "type": "string" }, - "zip": { "type": "string" } + "zip": { "type": "string" }, + "additionalProperties": { "type": "object" } }, "required": [ "email" @@ -96,6 +97,34 @@ "label": "Zip", "tooltip": "The contact's zip.", "index": 10 + }, + "additionalProperties": { + "type": "expression", + "label": "Additional Properties", + "tooltip": "Addition HubSpot or 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#additionalFieldsToSelectArray" + } + } + }, + "value": { + "type": "text", + "label": "Value", + "tooltip": "The value for the property.", + "index": 2 + } + } } } } @@ -118,7 +147,7 @@ ] } ], - "version": "1.0.1", + "version": "1.0.2", "dependencies": { "service": "1.0.1" } diff --git a/src/appmixer/hubspot/crm/CreateDeal/CreateDeal.js b/src/appmixer/hubspot/crm/CreateDeal/CreateDeal.js index d04e92236..dacbe69c5 100644 --- a/src/appmixer/hubspot/crm/CreateDeal/CreateDeal.js +++ b/src/appmixer/hubspot/crm/CreateDeal/CreateDeal.js @@ -17,13 +17,20 @@ module.exports = { const { auth } = context; const hs = new Hubspot(auth.accessToken, context.config); + const additionalPropertiesArray = context.messages.in.content.additionalProperties?.AND || []; + const additionalProperties = additionalPropertiesArray.reduce((acc, field) => { + acc[field.name] = field.value; + return acc; + }, {}); + const payload = { properties: { amount: amount || '', dealname: dealname, dealstage: dealstage || 'appointmentscheduled', hubspot_owner_id: hubSpotOwnerId, - pipeline: pipeline || 'default' + pipeline: pipeline || 'default', + ...additionalProperties } }; diff --git a/src/appmixer/hubspot/crm/CreateDeal/component.json b/src/appmixer/hubspot/crm/CreateDeal/component.json index 44a1ca799..e3bcfe884 100644 --- a/src/appmixer/hubspot/crm/CreateDeal/component.json +++ b/src/appmixer/hubspot/crm/CreateDeal/component.json @@ -1,9 +1,9 @@ { "name": "appmixer.hubspot.crm.CreateDeal", - "author": "Zulkafil Tabish ", + "author": "Appmixer ", "icon": "", "description": "Create a new deal.", - "version": "1.0.3", + "version": "1.0.4", "dependencies": { "service": "1.0.1" }, @@ -29,8 +29,8 @@ "pipeline": { "type": "string" }, "hubSpotOwnerId": { "type": "string" }, "closedate": { "type": "string" }, - "amount": { "type": "string" } - + "amount": { "type": "string" }, + "additionalProperties": { "type": "object" } }, "required": [ "dealname" @@ -97,6 +97,34 @@ "label": "Amount", "tooltip": "Amount.", "index": 6 + }, + "additionalProperties": { + "type": "expression", + "label": "Additional Properties", + "tooltip": "Addition HubSpot or custom properties to add to the deal.", + "exclusiveFields": ["name"], + "index": 7, + "levels": ["AND"], + "fields": { + "name": { + "type": "select", + "label": "Property", + "tooltip": "The property to add to the deal.", + "index": 1, + "source": { + "url": "/component/appmixer/hubspot/crm/GetDealsProperties?outPort=out", + "data": { + "transform": "./GetDealsProperties#additionalFieldsToSelectArray" + } + } + }, + "value": { + "type": "text", + "label": "Value", + "tooltip": "The value for the property.", + "index": 2 + } + } } }, "groups": { diff --git a/src/appmixer/hubspot/crm/GetContact/component.json b/src/appmixer/hubspot/crm/GetContact/component.json index 9da7b9f58..3c51a1d7e 100644 --- a/src/appmixer/hubspot/crm/GetContact/component.json +++ b/src/appmixer/hubspot/crm/GetContact/component.json @@ -3,7 +3,7 @@ "author": "Zulkafil Tabish ", "icon": "", "description": "Get a contact by Email or ID.", - "version": "1.0.2", + "version": "1.0.3", "private": false, "auth": { "service": "appmixer:hubspot" @@ -73,6 +73,9 @@ "source": { "url": "/component/appmixer/hubspot/crm/GetContactsProperties?outPort=out", "data": { + "messages": { + "in/properties": "inputs/in/properties" + }, "transform": "./transformers#contactToSelectArray" } } diff --git a/src/appmixer/hubspot/crm/GetContactsProperties/GetContactsProperties.js b/src/appmixer/hubspot/crm/GetContactsProperties/GetContactsProperties.js index f6b5bd785..dacd3c931 100644 --- a/src/appmixer/hubspot/crm/GetContactsProperties/GetContactsProperties.js +++ b/src/appmixer/hubspot/crm/GetContactsProperties/GetContactsProperties.js @@ -1,5 +1,7 @@ 'use strict'; + const Hubspot = require('../../Hubspot'); +const { getObjectProperties, WATCHED_PROPERTIES_CONTACT } = require('../../commons'); module.exports = { @@ -7,9 +9,28 @@ module.exports = { const { auth } = context; const hs = new Hubspot(auth.accessToken, context.config); - const { data } = await hs.call('get', 'crm/v3/properties/contacts'); + const properties = await getObjectProperties(context, hs, 'contacts', 'all'); + + const propertiesToOutput = context.messages.in.content?.properties; + if (propertiesToOutput) { + // We have a set of properties defined in the inspector. + // We only want to return these properties. See GetContact/component.json - outPorts + const propertiesToReturn = properties.filter((property) => propertiesToOutput.includes(property.name)); + + return context.sendJson(propertiesToReturn, 'out'); + } - return context.sendJson(data.results, 'out'); + return context.sendJson(properties, 'out'); + }, + + /** Returns properties that not hardcoded into the component. Both custom and HubSpot properties. */ + additionalFieldsToSelectArray(contactsProperties) { + return contactsProperties + .filter((property) => property.formField) + .filter((property) => !WATCHED_PROPERTIES_CONTACT.includes(property.name)) + .map((property) => { + return { label: property.label, value: property.name }; + }); }, contactsPropertiesToContactInspector(contactsProperties) { diff --git a/src/appmixer/hubspot/crm/GetContactsProperties/component.json b/src/appmixer/hubspot/crm/GetContactsProperties/component.json index 6ce7f14b5..139415f66 100644 --- a/src/appmixer/hubspot/crm/GetContactsProperties/component.json +++ b/src/appmixer/hubspot/crm/GetContactsProperties/component.json @@ -1,6 +1,6 @@ { "name": "appmixer.hubspot.crm.GetContactsProperties", - "author": "Zulkafil Tabish ", + "author": "Appmixer ", "icon": "", "description": "Get a list of all contacts properties.", "private": true, @@ -16,7 +16,7 @@ }, "inPorts": [ "in" ], "outPorts": [ "out" ], - "version": "1.1.0", + "version": "1.1.1", "dependencies": { "service": "1.0.1" } diff --git a/src/appmixer/hubspot/crm/GetDealsProperties/GetDealsProperties.js b/src/appmixer/hubspot/crm/GetDealsProperties/GetDealsProperties.js index 56db21bd8..2d98a7b17 100644 --- a/src/appmixer/hubspot/crm/GetDealsProperties/GetDealsProperties.js +++ b/src/appmixer/hubspot/crm/GetDealsProperties/GetDealsProperties.js @@ -1,5 +1,6 @@ 'use strict'; const Hubspot = require('../../Hubspot'); +const { getObjectProperties, WATCHED_PROPERTIES_DEAL } = require('../../commons'); module.exports = { @@ -7,9 +8,9 @@ module.exports = { const { auth } = context; const hs = new Hubspot(auth.accessToken, context.config); - const { data } = await hs.call('get', 'crm/v3/properties/deals'); + const properties = await getObjectProperties(context, hs, 'deals', 'all'); - return context.sendJson(data.results, 'out'); + return context.sendJson(properties, 'out'); }, dealsPropertiesToDealInspector(dealsProperties) { @@ -64,6 +65,17 @@ module.exports = { return inspector; }, + /** Returns properties that not hardcoded into the component. Both custom and HubSpot properties. */ + additionalFieldsToSelectArray(dealsProperties) { + + return dealsProperties + .filter((property) => property.formField) + .filter((property) => !WATCHED_PROPERTIES_DEAL.includes(property.name)) + .map((property) => { + return { label: property.label, value: property.name }; + }); + }, + dealToLabelNameArray(dealsProperties) { const transformed = []; diff --git a/src/appmixer/hubspot/crm/GetDealsProperties/component.json b/src/appmixer/hubspot/crm/GetDealsProperties/component.json index a8cf10d7d..2574e4245 100644 --- a/src/appmixer/hubspot/crm/GetDealsProperties/component.json +++ b/src/appmixer/hubspot/crm/GetDealsProperties/component.json @@ -1,6 +1,6 @@ { "name": "appmixer.hubspot.crm.GetDealsProperties", - "author": "Zulkafil Tabish ", + "author": "Appmixer ", "icon": "", "description": "Get a list of all deals properties.", "private": true, @@ -16,7 +16,7 @@ }, "inPorts": [ "in" ], "outPorts": [ "out" ], - "version": "1.0.1", + "version": "1.0.2", "dependencies": { "service": "1.0.1" } diff --git a/src/appmixer/hubspot/crm/NewContact/NewContact.js b/src/appmixer/hubspot/crm/NewContact/NewContact.js index 97c57d512..6b5598284 100644 --- a/src/appmixer/hubspot/crm/NewContact/NewContact.js +++ b/src/appmixer/hubspot/crm/NewContact/NewContact.js @@ -1,5 +1,6 @@ 'use strict'; const BaseSubscriptionComponent = require('../../BaseSubscriptionComponent'); +const { getObjectProperties } = require('../../commons'); const subscriptionType = 'contact.creation'; @@ -56,8 +57,7 @@ class NewContact extends BaseSubscriptionComponent { const { properties } = context.properties; if (!properties) { // Return all properties by default. - const { data } = await this.hubspot.call('get', 'crm/v3/properties/contacts'); - propertiesToReturn = data.results?.map((property) => property.name); + propertiesToReturn = await getObjectProperties(context, this.hubspot, 'contacts', 'names'); } else { propertiesToReturn = properties.split(','); } diff --git a/src/appmixer/hubspot/crm/NewDeal/NewDeal.js b/src/appmixer/hubspot/crm/NewDeal/NewDeal.js index 398339ba3..0b332968f 100644 --- a/src/appmixer/hubspot/crm/NewDeal/NewDeal.js +++ b/src/appmixer/hubspot/crm/NewDeal/NewDeal.js @@ -1,5 +1,6 @@ 'use strict'; const BaseSubscriptionComponent = require('../../BaseSubscriptionComponent'); +const { getObjectProperties } = require('../../commons'); const subscriptionType = 'deal.creation'; @@ -56,8 +57,7 @@ class NewDeal extends BaseSubscriptionComponent { const { properties } = context.properties; if (!properties) { // Return all properties by default. - const { data } = await this.hubspot.call('get', 'crm/v3/properties/deals'); - propertiesToReturn = data.results?.map((property) => property.name); + propertiesToReturn = await getObjectProperties(context, this.hubspot, 'deals', 'names'); } else { propertiesToReturn = properties.split(','); } diff --git a/src/appmixer/hubspot/crm/UpdateContact/UpdateContact.js b/src/appmixer/hubspot/crm/UpdateContact/UpdateContact.js index f5e607cc6..7bb988e22 100644 --- a/src/appmixer/hubspot/crm/UpdateContact/UpdateContact.js +++ b/src/appmixer/hubspot/crm/UpdateContact/UpdateContact.js @@ -22,6 +22,12 @@ module.exports = { const { auth } = context; const hs = new Hubspot(auth.accessToken, context.config); + const additionalPropertiesArray = context.messages.in.content.additionalProperties?.AND || []; + const additionalProperties = additionalPropertiesArray.reduce((acc, field) => { + acc[field.name] = field.value; + return acc; + }, {}); + const payload = { properties: { email, @@ -33,7 +39,8 @@ module.exports = { address: address, city: city, state: state, - zip: zip + zip: zip, + ...additionalProperties } }; Object.keys(payload.properties).forEach(property => { diff --git a/src/appmixer/hubspot/crm/UpdateContact/component.json b/src/appmixer/hubspot/crm/UpdateContact/component.json index 5862c8e24..bf0f25e31 100644 --- a/src/appmixer/hubspot/crm/UpdateContact/component.json +++ b/src/appmixer/hubspot/crm/UpdateContact/component.json @@ -1,9 +1,9 @@ { "name": "appmixer.hubspot.crm.UpdateContact", - "author": "Zulkafil Tabish ", + "author": "Appmixer ", "icon": "", "description": "Update a contact.", - "version": "1.0.2", + "version": "1.0.3", "private": false, "auth": { "service": "appmixer:hubspot" @@ -31,7 +31,8 @@ "address": { "type": "string" }, "city": { "type": "string" }, "state": { "type": "string" }, - "zip": { "type": "string" } + "zip": { "type": "string" }, + "additionalProperties": { "type": "object" } }, "anyOf": [ { "required": [ "contactId" ] }, @@ -105,6 +106,34 @@ "label": "Zip", "tooltip": "The contact's zip.", "index": 11 + }, + "additionalProperties": { + "type": "expression", + "label": "Additional Properties", + "tooltip": "Addition HubSpot or custom properties to add to the contact.", + "exclusiveFields": ["name"], + "index": 12, + "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#additionalFieldsToSelectArray" + } + } + }, + "value": { + "type": "text", + "label": "Value", + "tooltip": "The value for the property.", + "index": 2 + } + } } } } diff --git a/src/appmixer/hubspot/crm/UpdateDeal/UpdateDeal.js b/src/appmixer/hubspot/crm/UpdateDeal/UpdateDeal.js index c3db7a32d..507b6c049 100644 --- a/src/appmixer/hubspot/crm/UpdateDeal/UpdateDeal.js +++ b/src/appmixer/hubspot/crm/UpdateDeal/UpdateDeal.js @@ -18,13 +18,20 @@ module.exports = { const { auth } = context; const hs = new Hubspot(auth.accessToken, context.config); + const additionalPropertiesArray = context.messages.in.content.additionalProperties?.AND || []; + const additionalProperties = additionalPropertiesArray.reduce((acc, field) => { + acc[field.name] = field.value; + return acc; + }, {}); + const payload = { properties: { amount: amount, dealname: dealname, dealstage: dealstage, hubspot_owner_id: hubSpotOwnerId, - pipeline: pipeline + pipeline: pipeline, + ...additionalProperties } }; @@ -46,6 +53,5 @@ module.exports = { const { data } = await hs.call('patch', `crm/v3/objects/deals/${dealId}`, payload); return context.sendJson(data, 'updateDeal'); - } }; diff --git a/src/appmixer/hubspot/crm/UpdateDeal/component.json b/src/appmixer/hubspot/crm/UpdateDeal/component.json index 4753363fe..79c094809 100644 --- a/src/appmixer/hubspot/crm/UpdateDeal/component.json +++ b/src/appmixer/hubspot/crm/UpdateDeal/component.json @@ -1,9 +1,9 @@ { "name": "appmixer.hubspot.crm.UpdateDeal", - "author": "Zulkafil Tabish ", + "author": "Appmixer ", "icon": "", "description": "Update a deal.", - "version": "1.0.2", + "version": "1.0.3", "private": false, "auth": { "service": "appmixer:hubspot" @@ -27,7 +27,8 @@ "pipeline": { "type": "string" }, "hubSpotOwnerId": { "type": "string" }, "closedate": { "type": "string" }, - "amount": { "type": "string" } + "amount": { "type": "string" }, + "additionalProperties": { "type": "object" } }, "required": [ "dealId" @@ -95,6 +96,34 @@ "label": "Amount", "tooltip": "Amount.", "index": 7 + }, + "additionalProperties": { + "type": "expression", + "label": "Additional Properties", + "tooltip": "Addition HubSpot or custom properties to add to the deal.", + "exclusiveFields": ["name"], + "index": 8, + "levels": ["AND"], + "fields": { + "name": { + "type": "select", + "label": "Property", + "tooltip": "The property to add to the deal.", + "index": 1, + "source": { + "url": "/component/appmixer/hubspot/crm/GetDealsProperties?outPort=out", + "data": { + "transform": "./GetDealsProperties#additionalFieldsToSelectArray" + } + } + }, + "value": { + "type": "text", + "label": "Value", + "tooltip": "The value for the property.", + "index": 2 + } + } } }, "groups": { diff --git a/src/appmixer/hubspot/crm/UpdatedContact/UpdatedContact.js b/src/appmixer/hubspot/crm/UpdatedContact/UpdatedContact.js index 7345a8198..39efdfe02 100644 --- a/src/appmixer/hubspot/crm/UpdatedContact/UpdatedContact.js +++ b/src/appmixer/hubspot/crm/UpdatedContact/UpdatedContact.js @@ -1,6 +1,6 @@ 'use strict'; const BaseSubscriptionComponent = require('../../BaseSubscriptionComponent'); -const { WATCHED_PROPERTIES_CONTACT } = require('../../commons'); +const { WATCHED_PROPERTIES_CONTACT, getObjectProperties } = require('../../commons'); const subscriptionType = 'contact.propertyChange'; @@ -65,8 +65,7 @@ class UpdatedContact extends BaseSubscriptionComponent { const { properties } = context.properties; if (!properties) { // Return all properties by default. - const { data } = await this.hubspot.call('get', 'crm/v3/properties/contacts'); - propertiesToReturn = data.results?.map((property) => property.name); + propertiesToReturn = await getObjectProperties(context, this.hubspot, 'contacts', 'names'); } else { propertiesToReturn = properties.split(','); } diff --git a/src/appmixer/hubspot/crm/UpdatedDeal/UpdatedDeal.js b/src/appmixer/hubspot/crm/UpdatedDeal/UpdatedDeal.js index 0c4b45ac4..fa18b9cb4 100644 --- a/src/appmixer/hubspot/crm/UpdatedDeal/UpdatedDeal.js +++ b/src/appmixer/hubspot/crm/UpdatedDeal/UpdatedDeal.js @@ -1,6 +1,6 @@ 'use strict'; const BaseSubscriptionComponent = require('../../BaseSubscriptionComponent'); -const { WATCHED_PROPERTIES_DEAL } = require('../../commons'); +const { WATCHED_PROPERTIES_DEAL, getObjectProperties } = require('../../commons'); const subscriptionType = 'deal.propertyChange'; @@ -63,8 +63,7 @@ class UpdatedDeal extends BaseSubscriptionComponent { const { properties } = context.properties; if (!properties) { // Return all properties by default. - const { data } = await this.hubspot.call('get', 'crm/v3/properties/deals'); - propertiesToReturn = data.results?.map((property) => property.name); + propertiesToReturn = await getObjectProperties(context, this.hubspot, 'deals', 'names'); } else { propertiesToReturn = properties.split(','); }