-
Notifications
You must be signed in to change notification settings - Fork 44
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
feat: Cache identity responses (#819)
- Loading branch information
Showing
22 changed files
with
1,238 additions
and
90 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,197 @@ | ||
import Constants from './constants'; | ||
import { Dictionary, parseNumber } from './utils'; | ||
import { BaseVault } from './vault'; | ||
import Types from './types'; | ||
import { IdentityApiData, UserIdentities } from '@mparticle/web-sdk'; | ||
import { IdentityAPIMethod } from './sdkRuntimeModels'; | ||
import { isObject } from './utils'; | ||
const { Identify, Modify, Login, Logout } = Constants.IdentityMethods; | ||
import { ONE_DAY_IN_SECONDS, MILLIS_IN_ONE_SEC } from './constants'; | ||
|
||
export interface IKnownIdentities extends UserIdentities { | ||
device_application_stamp?: string; | ||
} | ||
|
||
export interface ICachedIdentityCall { | ||
responseText: string; | ||
status: number; | ||
expireTimestamp: number; | ||
} | ||
|
||
export const cacheOrClearIdCache = ( | ||
method: string, | ||
knownIdentities: IKnownIdentities, | ||
idCache: BaseVault<Dictionary<ICachedIdentityCall>>, | ||
xhr: XMLHttpRequest | ||
): void => { | ||
const CACHE_HEADER = 'x-mp-max-age'; | ||
|
||
// default the expire timestamp to one day in milliseconds unless a header comes back | ||
let expireTimestamp = new Date().getTime() + ONE_DAY_IN_SECONDS * MILLIS_IN_ONE_SEC; | ||
|
||
if (xhr.getAllResponseHeaders().includes(CACHE_HEADER)) { | ||
expireTimestamp = | ||
parseNumber(xhr.getResponseHeader(CACHE_HEADER)) * MILLIS_IN_ONE_SEC; | ||
} | ||
|
||
switch (method) { | ||
case Login: | ||
case Identify: | ||
cacheIdentityRequest( | ||
method, | ||
knownIdentities, | ||
expireTimestamp, | ||
idCache, | ||
xhr | ||
); | ||
break; | ||
case Modify: | ||
case Logout: | ||
idCache.purge(); | ||
break; | ||
} | ||
} | ||
|
||
export const cacheIdentityRequest = ( | ||
method: IdentityAPIMethod, | ||
identities: IKnownIdentities, | ||
expireTimestamp: number, | ||
idCache: BaseVault<Dictionary<ICachedIdentityCall>>, | ||
xhr: XMLHttpRequest | ||
): void => { | ||
const cache: Dictionary<ICachedIdentityCall> = idCache.retrieve() || ({} as Dictionary<ICachedIdentityCall>); | ||
const cacheKey = concatenateIdentities(method, identities); | ||
|
||
cache[cacheKey] = { responseText: xhr.responseText, status: xhr.status, expireTimestamp}; | ||
idCache.store(cache); | ||
}; | ||
|
||
// We need to ensure that identities are concatenated in a deterministic way, so | ||
// we sort the identities based on their enum. | ||
// we create an array, set the user identity at the index of the user identity type | ||
export const concatenateIdentities = ( | ||
method: IdentityAPIMethod, | ||
userIdentities: IKnownIdentities | ||
): string => { | ||
const DEVICE_APPLICATION_STAMP = 'device_application_stamp'; | ||
// set DAS first since it is not an official identity type | ||
let cacheKey: string = `${method}:${DEVICE_APPLICATION_STAMP}=${userIdentities.device_application_stamp};`; | ||
const idLength: number = Object.keys(userIdentities).length; | ||
let concatenatedIdentities: string = ''; | ||
|
||
if (idLength) { | ||
let userIDArray: Array<string> = new Array(); | ||
// create an array where each index is equal to the user identity type | ||
for (let key in userIdentities) { | ||
if (key === DEVICE_APPLICATION_STAMP) { | ||
continue; | ||
} else { | ||
userIDArray[Types.IdentityType.getIdentityType(key)] = | ||
userIdentities[key]; | ||
} | ||
} | ||
|
||
concatenatedIdentities = userIDArray.reduce( | ||
(prevValue: string, currentValue: string, index: number) => { | ||
const idName: string = Types.IdentityType.getIdentityName(index); | ||
return `${prevValue}${idName}=${currentValue};`; | ||
}, | ||
cacheKey | ||
); | ||
} | ||
|
||
return concatenatedIdentities; | ||
}; | ||
|
||
export const hasValidCachedIdentity = ( | ||
method: IdentityAPIMethod, | ||
proposedUserIdentities: IKnownIdentities, | ||
idCache?: BaseVault<Dictionary<ICachedIdentityCall>> | ||
): Boolean => { | ||
// There is an edge case where multiple identity calls are taking place | ||
// before identify fires, so there may not be a cache. See what happens when | ||
// the ? in idCache is removed to the following test | ||
// "queued events contain login mpid instead of identify mpid when calling | ||
// login immediately after mParticle initializes" | ||
const cache = idCache?.retrieve(); | ||
|
||
// if there is no cache, then there is no valid cached identity | ||
if (!cache) { | ||
return false; | ||
} | ||
|
||
const cacheKey: string = concatenateIdentities( | ||
method, | ||
proposedUserIdentities | ||
); | ||
|
||
// if cache doesn't have the cacheKey, there is no valid cached identity | ||
if (!cache.hasOwnProperty(cacheKey)) { | ||
return false; | ||
} | ||
|
||
// If there is a valid cache key, compare the expireTimestamp to the current time. | ||
// If the current time is greater than the expireTimestamp, it is not a valid | ||
// cached identity. | ||
const expireTimestamp = cache[cacheKey].expireTimestamp; | ||
|
||
if (expireTimestamp < new Date().getTime()) { | ||
return false; | ||
} else { | ||
return true; | ||
} | ||
}; | ||
|
||
export const getCachedIdentity = ( | ||
method: IdentityAPIMethod, | ||
proposedUserIdentities: IKnownIdentities, | ||
idCache: BaseVault<Dictionary<ICachedIdentityCall>> | ||
): ICachedIdentityCall | null => { | ||
const cacheKey: string = concatenateIdentities( | ||
method, | ||
proposedUserIdentities | ||
); | ||
|
||
const cache = idCache.retrieve(); | ||
const cachedIdentity = cache ? cache[cacheKey] : null; | ||
|
||
return cachedIdentity; | ||
}; | ||
|
||
// https://go.mparticle.com/work/SQDSDKS-6079 | ||
export const createKnownIdentities = ( | ||
identityApiData: IdentityApiData, | ||
deviceId: string | ||
): IKnownIdentities => { | ||
const identitiesResult: IKnownIdentities = {}; | ||
|
||
if ( | ||
identityApiData?.userIdentities && | ||
isObject(identityApiData.userIdentities) | ||
) { | ||
for (var identity in identityApiData.userIdentities) { | ||
identitiesResult[identity] = | ||
identityApiData.userIdentities[identity]; | ||
} | ||
} | ||
identitiesResult.device_application_stamp = deviceId; | ||
|
||
return identitiesResult; | ||
}; | ||
|
||
export const removeExpiredIdentityCacheDates = (idCache: BaseVault<Dictionary<ICachedIdentityCall>>) => { | ||
const cache: Dictionary<ICachedIdentityCall> = idCache.retrieve() || {} as Dictionary<ICachedIdentityCall>; | ||
|
||
idCache.purge(); | ||
|
||
const currentTime: number = new Date().getTime(); | ||
|
||
// Iterate over the cache and remove any key/value pairs that are expired | ||
for (let key in cache) { | ||
if (cache[key].expireTimestamp < currentTime) { | ||
delete cache[key]; | ||
} | ||
}; | ||
|
||
idCache.store(cache); | ||
} |
Oops, something went wrong.