Skip to content

Commit

Permalink
Merge branch 'main' into dn-feat/add-recording-status-prop
Browse files Browse the repository at this point in the history
  • Loading branch information
daibhin committed Nov 29, 2024
2 parents 04db518 + 60c4178 commit 1e20b09
Show file tree
Hide file tree
Showing 35 changed files with 1,043 additions and 169 deletions.
47 changes: 47 additions & 0 deletions CHANGELOG.md
Original file line number Diff line number Diff line change
@@ -1,3 +1,50 @@
## 1.193.1 - 2024-11-28

- fix: zone detection (#1570)

## 1.193.0 - 2024-11-28

- feat: allow decide to provide script name for recorder (#1509)

## 1.192.1 - 2024-11-28



## 1.192.0 - 2024-11-28

- feat: Start tracking timezone offset and language prefix (#1568)

## 1.191.0 - 2024-11-28

- feat: different default and max idle period (#1558)

## 1.190.2 - 2024-11-27

- fix: patch to angular detection in rrweb (#1566)
- chore: fix changelog (#1564)

## 1.190.1 - 2024-11-27

- fix: catch errors detecting dialog state when recording (#1562)

## 1.190.0 - 2024-11-27

- feat: Add initial person info to cookie when using localPlusCookieStore (#1559)

## 1.189.1 - 2024-11-27

- feat: hog site functions (#1546)

## 1.189.0 - 2024-11-26

- feat: Add better npm import, and script entrypoint for customizations (#1550)

## 1.188.1 - 2024-11-26

- fix: safari support requires we don't use (#1553)
- fix: endless capturing /s/ (#1551)
- chore: make platform more specific (#1549)

## 1.188.0 - 2024-11-22

- fix(surveys): Process feature_flag_keys on Survey object (#1548)
Expand Down
1 change: 1 addition & 0 deletions cypress/e2e/session-recording.cy.ts
Original file line number Diff line number Diff line change
Expand Up @@ -523,6 +523,7 @@ describe('Session recording', () => {
capturedSnapshot['properties']['$snapshot_data'][3],
capturedSnapshot['properties']['$snapshot_data'][4],
])

expectPageViewCustomEvent(customEvents[0])
expectPostHogConfigCustomEvent(customEvents[1])
expectSessionOptionsCustomEvent(customEvents[2])
Expand Down
2 changes: 1 addition & 1 deletion package.json
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
{
"name": "posthog-js",
"version": "1.188.0",
"version": "1.193.1",
"description": "Posthog-js allows you to automatically capture usage and send events to PostHog.",
"repository": "https://github.com/PostHog/posthog-js",
"author": "[email protected]",
Expand Down
56 changes: 43 additions & 13 deletions patches/@[email protected]
Original file line number Diff line number Diff line change
@@ -1,18 +1,18 @@
diff --git a/dist/record.js b/dist/record.js
index 46ec389fefb698243008b39db65470dbdf0a3857..891e1cf6439630d19e9b745ff428db438943a0b2 100644
index 46ec389fefb698243008b39db65470dbdf0a3857..a18724d8b6ba43a30935daf257127fbb0c898541 100644
--- a/dist/record.js
+++ b/dist/record.js
@@ -26,6 +26,14 @@ const testableMethods$1 = {
Element: [],
MutationObserver: ["constructor"]
};
+const isFunction = (x) => typeof x === 'function';
+const isAngularZonePatchedFunction = (x) => {
+ if (!isFunction(x)) {
+ return false;
+
+const isAngularZonePresent = () => {
+ try {
+ return !!globalThis.Zone
+ } catch {
+ return false
+ }
+ const prototypeKeys = Object.getOwnPropertyNames(x.prototype || {});
+ return prototypeKeys.some((key) => key.indexOf('__zone'));
+}
const untaintedBasePrototype$1 = {};
function getUntaintedPrototype$1(key) {
Expand All @@ -22,10 +22,23 @@ index 46ec389fefb698243008b39db65470dbdf0a3857..891e1cf6439630d19e9b745ff428db43
)
);
- if (isUntaintedAccessors && isUntaintedMethods) {
+ if (isUntaintedAccessors && isUntaintedMethods && !isAngularZonePatchedFunction(defaultObj)) {
+ if (isUntaintedAccessors && isUntaintedMethods && !isAngularZonePresent()) {
untaintedBasePrototype$1[key] = defaultObj.prototype;
return defaultObj.prototype;
}
@@ -65,10 +73,10 @@ function getUntaintedPrototype$1(key) {
if (!win) return defaultObj.prototype;
const untaintedObject = win[key].prototype;
document.body.removeChild(iframeEl);
- if (!untaintedObject) return defaultPrototype;
+ if (!untaintedObject) return defaultObj.prototype;
return untaintedBasePrototype$1[key] = untaintedObject;
} catch {
- return defaultPrototype;
+ return defaultObj.prototype;
}
}
const untaintedAccessorCache$1 = {};
@@ -246,6 +254,9 @@ function isCSSImportRule(rule2) {
function isCSSStyleRule(rule2) {
return "selectorText" in rule2;
Expand Down Expand Up @@ -54,7 +67,24 @@ index 46ec389fefb698243008b39db65470dbdf0a3857..891e1cf6439630d19e9b745ff428db43
let cssText = null;
if (stylesheet) {
cssText = stringifyStylesheet(stylesheet);
@@ -1116,7300 +1132,227 @@ function serializeNodeWithId(n2, options) {
@@ -855,7 +871,15 @@ function serializeElementNode(n2, options) {
}
}
if (tagName === "dialog" && n2.open) {
- attributes.rr_open_mode = n2.matches("dialog:modal") ? "modal" : "non-modal";
+ try {
+ attributes.rr_open_mode = n2.matches("dialog:modal") ? "modal" : "non-modal";
+ } catch {
+ // likely this is safari not able to deal with the `:modal` selector
+ // we can't detect whether the dialog is modal or non-modal open, so have to guess
+ // hopefully this is only safari 15.4 and 15.5
+ attributes.rr_open_mode = "modal"
+ attributes.ph_rr_could_not_detect_modal = true
+ }
}
if (tagName === "canvas" && recordCanvas) {
if (n2.__context === "2d") {
@@ -1116,7300 +1140,227 @@ function serializeNodeWithId(n2, options) {
keepIframeSrcFn
};
if (serializedNode.type === NodeType$2.Element && serializedNode.tagName === "textarea" && serializedNode.attributes.value !== void 0) ;
Expand Down Expand Up @@ -7564,16 +7594,16 @@ index 46ec389fefb698243008b39db65470dbdf0a3857..891e1cf6439630d19e9b745ff428db43
class BaseRRNode {
// eslint-disable-next-line @typescript-eslint/no-unused-vars, @typescript-eslint/no-explicit-any
constructor(..._args) {
@@ -8507,7 +1450,7 @@ function getUntaintedPrototype(key) {
@@ -8507,7 +1458,7 @@ function getUntaintedPrototype(key) {
}
)
);
- if (isUntaintedAccessors && isUntaintedMethods) {
+ if (isUntaintedAccessors && isUntaintedMethods && !isAngularZonePatchedFunction(defaultObj)) {
+ if (isUntaintedAccessors && isUntaintedMethods && !isAngularZonePresent()) {
untaintedBasePrototype[key] = defaultObj.prototype;
return defaultObj.prototype;
}
@@ -11382,11 +4325,19 @@ class CanvasManager {
@@ -11382,11 +4333,19 @@ class CanvasManager {
let rafId;
const getCanvas = () => {
const matchedCanvas = [];
Expand All @@ -7598,7 +7628,7 @@ index 46ec389fefb698243008b39db65470dbdf0a3857..891e1cf6439630d19e9b745ff428db43
return matchedCanvas;
};
const takeCanvasSnapshots = (timestamp) => {
@@ -11407,13 +4358,20 @@ class CanvasManager {
@@ -11407,13 +4366,20 @@ class CanvasManager {
context.clear(context.COLOR_BUFFER_BIT);
}
}
Expand Down
2 changes: 1 addition & 1 deletion playground/nextjs/pages/index.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -2,7 +2,7 @@
import { useActiveFeatureFlags, usePostHog } from 'posthog-js/react'
import { useEffect, useState } from 'react'
import { cookieConsentGiven, PERSON_PROCESSING_MODE } from '@/src/posthog'
import { setAllPersonProfilePropertiesAsPersonPropertiesForFlags } from 'posthog-js/lib/src/customizations/setAllPersonProfilePropertiesAsPersonPropertiesForFlags'
import { setAllPersonProfilePropertiesAsPersonPropertiesForFlags } from 'posthog-js/lib/src/customizations'
import { STORED_PERSON_PROPERTIES_KEY } from '../../../src/constants'

export default function Home() {
Expand Down
6 changes: 3 additions & 3 deletions pnpm-lock.yaml

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

31 changes: 0 additions & 31 deletions src/__tests__/decide.ts
Original file line number Diff line number Diff line change
@@ -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'
Expand Down Expand Up @@ -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/')
})
})
})
Original file line number Diff line number Diff line change
Expand Up @@ -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',
Expand Down Expand Up @@ -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',
Expand Down Expand Up @@ -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',
Expand Down Expand Up @@ -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',
Expand Down
36 changes: 30 additions & 6 deletions src/__tests__/extensions/replay/config.test.ts
Original file line number Diff line number Diff line change
Expand Up @@ -75,34 +75,58 @@ describe('config', () => {
{
name: 'https://app.posthog.com/api/feature_flag/',
},
undefined,
],
[
{
name: 'https://app.posthog.com/s/',
name: 'https://app.posthog.com/s/?ip=1&ver=123',
},
undefined,
undefined,
],
[
{
name: 'https://app.posthog.com/e/',
name: 'https://app.posthog.com/e/?ip=1&ver=123',
},
undefined,
undefined,
],
[
{
name: 'https://app.posthog.com/i/v0/e/',
name: 'https://app.posthog.com/i/v0/e/?ip=1&ver=123',
},
undefined,
undefined,
],
[
{
// even an imaginary future world of rust session replay capture
name: 'https://app.posthog.com/i/v0/s/',
name: 'https://app.posthog.com/i/v0/s/?ip=1&ver=123',
},
undefined,
undefined,
],
])('ignores ingestion paths', (capturedRequest, expected) => {
const networkOptions = buildNetworkRequestOptions(defaultConfig(), {})
[
{
// using a relative path as a reverse proxy api host
name: 'https://app.posthog.com/ingest/s/?ip=1&ver=123',
},
undefined,
'/ingest',
],
[
{
// using a reverse proxy with a path
name: 'https://app.posthog.com/ingest/s/?ip=1&ver=123',
},
undefined,
'https://app.posthog.com/ingest',
],
])('ignores ingestion paths', (capturedRequest, expected, apiHost?: string) => {
const networkOptions = buildNetworkRequestOptions(
{ ...defaultConfig(), api_host: apiHost || 'https://us.posthog.com' },
{}
)
const x = networkOptions.maskRequestFn!(capturedRequest as CapturedNetworkRequest)
expect(x).toEqual(expected)
})
Expand Down
Loading

0 comments on commit 1e20b09

Please sign in to comment.