Skip to content

Commit

Permalink
fix(hs): time stamp and contact not found issue (#2760)
Browse files Browse the repository at this point in the history
  • Loading branch information
mihir-4116 authored Oct 20, 2023
2 parents 4bb71f4 + 255e9aa commit f5a1d45
Show file tree
Hide file tree
Showing 7 changed files with 213 additions and 21 deletions.
5 changes: 4 additions & 1 deletion src/v0/destinations/hs/HSTransform-v1.js
Original file line number Diff line number Diff line change
Expand Up @@ -34,6 +34,7 @@ const {
getEmailAndUpdatedProps,
formatPropertyValueForIdentify,
getHsSearchId,
populateTraits,
} = require('./util');
const { JSON_MIME_TYPE } = require('../../util/constant');

Expand All @@ -52,7 +53,7 @@ const { JSON_MIME_TYPE } = require('../../util/constant');
*/
const processLegacyIdentify = async (message, destination, propertyMap) => {
const { Config } = destination;
const traits = getFieldValueFromMessage(message, 'traits');
let traits = getFieldValueFromMessage(message, 'traits');
const mappedToDestination = get(message, MappedToDestinationKey);
const operation = get(message, 'context.hubspotOperation');
// if mappedToDestination is set true, then add externalId to traits
Expand Down Expand Up @@ -80,6 +81,8 @@ const processLegacyIdentify = async (message, destination, propertyMap) => {
)}/${hsSearchId}`;
response.method = defaultPatchRequestConfig.requestMethod;
}

traits = await populateTraits(propertyMap, traits, destination);
response.body.JSON = removeUndefinedAndNullValues({ properties: traits });
response.source = 'rETL';
response.operation = operation;
Expand Down
4 changes: 3 additions & 1 deletion src/v0/destinations/hs/HSTransform-v2.js
Original file line number Diff line number Diff line change
Expand Up @@ -41,6 +41,7 @@ const {
searchContacts,
getEventAndPropertiesFromConfig,
getHsSearchId,
populateTraits,
} = require('./util');
const { JSON_MIME_TYPE } = require('../../util/constant');

Expand Down Expand Up @@ -69,7 +70,7 @@ const addHsAuthentication = (response, Config) => {
*/
const processIdentify = async (message, destination, propertyMap) => {
const { Config } = destination;
const traits = getFieldValueFromMessage(message, 'traits');
let traits = getFieldValueFromMessage(message, 'traits');
const mappedToDestination = get(message, MappedToDestinationKey);
const operation = get(message, 'context.hubspotOperation');
const externalIdObj = getDestinationExternalIDObjectForRetl(message, 'HS');
Expand Down Expand Up @@ -124,6 +125,7 @@ const processIdentify = async (message, destination, propertyMap) => {
response.method = defaultPatchRequestConfig.requestMethod;
}

traits = await populateTraits(propertyMap, traits, destination);
response.body.JSON = removeUndefinedAndNullValues({ properties: traits });
response.source = 'rETL';
response.operation = operation;
Expand Down
1 change: 1 addition & 0 deletions src/v0/destinations/hs/transform.js
Original file line number Diff line number Diff line change
Expand Up @@ -88,6 +88,7 @@ const processRouterDest = async (inputs, reqMetadata) => {
if (mappedToDestination && GENERIC_TRUE_VALUES.includes(mappedToDestination?.toString())) {
// skip splitting the batches to inserts and updates if object it is an association
if (objectType.toLowerCase() !== 'association') {
propertyMap = await getProperties(destination);
// get info about existing objects and splitting accordingly.
tempInputs = await splitEventsForCreateUpdate(tempInputs, destination);
}
Expand Down
72 changes: 54 additions & 18 deletions src/v0/destinations/hs/util.js
Original file line number Diff line number Diff line change
Expand Up @@ -158,23 +158,33 @@ const validatePayloadDataTypes = (propertyMap, hsSupportedKey, value, traitsKey)

if (propertyMap[hsSupportedKey] === 'bool' && typeof propValue === 'object') {
throw new InstrumentationError(
`Property ${traitsKey} data type ${typeof propValue} is not matching with Hubspot property data type ${
propertyMap[hsSupportedKey]
`Property ${traitsKey} data type ${typeof propValue} is not matching with Hubspot property data type ${propertyMap[hsSupportedKey]
}`,
);
}

if (propertyMap[hsSupportedKey] === 'number' && typeof propValue !== 'number') {
throw new InstrumentationError(
`Property ${traitsKey} data type ${typeof propValue} is not matching with Hubspot property data type ${
propertyMap[hsSupportedKey]
`Property ${traitsKey} data type ${typeof propValue} is not matching with Hubspot property data type ${propertyMap[hsSupportedKey]
}`,
);
}

return propValue;
};

/**
* Converts date to UTC Midnight TimeStamp
* @param {*} propValue
* @returns
*/
const getUTCMidnightTimeStampValue = (propValue) => {
const time = propValue;
const date = new Date(time);
date.setUTCHours(0, 0, 0, 0);
return date.getTime();
}

/**
* add addtional properties in the payload that is provided in traits
* only when it matches with HS properties (pre-defined/created from dashboard)
Expand Down Expand Up @@ -204,10 +214,7 @@ const getTransformedJSON = async (message, destination, propertyMap) => {
if (!rawPayload[traitsKey] && propertyMap[hsSupportedKey]) {
let propValue = traits[traitsKey];
if (propertyMap[hsSupportedKey] === 'date') {
const time = propValue;
const date = new Date(time);
date.setUTCHours(0, 0, 0, 0);
propValue = date.getTime();
propValue = getUTCMidnightTimeStampValue(propValue);
}

rawPayload[hsSupportedKey] = validatePayloadDataTypes(
Expand Down Expand Up @@ -459,7 +466,7 @@ const getEventAndPropertiesFromConfig = (message, destination, payload) => {
*/
const getExistingData = async (inputs, destination) => {
const { Config } = destination;
const values = [];
let values = [];
let searchResponse;
let updateHubspotIds = [];
const firstMessage = inputs[0].message;
Expand All @@ -478,8 +485,10 @@ const getExistingData = async (inputs, destination) => {
inputs.map(async (input) => {
const { message } = input;
const { destinationExternalId } = getDestinationExternalIDInfoForRetl(message, DESTINATION);
values.push(destinationExternalId);
values.push(destinationExternalId.toString().toLowerCase());
});

values = Array.from(new Set(values));
const requestData = {
filterGroups: [
{
Expand Down Expand Up @@ -523,15 +532,15 @@ const getExistingData = async (inputs, destination) => {
searchResponse =
Config.authorizationType === 'newPrivateAppApi'
? await httpPOST(url, requestData, requestOptions, {
destType: 'hs',
feature: 'transformation',
endpointPath,
})
destType: 'hs',
feature: 'transformation',
endpointPath,
})
: await httpPOST(url, requestData, {
destType: 'hs',
feature: 'transformation',
endpointPath,
});
destType: 'hs',
feature: 'transformation',
endpointPath,
});
searchResponse = processAxiosResponse(searchResponse);

if (searchResponse.status !== 200) {
Expand Down Expand Up @@ -626,6 +635,31 @@ const getHsSearchId = (message) => {
return { hsSearchId };
};

/**
* returns updated traits
* @param {*} propertyMap
* @param {*} traits
* @param {*} destination
*/
const populateTraits = async (propertyMap, traits, destination) => {
const populatedTraits = traits;
let propertyToTypeMap = propertyMap;
if (!propertyToTypeMap) {
// fetch HS properties
propertyToTypeMap = await getProperties(destination);
}

const keys = Object.keys(populatedTraits);
keys.forEach((key) => {
const value = populatedTraits[key];
if (propertyToTypeMap[key] === 'date') {
populatedTraits[key] = getUTCMidnightTimeStampValue(value);
}
})

return populatedTraits;
}

module.exports = {
validateDestinationConfig,
formatKey,
Expand All @@ -639,4 +673,6 @@ module.exports = {
splitEventsForCreateUpdate,
getHsSearchId,
validatePayloadDataTypes,
getUTCMidnightTimeStampValue,
populateTraits,
};
24 changes: 24 additions & 0 deletions test/__mocks__/data/hs/response.json
Original file line number Diff line number Diff line change
Expand Up @@ -3,6 +3,18 @@
{ "name": "company_size", "type": "string" },
{ "name": "date_of_birth", "type": "string" },
{ "name": "days_to_close", "type": "number" },
{
"name": "date_submitted",
"type": "date"
},
{
"name": "days_create",
"type": "date"
},
{
"name": "days_closed",
"type": "date"
},
{ "name": "degree", "type": "string" },
{ "name": "field_of_study", "type": "string" },
{ "name": "first_conversion_date", "type": "datetime" },
Expand Down Expand Up @@ -192,6 +204,18 @@
{ "name": "company_size", "type": "string" },
{ "name": "date_of_birth", "type": "string" },
{ "name": "days_to_close", "type": "number" },
{
"name": "date_submitted",
"type": "date"
},
{
"name": "date_created",
"type": "date"
},
{
"name": "date_closed",
"type": "date"
},
{ "name": "degree", "type": "string" },
{ "name": "field_of_study", "type": "string" },
{ "name": "first_conversion_date", "type": "datetime" },
Expand Down
112 changes: 112 additions & 0 deletions test/__tests__/data/hs_router_rETL_input.json
Original file line number Diff line number Diff line change
Expand Up @@ -216,5 +216,117 @@
"metadata": {
"jobId": 3
}
},
{
"message": {
"channel": "web",
"context": {
"mappedToDestination": true,
"externalId": [
{
"identifierType": "email",
"id": "[email protected]",
"type": "HS-lead"
}
],
"sources": {
"job_id": "24c5HJxHomh6YCngEOCgjS5r1KX/Syncher",
"task_id": "vw_rs_mailchimp_mocked_hg_data",
"version": "v1.8.1",
"batch_id": "f252c69d-c40d-450e-bcd2-2cf26cb62762",
"job_run_id": "c8el40l6e87v0c4hkbl0",
"task_run_id": "c8el40l6e87v0c4hkblg"
}
},
"type": "identify",
"traits": {
"firstname": "Test Hubspot",
"anonymousId": "123451",
"country": "India",
"date_submitted": "2023-09-25T17:31:04.128251Z",
"date_created": "2023-03-30T01:02:03.05Z",
"date_closed": "2023-10-18T04:38:59.229347Z"
},
"messageId": "50360b9c-ea8d-409c-b672-c9230f91cce5",
"originalTimestamp": "2019-10-15T09:35:31.288Z",
"anonymousId": "00000000000000000000000000",
"userId": "12345",
"integrations": {
"All": true
},
"sentAt": "2019-10-14T09:03:22.563Z"
},
"destination": {
"Config": {
"authorizationType": "newPrivateAppApi",
"accessToken": "dummy-access-token",
"hubID": "dummy-hubId",
"apiKey": "dummy-apikey",
"apiVersion": "newApi",
"lookupField": "lookupField",
"hubspotEvents": [
{
"rsEventName": "Purchase",
"hubspotEventName": "pedummy-hubId_rs_hub_test",
"eventProperties": [
{
"from": "Revenue",
"to": "value"
},
{
"from": "Price",
"to": "cost"
}
]
},
{
"rsEventName": "Order Complete",
"hubspotEventName": "pedummy-hubId_rs_hub_chair",
"eventProperties": [
{
"from": "firstName",
"to": "first_name"
},
{
"from": "lastName",
"to": "last_name"
}
]
}
],
"eventFilteringOption": "disable",
"blacklistedEvents": [
{
"eventName": ""
}
],
"whitelistedEvents": [
{
"eventName": ""
}
]
},
"secretConfig": {},
"ID": "1mMy5cqbtfuaKZv1IhVQKnBdVwe",
"name": "Hubspot",
"enabled": true,
"workspaceId": "1TSN08muJTZwH8iCDmnnRt1pmLd",
"deleted": false,
"createdAt": "2020-12-30T08:39:32.005Z",
"updatedAt": "2021-02-03T16:22:31.374Z",
"destinationDefinition": {
"id": "1aIXqM806xAVm92nx07YwKbRrO9",
"name": "HS",
"displayName": "Hubspot",
"createdAt": "2020-04-09T09:24:31.794Z",
"updatedAt": "2021-01-11T11:03:28.103Z"
},
"transformations": [],
"isConnectionEnabled": true,
"isProcessorEnabled": true
},
"metadata": {
"jobId": 4
}
}
]
16 changes: 15 additions & 1 deletion test/__tests__/data/hs_router_rETL_output.json
Original file line number Diff line number Diff line change
Expand Up @@ -20,6 +20,17 @@
"country": "India 1",
"email": "[email protected]"
}
},
{
"properties": {
"firstname": "Test Hubspot",
"anonymousId": "123451",
"country": "India",
"email": "[email protected]",
"date_closed": 1697587200000,
"date_created": 1680134400000,
"date_submitted": 1695600000000
}
}
]
},
Expand All @@ -32,6 +43,9 @@
"metadata": [
{
"jobId": 3
},
{
"jobId": 4
}
],
"batched": true,
Expand Down Expand Up @@ -214,4 +228,4 @@
"isProcessorEnabled": true
}
}
]
]

0 comments on commit f5a1d45

Please sign in to comment.