Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat: onboard source ortto #2693

Merged
merged 8 commits into from
Oct 13, 2023
Merged
Show file tree
Hide file tree
Changes from all 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
95 changes: 95 additions & 0 deletions src/v0/sources/ortto/event_mapping.json
Original file line number Diff line number Diff line change
@@ -0,0 +1,95 @@
{
"act::s": "Sent email",
"act::c": "Clicked email",
"act::o": "Opened email",
"act::r": "Received email",
"act::b": "Bounced email",
"act::sk": "Skipped email",
"act::rt": "Redacted email",
"act::f": "Forwarded email",
"act::i": "Invalid email",
"act::v": "Viewed email online",
"act::d": "Deferred email",

"act::ss": "Incident Triggered",
"act::ds": "Incident Unacknowledged",
"act::cs": "Service Created",
"act::is": "Service Deleted",
"act::bs": "Service Updated",
"act::des": "Service Updated",
"act::rs": "Service Updated",
"act::rtss": "Service Updated",
"act::ras": "Service Updated",
"act::cos": "Service Updated",
"act::rtns": "Service Updated",
"act::sf": "Submitted form",
"act::oos-a": "Opted out from SMS audience",
"act::ois-a": "Opted in to SMS audience",
"act::oos-all": "Opted out from all SMS",
"act::ros-all": "Opted in to all SMS",
"act::ss-ab": "SMS subscription abuse",
"act::scr": "SMS consent requested",
"act::scf": "SMS consent failed",

"act::sph": "Sent push",
"act::dph": "Delivered push",
"act::cph": "Clicked push",
"act::iph": "Invalid push",
"act::oiip-all": "Opted in to all iOS push",
"act::ooip-all": "Opted out of all iOS push",
"act::oiip-a": "Opted in to iOS push audience",
"act::oips-a": "Opted out of iOS push audience",
"act::oigp-all": "Opted in to all Android push",
"act::oogp-all": "Opted out of all Android push",
"act::oigp-a": "Opted out of Android push audience",
"act::oogs-a": "Opted in to Android push audience",
"act::oiwp-all": "Opted in to all web push",
"act::oowp-all": "Opted out of all web push",
"act::oiwp-a": "Opted in to web push audience",
"act::oows-a": "Opted out of to web push audience",
"act::iws-ab": "Web push subscription abuse",

"act::ws": "Website session",
"act::us": "User session",

"act::enter-audience": "Entered audience",
"act::leave-audience": "Left audience",
"act::ta": "Tag added",
"act::tr": "Tag removed",
"act::u-a": "Unsubscribe from audience",
"act::s-a": "Subscribe to audience",
"act::u-all": "Un subscribe from all email",
"act::s-all": "Resubscribe globally",
"act::s-ab": "Subscription abuse",

"act::ep": "Entered playbook",
"act::lp": "Left playbook",
"act::spa": "Successful playbook action",
"act::fpa": "Failed playbook action",

"act::ej": "Entered journey",
"act::lj": "Left journey",
"act::sja": "Successful journey action",
"act::fja": "Failed journey action",

"act::cw": "Clicked widget",
"act::sw": "Shown widget",
"act::dw": "Dismissed widget",
"act::rw": "Reacted to widget",
"act::swf": "Submitted widget form",
"act::rws": "Responded to widget survey",
"act::wwv": "Watched widget video",
"act::sww": "Spun widget wheel",

"act::cop": "Conversation opened",
"act::cro": "Conversation re-opened",
"act::cas": "Conversation assigned",
"act::clc": "Conversation closed",
"act::csn": "Conversation snoozed",
"act::crt": "Conversation rated",
"act::cms": "Conversation marked as spam",
"act::cvs": "Conversation voice call started",
"act::cve": "Conversation voice call ended",
"act::ctg": "Conversation tagged",
"act::cut": "Conversation un-tagged"
}
38 changes: 38 additions & 0 deletions src/v0/sources/ortto/mapping.json
Original file line number Diff line number Diff line change
@@ -0,0 +1,38 @@
[
{
"sourceKeys": "contact.email",
"destKeys": "context.traits.email"
},
{
"sourceKeys": "contact.external_id",
"destKeys": "userId"
},
{
"sourceKeys": "contact.country.name",
"destKeys": "context.traits.address.country"
},
{
"sourceKeys": "contact.city.name",
"destKeys": "context.traits.address.city"
},
{
"sourceKeys": "contact.postal",
"destKeys": "context.traits.address.postalCode"
},
{
"sourceKeys": "id",
"destKeys": "messageId"
},
{
"sourceKeys": "time",
"destKeys": "originalTimestamp"
},
{
"sourceKeys": "contact.first_name",
"destKeys": "context.traits.firstName"
},
{
"sourceKeys": "contact.last_name",
"destKeys": "context.traits.lastName"
}
]
114 changes: 114 additions & 0 deletions src/v0/sources/ortto/transform.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,114 @@
const path = require('path');
const fs = require('fs');
const { flattenJson, removeUndefinedAndNullValues, generateUUID } = require('../../util');
const Message = require('../message');
const eventMapping = require('./event_mapping.json');
const { JSON_MIME_TYPE } = require('../../util/constant');

// import mapping json using JSON.parse to preserve object key order
const mapping = JSON.parse(fs.readFileSync(path.resolve(__dirname, './mapping.json'), 'utf-8'));

function settingProperties(event, message) {
const messageReplica = message;

// flattening the event and assigning it to properties
messageReplica.properties = removeUndefinedAndNullValues(flattenJson(event));

// fields that are already mapped
const excludeFields = [
'contact.email',
'contact.contact_id',
'id',
'time',
'activity.field_id',
'contact.external_id',
'contact.country.name',
'contact.city.name',
'contact.postal',
'contact.first_name',
'contact.last_name',
'contact.birthday.day',
'contact.birthday.month',
'contact.birthday.year',
'contact.phone_number.c',
'contact.phone_number',
'contact.phone_number.n',
];

// deleting already mapped fields
excludeFields.forEach((field) => {
delete messageReplica.properties[field];
});

return message;
}

function process(event) {
let message = new Message(`ortto`);

// Here, we are checking for the test event to discard them
if (event.activity?.field_id === 'act::test_webhook') {
return {
outputToSource: {
body: Buffer.from(JSON.stringify(event)).toString('base64'),
contentType: JSON_MIME_TYPE,
},
statusCode: 200,
};
}

// we are setting event type as track always
message.setEventType('track');

message.setPropertiesV2(event, mapping);

// setting anonymousId
if (!message.userId) {
message.anonymousId = generateUUID();
}

if (event.contact?.birthday) {
const month =
event.contact.birthday?.month < 10
sanpj2292 marked this conversation as resolved.
Show resolved Hide resolved
? `0${event.contact.birthday?.month}`
: event.contact.birthday?.month;
const day =
event.contact.birthday?.day < 10
? `0${event.contact.birthday?.day}`
: event.contact.birthday?.day;
message.context.traits.birthday = `${event.contact.birthday?.year}-${month}-${day}`;
}

if (event.contact?.phone_number) {
message.context.traits.phone = `${event.contact.phone_number?.c}${event.contact.phone_number?.n}`;
}

// Updating timestamp to acceptable timestamp format ["2023-10-10T06:24:19.103820974Z" -> "2023-10-10T06:24:19.000Z"]
if (message.originalTimestamp) {
const date = `${Math.floor(new Date(message.originalTimestamp).getTime() / 1000)}`;
message.originalTimestamp = new Date(date * 1000).toISOString();
}

// setting event Name
message.setEventName(eventMapping[event.activity.field_id]);
ItsSudip marked this conversation as resolved.
Show resolved Hide resolved

if (!message.event) {
message.event = 'custom event triggered';
}

// setting up ortto contact.contact_id to externalId
if (event.contact?.contact_id) {
message.context.externalId = [
{
type: 'orttoPersonId',
id: event.contact.contact_id,
},
];
}

message = settingProperties(event, message);

return message;
}

module.exports = { process };
Loading
Loading