diff --git a/CHANGELOG.md b/CHANGELOG.md index 42b22827da..dbe8ae98fe 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -2,6 +2,13 @@ All notable changes to this project will be documented in this file. See [standard-version](https://github.com/conventional-changelog/standard-version) for commit guidelines. +### [1.69.1](https://github.com/rudderlabs/rudder-transformer/compare/v1.69.0...v1.69.1) (2024-06-25) + + +### Bug Fixes + +* enhancement: introduce user model for one signal ([#3499](https://github.com/rudderlabs/rudder-transformer/issues/3499)) ([1c8e950](https://github.com/rudderlabs/rudder-transformer/commit/1c8e950f3d8789b33bba69a30c9eb21c40ce3d04)) + ## [1.69.0](https://github.com/rudderlabs/rudder-transformer/compare/v1.68.2...v1.69.0) (2024-06-10) diff --git a/package-lock.json b/package-lock.json index 7fa0d6d3d4..c96e589aff 100644 --- a/package-lock.json +++ b/package-lock.json @@ -1,12 +1,12 @@ { "name": "rudder-transformer", - "version": "1.69.0", + "version": "1.69.1", "lockfileVersion": 3, "requires": true, "packages": { "": { "name": "rudder-transformer", - "version": "1.69.0", + "version": "1.69.1", "license": "ISC", "dependencies": { "@amplitude/ua-parser-js": "0.7.24", diff --git a/package.json b/package.json index 2c7c6711e0..89513532e0 100644 --- a/package.json +++ b/package.json @@ -1,6 +1,6 @@ { "name": "rudder-transformer", - "version": "1.69.0", + "version": "1.69.1", "description": "", "homepage": "https://github.com/rudderlabs/rudder-transformer#readme", "bugs": { diff --git a/src/v0/destinations/one_signal/config.js b/src/v0/destinations/one_signal/config.js index 1a58f3f91f..fdaa3ecd7f 100644 --- a/src/v0/destinations/one_signal/config.js +++ b/src/v0/destinations/one_signal/config.js @@ -1,6 +1,7 @@ const { getMappingConfig } = require('../../util'); const BASE_URL = 'https://onesignal.com/api/v1'; +const BASE_URL_V2 = 'https://api.onesignal.com/apps/{{app_id}}/users'; const ENDPOINTS = { IDENTIFY: { @@ -16,13 +17,33 @@ const ENDPOINTS = { const ConfigCategory = { IDENTIFY: { name: 'OneSignalIdentifyConfig', endpoint: '/players' }, + IDENTIFY_V2: { name: 'OneSignalIdentifyConfigV2' }, + SUBSCRIPTION: { name: 'OneSignalSubscriptionConfig' }, }; const mappingConfig = getMappingConfig(ConfigCategory, __dirname); +// Used for User Model (V2) +const deviceTypesV2Enums = [ + 'iOSPush', + 'email', + 'sms', + 'AndroidPush', + 'HuaweiPush', + 'FireOSPush', + 'WindowsPush', + 'macOSPush', + 'ChromeExtensionPush', + 'ChromePush', + 'SafariLegacyPush', + 'FirefoxPush', + 'SafariPush', +]; module.exports = { BASE_URL, + BASE_URL_V2, ENDPOINTS, ConfigCategory, mappingConfig, + deviceTypesV2Enums, }; diff --git a/src/v0/destinations/one_signal/data/OneSignalIdentifyConfigV2.json b/src/v0/destinations/one_signal/data/OneSignalIdentifyConfigV2.json new file mode 100644 index 0000000000..61ab6e0109 --- /dev/null +++ b/src/v0/destinations/one_signal/data/OneSignalIdentifyConfigV2.json @@ -0,0 +1,54 @@ +[ + { "sourceKeys": "context.locale", "destKey": "properties.laguage", "required": false }, + { "sourceKeys": "context.ip", "destKey": "properties.ip", "required": false }, + { "sourceKeys": "context.timezone", "destKey": "properties.timezone_id", "required": false }, + { "sourceKeys": "context.location.latitude", "destKey": "properties.lat", "required": false }, + { "sourceKeys": "context.location.longitude", "destKey": "properties.long", "required": false }, + { + "sourceKeys": "createdAt", + "destKey": "properties.created_at", + "sourceFromGenericMap": true, + "metadata": { + "type": "secondTimestamp" + }, + "required": false + }, + { + "sourceKeys": "createdAt", + "destKey": "properties.last_active", + "sourceFromGenericMap": true, + "metadata": { + "type": "secondTimestamp" + }, + "required": false + }, + { + "sourceKeys": [ + "traits.country", + "context.traits.country", + "traits.address.country", + "context.traits.address.country" + ], + "destKey": "properties.country", + "required": false + }, + { + "sourceKeys": [ + "traits.firstActive", + "context.traits.firstActive", + "traits.first_active", + "context.traits.first_active" + ], + "metadata": { + "type": "secondTimestamp" + }, + "destKey": "properties.first_active", + "required": false + }, + { + "sourceKeys": "userIdOnly", + "destKey": "identity.external_id", + "sourceFromGenericMap": true, + "required": false + } +] diff --git a/src/v0/destinations/one_signal/data/OneSignalSubscriptionConfig.json b/src/v0/destinations/one_signal/data/OneSignalSubscriptionConfig.json new file mode 100644 index 0000000000..4a7a877daa --- /dev/null +++ b/src/v0/destinations/one_signal/data/OneSignalSubscriptionConfig.json @@ -0,0 +1,8 @@ +[ + { "sourceKeys": "enabled", "destKey": "enabled", "required": false }, + { "sourceKeys": "notification_types", "destKey": "notification_types", "required": false }, + { "sourceKeys": "session_time", "destKey": "session_time", "required": false }, + { "sourceKeys": "session_count", "destKey": "session_count", "required": false }, + { "sourceKeys": "app_version", "destKey": "app_version", "required": false }, + { "sourceKeys": "test_type", "destKey": "test_type", "required": false } +] diff --git a/src/v0/destinations/one_signal/transform.js b/src/v0/destinations/one_signal/transform.js index b025660fa4..aac48e3b4e 100644 --- a/src/v0/destinations/one_signal/transform.js +++ b/src/v0/destinations/one_signal/transform.js @@ -4,6 +4,7 @@ const { TransformationError, InstrumentationError, } = require('@rudderstack/integrations-lib'); +const { process: processV2 } = require('./transformV2'); const { EventType } = require('../../../constants'); const { ConfigCategory, mappingConfig, BASE_URL, ENDPOINTS } = require('./config'); const { @@ -186,10 +187,16 @@ const groupResponseBuilder = (message, { Config }) => { }; const processEvent = (message, destination) => { + const { Config } = destination; + const { version, appId } = Config; + if (version === 'V2') { + // This version is used to direct the request to user centric model + return processV2(message, destination); + } if (!message.type) { throw new InstrumentationError('Event type is required'); } - if (!destination.Config.appId) { + if (!appId) { throw new ConfigurationError('appId is a required field'); } const messageType = message.type.toLowerCase(); diff --git a/src/v0/destinations/one_signal/transformV2.js b/src/v0/destinations/one_signal/transformV2.js new file mode 100644 index 0000000000..3d084e3c8a --- /dev/null +++ b/src/v0/destinations/one_signal/transformV2.js @@ -0,0 +1,159 @@ +const get = require('get-value'); +const { + ConfigurationError, + TransformationError, + InstrumentationError, +} = require('@rudderstack/integrations-lib'); +const { EventType } = require('../../../constants'); +const { ConfigCategory, mappingConfig, BASE_URL_V2 } = require('./config'); +const { + defaultRequestConfig, + getFieldValueFromMessage, + constructPayload, + defaultPostRequestConfig, + removeUndefinedAndNullValues, +} = require('../../util'); +const { + populateTags, + getProductPurchasesDetails, + getSubscriptions, + getOneSignalAliases, +} = require('./util'); +const { JSON_MIME_TYPE } = require('../../util/constant'); + +const responseBuilder = (payload, Config) => { + const { appId } = Config; + if (payload) { + const response = defaultRequestConfig(); + response.endpoint = `${BASE_URL_V2.replace('{{app_id}}', appId)}`; + response.headers = { + Accept: JSON_MIME_TYPE, + 'Content-Type': JSON_MIME_TYPE, + }; + response.method = defaultPostRequestConfig.requestMethod; + response.body.JSON = removeUndefinedAndNullValues(payload); + return response; + } + throw new TransformationError('Payload could not be populated due to wrong input'); +}; + +/** + * This function is used for creating response for identify call, to create a new user or update an existing user. + * a responseArray for creating/updating user is being prepared. + * If the value of emailDeviceType/smsDeviceType(toggle in dashboard) is true, separate responses will also be created + * for new subscriptions to be added to user with email/sms as token. + * @param {*} message + * @param {*} param1 + * @returns + */ +const identifyResponseBuilder = (message, { Config }) => { + // Populating the tags + const tags = populateTags(message); + + const payload = constructPayload(message, mappingConfig[ConfigCategory.IDENTIFY_V2.name]); + if (!payload?.identity?.external_id) { + const alias = getOneSignalAliases(message); + if (Object.keys(alias).length === 0) { + throw new InstrumentationError('userId or any other alias is required for identify'); + } + payload.identity = alias; + } + // Following check is to intialise properties object in case we don't get properties object from construct payload + if (!payload.properties) { + payload.properties = {}; + } + payload.subscriptions = getSubscriptions(message, Config); + payload.properties.tags = tags; + return responseBuilder(removeUndefinedAndNullValues(payload), Config); +}; + +/** + * This function is used to build the response for track call and Group call. + * It is used to edit the OneSignal tags using external_id. + * It edits tags[event] as true for track call + * @param {*} message + * @param {*} param1 + * @returns + */ +const trackOrGroupResponseBuilder = (message, { Config }, msgtype) => { + const { eventAsTags, allowedProperties } = Config; + const event = get(message, 'event'); + const groupId = getFieldValueFromMessage(message, 'groupId'); + // validation and adding tags for track and group call respectively + const tags = {}; + const payload = { properties: {} }; + if (msgtype === EventType.TRACK) { + if (!event) { + throw new InstrumentationError('Event is not present in the input payloads'); + } + /* Populating event as true in tags. + eg. tags: { + "event_name": true + } + */ + tags[event] = true; + payload.properties.purchases = getProductPurchasesDetails(message); + } + if (msgtype === EventType.GROUP) { + if (!groupId) { + throw new InstrumentationError('groupId is required for group events'); + } + tags.groupId = groupId; + } + + const externalUserId = getFieldValueFromMessage(message, 'userIdOnly'); + if (!externalUserId) { + const alias = getOneSignalAliases(message); + if (Object.keys(alias).length === 0) { + throw new InstrumentationError('userId or any other alias is required for track and group'); + } + payload.identity = alias; + } else { + payload.identity = { + external_id: externalUserId, + }; + } + + // Populating tags using allowed properties(from dashboard) + const properties = get(message, 'properties'); + if (properties && allowedProperties && Array.isArray(allowedProperties)) { + allowedProperties.forEach((item) => { + if (typeof properties[item.propertyName] === 'string') { + const tagName = + event && eventAsTags ? `${event}_${[item.propertyName]}` : item.propertyName; + tags[tagName] = properties[item.propertyName]; + } + }); + } + payload.properties.tags = tags; + return responseBuilder(removeUndefinedAndNullValues(payload), Config); +}; + +const processEvent = (message, destination) => { + if (!message.type) { + throw new InstrumentationError('Event type is required'); + } + if (!destination.Config.appId) { + throw new ConfigurationError('appId is a required field'); + } + const messageType = message.type.toLowerCase(); + let response; + switch (messageType) { + case EventType.IDENTIFY: + response = identifyResponseBuilder(message, destination); + break; + case EventType.TRACK: + response = trackOrGroupResponseBuilder(message, destination, EventType.TRACK); + break; + case EventType.GROUP: + response = trackOrGroupResponseBuilder(message, destination, EventType.GROUP); + break; + default: + throw new InstrumentationError(`Message type ${messageType} is not supported`); + } + return response; +}; + +const process = (message, destination) => processEvent(message, destination); + +module.exports = { process }; diff --git a/src/v0/destinations/one_signal/util.js b/src/v0/destinations/one_signal/util.js index 2de57de1b4..69cbd5440c 100644 --- a/src/v0/destinations/one_signal/util.js +++ b/src/v0/destinations/one_signal/util.js @@ -1,6 +1,13 @@ const { InstrumentationError } = require('@rudderstack/integrations-lib'); -const { getIntegrationsObj, getFieldValueFromMessage, getBrowserInfo } = require('../../util'); - +const { + getIntegrationsObj, + getFieldValueFromMessage, + getBrowserInfo, + constructPayload, + removeUndefinedAndNullValues, +} = require('../../util'); +const { ConfigCategory, mappingConfig, deviceTypesV2Enums } = require('./config'); +const { isDefinedAndNotNullAndNotEmpty } = require('../../util'); // For mapping device_type value const deviceTypeMapping = { android: 1, @@ -45,7 +52,7 @@ const populateTags = (message) => { const populateDeviceType = (message, payload) => { const integrationsObj = getIntegrationsObj(message, 'one_signal'); const devicePayload = payload; - if (integrationsObj && integrationsObj.deviceType && integrationsObj.identifier) { + if (integrationsObj?.deviceType && integrationsObj?.identifier) { devicePayload.device_type = parseInt(integrationsObj.deviceType, 10); devicePayload.identifier = integrationsObj.identifier; if (!validateDeviceType(devicePayload.device_type)) { @@ -72,4 +79,147 @@ const populateDeviceType = (message, payload) => { } }; -module.exports = { populateDeviceType, populateTags }; +/** + * This function is used to populate device type required for creating a subscription + * it checks from integrations object and fall back to message.channel and fif nothing is given it return a n empty object + * @param {*} message + * @param {*} payload + * returns Object + */ +const getDeviceDetails = (message) => { + const integrationsObj = getIntegrationsObj(message, 'one_signal'); + const devicePayload = {}; + if (integrationsObj?.deviceType && integrationsObj?.identifier) { + devicePayload.type = integrationsObj.deviceType; + devicePayload.token = integrationsObj.token || integrationsObj.identifier; + } + // Mapping device type when it is not present in the integrationsObject + if (!devicePayload.type) { + // if channel is mobile, checking for type from `context.device.type` + if (message.channel === 'mobile') { + devicePayload.type = message.context?.device?.type; + devicePayload.token = message.context?.device?.token + ? message.context.device.token + : message.context?.device?.id; + } + // Parsing the UA to get the browser info to map the device_type + if (message.channel === 'web' && message.context?.userAgent) { + const browser = getBrowserInfo(message.context.userAgent); + devicePayload.type = `${browser.name}Push`; // For chrome it would be like ChromePush + devicePayload.token = message.anonymousId; + } + } + if (!deviceTypesV2Enums.includes(devicePayload.type)) { + return {}; // No device related information available + } + return devicePayload; +}; +/** + * This function maps and returns the product purchases details built from input message.properties.products + * @param {*} message + * @returns + */ +const getProductPurchasesDetails = (message) => { + const { properties } = message; + const purchases = properties?.products; + if (purchases && Array.isArray(purchases)) { + return purchases.map((product) => ({ + sku: product.sku, + iso: product.iso, + count: product.quantity, + amount: product.amount, + })); + } + const purchaseObject = removeUndefinedAndNullValues({ + sku: properties?.sku, + iso: properties?.iso, + count: properties?.quantity, + amount: properties?.amount, + }); + return Object.keys(purchaseObject).length > 0 ? [purchaseObject] : []; +}; + +/** + * This function generates the subscriptions Payload for the given deviceType and token + * https://documentation.onesignal.com/reference/create-user#:~:text=string-,subscriptions,-array%20of%20objects + * @param {*} message + * @param {*} deviceType + * @param {*} token + * @returns + */ +const constructSubscription = (message, subscriptionType, token, subscriptionField) => { + const deviceModel = message.context?.device?.model; + const deviceOs = message.context?.os?.version; + let deviceSubscriptionPayload = { + type: subscriptionType, + token, + device_model: deviceModel, + device_os: deviceOs, + }; + // Following mapping is used to do paticular and specific property mapping for subscription + const traits = message.context?.traits || message.traits; + if (traits?.subscriptions?.[subscriptionField]) { + deviceSubscriptionPayload = { + ...deviceSubscriptionPayload, + ...constructPayload( + traits.subscriptions[subscriptionField], + mappingConfig[ConfigCategory.SUBSCRIPTION.name], + ), + }; + } + return deviceSubscriptionPayload; +}; + +/** + * This function constructs subscriptions list from message and returns subscriptions list + * @param {*} message + * @param {*} Config + * @returns + */ +const getSubscriptions = (message, Config) => { + const { emailDeviceType, smsDeviceType } = Config; + // Creating response for creation of new device or updation of an existing device + const subscriptions = []; + const deviceTypeSubscription = getDeviceDetails(message); + if (deviceTypeSubscription.token) { + subscriptions.push( + constructSubscription(message, deviceTypeSubscription.type, deviceTypeSubscription.token), + ); + } + + // Creating a device with email as an identifier + if (emailDeviceType) { + const token = getFieldValueFromMessage(message, 'email'); + if (isDefinedAndNotNullAndNotEmpty(token)) { + subscriptions.push(constructSubscription(message, 'Email', token, 'email')); + } + } + // Creating a device with phone as an identifier + if (smsDeviceType) { + const token = getFieldValueFromMessage(message, 'phone'); + if (isDefinedAndNotNullAndNotEmpty(token)) { + subscriptions.push(constructSubscription(message, 'SMS', token, 'phone')); + } + } + return subscriptions.length > 0 ? subscriptions : undefined; +}; + +/** + * This function fetched all the aliases to be passed to one signal from integrations object + * @param {*} message + * @returns object + */ +const getOneSignalAliases = (message) => { + const integrationsObj = getIntegrationsObj(message, 'one_signal'); + if (integrationsObj?.aliases) { + return integrationsObj.aliases; + } + return {}; +}; +module.exports = { + populateDeviceType, + populateTags, + getProductPurchasesDetails, + getSubscriptions, + getOneSignalAliases, +}; diff --git a/src/v0/destinations/one_signal/utils.test.js b/src/v0/destinations/one_signal/utils.test.js new file mode 100644 index 0000000000..afcf746ab6 --- /dev/null +++ b/src/v0/destinations/one_signal/utils.test.js @@ -0,0 +1,35 @@ +const { getOneSignalAliases } = require('./util'); + +describe('getOneSignalAliases', () => { + // returns aliases when integrationsObj contains aliases + it('should return aliases when integrationsObj contains aliases', () => { + const message = { + someKey: 'someValue', + integrations: { one_signal: { aliases: { alias1: 'value1' } } }, + }; + const result = getOneSignalAliases(message); + expect(result).toEqual({ alias1: 'value1' }); + }); + + // handles null or undefined message parameter gracefully + it('should handle null or undefined message parameter gracefully', () => { + let result = getOneSignalAliases(null); + expect(result).toEqual({}); + result = getOneSignalAliases(undefined); + expect(result).toEqual({}); + }); + + // returns an empty object when integrationsObj does not contain aliases + it('should return an empty object when integrationsObj does not contain aliases', () => { + const message = { someKey: 'someValue', integrations: { one_signal: {} } }; + const result = getOneSignalAliases(message); + expect(result).toEqual({}); + }); + + // handles message parameter with unexpected structure + it('should handle message parameter with unexpected structure', () => { + const message = null; + const result = getOneSignalAliases(message); + expect(result).toEqual({}); + }); +}); diff --git a/test/integrations/destinations/one_signal/processor/commonConfig.ts b/test/integrations/destinations/one_signal/processor/commonConfig.ts new file mode 100644 index 0000000000..cdef3dbfb4 --- /dev/null +++ b/test/integrations/destinations/one_signal/processor/commonConfig.ts @@ -0,0 +1,45 @@ +export const destination = { + Config: { + appId: 'random-818c-4a28-b98e-6cd8a994eb22', + emailDeviceType: true, + smsDeviceType: true, + eventAsTags: false, + allowedProperties: [ + { propertyName: 'brand' }, + { propertyName: 'firstName' }, + { propertyName: 'price' }, + ], + version: 'V2', + }, +}; + +export const endpoint = 'https://api.onesignal.com/apps/random-818c-4a28-b98e-6cd8a994eb22/users'; + +export const headers = { Accept: 'application/json', 'Content-Type': 'application/json' }; + +export const commonTraits = { + brand: 'John Players', + price: '15000', + firstName: 'Test', +}; +export const commonTags = { + brand: 'John Players', + price: '15000', + firstName: 'Test', + anonymousId: '97c46c81-3140-456d-b2a9-690d70aaca35', +}; + +export const commonProperties = { + products: [ + { + sku: 3, + iso: 'iso', + quantity: 2, + amount: 100, + }, + ], + brand: 'John Players', + price: '15000', + firstName: 'Test', + customKey: 'customVal', +}; diff --git a/test/integrations/destinations/one_signal/processor/data.ts b/test/integrations/destinations/one_signal/processor/data.ts index 4171157aef..13aa6043f1 100644 --- a/test/integrations/destinations/one_signal/processor/data.ts +++ b/test/integrations/destinations/one_signal/processor/data.ts @@ -1,3 +1,4 @@ +import { data_v2 } from './data_v2'; export const data = [ { name: 'one_signal', @@ -1542,4 +1543,5 @@ export const data = [ }, }, }, + ...data_v2, ]; diff --git a/test/integrations/destinations/one_signal/processor/data_v2.ts b/test/integrations/destinations/one_signal/processor/data_v2.ts new file mode 100644 index 0000000000..325dca336d --- /dev/null +++ b/test/integrations/destinations/one_signal/processor/data_v2.ts @@ -0,0 +1,5 @@ +import { identifyTests } from './identify'; +import { trackTests } from './track'; +import { validations } from './validations'; +import { groupTests } from './group'; +export const data_v2 = [...identifyTests, ...trackTests, ...validations, ...groupTests]; diff --git a/test/integrations/destinations/one_signal/processor/group.ts b/test/integrations/destinations/one_signal/processor/group.ts new file mode 100644 index 0000000000..e16d1e4fe6 --- /dev/null +++ b/test/integrations/destinations/one_signal/processor/group.ts @@ -0,0 +1,121 @@ +import { destination, endpoint, headers } from './commonConfig'; +export const groupTests = [ + { + name: 'one_signal', + id: 'One Signal V2-test-group-success-1', + description: + 'Group call for adding a tag groupId with value as group id with no userId available', + module: 'destination', + successCriteria: 'Request gets passed with 200 Status Code with userId mapped to external_id', + feature: 'processor', + scenario: 'Framework+Buisness', + version: 'v0', + input: { + request: { + body: [ + { + destination, + message: { + integrations: { + one_signal: { + aliases: { custom_alias: 'custom_alias_identifier' }, + }, + }, + type: 'group', + channel: 'server', + groupId: 'group@27', + rudderId: '8f8fa6b5-8e24-489c-8e22-61f23f2e364f', + messageId: '2116ef8c-efc3-4ca4-851b-02ee60dad6ff', + originalTimestamp: '2021-01-03T17:02:53.193Z', + }, + }, + ], + method: 'POST', + }, + pathSuffix: '', + }, + output: { + response: { + status: 200, + body: [ + { + output: { + version: '1', + type: 'REST', + method: 'POST', + endpoint, + headers, + params: {}, + body: { + FORM: {}, + JSON: { + properties: { + tags: { + groupId: 'group@27', + }, + }, + identity: { + custom_alias: 'custom_alias_identifier', + }, + }, + JSON_ARRAY: {}, + XML: {}, + }, + files: {}, + userId: '', + }, + statusCode: 200, + }, + ], + }, + }, + }, + { + name: 'one_signal', + id: 'One Signal V2-test-group-failure-1', + description: 'V2-> No Group Id Passes', + module: 'destination', + successCriteria: + 'Request gets passed with 200 Status Code and failure happened due no group id available', + feature: 'processor', + scenario: 'Framework', + version: 'v0', + input: { + request: { + body: [ + { + destination, + message: { + type: 'group', + rudderId: '8f8fa6b5-8e24-489c-8e22-61f23f2e364f', + messageId: '2116ef8c-efc3-4ca4-851b-02ee60dad6ff', + anonymousId: '97c46c81-3140-456d-b2a9-690d70aaca35', + originalTimestamp: '2021-01-03T17:02:53.193Z', + }, + }, + ], + method: 'POST', + }, + pathSuffix: '', + }, + output: { + response: { + status: 200, + body: [ + { + error: 'groupId is required for group events', + statTags: { + destType: 'ONE_SIGNAL', + errorCategory: 'dataValidation', + errorType: 'instrumentation', + feature: 'processor', + implementation: 'native', + module: 'destination', + }, + statusCode: 400, + }, + ], + }, + }, + }, +]; diff --git a/test/integrations/destinations/one_signal/processor/identify.ts b/test/integrations/destinations/one_signal/processor/identify.ts new file mode 100644 index 0000000000..175c18a8fa --- /dev/null +++ b/test/integrations/destinations/one_signal/processor/identify.ts @@ -0,0 +1,341 @@ +import { commonTags, commonTraits, destination, endpoint, headers } from './commonConfig'; + +export const identifyTests = [ + { + name: 'one_signal', + id: 'One Signal V2-test-identify-success-1', + description: + 'V2-> Identify call for creating new user with userId only available and no subscriptions available', + module: 'destination', + successCriteria: 'Request gets passed with 200 Status Code with userId mapped to external_id', + feature: 'processor', + scenario: 'Framework+Buisness', + version: 'v0', + input: { + request: { + body: [ + { + destination, + message: { + type: 'identify', + userId: 'user@27', + channel: 'server', + context: { + app: { + version: '1.1.11', + }, + traits: commonTraits, + locale: 'en-US', + screen: { density: 2 }, + userAgent: + 'Mozilla/5.0 (Macintosh; Intel Mac OS X 10.16; rv:84.0) Gecko/20100101 Firefox/84.0', + }, + rudderId: '8f8fa6b5-8e24-489c-8e22-61f23f2e364f', + messageId: '2116ef8c-efc3-4ca4-851b-02ee60dad6ff', + anonymousId: '97c46c81-3140-456d-b2a9-690d70aaca35', + originalTimestamp: '2021-01-03T17:02:53.193Z', + }, + }, + ], + method: 'POST', + }, + pathSuffix: '', + }, + output: { + response: { + status: 200, + body: [ + { + output: { + version: '1', + type: 'REST', + method: 'POST', + endpoint, + headers, + params: {}, + body: { + FORM: {}, + JSON: { + properties: { + tags: commonTags, + laguage: 'en-US', + created_at: 1609693373, + last_active: 1609693373, + }, + identity: { + external_id: 'user@27', + }, + }, + JSON_ARRAY: {}, + XML: {}, + }, + files: {}, + userId: '', + }, + statusCode: 200, + }, + ], + }, + }, + }, + { + name: 'one_signal', + id: 'One Signal V2-test-identify-success-2', + description: + 'V2-> Identify call for creating new user with userId and one device subscription available', + module: 'destination', + successCriteria: + 'Request gets passed with 200 Status Code with userId mapped to external_id and one subscription for device where identifier is mapped from anonymousId', + feature: 'processor', + scenario: 'Framework+Buisness', + version: 'v0', + input: { + request: { + body: [ + { + destination, + message: { + type: 'identify', + userId: 'user@27', + channel: 'web', + context: { + app: { + name: 'RudderLabs JavaScript SDK', + build: '1.0.0', + version: '1.1.11', + namespace: 'com.rudderlabs.javascript', + }, + traits: commonTraits, + locale: 'en-US', + campaign: {}, + userAgent: + 'Mozilla/5.0 (Macintosh; Intel Mac OS X 10.16; rv:84.0) Gecko/20100101 Firefox/84.0', + }, + rudderId: '8f8fa6b5-8e24-489c-8e22-61f23f2e364f', + messageId: '2116ef8c-efc3-4ca4-851b-02ee60dad6ff', + anonymousId: '97c46c81-3140-456d-b2a9-690d70aaca35', + originalTimestamp: '2021-01-03T17:02:53.193Z', + }, + }, + ], + method: 'POST', + }, + pathSuffix: '', + }, + output: { + response: { + status: 200, + body: [ + { + output: { + version: '1', + type: 'REST', + method: 'POST', + endpoint, + headers, + params: {}, + body: { + FORM: {}, + JSON: { + properties: { + tags: commonTags, + laguage: 'en-US', + created_at: 1609693373, + last_active: 1609693373, + }, + subscriptions: [ + { token: '97c46c81-3140-456d-b2a9-690d70aaca35', type: 'FirefoxPush' }, + ], + identity: { + external_id: 'user@27', + }, + }, + JSON_ARRAY: {}, + XML: {}, + }, + files: {}, + userId: '', + }, + statusCode: 200, + }, + ], + }, + }, + }, + { + name: 'one_signal', + id: 'One Signal V2-test-identify-success-3', + description: + 'V2-> Identify call for creating new user with userId and three device subscription available', + module: 'destination', + successCriteria: + 'Request gets passed with 200 Status Code with userId mapped to external_id and three subscription for device where one is mapped from anonymousId, one from email and one from phone', + feature: 'processor', + scenario: 'Framework+Buisness', + version: 'v0', + input: { + request: { + body: [ + { + destination, + message: { + type: 'identify', + integrations: { + one_signal: { + aliases: { custom_alias: 'custom_alias_identifier' }, + }, + }, + channel: 'web', + context: { + app: { + name: 'RudderLabs JavaScript SDK', + build: '1.0.0', + version: '1.1.11', + namespace: 'com.rudderlabs.javascript', + }, + device: { + model: 'dummy model', + }, + os: { version: '1.0.0' }, + traits: { + ...commonTraits, + email: 'example@abc.com', + phone: '12345678', + subscriptions: { + email: { + enabled: true, + notification_types: 'SMS', + session_time: 123456, + session_count: 22, + app_version: '1.0.0', + test_type: 'dev', + }, + }, + }, + locale: 'en-US', + campaign: {}, + userAgent: + 'Mozilla/5.0 (Macintosh; Intel Mac OS X 10.16; rv:84.0) Gecko/20100101 Firefox/84.0', + }, + rudderId: '8f8fa6b5-8e24-489c-8e22-61f23f2e364f', + messageId: '2116ef8c-efc3-4ca4-851b-02ee60dad6ff', + anonymousId: '97c46c81-3140-456d-b2a9-690d70aaca35', + originalTimestamp: '2021-01-03T17:02:53.193Z', + }, + }, + ], + method: 'POST', + }, + pathSuffix: '', + }, + output: { + response: { + status: 200, + body: [ + { + output: { + version: '1', + type: 'REST', + method: 'POST', + endpoint, + headers, + params: {}, + body: { + FORM: {}, + JSON: { + properties: { + tags: { ...commonTags, email: 'example@abc.com', phone: '12345678' }, + laguage: 'en-US', + created_at: 1609693373, + last_active: 1609693373, + }, + subscriptions: [ + { + device_model: 'dummy model', + device_os: '1.0.0', + token: '97c46c81-3140-456d-b2a9-690d70aaca35', + type: 'FirefoxPush', + }, + { + app_version: '1.0.0', + device_model: 'dummy model', + device_os: '1.0.0', + enabled: true, + notification_types: 'SMS', + session_count: 22, + session_time: 123456, + test_type: 'dev', + token: 'example@abc.com', + type: 'Email', + }, + { + device_model: 'dummy model', + device_os: '1.0.0', + token: '12345678', + type: 'SMS', + }, + ], + identity: { custom_alias: 'custom_alias_identifier' }, + }, + JSON_ARRAY: {}, + XML: {}, + }, + files: {}, + userId: '', + }, + statusCode: 200, + }, + ], + }, + }, + }, + { + name: 'one_signal', + id: 'One Signal V2-test-identify-failure-1', + description: 'V2-> Identify call without any aliases', + module: 'destination', + successCriteria: + 'Request gets passed with 200 Status Code and failure happened due no aliases present', + feature: 'processor', + scenario: 'Framework', + version: 'v0', + input: { + request: { + body: [ + { + destination, + message: { + type: 'identify', + channel: 'server', + rudderId: '8f8fa6b5-8e24-489c-8e22-61f23f2e364f', + messageId: '2116ef8c-efc3-4ca4-851b-02ee60dad6ff', + anonymousId: '97c46c81-3140-456d-b2a9-690d70aaca35', + originalTimestamp: '2021-01-03T17:02:53.193Z', + }, + }, + ], + method: 'POST', + }, + pathSuffix: '', + }, + output: { + response: { + status: 200, + body: [ + { + error: 'userId or any other alias is required for identify', + statTags: { + destType: 'ONE_SIGNAL', + errorCategory: 'dataValidation', + errorType: 'instrumentation', + feature: 'processor', + implementation: 'native', + module: 'destination', + }, + statusCode: 400, + }, + ], + }, + }, + }, +]; diff --git a/test/integrations/destinations/one_signal/processor/track.ts b/test/integrations/destinations/one_signal/processor/track.ts new file mode 100644 index 0000000000..41735fc7a2 --- /dev/null +++ b/test/integrations/destinations/one_signal/processor/track.ts @@ -0,0 +1,249 @@ +import { commonProperties, destination, endpoint, headers } from './commonConfig'; + +const commonTrackTags = { + brand: 'John Players', + price: '15000', + firstName: 'Test', +}; + +const purchases = [ + { + sku: 3, + iso: 'iso', + count: 2, + amount: 100, + }, +]; + +export const trackTests = [ + { + name: 'one_signal', + id: 'One Signal V2-test-track-success-1', + description: + 'V2-> Track call for updating user tags with userId available and products details available', + module: 'destination', + successCriteria: + 'Request gets passed with 200 Status Code with userId mapped to external_id and properties mapped to tags', + feature: 'processor', + scenario: 'Framework+Buisness', + version: 'v0', + input: { + request: { + body: [ + { + destination, + message: { + type: 'track', + event: 'Order Completed', + userId: 'user@27', + channel: 'server', + properties: commonProperties, + rudderId: '8f8fa6b5-8e24-489c-8e22-61f23f2e364f', + messageId: '2116ef8c-efc3-4ca4-851b-02ee60dad6ff', + anonymousId: '97c46c81-3140-456d-b2a9-690d70aaca35', + originalTimestamp: '2021-01-03T17:02:53.193Z', + }, + }, + ], + method: 'POST', + }, + pathSuffix: '', + }, + output: { + response: { + status: 200, + body: [ + { + output: { + version: '1', + type: 'REST', + method: 'POST', + endpoint, + headers, + params: {}, + body: { + FORM: {}, + JSON: { + properties: { purchases, tags: { ...commonTrackTags, 'Order Completed': true } }, + + identity: { + external_id: 'user@27', + }, + }, + JSON_ARRAY: {}, + XML: {}, + }, + files: {}, + userId: '', + }, + statusCode: 200, + }, + ], + }, + }, + }, + { + name: 'one_signal', + id: 'One Signal V2-test-track-success-2', + description: 'V2-> Track call for products details available in properties directly', + module: 'destination', + successCriteria: + 'Request gets passed with 200 Status Code with userId mapped to external_id and purchases mapped from proeprties mapped to tags', + feature: 'processor', + scenario: 'Framework+Buisness', + version: 'v0', + input: { + request: { + body: [ + { + destination, + message: { + type: 'track', + event: 'Order Completed', + userId: 'user@27', + channel: 'server', + properties: {}, + rudderId: '8f8fa6b5-8e24-489c-8e22-61f23f2e364f', + messageId: '2116ef8c-efc3-4ca4-851b-02ee60dad6ff', + anonymousId: '97c46c81-3140-456d-b2a9-690d70aaca35', + originalTimestamp: '2021-01-03T17:02:53.193Z', + }, + }, + ], + method: 'POST', + }, + pathSuffix: '', + }, + output: { + response: { + status: 200, + body: [ + { + output: { + version: '1', + type: 'REST', + method: 'POST', + endpoint, + headers, + params: {}, + body: { + FORM: {}, + JSON: { + properties: { purchases: [], tags: { 'Order Completed': true } }, + identity: { + external_id: 'user@27', + }, + }, + JSON_ARRAY: {}, + XML: {}, + }, + files: {}, + userId: '', + }, + statusCode: 200, + }, + ], + }, + }, + }, + { + name: 'one_signal', + id: 'One Signal V2-test-track-failure-1', + description: 'V2-> Track call without event name', + module: 'destination', + successCriteria: + 'Request gets passed with 200 Status Code and failure happened due instrumentation error', + feature: 'processor', + scenario: 'Framework+Buisness', + version: 'v0', + input: { + request: { + body: [ + { + destination, + message: { + type: 'track', + userId: 'user@27', + channel: 'server', + rudderId: '8f8fa6b5-8e24-489c-8e22-61f23f2e364f', + messageId: '2116ef8c-efc3-4ca4-851b-02ee60dad6ff', + anonymousId: '97c46c81-3140-456d-b2a9-690d70aaca35', + originalTimestamp: '2021-01-03T17:02:53.193Z', + }, + }, + ], + method: 'POST', + }, + pathSuffix: '', + }, + output: { + response: { + status: 200, + body: [ + { + error: 'Event is not present in the input payloads', + statTags: { + destType: 'ONE_SIGNAL', + errorCategory: 'dataValidation', + errorType: 'instrumentation', + feature: 'processor', + implementation: 'native', + module: 'destination', + }, + statusCode: 400, + }, + ], + }, + }, + }, + { + name: 'one_signal', + id: 'One Signal V2-test-track-failure-2', + description: 'V2-> Track call without any aliases', + module: 'destination', + successCriteria: + 'Request gets passed with 200 Status Code and failure happened due no aliases present', + feature: 'processor', + scenario: 'Framework', + version: 'v0', + input: { + request: { + body: [ + { + destination, + message: { + type: 'track', + event: 'dummy event', + channel: 'server', + rudderId: '8f8fa6b5-8e24-489c-8e22-61f23f2e364f', + messageId: '2116ef8c-efc3-4ca4-851b-02ee60dad6ff', + anonymousId: '97c46c81-3140-456d-b2a9-690d70aaca35', + originalTimestamp: '2021-01-03T17:02:53.193Z', + }, + }, + ], + method: 'POST', + }, + pathSuffix: '', + }, + output: { + response: { + status: 200, + body: [ + { + error: 'userId or any other alias is required for track and group', + statTags: { + destType: 'ONE_SIGNAL', + errorCategory: 'dataValidation', + errorType: 'instrumentation', + feature: 'processor', + implementation: 'native', + module: 'destination', + }, + statusCode: 400, + }, + ], + }, + }, + }, +]; diff --git a/test/integrations/destinations/one_signal/processor/validations.ts b/test/integrations/destinations/one_signal/processor/validations.ts new file mode 100644 index 0000000000..7cfa158eee --- /dev/null +++ b/test/integrations/destinations/one_signal/processor/validations.ts @@ -0,0 +1,149 @@ +import { destination } from './commonConfig'; +export const validations = [ + { + name: 'one_signal', + id: 'One Signal V2-test-validation-failure-1', + description: 'V2-> No Message type passed', + module: 'destination', + successCriteria: + 'Request gets passed with 200 Status Code and failure happened due no message type present', + feature: 'processor', + scenario: 'Framework', + version: 'v0', + input: { + request: { + body: [ + { + destination, + message: { + userId: 'user@27', + rudderId: '8f8fa6b5-8e24-489c-8e22-61f23f2e364f', + messageId: '2116ef8c-efc3-4ca4-851b-02ee60dad6ff', + anonymousId: '97c46c81-3140-456d-b2a9-690d70aaca35', + originalTimestamp: '2021-01-03T17:02:53.193Z', + }, + }, + ], + method: 'POST', + }, + pathSuffix: '', + }, + output: { + response: { + status: 200, + body: [ + { + error: 'Event type is required', + statTags: { + destType: 'ONE_SIGNAL', + errorCategory: 'dataValidation', + errorType: 'instrumentation', + feature: 'processor', + implementation: 'native', + module: 'destination', + }, + statusCode: 400, + }, + ], + }, + }, + }, + { + name: 'one_signal', + id: 'One Signal V2-test-validation-failure-2', + description: 'V2-> invalid Message type passed', + module: 'destination', + successCriteria: + 'Request gets passed with 200 Status Code and failure happened due invalid message type present', + feature: 'processor', + scenario: 'Framework', + version: 'v0', + input: { + request: { + body: [ + { + destination, + message: { + type: 'invalid', + userId: 'user@27', + rudderId: '8f8fa6b5-8e24-489c-8e22-61f23f2e364f', + messageId: '2116ef8c-efc3-4ca4-851b-02ee60dad6ff', + anonymousId: '97c46c81-3140-456d-b2a9-690d70aaca35', + originalTimestamp: '2021-01-03T17:02:53.193Z', + }, + }, + ], + method: 'POST', + }, + pathSuffix: '', + }, + output: { + response: { + status: 200, + body: [ + { + error: 'Message type invalid is not supported', + statTags: { + destType: 'ONE_SIGNAL', + errorCategory: 'dataValidation', + errorType: 'instrumentation', + feature: 'processor', + implementation: 'native', + module: 'destination', + }, + statusCode: 400, + }, + ], + }, + }, + }, + { + name: 'one_signal', + id: 'One Signal V2-test-validation-failure-3', + description: 'V2-> No App Id Present in destination Config', + module: 'destination', + successCriteria: + 'Request gets passed with 200 Status Code and failure happened due no Configuration Error', + feature: 'processor', + scenario: 'Framework', + version: 'v0', + input: { + request: { + body: [ + { + destination: { Config: {} }, + message: { + type: 'invalid', + userId: 'user@27', + rudderId: '8f8fa6b5-8e24-489c-8e22-61f23f2e364f', + messageId: '2116ef8c-efc3-4ca4-851b-02ee60dad6ff', + anonymousId: '97c46c81-3140-456d-b2a9-690d70aaca35', + originalTimestamp: '2021-01-03T17:02:53.193Z', + }, + }, + ], + method: 'POST', + }, + pathSuffix: '', + }, + output: { + response: { + status: 200, + body: [ + { + error: 'appId is a required field', + statTags: { + destType: 'ONE_SIGNAL', + errorCategory: 'dataValidation', + errorType: 'configuration', + feature: 'processor', + implementation: 'native', + module: 'destination', + }, + statusCode: 400, + }, + ], + }, + }, + }, +];