Skip to content

Commit

Permalink
WIP add store migration
Browse files Browse the repository at this point in the history
  • Loading branch information
robbie-c committed Nov 21, 2023
1 parent 7e67d7c commit bc62de0
Show file tree
Hide file tree
Showing 4 changed files with 76 additions and 9 deletions.
2 changes: 1 addition & 1 deletion src/posthog-core.ts
Original file line number Diff line number Diff line change
Expand Up @@ -108,7 +108,7 @@ export const defaultConfig = (): PostHogConfig => ({
autocapture: true,
rageclick: true,
cross_subdomain_cookie: isCrossDomainCookie(document?.location),
persistence: 'cookie',
persistence: 'default',
persistence_name: '',
cookie_name: '',
loaded: __NOOP,
Expand Down
79 changes: 73 additions & 6 deletions src/posthog-persistence.ts
Original file line number Diff line number Diff line change
@@ -1,12 +1,19 @@
/* eslint camelcase: "off" */

import { _each, _extend, _include, _strip_empty_properties } from './utils'
import { cookieStore, localStore, localPlusCookieStore, memoryStore, sessionStore } from './storage'
import {
COOKIE_PERSISTED_PROPERTIES,
cookieStore,
localPlusCookieStore,
localStore,
memoryStore,
sessionStore,
} from './storage'
import { PersistentStore, PostHogConfig, Properties } from './types'
import {
PERSISTENCE_RESERVED_PROPERTIES,
EVENT_TIMERS_KEY,
ENABLED_FEATURE_FLAGS,
EVENT_TIMERS_KEY,
PERSISTENCE_RESERVED_PROPERTIES,
POSTHOG_QUOTA_LIMITED,
USER_STATE,
} from './constants'
Expand All @@ -21,6 +28,7 @@ const CASE_INSENSITIVE_PERSISTENCE_TYPES: readonly Lowercase<PostHogConfig['pers
'localstorage+cookie',
'sessionstorage',
'memory',
'default',
]

/**
Expand Down Expand Up @@ -64,8 +72,8 @@ export class PostHogPersistence {
config['persistence'].toLowerCase() as Lowercase<PostHogConfig['persistence']>
) === -1
) {
logger.critical('Unknown persistence type ' + config['persistence'] + '; falling back to cookie')
config['persistence'] = 'cookie'
logger.critical('Unknown persistence type ' + config['persistence'] + '; falling back to default')
config['persistence'] = 'default'
}
// We handle storage type in a case-insensitive way for backwards compatibility
const storage_type = config['persistence'].toLowerCase() as Lowercase<PostHogConfig['persistence']>
Expand All @@ -77,8 +85,11 @@ export class PostHogPersistence {
this.storage = sessionStore
} else if (storage_type === 'memory') {
this.storage = memoryStore
} else {
} else if (storage_type === 'cookie') {
this.storage = cookieStore
} else {
const { storage } = this._getAndMigrateToDefaultStore()
this.storage = storage
}

this.user_state = 'anonymous'
Expand Down Expand Up @@ -314,4 +325,60 @@ export class PostHogPersistence {
this.props[POSTHOG_QUOTA_LIMITED] = state
this.save()
}

_getAndMigrateToDefaultStore = (): { storage: PersistentStore; wasPreviouslyDifferentStore: boolean } => {
// Check if there's any data in local/cookie/session storage, and if needed,
// migrate to cookie+localstorage
const defaultStore = localPlusCookieStore
const currentStore = this._getCurrentStore()

if (currentStore && currentStore !== defaultStore) {
this._migrateStoreToDefaultStore(currentStore, defaultStore)
return { storage: defaultStore, wasPreviouslyDifferentStore: !!currentStore }
} else {
return { storage: defaultStore, wasPreviouslyDifferentStore: false }
}
}

_getCurrentStore = () => {
// Test by picking a property that is stored in the cookie for local+cookie
const testKey = COOKIE_PERSISTED_PROPERTIES[0]

// If that key exists in the local store, then we're definitely using
// local storage
const isLocalStore = localStore.is_supported() && localStore.get(this.name)?.[testKey]
if (isLocalStore) {
return localStore
}
// If not using pure localStore, we must be using localPlusCookie if
// there's data in both
const isLocalPlusCookie =
localPlusCookieStore.is_supported() &&
localStore.is_supported() &&
localStore.get(this.name) &&
cookieStore.is_supported() &&
cookieStore.get(this.name)
if (isLocalPlusCookie) {
return localPlusCookieStore
}
// If there's any data in cookieStore at this point then we're on cookieStore
const isCookie = cookieStore.is_supported() && cookieStore.get(this.name)
if (isCookie) {
return cookieStore
}

const isSession = sessionStore.is_supported() && sessionStore.get(this.name)
if (isSession) {
return sessionStore
}

// memory store, no existing store, or existing store no longer supported
return undefined
}

_migrateStoreToDefaultStore(store: PersistentStore, defaultStore: PersistentStore) {
const values = store.get(this.name)
store.remove(this.name, this.cross_subdomain)
defaultStore.set(this.name, values, this.expire_days, this.cross_subdomain, this.secure)
}
}
2 changes: 1 addition & 1 deletion src/storage.ts
Original file line number Diff line number Diff line change
Expand Up @@ -236,7 +236,7 @@ export const localStore: PersistentStore = {
// Use localstorage for most data but still use cookie for COOKIE_PERSISTED_PROPERTIES
// This solves issues with cookies having too much data in them causing headers too large
// Also makes sure we don't have to send a ton of data to the server
const COOKIE_PERSISTED_PROPERTIES = [DISTINCT_ID, SESSION_ID, SESSION_RECORDING_IS_SAMPLED]
export const COOKIE_PERSISTED_PROPERTIES = [DISTINCT_ID, SESSION_ID, SESSION_RECORDING_IS_SAMPLED]

export const localPlusCookieStore: PersistentStore = {
...localStore,
Expand Down
2 changes: 1 addition & 1 deletion src/types.ts
Original file line number Diff line number Diff line change
Expand Up @@ -64,7 +64,7 @@ export interface PostHogConfig {
autocapture: boolean | AutocaptureConfig
rageclick: boolean
cross_subdomain_cookie: boolean
persistence: 'localStorage' | 'cookie' | 'memory' | 'localStorage+cookie' | 'sessionStorage'
persistence: 'localStorage' | 'cookie' | 'memory' | 'localStorage+cookie' | 'sessionStorage' | 'default'
persistence_name: string
cookie_name: string
loaded: (posthog_instance: PostHog) => void
Expand Down

0 comments on commit bc62de0

Please sign in to comment.