From 4b6983da6a5aa958c02158e896d5b5a0352cfe36 Mon Sep 17 00:00:00 2001 From: Marius Andra Date: Fri, 22 Nov 2024 11:24:57 +0100 Subject: [PATCH 1/7] site functions --- src/autocapture.ts | 11 +- src/decide.ts | 17 +-- src/entrypoints/external-scripts-loader.ts | 16 +-- src/posthog-core.ts | 6 ++ src/site-apps.ts | 118 +++++++++++++++++++++ src/types.ts | 2 +- src/utils/globals.ts | 2 - 7 files changed, 135 insertions(+), 37 deletions(-) create mode 100644 src/site-apps.ts diff --git a/src/autocapture.ts b/src/autocapture.ts index cf9289e3a..a4cb4078f 100644 --- a/src/autocapture.ts +++ b/src/autocapture.ts @@ -213,13 +213,10 @@ export function autocapturePropertiesForElement( const props = extend( getDefaultProperties(e.type), - elementsChainAsString - ? { - $elements_chain: getElementsChainString(elementsJson), - } - : { - $elements: elementsJson, - }, + // Sending "$elements" is deprecated. Only one client on US cloud uses this. + !elementsChainAsString ? { $elements: elementsJson } : {}, + // Always send $elements_chain, as it's needed downstream in site app filtering + { $elements_chain: getElementsChainString(elementsJson) }, elementsJson[0]?.['$el_text'] ? { $el_text: elementsJson[0]?.['$el_text'] } : {}, externalHref && e.type === 'click' ? { $external_click_url: externalHref } : {}, autocaptureAugmentProperties diff --git a/src/decide.ts b/src/decide.ts index 9ccb59819..88c7ab92d 100644 --- a/src/decide.ts +++ b/src/decide.ts @@ -3,7 +3,7 @@ import { Compression, DecideResponse } from './types' import { STORED_GROUP_PROPERTIES_KEY, STORED_PERSON_PROPERTIES_KEY } from './constants' import { logger } from './utils/logger' -import { document, assignableWindow } from './utils/globals' +import { document } from './utils/globals' export class Decide { constructor(private readonly instance: PostHog) { @@ -64,20 +64,5 @@ export class Decide { } this.instance._afterDecideResponse(response) - - if (response['siteApps']) { - if (this.instance.config.opt_in_site_apps) { - for (const { id, url } of response['siteApps']) { - assignableWindow[`__$$ph_site_app_${id}`] = this.instance - assignableWindow.__PosthogExtensions__?.loadSiteApp?.(this.instance, url, (err) => { - if (err) { - return logger.error(`Error while initializing PostHog app with config id ${id}`, err) - } - }) - } - } else if (response['siteApps'].length > 0) { - logger.error('PostHog site apps are disabled. Enable the "opt_in_site_apps" config to proceed.') - } - } } } diff --git a/src/entrypoints/external-scripts-loader.ts b/src/entrypoints/external-scripts-loader.ts index 084b47d82..788f64def 100644 --- a/src/entrypoints/external-scripts-loader.ts +++ b/src/entrypoints/external-scripts-loader.ts @@ -2,7 +2,11 @@ import type { PostHog } from '../posthog-core' import { assignableWindow, document, PostHogExtensionKind } from '../utils/globals' import { logger } from '../utils/logger' -const loadScript = (posthog: PostHog, url: string, callback: (error?: string | Event, event?: Event) => void) => { +export const loadScript = ( + posthog: PostHog, + url: string, + callback: (error?: string | Event, event?: Event) => void +) => { if (posthog.config.disable_external_dependency_loading) { logger.warn(`${url} was requested but loading of external scripts is disabled.`) return callback('Loading of external scripts is disabled') @@ -56,13 +60,3 @@ assignableWindow.__PosthogExtensions__.loadExternalDependency = ( loadScript(posthog, url, callback) } - -assignableWindow.__PosthogExtensions__.loadSiteApp = ( - posthog: PostHog, - url: string, - callback: (error?: string | Event, event?: Event) => void -): void => { - const scriptUrl = posthog.requestRouter.endpointFor('api', url) - - loadScript(posthog, scriptUrl, callback) -} diff --git a/src/posthog-core.ts b/src/posthog-core.ts index db362d2af..49e21a25f 100644 --- a/src/posthog-core.ts +++ b/src/posthog-core.ts @@ -80,6 +80,7 @@ import { ExceptionObserver } from './extensions/exception-autocapture' import { WebVitalsAutocapture } from './extensions/web-vitals' import { WebExperiments } from './web-experiments' import { PostHogExceptions } from './posthog-exceptions' +import { SiteApps } from './site-apps' import { DeadClicksAutocapture, isDeadClicksEnabledForAutocapture } from './extensions/dead-clicks-autocapture' /* @@ -258,6 +259,7 @@ export class PostHog { sessionManager?: SessionIdManager sessionPropsManager?: SessionPropsManager requestRouter: RequestRouter + siteApps?: SiteApps autocapture?: Autocapture heatmaps?: Heatmaps webVitalsAutocapture?: WebVitalsAutocapture @@ -432,6 +434,9 @@ export class PostHog { new TracingHeaders(this).startIfEnabledOrStop() + this.siteApps = new SiteApps(this) + this.siteApps?.init() + this.sessionRecording = new SessionRecording(this) this.sessionRecording.startIfEnabledOrStop() @@ -562,6 +567,7 @@ export class PostHog { : 'always', }) + this.siteApps?.afterDecideResponse(response) this.sessionRecording?.afterDecideResponse(response) this.autocapture?.afterDecideResponse(response) this.heatmaps?.afterDecideResponse(response) diff --git a/src/site-apps.ts b/src/site-apps.ts new file mode 100644 index 000000000..b0bcc65f0 --- /dev/null +++ b/src/site-apps.ts @@ -0,0 +1,118 @@ +import { loadScript } from './entrypoints/external-scripts-loader' +import { PostHog } from './posthog-core' +import { CaptureResult, DecideResponse } from './types' +import { assignableWindow } from './utils/globals' +import { logger } from './utils/logger' + +export class SiteApps { + instance: PostHog + enabled: boolean + missedInvocations: Record[] + loaded: boolean + appsLoading: Set + + constructor(instance: PostHog) { + this.instance = instance + // can't use if site apps are disabled, or if we're not asking /decide for site apps + this.enabled = !!this.instance.config.opt_in_site_apps && !this.instance.config.advanced_disable_decide + // events captured between loading posthog-js and the site app; up to 1000 events + this.missedInvocations = [] + // capture events until loaded + this.loaded = false + this.appsLoading = new Set() + } + + eventCollector(_eventName: string, eventPayload?: CaptureResult | undefined) { + if (!this.enabled) { + return + } + if (!this.loaded && eventPayload) { + const globals = this.globalsForEvent(eventPayload) + this.missedInvocations.push(globals) + if (this.missedInvocations.length > 1000) { + this.missedInvocations = this.missedInvocations.slice(990) + } + } + } + + init() { + this.instance?._addCaptureHook(this.eventCollector.bind(this)) + } + + globalsForEvent(event: CaptureResult): Record { + if (!event) { + throw new Error('Event payload is required') + } + const groups: Record> = {} + const groupIds = this.instance.get_property('$groups') || [] + const groupProperties = this.instance.get_property('$stored_group_properties') || {} + for (const [type, properties] of Object.entries(groupProperties)) { + groups[type] = { id: groupIds[type], type, properties } + } + const { $set_once, $set, ..._event } = event + const globals = { + event: { + ..._event, + properties: { + ...event.properties, + ...($set ? { $set: { ...(event.properties?.$set ?? {}), ...$set } } : {}), + ...($set_once ? { $set_once: { ...(event.properties?.$set_once ?? {}), ...$set_once } } : {}), + }, + elements_chain: event.properties?.['$elements_chain'] ?? '', + // TODO: + // - elements_chain_href: '', + // - elements_chain_texts: [] as string[], + // - elements_chain_ids: [] as string[], + // - elements_chain_elements: [] as string[], + distinct_id: event.properties?.['distinct_id'], + }, + person: { + properties: this.instance.get_property('$stored_person_properties'), + }, + groups, + } + return globals + } + + loadSiteApp(posthog: PostHog, url: string, callback: (error?: string | Event, event?: Event) => void) { + const scriptUrl = posthog.requestRouter.endpointFor('api', url) + loadScript(posthog, scriptUrl, callback) + } + + afterDecideResponse(response?: DecideResponse): void { + if (response?.['siteApps']) { + if (this.enabled && this.instance.config.opt_in_site_apps) { + const checkIfAllLoaded = () => { + // Stop collecting events once all site apps are loaded + if (this.appsLoading.size === 0) { + this.loaded = true + this.missedInvocations = [] + } + } + for (const { id, url } of response['siteApps']) { + this.appsLoading.add(id) + assignableWindow[`__$$ph_site_app_${id}_posthog`] = this.instance + assignableWindow[`__$$ph_site_app_${id}_missed_invocations`] = () => this.missedInvocations + assignableWindow[`__$$ph_site_app_${id}_callback`] = () => { + this.appsLoading.delete(id) + checkIfAllLoaded() + } + this.loadSiteApp(this.instance, url, (err) => { + if (err) { + this.appsLoading.delete(id) + checkIfAllLoaded() + return logger.error(`Error while initializing PostHog app with config id ${id}`, err) + } + }) + } + } else if (response['siteApps'].length > 0) { + logger.error('PostHog site apps are disabled. Enable the "opt_in_site_apps" config to proceed.') + } + } else { + this.loaded = true + this.enabled = false + } + } + + // TODO: opting out of stuff should disable this +} diff --git a/src/types.ts b/src/types.ts index d7994968a..3044e10df 100644 --- a/src/types.ts +++ b/src/types.ts @@ -523,7 +523,7 @@ export interface DecideResponse { editorParams?: ToolbarParams /** @deprecated, renamed to toolbarParams, still present on older API responses */ toolbarVersion: 'toolbar' /** @deprecated, moved to toolbarParams */ isAuthenticated: boolean - siteApps: { id: number; url: string }[] + siteApps: { id: string; url: string }[] heatmaps?: boolean defaultIdentifiedOnly?: boolean captureDeadClicks?: boolean diff --git a/src/utils/globals.ts b/src/utils/globals.ts index 377fe22b5..819a7e59a 100644 --- a/src/utils/globals.ts +++ b/src/utils/globals.ts @@ -48,8 +48,6 @@ interface PostHogExtensions { callback: (error?: string | Event, event?: Event) => void ) => void - loadSiteApp?: (posthog: PostHog, appUrl: string, callback: (error?: string | Event, event?: Event) => void) => void - parseErrorAsProperties?: ( [event, source, lineno, colno, error]: ErrorEventArgs, metadata?: ErrorMetadata From 7473b9bff6d6dc26d1c162cbd87a72908334530c Mon Sep 17 00:00:00 2001 From: Marius Andra Date: Mon, 25 Nov 2024 15:13:07 +0100 Subject: [PATCH 2/7] test fixes --- ...azy-loaded-dead-clicks-autocapture.test.ts | 4 + src/__tests__/posthog-persistence.test.ts | 6 +- src/__tests__/site-apps.ts | 81 +++++++++++++++++++ 3 files changed, 88 insertions(+), 3 deletions(-) create mode 100644 src/__tests__/site-apps.ts diff --git a/src/__tests__/entrypoints/lazy-loaded-dead-clicks-autocapture.test.ts b/src/__tests__/entrypoints/lazy-loaded-dead-clicks-autocapture.test.ts index b206acd62..07a51086c 100644 --- a/src/__tests__/entrypoints/lazy-loaded-dead-clicks-autocapture.test.ts +++ b/src/__tests__/entrypoints/lazy-loaded-dead-clicks-autocapture.test.ts @@ -228,6 +228,7 @@ describe('LazyLoadedDeadClicksAutocapture', () => { $dead_click_selection_changed_timeout: true, $ce_version: 1, $el_text: 'text', + $elements_chain: 'body:text="text"nth-child="2"nth-of-type="1"', $elements: [ { $el_text: 'text', @@ -270,6 +271,7 @@ describe('LazyLoadedDeadClicksAutocapture', () => { $dead_click_selection_changed_timeout: false, $ce_version: 1, $el_text: 'text', + $elements_chain: 'body:text="text"nth-child="2"nth-of-type="1"', $elements: [ { $el_text: 'text', @@ -312,6 +314,7 @@ describe('LazyLoadedDeadClicksAutocapture', () => { $dead_click_selection_changed_timeout: false, $ce_version: 1, $el_text: 'text', + $elements_chain: 'body:text="text"nth-child="2"nth-of-type="1"', $elements: [ { $el_text: 'text', @@ -354,6 +357,7 @@ describe('LazyLoadedDeadClicksAutocapture', () => { $dead_click_selection_changed_timeout: false, $ce_version: 1, $el_text: 'text', + $elements_chain: 'body:text="text"nth-child="2"nth-of-type="1"', $elements: [ { $el_text: 'text', diff --git a/src/__tests__/posthog-persistence.test.ts b/src/__tests__/posthog-persistence.test.ts index 086246e24..d837b8a2c 100644 --- a/src/__tests__/posthog-persistence.test.ts +++ b/src/__tests__/posthog-persistence.test.ts @@ -120,13 +120,13 @@ describe('persistence', () => { it('should migrate data from cookies to localStorage', () => { const lib = new PostHogPersistence(makePostHogConfig('bla', 'cookie')) lib.register_once({ distinct_id: 'testy', test_prop: 'test_value' }, undefined, undefined) - expect(document.cookie).toEqual( + expect(document.cookie).toContain( 'ph__posthog=%7B%22distinct_id%22%3A%22testy%22%2C%22test_prop%22%3A%22test_value%22%7D' ) const lib2 = new PostHogPersistence(makePostHogConfig('bla', 'localStorage+cookie')) - expect(document.cookie).toEqual('ph__posthog=%7B%22distinct_id%22%3A%22testy%22%7D') + expect(document.cookie).toContain('ph__posthog=%7B%22distinct_id%22%3A%22testy%22%7D') lib2.register({ test_prop2: 'test_val', distinct_id: 'test2' }) - expect(document.cookie).toEqual('ph__posthog=%7B%22distinct_id%22%3A%22test2%22%7D') + expect(document.cookie).toContain('ph__posthog=%7B%22distinct_id%22%3A%22test2%22%7D') expect(lib2.props).toEqual({ distinct_id: 'test2', test_prop: 'test_value', test_prop2: 'test_val' }) lib2.remove() expect(localStorage.getItem('ph__posthog')).toEqual(null) diff --git a/src/__tests__/site-apps.ts b/src/__tests__/site-apps.ts new file mode 100644 index 000000000..084aefc4b --- /dev/null +++ b/src/__tests__/site-apps.ts @@ -0,0 +1,81 @@ +import { SiteApps } from '../site-apps' +import { PostHogPersistence } from '../posthog-persistence' +import { RequestRouter } from '../utils/request-router' +import { expectScriptToExist, expectScriptToNotExist } from './helpers/script-utils' +import { PostHog } from '../posthog-core' +import { DecideResponse, PostHogConfig, Properties } from '../types' +import '../entrypoints/external-scripts-loader' + +describe('SiteApps', () => { + let posthog: PostHog + + const siteApps = () => new SiteApps(posthog) + + const defaultConfig: Partial = { + token: 'testtoken', + api_host: 'https://test.com', + persistence: 'memory', + } + + beforeEach(() => { + // clean the JSDOM to prevent interdependencies between tests + document.body.innerHTML = '' + document.head.innerHTML = '' + jest.spyOn(window.console, 'error').mockImplementation() + + posthog = { + config: defaultConfig, + persistence: new PostHogPersistence(defaultConfig as PostHogConfig), + register: (props: Properties) => posthog.persistence!.register(props), + unregister: (key: string) => posthog.persistence!.unregister(key), + get_property: (key: string) => posthog.persistence!.props[key], + capture: jest.fn(), + _addCaptureHook: jest.fn(), + _afterDecideResponse: jest.fn(), + get_distinct_id: jest.fn().mockImplementation(() => 'distinctid'), + _send_request: jest.fn().mockImplementation(({ callback }) => callback?.({ config: {} })), + featureFlags: { + receivedFeatureFlags: jest.fn(), + setReloadingPaused: jest.fn(), + _startReloadTimer: jest.fn(), + }, + requestRouter: new RequestRouter({ config: defaultConfig } as unknown as PostHog), + _hasBootstrappedFeatureFlags: jest.fn(), + getGroups: () => ({ organization: '5' }), + } as unknown as PostHog + }) + + describe('afterDecideResponse', () => { + const subject = (decideResponse: DecideResponse) => siteApps().afterDecideResponse(decideResponse) + + it('runs site apps if opted in', () => { + posthog.config = { + api_host: 'https://test.com', + opt_in_site_apps: true, + persistence: 'memory', + } as PostHogConfig + + subject({ siteApps: [{ id: 1, url: '/site_app/1/tokentoken/hash/' }] } as DecideResponse) + + expectScriptToExist('https://test.com/site_app/1/tokentoken/hash/') + }) + + it('does not run site apps code if not opted in', () => { + ;(window as any).POSTHOG_DEBUG = true + // don't technically need to run this but this test assumes opt_in_site_apps is false, let's make that explicit + posthog.config = { + api_host: 'https://test.com', + opt_in_site_apps: false, + persistence: 'memory', + } as unknown as PostHogConfig + + subject({ siteApps: [{ id: 1, url: '/site_app/1/tokentoken/hash/' }] } as DecideResponse) + + expect(console.error).toHaveBeenCalledWith( + '[PostHog.js]', + 'PostHog site apps are disabled. Enable the "opt_in_site_apps" config to proceed.' + ) + expectScriptToNotExist('https://test.com/site_app/1/tokentoken/hash/') + }) + }) +}) From 3a9a31702faf9fa49ccac7d5836bd444a1b20bdc Mon Sep 17 00:00:00 2001 From: Marius Andra Date: Mon, 25 Nov 2024 15:31:18 +0100 Subject: [PATCH 3/7] more tests --- src/__tests__/site-apps.ts | 306 ++++++++++++++++++++++++++++++++++--- src/site-apps.ts | 5 +- 2 files changed, 287 insertions(+), 24 deletions(-) diff --git a/src/__tests__/site-apps.ts b/src/__tests__/site-apps.ts index 084aefc4b..532a1d7a7 100644 --- a/src/__tests__/site-apps.ts +++ b/src/__tests__/site-apps.ts @@ -1,16 +1,21 @@ +// __tests__/site-apps.ts + import { SiteApps } from '../site-apps' import { PostHogPersistence } from '../posthog-persistence' import { RequestRouter } from '../utils/request-router' -import { expectScriptToExist, expectScriptToNotExist } from './helpers/script-utils' import { PostHog } from '../posthog-core' -import { DecideResponse, PostHogConfig, Properties } from '../types' +import { DecideResponse, PostHogConfig, Properties, CaptureResult } from '../types' +import { assignableWindow } from '../utils/globals' import '../entrypoints/external-scripts-loader' +import { logger } from '../utils/logger' + +jest.mock('../entrypoints/external-scripts-loader', () => ({ + loadScript: jest.fn(), +})) describe('SiteApps', () => { let posthog: PostHog - const siteApps = () => new SiteApps(posthog) - const defaultConfig: Partial = { token: 'testtoken', api_host: 'https://test.com', @@ -24,7 +29,7 @@ describe('SiteApps', () => { jest.spyOn(window.console, 'error').mockImplementation() posthog = { - config: defaultConfig, + config: { ...defaultConfig }, persistence: new PostHogPersistence(defaultConfig as PostHogConfig), register: (props: Properties) => posthog.persistence!.register(props), unregister: (key: string) => posthog.persistence!.unregister(key), @@ -45,37 +50,292 @@ describe('SiteApps', () => { } as unknown as PostHog }) - describe('afterDecideResponse', () => { - const subject = (decideResponse: DecideResponse) => siteApps().afterDecideResponse(decideResponse) - - it('runs site apps if opted in', () => { + describe('constructor', () => { + it('sets enabled to true when opt_in_site_apps is true and advanced_disable_decide is false', () => { posthog.config = { - api_host: 'https://test.com', + ...defaultConfig, opt_in_site_apps: true, - persistence: 'memory', + advanced_disable_decide: false, } as PostHogConfig - subject({ siteApps: [{ id: 1, url: '/site_app/1/tokentoken/hash/' }] } as DecideResponse) + const siteAppsInstance = new SiteApps(posthog) - expectScriptToExist('https://test.com/site_app/1/tokentoken/hash/') + expect(siteAppsInstance.enabled).toBe(true) }) - it('does not run site apps code if not opted in', () => { - ;(window as any).POSTHOG_DEBUG = true - // don't technically need to run this but this test assumes opt_in_site_apps is false, let's make that explicit + it('sets enabled to false when opt_in_site_apps is false', () => { posthog.config = { - api_host: 'https://test.com', + ...defaultConfig, opt_in_site_apps: false, - persistence: 'memory', - } as unknown as PostHogConfig + advanced_disable_decide: false, + } as PostHogConfig + + const siteAppsInstance = new SiteApps(posthog) + + expect(siteAppsInstance.enabled).toBe(false) + }) + + it('sets enabled to false when advanced_disable_decide is true', () => { + posthog.config = { + ...defaultConfig, + opt_in_site_apps: true, + advanced_disable_decide: true, + } as PostHogConfig + + const siteAppsInstance = new SiteApps(posthog) + + expect(siteAppsInstance.enabled).toBe(false) + }) + + it('initializes missedInvocations, loaded, appsLoading correctly', () => { + const siteAppsInstance = new SiteApps(posthog) + + expect(siteAppsInstance.missedInvocations).toEqual([]) + expect(siteAppsInstance.loaded).toBe(false) + expect(siteAppsInstance.appsLoading).toEqual(new Set()) + }) + }) + + describe('init', () => { + it('adds eventCollector as a capture hook', () => { + const siteAppsInstance = new SiteApps(posthog) + siteAppsInstance.init() + + expect(posthog._addCaptureHook).toHaveBeenCalledWith(expect.any(Function)) + }) + }) + + describe('eventCollector', () => { + let siteAppsInstance: SiteApps + + beforeEach(() => { + siteAppsInstance = new SiteApps(posthog) + }) + + it('does nothing if enabled is false', () => { + siteAppsInstance.enabled = false + siteAppsInstance.eventCollector('event_name', {} as CaptureResult) + + expect(siteAppsInstance.missedInvocations.length).toBe(0) + }) + + it('collects event if enabled and loaded is false', () => { + siteAppsInstance.enabled = true + siteAppsInstance.loaded = false + + const eventPayload = { event: 'test_event', properties: { prop1: 'value1' } } as CaptureResult + + jest.spyOn(siteAppsInstance, 'globalsForEvent').mockReturnValue({ some: 'globals' }) + + siteAppsInstance.eventCollector('test_event', eventPayload) + + expect(siteAppsInstance.globalsForEvent).toHaveBeenCalledWith(eventPayload) + expect(siteAppsInstance.missedInvocations).toEqual([{ some: 'globals' }]) + }) + + it('trims missedInvocations to last 990 when exceeding 1000', () => { + siteAppsInstance.enabled = true + siteAppsInstance.loaded = false + + siteAppsInstance.missedInvocations = new Array(1000).fill({}).map((_, index) => ({ index })) + + const eventPayload = { event: 'test_event', properties: { prop1: 'value1' } } as CaptureResult + + jest.spyOn(siteAppsInstance, 'globalsForEvent').mockReturnValue({ some: 'globals' }) + + siteAppsInstance.eventCollector('test_event', eventPayload) + + expect(siteAppsInstance.missedInvocations.length).toBe(991) + // Ensure that the first 10 events were trimmed + expect(siteAppsInstance.missedInvocations[0]).toEqual({ index: 10 }) + expect(siteAppsInstance.missedInvocations[990]).toEqual({ some: 'globals' }) + }) + }) + + describe('globalsForEvent', () => { + let siteAppsInstance: SiteApps + + beforeEach(() => { + siteAppsInstance = new SiteApps(posthog) + }) + + it('throws an error if event is undefined', () => { + expect(() => siteAppsInstance.globalsForEvent(undefined as any)).toThrow('Event payload is required') + }) + + it('constructs globals object correctly', () => { + jest.spyOn(posthog, 'get_property').mockImplementation((key) => { + if (key === '$groups') { + return { groupType: 'groupId' } + } else if (key === '$stored_group_properties') { + return { groupType: { prop1: 'value1' } } + } else if (key === '$stored_person_properties') { + return { personProp: 'personValue' } + } + }) + + const eventPayload = { + uuid: 'test_uuid', + event: 'test_event', + properties: { + prop1: 'value1', + distinct_id: 'test_distinct_id', + $elements_chain: 'elements_chain_value', + }, + $set: { setProp: 'setValue' }, + $set_once: { setOnceProp: 'setOnceValue' }, + } as CaptureResult + + const globals = siteAppsInstance.globalsForEvent(eventPayload) + + expect(globals).toEqual({ + event: { + uuid: 'test_uuid', + event: 'test_event', + properties: { + $elements_chain: 'elements_chain_value', + prop1: 'value1', + distinct_id: 'test_distinct_id', + $set: { setProp: 'setValue' }, + $set_once: { setOnceProp: 'setOnceValue' }, + }, + elements_chain: 'elements_chain_value', + distinct_id: 'test_distinct_id', + }, + person: { + properties: { personProp: 'personValue' }, + }, + groups: { + groupType: { + id: 'groupId', + type: 'groupType', + properties: { prop1: 'value1' }, + }, + }, + }) + }) + }) + + describe('loadSiteApp', () => { + let siteAppsInstance: SiteApps + let loadScript: jest.Mock + + beforeEach(() => { + siteAppsInstance = new SiteApps(posthog) + // eslint-disable-next-line @typescript-eslint/no-require-imports + loadScript = require('../entrypoints/external-scripts-loader').loadScript as jest.Mock + loadScript.mockClear() + }) + + it('calls loadScript with correct parameters', () => { + const callback = jest.fn() + jest.spyOn(posthog.requestRouter, 'endpointFor').mockReturnValue('https://test.com/script.js') + + siteAppsInstance.loadSiteApp(posthog, '/site_app/1', callback) + + expect(posthog.requestRouter.endpointFor).toHaveBeenCalledWith('api', '/site_app/1') + expect(loadScript).toHaveBeenCalledWith(posthog, 'https://test.com/script.js', callback) + }) + }) + + describe('afterDecideResponse', () => { + let siteAppsInstance: SiteApps + + beforeEach(() => { + siteAppsInstance = new SiteApps(posthog) + jest.spyOn(siteAppsInstance, 'loadSiteApp').mockImplementation((posthog, url, callback) => { + // Simulate loading + callback() + }) + }) + + it('sets loaded to true and enabled to false when response is undefined', () => { + siteAppsInstance.afterDecideResponse(undefined) + + expect(siteAppsInstance.loaded).toBe(true) + expect(siteAppsInstance.enabled).toBe(false) + }) + + it('loads site apps when enabled and opt_in_site_apps is true', () => { + posthog.config.opt_in_site_apps = true + siteAppsInstance.enabled = true + const response = { + siteApps: [ + { id: '1', url: '/site_app/1' }, + { id: '2', url: '/site_app/2' }, + ], + } as DecideResponse + + siteAppsInstance.afterDecideResponse(response) + + expect(siteAppsInstance.appsLoading.size).toBe(2) + expect(siteAppsInstance.loaded).toBe(false) + expect(siteAppsInstance.loadSiteApp).toHaveBeenCalledTimes(2) + }) + + it('does not load site apps when enabled is false', () => { + siteAppsInstance.enabled = false + posthog.config.opt_in_site_apps = false + const response = { + siteApps: [{ id: '1', url: '/site_app/1' }], + } as DecideResponse + + siteAppsInstance.afterDecideResponse(response) + + expect(siteAppsInstance.loaded).toBe(true) + expect(siteAppsInstance.enabled).toBe(false) + expect(siteAppsInstance.loadSiteApp).not.toHaveBeenCalled() + }) + + it('clears missedInvocations when all apps are loaded', () => { + posthog.config.opt_in_site_apps = true + siteAppsInstance.enabled = true + siteAppsInstance.missedInvocations = [{ some: 'data' }] + const response = { + siteApps: [{ id: '1', url: '/site_app/1' }], + } as DecideResponse + + siteAppsInstance.afterDecideResponse(response) + + // Simulate app loaded + siteAppsInstance.appsLoading.delete('1') + // Manually invoke the checkIfAllLoaded function + if (siteAppsInstance.appsLoading.size === 0) { + siteAppsInstance.loaded = true + siteAppsInstance.missedInvocations = [] + } + + expect(siteAppsInstance.loaded).toBe(true) + expect(siteAppsInstance.missedInvocations).toEqual([]) + }) + + it('sets assignableWindow properties for each site app', () => { + posthog.config.opt_in_site_apps = true + siteAppsInstance.enabled = true + const response = { + siteApps: [{ id: '1', url: '/site_app/1' }], + } as DecideResponse + + siteAppsInstance.afterDecideResponse(response) + + expect(assignableWindow['__$$ph_site_app_1_posthog']).toBe(posthog) + expect(typeof assignableWindow['__$$ph_site_app_1_missed_invocations']).toBe('function') + expect(typeof assignableWindow['__$$ph_site_app_1_callback']).toBe('function') + }) + + it('logs error if site apps are disabled but response contains site apps', () => { + posthog.config.opt_in_site_apps = false + siteAppsInstance.enabled = false + const response = { + siteApps: [{ id: '1', url: '/site_app/1' }], + } as DecideResponse - subject({ siteApps: [{ id: 1, url: '/site_app/1/tokentoken/hash/' }] } as DecideResponse) + jest.spyOn(logger, 'error').mockImplementation() + siteAppsInstance.afterDecideResponse(response) - expect(console.error).toHaveBeenCalledWith( - '[PostHog.js]', + expect(logger.error).toHaveBeenCalledWith( 'PostHog site apps are disabled. Enable the "opt_in_site_apps" config to proceed.' ) - expectScriptToNotExist('https://test.com/site_app/1/tokentoken/hash/') }) }) }) diff --git a/src/site-apps.ts b/src/site-apps.ts index b0bcc65f0..f1a44c723 100644 --- a/src/site-apps.ts +++ b/src/site-apps.ts @@ -30,7 +30,7 @@ export class SiteApps { const globals = this.globalsForEvent(eventPayload) this.missedInvocations.push(globals) if (this.missedInvocations.length > 1000) { - this.missedInvocations = this.missedInvocations.slice(990) + this.missedInvocations = this.missedInvocations.slice(10) } } } @@ -107,6 +107,9 @@ export class SiteApps { } } else if (response['siteApps'].length > 0) { logger.error('PostHog site apps are disabled. Enable the "opt_in_site_apps" config to proceed.') + this.loaded = true + } else { + this.loaded = true } } else { this.loaded = true From a365dfb99954b4a7d6b75b9f03671801d704c5d1 Mon Sep 17 00:00:00 2001 From: Marius Andra Date: Mon, 25 Nov 2024 17:10:26 +0100 Subject: [PATCH 4/7] fix some break others --- src/__tests__/decide.ts | 31 ---------------------- src/entrypoints/external-scripts-loader.ts | 16 +++++++---- src/site-apps.ts | 11 +++----- src/utils/globals.ts | 2 ++ 4 files changed, 16 insertions(+), 44 deletions(-) diff --git a/src/__tests__/decide.ts b/src/__tests__/decide.ts index 534d64c65..026f6d0b5 100644 --- a/src/__tests__/decide.ts +++ b/src/__tests__/decide.ts @@ -1,7 +1,6 @@ import { Decide } from '../decide' import { PostHogPersistence } from '../posthog-persistence' import { RequestRouter } from '../utils/request-router' -import { expectScriptToExist, expectScriptToNotExist } from './helpers/script-utils' import { PostHog } from '../posthog-core' import { DecideResponse, PostHogConfig, Properties } from '../types' import '../entrypoints/external-scripts-loader' @@ -246,35 +245,5 @@ describe('Decide', () => { expect(posthog._afterDecideResponse).toHaveBeenCalledWith(decideResponse) expect(posthog.featureFlags.receivedFeatureFlags).not.toHaveBeenCalled() }) - - it('runs site apps if opted in', () => { - posthog.config = { - api_host: 'https://test.com', - opt_in_site_apps: true, - persistence: 'memory', - } as PostHogConfig - - subject({ siteApps: [{ id: 1, url: '/site_app/1/tokentoken/hash/' }] } as DecideResponse) - - expectScriptToExist('https://test.com/site_app/1/tokentoken/hash/') - }) - - it('does not run site apps code if not opted in', () => { - ;(window as any).POSTHOG_DEBUG = true - // don't technically need to run this but this test assumes opt_in_site_apps is false, let's make that explicit - posthog.config = { - api_host: 'https://test.com', - opt_in_site_apps: false, - persistence: 'memory', - } as unknown as PostHogConfig - - subject({ siteApps: [{ id: 1, url: '/site_app/1/tokentoken/hash/' }] } as DecideResponse) - - expect(console.error).toHaveBeenCalledWith( - '[PostHog.js]', - 'PostHog site apps are disabled. Enable the "opt_in_site_apps" config to proceed.' - ) - expectScriptToNotExist('https://test.com/site_app/1/tokentoken/hash/') - }) }) }) diff --git a/src/entrypoints/external-scripts-loader.ts b/src/entrypoints/external-scripts-loader.ts index 788f64def..084b47d82 100644 --- a/src/entrypoints/external-scripts-loader.ts +++ b/src/entrypoints/external-scripts-loader.ts @@ -2,11 +2,7 @@ import type { PostHog } from '../posthog-core' import { assignableWindow, document, PostHogExtensionKind } from '../utils/globals' import { logger } from '../utils/logger' -export const loadScript = ( - posthog: PostHog, - url: string, - callback: (error?: string | Event, event?: Event) => void -) => { +const loadScript = (posthog: PostHog, url: string, callback: (error?: string | Event, event?: Event) => void) => { if (posthog.config.disable_external_dependency_loading) { logger.warn(`${url} was requested but loading of external scripts is disabled.`) return callback('Loading of external scripts is disabled') @@ -60,3 +56,13 @@ assignableWindow.__PosthogExtensions__.loadExternalDependency = ( loadScript(posthog, url, callback) } + +assignableWindow.__PosthogExtensions__.loadSiteApp = ( + posthog: PostHog, + url: string, + callback: (error?: string | Event, event?: Event) => void +): void => { + const scriptUrl = posthog.requestRouter.endpointFor('api', url) + + loadScript(posthog, scriptUrl, callback) +} diff --git a/src/site-apps.ts b/src/site-apps.ts index f1a44c723..584679275 100644 --- a/src/site-apps.ts +++ b/src/site-apps.ts @@ -1,8 +1,8 @@ -import { loadScript } from './entrypoints/external-scripts-loader' import { PostHog } from './posthog-core' import { CaptureResult, DecideResponse } from './types' import { assignableWindow } from './utils/globals' import { logger } from './utils/logger' +import { isArray } from './utils/type-utils' export class SiteApps { instance: PostHog @@ -74,13 +74,8 @@ export class SiteApps { return globals } - loadSiteApp(posthog: PostHog, url: string, callback: (error?: string | Event, event?: Event) => void) { - const scriptUrl = posthog.requestRouter.endpointFor('api', url) - loadScript(posthog, scriptUrl, callback) - } - afterDecideResponse(response?: DecideResponse): void { - if (response?.['siteApps']) { + if (isArray(response?.siteApps) && response.siteApps.length > 0) { if (this.enabled && this.instance.config.opt_in_site_apps) { const checkIfAllLoaded = () => { // Stop collecting events once all site apps are loaded @@ -97,7 +92,7 @@ export class SiteApps { this.appsLoading.delete(id) checkIfAllLoaded() } - this.loadSiteApp(this.instance, url, (err) => { + assignableWindow.__PosthogExtensions__?.loadSiteApp?.(this.instance, url, (err) => { if (err) { this.appsLoading.delete(id) checkIfAllLoaded() diff --git a/src/utils/globals.ts b/src/utils/globals.ts index 819a7e59a..377fe22b5 100644 --- a/src/utils/globals.ts +++ b/src/utils/globals.ts @@ -48,6 +48,8 @@ interface PostHogExtensions { callback: (error?: string | Event, event?: Event) => void ) => void + loadSiteApp?: (posthog: PostHog, appUrl: string, callback: (error?: string | Event, event?: Event) => void) => void + parseErrorAsProperties?: ( [event, source, lineno, colno, error]: ErrorEventArgs, metadata?: ErrorMetadata From 45660382c16cc7b9d1c499c30413fabe3778c19b Mon Sep 17 00:00:00 2001 From: Marius Andra Date: Tue, 26 Nov 2024 10:07:38 +0100 Subject: [PATCH 5/7] tests --- src/__tests__/site-apps.ts | 141 ++++++++++++++++++------------------- 1 file changed, 69 insertions(+), 72 deletions(-) diff --git a/src/__tests__/site-apps.ts b/src/__tests__/site-apps.ts index 532a1d7a7..ec13259a1 100644 --- a/src/__tests__/site-apps.ts +++ b/src/__tests__/site-apps.ts @@ -6,15 +6,19 @@ import { RequestRouter } from '../utils/request-router' import { PostHog } from '../posthog-core' import { DecideResponse, PostHogConfig, Properties, CaptureResult } from '../types' import { assignableWindow } from '../utils/globals' -import '../entrypoints/external-scripts-loader' import { logger } from '../utils/logger' +import '../entrypoints/external-scripts-loader' +import { isFunction } from '../utils/type-utils' -jest.mock('../entrypoints/external-scripts-loader', () => ({ - loadScript: jest.fn(), +jest.mock('../utils/logger', () => ({ + logger: { + error: jest.fn(), + }, })) describe('SiteApps', () => { let posthog: PostHog + let siteAppsInstance: SiteApps const defaultConfig: Partial = { token: 'testtoken', @@ -23,11 +27,25 @@ describe('SiteApps', () => { } beforeEach(() => { - // clean the JSDOM to prevent interdependencies between tests + // Clean the JSDOM to prevent interdependencies between tests document.body.innerHTML = '' document.head.innerHTML = '' jest.spyOn(window.console, 'error').mockImplementation() + // Reset assignableWindow properties + assignableWindow.__PosthogExtensions__ = { + loadSiteApp: jest.fn().mockImplementation((_instance, _url, callback) => { + // Simulate async loading + setTimeout(() => { + const id = _url.split('/').pop() + if (isFunction(assignableWindow[`__$$ph_site_app_${id}_callback`])) { + assignableWindow[`__$$ph_site_app_${id}_callback`]() + } + callback() + }, 0) + }), + } + posthog = { config: { ...defaultConfig }, persistence: new PostHogPersistence(defaultConfig as PostHogConfig), @@ -48,6 +66,12 @@ describe('SiteApps', () => { _hasBootstrappedFeatureFlags: jest.fn(), getGroups: () => ({ organization: '5' }), } as unknown as PostHog + + siteAppsInstance = new SiteApps(posthog) + }) + + afterEach(() => { + jest.clearAllMocks() }) describe('constructor', () => { @@ -58,7 +82,7 @@ describe('SiteApps', () => { advanced_disable_decide: false, } as PostHogConfig - const siteAppsInstance = new SiteApps(posthog) + siteAppsInstance = new SiteApps(posthog) expect(siteAppsInstance.enabled).toBe(true) }) @@ -70,7 +94,7 @@ describe('SiteApps', () => { advanced_disable_decide: false, } as PostHogConfig - const siteAppsInstance = new SiteApps(posthog) + siteAppsInstance = new SiteApps(posthog) expect(siteAppsInstance.enabled).toBe(false) }) @@ -82,14 +106,12 @@ describe('SiteApps', () => { advanced_disable_decide: true, } as PostHogConfig - const siteAppsInstance = new SiteApps(posthog) + siteAppsInstance = new SiteApps(posthog) expect(siteAppsInstance.enabled).toBe(false) }) it('initializes missedInvocations, loaded, appsLoading correctly', () => { - const siteAppsInstance = new SiteApps(posthog) - expect(siteAppsInstance.missedInvocations).toEqual([]) expect(siteAppsInstance.loaded).toBe(false) expect(siteAppsInstance.appsLoading).toEqual(new Set()) @@ -98,7 +120,6 @@ describe('SiteApps', () => { describe('init', () => { it('adds eventCollector as a capture hook', () => { - const siteAppsInstance = new SiteApps(posthog) siteAppsInstance.init() expect(posthog._addCaptureHook).toHaveBeenCalledWith(expect.any(Function)) @@ -106,12 +127,6 @@ describe('SiteApps', () => { }) describe('eventCollector', () => { - let siteAppsInstance: SiteApps - - beforeEach(() => { - siteAppsInstance = new SiteApps(posthog) - }) - it('does nothing if enabled is false', () => { siteAppsInstance.enabled = false siteAppsInstance.eventCollector('event_name', {} as CaptureResult) @@ -137,7 +152,7 @@ describe('SiteApps', () => { siteAppsInstance.enabled = true siteAppsInstance.loaded = false - siteAppsInstance.missedInvocations = new Array(1000).fill({}).map((_, index) => ({ index })) + siteAppsInstance.missedInvocations = new Array(1000).fill({}) const eventPayload = { event: 'test_event', properties: { prop1: 'value1' } } as CaptureResult @@ -146,19 +161,12 @@ describe('SiteApps', () => { siteAppsInstance.eventCollector('test_event', eventPayload) expect(siteAppsInstance.missedInvocations.length).toBe(991) - // Ensure that the first 10 events were trimmed - expect(siteAppsInstance.missedInvocations[0]).toEqual({ index: 10 }) + expect(siteAppsInstance.missedInvocations[0]).toEqual({}) expect(siteAppsInstance.missedInvocations[990]).toEqual({ some: 'globals' }) }) }) describe('globalsForEvent', () => { - let siteAppsInstance: SiteApps - - beforeEach(() => { - siteAppsInstance = new SiteApps(posthog) - }) - it('throws an error if event is undefined', () => { expect(() => siteAppsInstance.globalsForEvent(undefined as any)).toThrow('Event payload is required') }) @@ -216,39 +224,7 @@ describe('SiteApps', () => { }) }) - describe('loadSiteApp', () => { - let siteAppsInstance: SiteApps - let loadScript: jest.Mock - - beforeEach(() => { - siteAppsInstance = new SiteApps(posthog) - // eslint-disable-next-line @typescript-eslint/no-require-imports - loadScript = require('../entrypoints/external-scripts-loader').loadScript as jest.Mock - loadScript.mockClear() - }) - - it('calls loadScript with correct parameters', () => { - const callback = jest.fn() - jest.spyOn(posthog.requestRouter, 'endpointFor').mockReturnValue('https://test.com/script.js') - - siteAppsInstance.loadSiteApp(posthog, '/site_app/1', callback) - - expect(posthog.requestRouter.endpointFor).toHaveBeenCalledWith('api', '/site_app/1') - expect(loadScript).toHaveBeenCalledWith(posthog, 'https://test.com/script.js', callback) - }) - }) - describe('afterDecideResponse', () => { - let siteAppsInstance: SiteApps - - beforeEach(() => { - siteAppsInstance = new SiteApps(posthog) - jest.spyOn(siteAppsInstance, 'loadSiteApp').mockImplementation((posthog, url, callback) => { - // Simulate loading - callback() - }) - }) - it('sets loaded to true and enabled to false when response is undefined', () => { siteAppsInstance.afterDecideResponse(undefined) @@ -256,7 +232,7 @@ describe('SiteApps', () => { expect(siteAppsInstance.enabled).toBe(false) }) - it('loads site apps when enabled and opt_in_site_apps is true', () => { + it('loads site apps when enabled and opt_in_site_apps is true', (done) => { posthog.config.opt_in_site_apps = true siteAppsInstance.enabled = true const response = { @@ -270,7 +246,14 @@ describe('SiteApps', () => { expect(siteAppsInstance.appsLoading.size).toBe(2) expect(siteAppsInstance.loaded).toBe(false) - expect(siteAppsInstance.loadSiteApp).toHaveBeenCalledTimes(2) + + // Wait for the simulated async loading to complete + setTimeout(() => { + expect(assignableWindow.__PosthogExtensions__?.loadSiteApp).toHaveBeenCalledTimes(2) + expect(siteAppsInstance.appsLoading.size).toBe(0) + expect(siteAppsInstance.loaded).toBe(true) + done() + }, 10) }) it('does not load site apps when enabled is false', () => { @@ -284,10 +267,10 @@ describe('SiteApps', () => { expect(siteAppsInstance.loaded).toBe(true) expect(siteAppsInstance.enabled).toBe(false) - expect(siteAppsInstance.loadSiteApp).not.toHaveBeenCalled() + expect(assignableWindow.__PosthogExtensions__?.loadSiteApp).not.toHaveBeenCalled() }) - it('clears missedInvocations when all apps are loaded', () => { + it('clears missedInvocations when all apps are loaded', (done) => { posthog.config.opt_in_site_apps = true siteAppsInstance.enabled = true siteAppsInstance.missedInvocations = [{ some: 'data' }] @@ -297,16 +280,12 @@ describe('SiteApps', () => { siteAppsInstance.afterDecideResponse(response) - // Simulate app loaded - siteAppsInstance.appsLoading.delete('1') - // Manually invoke the checkIfAllLoaded function - if (siteAppsInstance.appsLoading.size === 0) { - siteAppsInstance.loaded = true - siteAppsInstance.missedInvocations = [] - } - - expect(siteAppsInstance.loaded).toBe(true) - expect(siteAppsInstance.missedInvocations).toEqual([]) + // Wait for the simulated async loading to complete + setTimeout(() => { + expect(siteAppsInstance.loaded).toBe(true) + expect(siteAppsInstance.missedInvocations).toEqual([]) + done() + }, 10) }) it('sets assignableWindow properties for each site app', () => { @@ -321,6 +300,11 @@ describe('SiteApps', () => { expect(assignableWindow['__$$ph_site_app_1_posthog']).toBe(posthog) expect(typeof assignableWindow['__$$ph_site_app_1_missed_invocations']).toBe('function') expect(typeof assignableWindow['__$$ph_site_app_1_callback']).toBe('function') + expect(assignableWindow.__PosthogExtensions__?.loadSiteApp).toHaveBeenCalledWith( + posthog, + '/site_app/1', + expect.any(Function) + ) }) it('logs error if site apps are disabled but response contains site apps', () => { @@ -330,12 +314,25 @@ describe('SiteApps', () => { siteApps: [{ id: '1', url: '/site_app/1' }], } as DecideResponse - jest.spyOn(logger, 'error').mockImplementation() siteAppsInstance.afterDecideResponse(response) expect(logger.error).toHaveBeenCalledWith( 'PostHog site apps are disabled. Enable the "opt_in_site_apps" config to proceed.' ) + expect(siteAppsInstance.loaded).toBe(true) + }) + + it('sets loaded to true if response.siteApps is empty', () => { + siteAppsInstance.enabled = true + posthog.config.opt_in_site_apps = true + const response = { + siteApps: [], + } as DecideResponse + + siteAppsInstance.afterDecideResponse(response) + + expect(siteAppsInstance.loaded).toBe(true) + expect(siteAppsInstance.enabled).toBe(false) }) }) }) From c942a923d044a2a691c4030b55c129f0a5209f69 Mon Sep 17 00:00:00 2001 From: MarconLP <13001502+MarconLP@users.noreply.github.com> Date: Tue, 26 Nov 2024 13:27:23 +0100 Subject: [PATCH 6/7] only load site apps when consent is given --- src/site-apps.ts | 13 +++++++++---- 1 file changed, 9 insertions(+), 4 deletions(-) diff --git a/src/site-apps.ts b/src/site-apps.ts index 584679275..c0aba3289 100644 --- a/src/site-apps.ts +++ b/src/site-apps.ts @@ -14,7 +14,10 @@ export class SiteApps { constructor(instance: PostHog) { this.instance = instance // can't use if site apps are disabled, or if we're not asking /decide for site apps - this.enabled = !!this.instance.config.opt_in_site_apps && !this.instance.config.advanced_disable_decide + this.enabled = + this.instance.config.opt_in_site_apps && + !this.instance.config.advanced_disable_decide && + this.instance.consent.isOptedIn() // events captured between loading posthog-js and the site app; up to 1000 events this.missedInvocations = [] // capture events until loaded @@ -76,7 +79,7 @@ export class SiteApps { afterDecideResponse(response?: DecideResponse): void { if (isArray(response?.siteApps) && response.siteApps.length > 0) { - if (this.enabled && this.instance.config.opt_in_site_apps) { + if (this.enabled) { const checkIfAllLoaded = () => { // Stop collecting events once all site apps are loaded if (this.appsLoading.size === 0) { @@ -100,6 +103,10 @@ export class SiteApps { } }) } + } else if (!this.instance.consent.isOptedIn()) { + logger.warn( + 'PostHog site apps are disabled. User has opted out. Call "posthog.opt_in_capturing()" to proceed.' + ) } else if (response['siteApps'].length > 0) { logger.error('PostHog site apps are disabled. Enable the "opt_in_site_apps" config to proceed.') this.loaded = true @@ -111,6 +118,4 @@ export class SiteApps { this.enabled = false } } - - // TODO: opting out of stuff should disable this } From b75f9e745b4b80e5d91a342195ff52d9d60870d2 Mon Sep 17 00:00:00 2001 From: MarconLP <13001502+MarconLP@users.noreply.github.com> Date: Wed, 27 Nov 2024 10:21:38 +0100 Subject: [PATCH 7/7] add disable logic --- playground/nextjs/src/posthog.ts | 2 ++ src/posthog-core.ts | 4 ++- src/site-apps.ts | 45 ++++++++++++++++++-------------- src/types.ts | 3 ++- 4 files changed, 32 insertions(+), 22 deletions(-) diff --git a/playground/nextjs/src/posthog.ts b/playground/nextjs/src/posthog.ts index 6a4062a34..342904b9a 100644 --- a/playground/nextjs/src/posthog.ts +++ b/playground/nextjs/src/posthog.ts @@ -28,6 +28,7 @@ export const configForConsent = (): Partial => { return { persistence: consentGiven ? 'localStorage+cookie' : 'memory', disable_surveys: !consentGiven, + disable_site_apps_destinations: !consentGiven, autocapture: consentGiven, disable_session_recording: !consentGiven, } @@ -49,6 +50,7 @@ if (typeof window !== 'undefined') { session_recording: { recordCrossOriginIframes: true, }, + opt_in_site_apps: true, debug: true, disable_web_experiments: false, scroll_root_selector: ['#scroll_element', 'html'], diff --git a/src/posthog-core.ts b/src/posthog-core.ts index 49e21a25f..1e2617f63 100644 --- a/src/posthog-core.ts +++ b/src/posthog-core.ts @@ -145,6 +145,7 @@ export const defaultConfig = (): PostHogConfig => ({ disable_persistence: false, disable_web_experiments: true, // disabled in beta. disable_surveys: false, + disable_site_apps_destinations: false, enable_recording_console_log: undefined, // When undefined, it falls back to the server-side setting secure_cookie: window?.location?.protocol === 'https:', ip: true, @@ -308,6 +309,7 @@ export class PostHog { this.scrollManager = new ScrollManager(this) this.pageViewManager = new PageViewManager(this) this.surveys = new PostHogSurveys(this) + this.siteApps = new SiteApps(this) this.experiments = new WebExperiments(this) this.exceptions = new PostHogExceptions(this) this.rateLimiter = new RateLimiter(this) @@ -434,7 +436,6 @@ export class PostHog { new TracingHeaders(this).startIfEnabledOrStop() - this.siteApps = new SiteApps(this) this.siteApps?.init() this.sessionRecording = new SessionRecording(this) @@ -1828,6 +1829,7 @@ export class PostHog { this.autocapture?.startIfEnabled() this.heatmaps?.startIfEnabled() this.surveys.loadIfEnabled() + this.siteApps?.loadIfEnabled() this._sync_opt_out_with_persistence() } } diff --git a/src/site-apps.ts b/src/site-apps.ts index c0aba3289..394cccf65 100644 --- a/src/site-apps.ts +++ b/src/site-apps.ts @@ -5,19 +5,12 @@ import { logger } from './utils/logger' import { isArray } from './utils/type-utils' export class SiteApps { - instance: PostHog - enabled: boolean + private _decideServerResponse?: DecideResponse['siteApps'] missedInvocations: Record[] loaded: boolean appsLoading: Set - constructor(instance: PostHog) { - this.instance = instance - // can't use if site apps are disabled, or if we're not asking /decide for site apps - this.enabled = - this.instance.config.opt_in_site_apps && - !this.instance.config.advanced_disable_decide && - this.instance.consent.isOptedIn() + constructor(private readonly instance: PostHog) { // events captured between loading posthog-js and the site app; up to 1000 events this.missedInvocations = [] // capture events until loaded @@ -26,7 +19,9 @@ export class SiteApps { } eventCollector(_eventName: string, eventPayload?: CaptureResult | undefined) { - if (!this.enabled) { + // can't use if site apps are disabled, or if we're not asking /decide for site apps + const enabled = this.instance.config.opt_in_site_apps && !this.instance.config.advanced_disable_decide + if (!enabled) { return } if (!this.loaded && eventPayload) { @@ -77,9 +72,15 @@ export class SiteApps { return globals } - afterDecideResponse(response?: DecideResponse): void { - if (isArray(response?.siteApps) && response.siteApps.length > 0) { - if (this.enabled) { + loadIfEnabled() { + if ( + this._decideServerResponse && + isArray(this._decideServerResponse) && + this._decideServerResponse.length > 0 + ) { + // can't use if site apps are disabled, or if we're not asking /decide for site apps + const enabled = this.instance.config.opt_in_site_apps && !this.instance.config.advanced_disable_decide + if (enabled) { const checkIfAllLoaded = () => { // Stop collecting events once all site apps are loaded if (this.appsLoading.size === 0) { @@ -87,7 +88,11 @@ export class SiteApps { this.missedInvocations = [] } } - for (const { id, url } of response['siteApps']) { + for (const { id, type, url } of this._decideServerResponse) { + if (this.instance.config.disable_site_apps_destinations && type === 'site_destination') continue + // if the site app is already loaded, skip it + // eslint-disable-next-line no-restricted-globals + if (`__$$ph_site_app_${id}_posthog` in window) continue this.appsLoading.add(id) assignableWindow[`__$$ph_site_app_${id}_posthog`] = this.instance assignableWindow[`__$$ph_site_app_${id}_missed_invocations`] = () => this.missedInvocations @@ -103,11 +108,7 @@ export class SiteApps { } }) } - } else if (!this.instance.consent.isOptedIn()) { - logger.warn( - 'PostHog site apps are disabled. User has opted out. Call "posthog.opt_in_capturing()" to proceed.' - ) - } else if (response['siteApps'].length > 0) { + } else if (this._decideServerResponse.length > 0) { logger.error('PostHog site apps are disabled. Enable the "opt_in_site_apps" config to proceed.') this.loaded = true } else { @@ -115,7 +116,11 @@ export class SiteApps { } } else { this.loaded = true - this.enabled = false } } + + afterDecideResponse(response: DecideResponse): void { + this._decideServerResponse = response['siteApps'] + this.loadIfEnabled() + } } diff --git a/src/types.ts b/src/types.ts index 3044e10df..24a379791 100644 --- a/src/types.ts +++ b/src/types.ts @@ -248,6 +248,7 @@ export interface PostHogConfig { /** @deprecated - use `disable_persistence` instead */ disable_cookie?: boolean disable_surveys: boolean + disable_site_apps_destinations: boolean disable_web_experiments: boolean /** If set, posthog-js will never load external scripts such as those needed for Session Replay or Surveys. */ disable_external_dependency_loading?: boolean @@ -523,7 +524,7 @@ export interface DecideResponse { editorParams?: ToolbarParams /** @deprecated, renamed to toolbarParams, still present on older API responses */ toolbarVersion: 'toolbar' /** @deprecated, moved to toolbarParams */ isAuthenticated: boolean - siteApps: { id: string; url: string }[] + siteApps: { id: string; type: string; url: string }[] heatmaps?: boolean defaultIdentifiedOnly?: boolean captureDeadClicks?: boolean