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

Add lotame id system #5388

Merged
merged 18 commits into from
Jul 9, 2020
Merged
Show file tree
Hide file tree
Changes from 14 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 integrationExamples/gpt/userId_example.html
Original file line number Diff line number Diff line change
Expand Up @@ -179,7 +179,8 @@
name: 'idl_env',
expires: 30
}
}, {
},
{
name: "sharedId",
params: {
syncTime: 60 // in seconds, default is 24 hours
Expand All @@ -189,6 +190,9 @@
name: "sharedid",
expires: 28
}
},
{
name: 'lotamePanoramaId'
}],
syncDelay: 5000,
auctionDelay: 1000
Expand Down
1 change: 1 addition & 0 deletions modules/.submodules.json
Original file line number Diff line number Diff line change
Expand Up @@ -7,6 +7,7 @@
"parrableIdSystem",
"britepoolIdSystem",
"liveIntentIdSystem",
"lotamePanoramaId",
"criteoIdSystem",
"netIdSystem",
"identityLinkIdSystem",
Expand Down
243 changes: 243 additions & 0 deletions modules/lotamePanoramaIdSystem.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,243 @@
/**
* This module adds LotamePanoramaId to the User ID module
* The {@link module:modules/userId} module is required
* @module modules/lotamePanoramaId
* @requires module:modules/userId
*/
import * as utils from '../src/utils.js';
import { ajax } from '../src/ajax.js';
import { submodule } from '../src/hook.js';
import { getStorageManager } from '../src/storageManager.js';

const KEY_ID = 'panoramaId';
const KEY_EXPIRY = `${KEY_ID}_expiry`;
const KEY_PROFILE = '_cc_id';
const MODULE_NAME = 'lotamePanoramaId';
const NINE_MONTHS_MS = 23328000 * 1000;
const DAYS_TO_CACHE = 7;
const DAY_MS = 60 * 60 * 24 * 1000;

export const storage = getStorageManager(null, MODULE_NAME);

/**
* Set the Lotame First Party Profile ID in the first party namespace
* @param {String} profileId
*/
function setProfileId(profileId) {
if (storage.cookiesAreEnabled()) {
let expirationDate = new Date(utils.timestamp() + NINE_MONTHS_MS).toUTCString();
storage.setCookie(KEY_PROFILE, profileId, expirationDate, 'Lax', undefined, undefined);
}
if (storage.hasLocalStorage()) {
storage.setDataInLocalStorage(KEY_PROFILE, profileId, undefined);
}
}

/**
* Get the Lotame profile id by checking cookies first and then local storage
*/
function getProfileId() {
if (storage.cookiesAreEnabled()) {
return storage.getCookie(KEY_PROFILE, undefined);
}
if (storage.hasLocalStorage()) {
return storage.getDataFromLocalStorage(KEY_PROFILE, undefined);
}
}

/**
* Get a value from browser storage by checking cookies first and then local storage
* @param {String} key
*/
function getFromStorage(key) {
let value = null;
if (storage.cookiesAreEnabled()) {
value = storage.getCookie(key, undefined);
}
if (storage.hasLocalStorage() && value === null) {
const storedValueExp = storage.getDataFromLocalStorage(
`${key}_exp`, undefined
);
if (storedValueExp === '') {
value = storage.getDataFromLocalStorage(key, undefined);
} else if (storedValueExp) {
if ((new Date(storedValueExp)).getTime() - Date.now() > 0) {
value = storage.getDataFromLocalStorage(key, undefined);
}
}
}
return value;
}

/**
* Save a key/value pair to the browser cache (cookies and local storage)
* @param {String} key
* @param {String} value
* @param {Number} expirationTimestamp
*/
function saveLotameCache(
key,
value,
expirationTimestamp = utils.timestamp() + DAYS_TO_CACHE * DAY_MS
) {
if (key && value) {
let expirationDate = new Date(expirationTimestamp).toUTCString();
if (storage.cookiesAreEnabled()) {
storage.setCookie(
key,
value,
expirationDate,
'Lax',
undefined,
undefined
);
}
if (storage.hasLocalStorage()) {
storage.setDataInLocalStorage(
`${key}_exp`,
String(expirationTimestamp),
undefined
);
storage.setDataInLocalStorage(key, value, undefined);
}
}
}

/**
* Retrieve all the cached values from cookies and/or local storage
*/
function getLotameLocalCache() {
let cache = {
data: getFromStorage(KEY_ID),
expiryTimestampMs: 0,
};

try {
const rawExpiry = getFromStorage(KEY_EXPIRY);
if (utils.isStr(rawExpiry)) {
cache.expiryTimestampMs = parseInt(rawExpiry, 0);
}
} catch (error) {
utils.logError(error);
}

return cache;
}

/**
* Clear a cached value from cookies and local storage
* @param {String} key
*/
function clearLotameCache(key) {
if (key) {
if (storage.cookiesAreEnabled()) {
let expirationDate = new Date(0).toUTCString();
storage.setCookie(key, '', expirationDate, 'Lax', undefined, undefined);
}
if (storage.hasLocalStorage()) {
storage.removeDataFromLocalStorage(key, undefined);
}
}
}
/** @type {Submodule} */
export const lotamePanoramaIdSubmodule = {

/**
* used to link submodule with config
* @type {string}
*/
name: MODULE_NAME,

/**
* Decode the stored id value for passing to bid requests
* @function decode
* @param {(Object|string)} value
* @returns {(Object|undefined)}
*/
decode(value, configParams) {
return utils.isStr(value) ? { 'lotamePanoramaId': value } : undefined;
},

/**
* Retrieve the Lotame Panorama Id
* @function
* @param {SubmoduleParams} [configParams]
* @param {ConsentData} [consentData]
* @param {(Object|undefined)} cacheIdObj
* @returns {IdResponse|undefined}
*/
getId(configParams, consentData, cacheIdObj) {
let localCache = getLotameLocalCache();

let refreshNeeded = Date.now() > localCache.expiryTimestampMs;

if (!refreshNeeded) {
return {
id: localCache.data
};
}

const storedUserId = getProfileId();

const resolveIdFunction = function (callback) {
let queryParams = {};
if (storedUserId) {
queryParams.fp = storedUserId
}

if (consentData && utils.isBoolean(consentData.gdprApplies)) {
queryParams.gdpr_applies = consentData.gdprApplies;
if (consentData.gdprApplies) {
queryParams.gdpr_consent = consentData.consentString;
}
}
const url = utils.buildUrl({
protocol: 'https',
host: `id.crwdcntrl.net`,
pathname: '/id',
search: utils.isEmpty(queryParams) ? undefined : queryParams,
});
ajax(
url,
(response) => {
let responseObj = {};
if (response) {
try {
responseObj = JSON.parse(response);
saveLotameCache(KEY_EXPIRY, responseObj.expiry_ts);

if (utils.isStr(responseObj.core_id)) {
saveLotameCache(
KEY_ID,
responseObj.core_id,
responseObj.expiry_ts
);
} else {
clearLotameCache(KEY_ID);
}

if (utils.isStr(responseObj.profile_id)) {
setProfileId(responseObj.profile_id);
} else {
clearLotameCache(KEY_PROFILE);
clearLotameCache(KEY_ID);
smenzer marked this conversation as resolved.
Show resolved Hide resolved
}
} catch (error) {
utils.logError(error);
}
}
callback(responseObj);
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

again, i may be missing something, but here you are storing to the prebid cache object the entire responseObj, which seems to contain several keys, including the core_id and profile_id. however, in your decode function on line 157, you are checking if the value is a string, otherwise you return undefined. my understanding is that the value you receive in decode() will be the same thing you pass in the callback() here...which means decode would always return undefined. again, i'm sure i'm missing something here since you're doing some of your own cache management, but wanted to flag this as it doesn't seem correct.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Looking into this part as it was tricky to trace what decode will always get as there are a couple of different paths

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Pushed up a fix, thanks for catching this!

},
undefined,
{
method: 'GET',
withCredentials: true
}
);
};

return { callback: resolveIdFunction };
},
};

submodule('userId', lotamePanoramaIdSubmodule);
25 changes: 25 additions & 0 deletions modules/lotamePanoramaIdSystem.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,25 @@
# Overview

```
Module Name: Lotame Panorama Id System
Module Type: Id System
Maintainer: [email protected]
```

# Description

Retrieve the Lotame Panorama Id

# Usage

```
pbjs.que.push(function() {
pbjs.setConfig({
usersync: {
userIds: [
{
name: 'lotamePanoramaId' // The only parameter that is needed
}],
}
});
```
6 changes: 6 additions & 0 deletions modules/userId/eids.js
Original file line number Diff line number Diff line change
Expand Up @@ -62,6 +62,12 @@ const USER_IDS_CONFIG = {
atype: 1
},

// lotamePanoramaId
lotamePanoramaId: {
source: 'crwdcntrl.net',
atype: 1,
},

// DigiTrust
'digitrustid': {
getValue: function (data) {
Expand Down
Loading