From 4e1e418c5c17440d2aded1205446ec19188ac2ee Mon Sep 17 00:00:00 2001 From: Paul D'Ambra Date: Tue, 17 Dec 2024 09:27:25 +0100 Subject: [PATCH 1/9] fix: fewer version warnings (#26942) --- .../versionCheckerLogic.test.ts | 41 +++++++++++-------- .../VersionChecker/versionCheckerLogic.ts | 19 ++++----- 2 files changed, 32 insertions(+), 28 deletions(-) diff --git a/frontend/src/lib/components/VersionChecker/versionCheckerLogic.test.ts b/frontend/src/lib/components/VersionChecker/versionCheckerLogic.test.ts index 103e89b2bfed9..77e7e0e9eb52e 100644 --- a/frontend/src/lib/components/VersionChecker/versionCheckerLogic.test.ts +++ b/frontend/src/lib/components/VersionChecker/versionCheckerLogic.test.ts @@ -73,35 +73,41 @@ describe('versionCheckerLogic', () => { { versionCount: 1, expectation: null }, { versionCount: 11, + expectation: null, + }, + { + versionCount: 51, expectation: { latestUsedVersion: '1.0.0', - latestAvailableVersion: '1.0.10', - numVersionsBehind: 10, - level: 'info', + latestAvailableVersion: '1.0.50', + numVersionsBehind: 50, + level: 'error', }, }, { - versionCount: 15, + minorUsedVersion: 40, + versionCount: 1, expectation: { latestUsedVersion: '1.0.0', - latestAvailableVersion: '1.0.14', - numVersionsBehind: 14, - level: 'info', + latestAvailableVersion: '1.40.0', + numVersionsBehind: 40, + level: 'warning', }, }, { - versionCount: 25, + majorUsedVersion: 2, + versionCount: 1, expectation: { latestUsedVersion: '1.0.0', - latestAvailableVersion: '1.0.24', - numVersionsBehind: 24, - level: 'error', + latestAvailableVersion: '2.0.0', + numVersionsBehind: 1, + level: 'info', }, }, ])('return a version warning if diff is great enough', async (options) => { // TODO: How do we clear the persisted value? const versionsList = Array.from({ length: options.versionCount }, (_, i) => ({ - version: `1.0.${i}`, + version: `${options.majorUsedVersion || 1}.${options.minorUsedVersion || 0}.${i}`, })).reverse() useMockedVersions( @@ -143,13 +149,14 @@ describe('versionCheckerLogic', () => { }, { usedVersions: [ - { version: '1.80.0', timestamp: '2023-01-01T12:00:00Z' }, - { version: '1.83.1-beta', timestamp: '2023-01-01T10:00:00Z' }, - { version: '1.84.0-delta', timestamp: '2023-01-01T08:00:00Z' }, + { version: '1.40.0', timestamp: '2023-01-01T12:00:00Z' }, + { version: '1.41.1-beta', timestamp: '2023-01-01T10:00:00Z' }, + { version: '1.42.0', timestamp: '2023-01-01T08:00:00Z' }, + { version: '1.42.0-delta', timestamp: '2023-01-01T08:00:00Z' }, ], expectation: { - latestUsedVersion: '1.84.0-delta', - numVersionsBehind: 1, + latestUsedVersion: '1.42.0', + numVersionsBehind: 42, latestAvailableVersion: '1.84.0', level: 'warning', }, diff --git a/frontend/src/lib/components/VersionChecker/versionCheckerLogic.ts b/frontend/src/lib/components/VersionChecker/versionCheckerLogic.ts index 7ffecbbf89c82..4c6067adf4afc 100644 --- a/frontend/src/lib/components/VersionChecker/versionCheckerLogic.ts +++ b/frontend/src/lib/components/VersionChecker/versionCheckerLogic.ts @@ -174,6 +174,7 @@ export const versionCheckerLogic = kea([ if (!warning && sdkVersions && latestAvailableVersion) { const diff = diffVersions(latestAvailableVersion, latestUsedVersion) + if (diff && diff.diff > 0) { // there's a difference between the latest used version and the latest available version @@ -188,18 +189,14 @@ export const versionCheckerLogic = kea([ } let level: 'warning' | 'info' | 'error' | undefined - if (diff.kind === 'major' || numVersionsBehind >= 20) { + if (diff.kind === 'major') { + level = 'info' // it is desirable to be on the latest major version, but not critical + } else if (diff.kind === 'minor') { + level = numVersionsBehind >= 40 ? 'warning' : undefined + } + + if (level === undefined && numVersionsBehind >= 50) { level = 'error' - } else if (diff.kind === 'minor' && diff.diff >= 15) { - level = 'warning' - } else if ((diff.kind === 'minor' && diff.diff >= 10) || numVersionsBehind >= 10) { - level = 'info' - } else if (latestUsedVersion.extra) { - // if we have an extra (alpha/beta/rc/etc.) version, we should always show a warning if they aren't on the latest - level = 'warning' - } else { - // don't warn for a small number of patch versions behind - level = undefined } // we check if there is a "latest user version string" to avoid returning odd data in unexpected cases From daa0a27ab741c1c902e0c5e74d9bdd55701a4f9a Mon Sep 17 00:00:00 2001 From: Marius Andra Date: Tue, 17 Dec 2024 09:35:33 +0100 Subject: [PATCH 2/9] feat(cdp): test site function javascript (#26943) Co-authored-by: github-actions <41898282+github-actions[bot]@users.noreply.github.com> --- posthog/cdp/site_functions.py | 4 +- posthog/cdp/test/test_site_functions.py | 73 +++++++++++++++++-- .../test/__snapshots__/test_in_cohort.ambr | 4 +- posthog/models/test/test_remote_config.py | 8 +- 4 files changed, 75 insertions(+), 14 deletions(-) diff --git a/posthog/cdp/site_functions.py b/posthog/cdp/site_functions.py index 690dc136ea577..f6ece18e28792 100644 --- a/posthog/cdp/site_functions.py +++ b/posthog/cdp/site_functions.py @@ -92,7 +92,7 @@ def get_transpiled_function(hog_function: HogFunction) -> str: """ let processEvent = undefined; if ('onEvent' in source) { - processEvent = function processEvent(globals) { + processEvent = function processEvent(globals, posthog) { if (!('onEvent' in source)) { return; }; const inputs = buildInputs(globals); const filterGlobals = { ...globals.groups, ...globals.event, person: globals.person, inputs, pdi: { distinct_id: globals.event.distinct_id, person: globals.person } }; @@ -123,7 +123,7 @@ def get_transpiled_function(hog_function: HogFunction) -> str: } return { - processEvent: processEvent + processEvent: (globals) => processEvent(globals, posthog) } } diff --git a/posthog/cdp/test/test_site_functions.py b/posthog/cdp/test/test_site_functions.py index 9370cb7266740..0b6c0bc1bb8a6 100644 --- a/posthog/cdp/test/test_site_functions.py +++ b/posthog/cdp/test/test_site_functions.py @@ -1,3 +1,4 @@ +import json import subprocess import tempfile from inline_snapshot import snapshot @@ -71,7 +72,7 @@ def test_get_transpiled_function_basic(self): };return exports;})(); let processEvent = undefined; if ('onEvent' in source) { - processEvent = function processEvent(globals) { + processEvent = function processEvent(globals, posthog) { if (!('onEvent' in source)) { return; }; const inputs = buildInputs(globals); const filterGlobals = { ...globals.groups, ...globals.event, person: globals.person, inputs, pdi: { distinct_id: globals.event.distinct_id, person: globals.person } }; @@ -98,7 +99,7 @@ def test_get_transpiled_function_basic(self): } return { - processEvent: processEvent + processEvent: (globals) => processEvent(globals, posthog) } } @@ -129,12 +130,12 @@ def test_get_transpiled_function_with_template_input(self): assert '__getGlobal("person")' in result def test_get_transpiled_function_with_filters(self): - self.hog_function.hog = "export function onEvent(event) { console.log(event.event); }" + self.hog_function.hog = "export function onEvent(globals) { console.log(globals); }" self.hog_function.filters = {"events": [{"id": "$pageview", "name": "$pageview", "type": "events", "order": 0}]} result = self.compile_and_run() - assert "console.log(event.event);" in result + assert "console.log(globals);" in result assert "const filterMatches = " in result assert '__getGlobal("event") == "$pageview"' in result assert "const filterMatches = !!(!!((__getGlobal" in result @@ -249,7 +250,7 @@ def test_get_transpiled_function_with_complex_filters(self): action.steps = [{"event": "$pageview", "url": "https://example.com"}] # type: ignore action.save() - self.hog_function.hog = "export function onEvent(event) { console.log(event.event); }" + self.hog_function.hog = "export function onEvent(globals) { console.log(globals); }" self.hog_function.filters = { "events": [{"id": "$pageview", "name": "$pageview", "type": "events"}], "actions": [{"id": str(action.pk), "name": "Test Action", "type": "actions"}], @@ -258,7 +259,7 @@ def test_get_transpiled_function_with_complex_filters(self): result = self.compile_and_run() - assert "console.log(event.event);" in result + assert "console.log(globals);" in result assert "const filterMatches = " in result assert '__getGlobal("event") == "$pageview"' in result assert "https://example.com" in result @@ -283,3 +284,63 @@ def test_get_transpiled_function_with_mappings(self): assert 'if (!!(!!((__getGlobal("event") == "$autocapture")))) {' in result assert "const newInputs = structuredClone(inputs);" in result assert 'newInputs["greeting"] = concat("Hallo, ", __getProperty' in result + + def test_run_function_onload(self): + self.hog_function.hog = "export function onLoad({ inputs, posthog }) { console.log(inputs.message); }" + self.hog_function.filters = {"events": [{"id": "$pageview", "name": "$pageview", "type": "events", "order": 0}]} + self.hog_function.inputs = {"message": {"value": "Hello World {person.properties.name}"}} + + result = self.compile_and_run() + assert "Hello World" in result + + response = self._execute_javascript( + result + + "().init({ posthog: { get_property: () => ({name: 'Bob'}) }, callback: () => { console.log('Loaded') } })" + ) + assert "Hello World Bob\nLoaded" == response.strip() + + def test_run_function_onevent(self): + self.hog_function.hog = "export function onEvent({ inputs }) { console.log(inputs.message); }" + # self.hog_function.filters = {"events": [{"id": "$pageview", "name": "$pageview", "type": "events", "order": 0}]} + self.hog_function.inputs = {"message": {"value": "Hello World {event.properties.id}"}} + self.hog_function.mappings = [ + { + "inputs": {"greeting": {"value": "Hallo, {person.properties.nonexistent_property}!"}}, + "filters": {"events": [{"id": "$pageview", "name": "$pageview", "type": "events"}]}, + } + ] + + result = self.compile_and_run() + assert "Hello World" in result + + globals = { + "event": {"event": "$pageview", "properties": {"id": "banana"}}, + "groups": {}, + "person": {"properties": {"name": "Bob"}}, + } + response = self._execute_javascript( + result + + "().init({ posthog: { get_property: () => ({name: 'Bob'}) }, callback: () => { console.log('Loaded') } }).processEvent(" + + json.dumps(globals) + + ")" + ) + assert "Loaded\nHello World banana" == response.strip() + + globals = { + "event": {"event": "$autocapture", "properties": {"id": "banana"}}, + "groups": {}, + "person": {"properties": {"name": "Bob"}}, + } + response = self._execute_javascript( + result + + "().init({ posthog: { get_property: () => ({name: 'Bob'}) }, callback: () => { console.log('Loaded') } }).processEvent(" + + json.dumps(globals) + + ")" + ) + assert "Loaded" == response.strip() + + def _execute_javascript(self, js) -> str: + with tempfile.NamedTemporaryFile(delete=False) as f: + f.write(js.encode("utf-8")) + f.flush() + return subprocess.check_output(["node", f.name]).decode("utf-8") diff --git a/posthog/hogql/transforms/test/__snapshots__/test_in_cohort.ambr b/posthog/hogql/transforms/test/__snapshots__/test_in_cohort.ambr index ce7faf7612056..804183be31efe 100644 --- a/posthog/hogql/transforms/test/__snapshots__/test_in_cohort.ambr +++ b/posthog/hogql/transforms/test/__snapshots__/test_in_cohort.ambr @@ -42,7 +42,7 @@ FROM events LEFT JOIN ( SELECT person_id AS cohort_person_id, 1 AS matched, cohort_id FROM static_cohort_people - WHERE in(cohort_id, [8])) AS __in_cohort ON equals(__in_cohort.cohort_person_id, person_id) + WHERE in(cohort_id, [11])) AS __in_cohort ON equals(__in_cohort.cohort_person_id, person_id) WHERE and(1, equals(__in_cohort.matched, 1)) LIMIT 100 ''' @@ -66,7 +66,7 @@ FROM events LEFT JOIN ( SELECT person_id AS cohort_person_id, 1 AS matched, cohort_id FROM static_cohort_people - WHERE in(cohort_id, [9])) AS __in_cohort ON equals(__in_cohort.cohort_person_id, person_id) + WHERE in(cohort_id, [12])) AS __in_cohort ON equals(__in_cohort.cohort_person_id, person_id) WHERE and(1, equals(__in_cohort.matched, 1)) LIMIT 100 ''' diff --git a/posthog/models/test/test_remote_config.py b/posthog/models/test/test_remote_config.py index 7bb985b78de6c..d9565e2422ddb 100644 --- a/posthog/models/test/test_remote_config.py +++ b/posthog/models/test/test_remote_config.py @@ -700,7 +700,7 @@ def test_renders_js_including_site_functions(self): const source = (function () {let exports={};"use strict";;return exports;})(); let processEvent = undefined; if ('onEvent' in source) { - processEvent = function processEvent(globals) { + processEvent = function processEvent(globals, posthog) { if (!('onEvent' in source)) { return; }; const inputs = buildInputs(globals); const filterGlobals = { ...globals.groups, ...globals.event, person: globals.person, inputs, pdi: { distinct_id: globals.event.distinct_id, person: globals.person } }; @@ -727,7 +727,7 @@ def test_renders_js_including_site_functions(self): } return { - processEvent: processEvent + processEvent: (globals) => processEvent(globals, posthog) } } @@ -746,7 +746,7 @@ def test_renders_js_including_site_functions(self): const source = (function () {let exports={};"use strict";;return exports;})(); let processEvent = undefined; if ('onEvent' in source) { - processEvent = function processEvent(globals) { + processEvent = function processEvent(globals, posthog) { if (!('onEvent' in source)) { return; }; const inputs = buildInputs(globals); const filterGlobals = { ...globals.groups, ...globals.event, person: globals.person, inputs, pdi: { distinct_id: globals.event.distinct_id, person: globals.person } }; @@ -773,7 +773,7 @@ def test_renders_js_including_site_functions(self): } return { - processEvent: processEvent + processEvent: (globals) => processEvent(globals, posthog) } } From 257eae155149c17c162a6291f1b8bd0322b18abc Mon Sep 17 00:00:00 2001 From: Marius Andra Date: Tue, 17 Dec 2024 11:14:59 +0100 Subject: [PATCH 3/9] feat(cdp): log filtering errors (#26966) --- plugin-server/src/cdp/cdp-consumers.ts | 27 ++++++++-- plugin-server/src/cdp/hog-executor.ts | 22 +++++--- .../cdp/cdp-processed-events-consumer.test.ts | 51 +++++++++++++++++++ 3 files changed, 89 insertions(+), 11 deletions(-) diff --git a/plugin-server/src/cdp/cdp-consumers.ts b/plugin-server/src/cdp/cdp-consumers.ts index dbbd163c72ae8..f738b559a0523 100644 --- a/plugin-server/src/cdp/cdp-consumers.ts +++ b/plugin-server/src/cdp/cdp-consumers.ts @@ -28,7 +28,7 @@ import { createKafkaProducerWrapper } from '../utils/db/hub' import { KafkaProducerWrapper } from '../utils/db/kafka-producer-wrapper' import { safeClickhouseString } from '../utils/db/utils' import { status } from '../utils/status' -import { castTimestampOrNow } from '../utils/utils' +import { castTimestampOrNow, UUIDT } from '../utils/utils' import { RustyHook } from '../worker/rusty-hook' import { FetchExecutor } from './fetch-executor' import { GroupsManager } from './groups-manager' @@ -43,7 +43,9 @@ import { HogFunctionInvocationResult, HogFunctionInvocationSerialized, HogFunctionInvocationSerializedCompressed, + HogFunctionLogEntrySerialized, HogFunctionMessageToProduce, + HogFunctionType, HogHooksFetchResponse, } from './types' import { @@ -199,6 +201,24 @@ abstract class CdpConsumerBase { }) } + protected logFilteringError(item: HogFunctionType, error: string) { + const logEntry: HogFunctionLogEntrySerialized = { + team_id: item.team_id, + log_source: 'hog_function', + log_source_id: item.id, + instance_id: new UUIDT().toString(), // random UUID, like it would be for an invocation + timestamp: castTimestampOrNow(null, TimestampFormat.ClickHouse), + level: 'error', + message: error, + } + + this.messagesToProduce.push({ + topic: KAFKA_LOG_ENTRIES, + value: logEntry, + key: logEntry.instance_id, + }) + } + // NOTE: These will be removed once we are only on Cyclotron protected async queueInvocationsToKafka(invocation: HogFunctionInvocation[]) { await Promise.all( @@ -479,7 +499,7 @@ export class CdpProcessedEventsConsumer extends CdpConsumerBase { }) ) - erroredFunctions.forEach((item) => + erroredFunctions.forEach(([item, error]) => { this.produceAppMetric({ team_id: item.team_id, app_source_id: item.id, @@ -487,7 +507,8 @@ export class CdpProcessedEventsConsumer extends CdpConsumerBase { metric_name: 'filtering_failed', count: 1, }) - ) + this.logFilteringError(item, error) + }) }) const states = await this.hogWatcher.getStates(possibleInvocations.map((x) => x.hogFunction.id)) diff --git a/plugin-server/src/cdp/hog-executor.ts b/plugin-server/src/cdp/hog-executor.ts index 15e147f022b7f..631f8c8438444 100644 --- a/plugin-server/src/cdp/hog-executor.ts +++ b/plugin-server/src/cdp/hog-executor.ts @@ -110,17 +110,17 @@ export class HogExecutor { this.telemetryMatcher = buildIntegerMatcher(this.hub.CDP_HOG_FILTERS_TELEMETRY_TEAMS, true) } - findMatchingFunctions(event: HogFunctionInvocationGlobals): { + findMatchingFunctions(globals: HogFunctionInvocationGlobals): { matchingFunctions: HogFunctionType[] nonMatchingFunctions: HogFunctionType[] - erroredFunctions: HogFunctionType[] + erroredFunctions: [HogFunctionType, string][] } { - const allFunctionsForTeam = this.hogFunctionManager.getTeamHogDestinations(event.project.id) - const filtersGlobals = convertToHogFunctionFilterGlobal(event) + const allFunctionsForTeam = this.hogFunctionManager.getTeamHogDestinations(globals.project.id) + const filtersGlobals = convertToHogFunctionFilterGlobal(globals) const nonMatchingFunctions: HogFunctionType[] = [] const matchingFunctions: HogFunctionType[] = [] - const erroredFunctions: HogFunctionType[] = [] + const erroredFunctions: [HogFunctionType, string][] = [] // Filter all functions based on the invocation allFunctionsForTeam.forEach((hogFunction) => { @@ -143,7 +143,10 @@ export class HogExecutor { error: filterResult.error.message, result: filterResult, }) - erroredFunctions.push(hogFunction) + erroredFunctions.push([ + hogFunction, + `Error filtering event ${globals.event.uuid}: ${filterResult.error.message}`, + ]) return } } catch (error) { @@ -153,7 +156,10 @@ export class HogExecutor { teamId: hogFunction.team_id, error: error.message, }) - erroredFunctions.push(hogFunction) + erroredFunctions.push([ + hogFunction, + `Error filtering event ${globals.event.uuid}: ${error.message}`, + ]) return } finally { const duration = performance.now() - start @@ -165,7 +171,7 @@ export class HogExecutor { hogFunctionName: hogFunction.name, teamId: hogFunction.team_id, duration, - eventId: event.event.uuid, + eventId: globals.event.uuid, }) } } diff --git a/plugin-server/tests/cdp/cdp-processed-events-consumer.test.ts b/plugin-server/tests/cdp/cdp-processed-events-consumer.test.ts index 4bd6eb339c5cf..c559a4240fca4 100644 --- a/plugin-server/tests/cdp/cdp-processed-events-consumer.test.ts +++ b/plugin-server/tests/cdp/cdp-processed-events-consumer.test.ts @@ -333,5 +333,56 @@ describe('CDP Processed Events Consumer', () => { ]) }) }) + + describe('filtering errors', () => { + let globals: HogFunctionInvocationGlobals + + beforeEach(() => { + globals = createHogExecutionGlobals({ + project: { + id: team.id, + } as any, + event: { + uuid: 'b3a1fe86-b10c-43cc-acaf-d208977608d0', + event: '$pageview', + properties: { + $current_url: 'https://posthog.com', + $lib_version: '1.0.0', + }, + } as any, + }) + }) + + it('should filter out functions that error while filtering', async () => { + const erroringFunction = await insertHogFunction({ + ...HOG_EXAMPLES.input_printer, + ...HOG_INPUTS_EXAMPLES.secret_inputs, + ...HOG_FILTERS_EXAMPLES.broken_filters, + }) + await processor.processBatch([globals]) + expect(decodeAllKafkaMessages()).toMatchObject([ + { + key: expect.any(String), + topic: 'clickhouse_app_metrics2_test', + value: { + app_source: 'hog_function', + app_source_id: erroringFunction.id, + count: 1, + metric_kind: 'other', + metric_name: 'filtering_failed', + team_id: 2, + timestamp: expect.any(String), + }, + }, + { + topic: 'log_entries_test', + value: { + message: + 'Error filtering event b3a1fe86-b10c-43cc-acaf-d208977608d0: Invalid HogQL bytecode, stack is empty, can not pop', + }, + }, + ]) + }) + }) }) }) From b131457da5712de82ff40e7cb4da00206b894fd8 Mon Sep 17 00:00:00 2001 From: Marius Andra Date: Tue, 17 Dec 2024 11:23:05 +0100 Subject: [PATCH 4/9] fix(cdp): batch exports only in destinations list (#26940) --- .../src/scenes/pipeline/destinations/destinationsLogic.tsx | 6 ++++-- 1 file changed, 4 insertions(+), 2 deletions(-) diff --git a/frontend/src/scenes/pipeline/destinations/destinationsLogic.tsx b/frontend/src/scenes/pipeline/destinations/destinationsLogic.tsx index 42f8112f99b6c..ea6bf3346687c 100644 --- a/frontend/src/scenes/pipeline/destinations/destinationsLogic.tsx +++ b/frontend/src/scenes/pipeline/destinations/destinationsLogic.tsx @@ -338,10 +338,12 @@ export const pipelineDestinationsLogic = kea([ }, })), - afterMount(({ actions }) => { + afterMount(({ actions, props }) => { actions.loadPlugins() actions.loadPluginConfigs() - actions.loadBatchExports() + if (props.types.includes('destination')) { + actions.loadBatchExports() + } actions.loadHogFunctions() }), ]) From 14578b02578bedef7f5bd4fa1ed6f9246cb740e8 Mon Sep 17 00:00:00 2001 From: Rafael Audibert <32079912+rafaeelaudibert@users.noreply.github.com> Date: Tue, 17 Dec 2024 07:42:43 -0300 Subject: [PATCH 5/9] feat: Mention `pdi` table on hogql editor (#26938) --- ...nents-hogqleditor--hog-ql-editor--dark.png | Bin 26003 -> 28224 bytes ...ents-hogqleditor--hog-ql-editor--light.png | Bin 29161 -> 28660 bytes ...components-hogqleditor--no-value--dark.png | Bin 17422 -> 18348 bytes ...omponents-hogqleditor--no-value--light.png | Bin 20100 -> 18861 bytes ...value-person-properties-disabled--dark.png | Bin 17422 -> 18348 bytes ...alue-person-properties-disabled--light.png | Bin 20100 -> 18861 bytes .../components/HogQLEditor/HogQLEditor.tsx | 2 +- 7 files changed, 1 insertion(+), 1 deletion(-) diff --git a/frontend/__snapshots__/components-hogqleditor--hog-ql-editor--dark.png b/frontend/__snapshots__/components-hogqleditor--hog-ql-editor--dark.png index 2caeb97213e25d0c5790e9682ee0bbc2f73ea602..25a32339e8edb042393bcdc164d8c4be9ee3f63e 100644 GIT binary patch literal 28224 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 literal 26003 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 z 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Ǐ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 W-hFf2<|qG4RY*x*4Gfnxd-FdfrB^io diff --git a/frontend/__snapshots__/components-hogqleditor--hog-ql-editor--light.png b/frontend/__snapshots__/components-hogqleditor--hog-ql-editor--light.png index ac6439a3b0f62122e9e3d4b9bf5e7bf788f2a209..513b3ed148bb76244e52709afea50b836ce01d9c 100644 GIT binary patch literal 28660 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 diff --git a/frontend/__snapshots__/components-hogqleditor--no-value--dark.png b/frontend/__snapshots__/components-hogqleditor--no-value--dark.png index 1c538dbd29f445b2576cd2f098cd13ea5d7f0d85..5ceff04a1d206a1eab7721b68365b1c811ec92de 100644 GIT binary patch literal 18348 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 diff --git a/frontend/__snapshots__/components-hogqleditor--no-value--light.png b/frontend/__snapshots__/components-hogqleditor--no-value--light.png index 401a4e98e08ce507a84c65321f8a4a219dafd4cf..2038e29ae408704575f06be5f54147232215e4b2 100644 GIT binary patch literal 18861 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 diff --git a/frontend/__snapshots__/components-hogqleditor--no-value-person-properties-disabled--light.png b/frontend/__snapshots__/components-hogqleditor--no-value-person-properties-disabled--light.png index 401a4e98e08ce507a84c65321f8a4a219dafd4cf..2038e29ae408704575f06be5f54147232215e4b2 100644 GIT binary patch literal 18861 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')\n- is_identified ? 'user' : 'anon'" - : "Enter HogQL Expression, such as:\n- properties.$current_url\n- person.properties.$geoip_country_name\n- toInt(properties.`Long Field Name`) * 10\n- concat(event, ' ', distinct_id)\n- if(1 < 2, 'small', 'large')")} + : "Enter HogQL Expression, such as:\n- properties.$current_url\n- person.properties.$geoip_country_name\n- pdi.person.properties.email\n- toInt(properties.`Long Field Name`) * 10\n- concat(event, ' ', distinct_id)\n- if(1 < 2, 'small', 'large')")} Date: Tue, 17 Dec 2024 11:52:37 +0100 Subject: [PATCH 6/9] fix: Modify remote config to scope by token (#26964) --- posthog/api/test/test_remote_config.py | 4 +-- posthog/models/remote_config.py | 7 ++-- posthog/models/test/test_remote_config.py | 39 ++++++++++++++++------- 3 files changed, 34 insertions(+), 16 deletions(-) diff --git a/posthog/api/test/test_remote_config.py b/posthog/api/test/test_remote_config.py index c5fb3a53a1173..637030f5fe474 100644 --- a/posthog/api/test/test_remote_config.py +++ b/posthog/api/test/test_remote_config.py @@ -111,7 +111,7 @@ def test_valid_config_js(self): assert response.headers["Content-Type"] == "application/javascript" assert response.content == snapshot( - b'(function() {\n window._POSTHOG_CONFIG = {"token": "token123", "supportedCompression": ["gzip", "gzip-js"], "hasFeatureFlags": false, "captureDeadClicks": false, "capturePerformance": {"network_timing": true, "web_vitals": false, "web_vitals_allowed_metrics": null}, "autocapture_opt_out": false, "autocaptureExceptions": false, "analytics": {"endpoint": "/i/v0/e/"}, "elementsChainAsString": true, "sessionRecording": {"endpoint": "/s/", "consoleLogRecordingEnabled": true, "recorderVersion": "v2", "sampleRate": null, "minimumDurationMilliseconds": null, "linkedFlag": null, "networkPayloadCapture": null, "urlTriggers": [], "urlBlocklist": [], "eventTriggers": [], "scriptConfig": null}, "heatmaps": false, "surveys": [], "defaultIdentifiedOnly": true};\n window._POSTHOG_JS_APPS = [];\n})();' + b'(function() {\n window._POSTHOG_REMOTE_CONFIG = window._POSTHOG_REMOTE_CONFIG || {};\n window._POSTHOG_REMOTE_CONFIG[\'token123\'] = {\n config: {"token": "token123", "supportedCompression": ["gzip", "gzip-js"], "hasFeatureFlags": false, "captureDeadClicks": false, "capturePerformance": {"network_timing": true, "web_vitals": false, "web_vitals_allowed_metrics": null}, "autocapture_opt_out": false, "autocaptureExceptions": false, "analytics": {"endpoint": "/i/v0/e/"}, "elementsChainAsString": true, "sessionRecording": {"endpoint": "/s/", "consoleLogRecordingEnabled": true, "recorderVersion": "v2", "sampleRate": null, "minimumDurationMilliseconds": null, "linkedFlag": null, "networkPayloadCapture": null, "urlTriggers": [], "urlBlocklist": [], "eventTriggers": [], "scriptConfig": null}, "heatmaps": false, "surveys": [], "defaultIdentifiedOnly": true},\n siteApps: []\n }\n})();' ) @patch("posthog.models.remote_config.get_array_js_content", return_value="[MOCKED_ARRAY_JS_CONTENT]") @@ -126,7 +126,7 @@ def test_valid_array_js(self, mock_get_array_js_content): assert response.content assert response.content == snapshot( - b'[MOCKED_ARRAY_JS_CONTENT]\n\n(function() {\n window._POSTHOG_CONFIG = {"token": "token123", "supportedCompression": ["gzip", "gzip-js"], "hasFeatureFlags": false, "captureDeadClicks": false, "capturePerformance": {"network_timing": true, "web_vitals": false, "web_vitals_allowed_metrics": null}, "autocapture_opt_out": false, "autocaptureExceptions": false, "analytics": {"endpoint": "/i/v0/e/"}, "elementsChainAsString": true, "sessionRecording": {"endpoint": "/s/", "consoleLogRecordingEnabled": true, "recorderVersion": "v2", "sampleRate": null, "minimumDurationMilliseconds": null, "linkedFlag": null, "networkPayloadCapture": null, "urlTriggers": [], "urlBlocklist": [], "eventTriggers": [], "scriptConfig": null}, "heatmaps": false, "surveys": [], "defaultIdentifiedOnly": true};\n window._POSTHOG_JS_APPS = [];\n})();' + b'[MOCKED_ARRAY_JS_CONTENT]\n\n(function() {\n window._POSTHOG_REMOTE_CONFIG = window._POSTHOG_REMOTE_CONFIG || {};\n window._POSTHOG_REMOTE_CONFIG[\'token123\'] = {\n config: {"token": "token123", "supportedCompression": ["gzip", "gzip-js"], "hasFeatureFlags": false, "captureDeadClicks": false, "capturePerformance": {"network_timing": true, "web_vitals": false, "web_vitals_allowed_metrics": null}, "autocapture_opt_out": false, "autocaptureExceptions": false, "analytics": {"endpoint": "/i/v0/e/"}, "elementsChainAsString": true, "sessionRecording": {"endpoint": "/s/", "consoleLogRecordingEnabled": true, "recorderVersion": "v2", "sampleRate": null, "minimumDurationMilliseconds": null, "linkedFlag": null, "networkPayloadCapture": null, "urlTriggers": [], "urlBlocklist": [], "eventTriggers": [], "scriptConfig": null}, "heatmaps": false, "surveys": [], "defaultIdentifiedOnly": true},\n siteApps: []\n }\n})();' ) # NOT actually testing the content here as it will change dynamically diff --git a/posthog/models/remote_config.py b/posthog/models/remote_config.py index 5ffc726d0d1c2..fd6d9cdda14d1 100644 --- a/posthog/models/remote_config.py +++ b/posthog/models/remote_config.py @@ -335,8 +335,11 @@ def get_config_js_via_token(cls, token: str, request: Optional[HttpRequest] = No config = sanitize_config_for_public_cdn(config, request=request) js_content = f"""(function() {{ - window._POSTHOG_CONFIG = {json.dumps(config)}; - window._POSTHOG_JS_APPS = [{','.join(site_apps_js)}]; + window._POSTHOG_REMOTE_CONFIG = window._POSTHOG_REMOTE_CONFIG || {{}}; + window._POSTHOG_REMOTE_CONFIG['{token}'] = {{ + config: {json.dumps(config)}, + siteApps: [{','.join(site_apps_js)}] + }} }})(); """.strip() diff --git a/posthog/models/test/test_remote_config.py b/posthog/models/test/test_remote_config.py index d9565e2422ddb..9afe0a99e1370 100644 --- a/posthog/models/test/test_remote_config.py +++ b/posthog/models/test/test_remote_config.py @@ -318,8 +318,11 @@ def _assert_matches_config_js(self, data): assert data == snapshot( """\ (function() { - window._POSTHOG_CONFIG = {"token": "phc_12345", "supportedCompression": ["gzip", "gzip-js"], "hasFeatureFlags": false, "captureDeadClicks": false, "capturePerformance": {"network_timing": true, "web_vitals": false, "web_vitals_allowed_metrics": null}, "autocapture_opt_out": false, "autocaptureExceptions": false, "analytics": {"endpoint": "/i/v0/e/"}, "elementsChainAsString": true, "sessionRecording": {"endpoint": "/s/", "consoleLogRecordingEnabled": true, "recorderVersion": "v2", "sampleRate": null, "minimumDurationMilliseconds": null, "linkedFlag": null, "networkPayloadCapture": null, "urlTriggers": [], "urlBlocklist": [], "eventTriggers": [], "scriptConfig": null}, "heatmaps": false, "surveys": [], "defaultIdentifiedOnly": true}; - window._POSTHOG_JS_APPS = []; + window._POSTHOG_REMOTE_CONFIG = window._POSTHOG_REMOTE_CONFIG || {}; + window._POSTHOG_REMOTE_CONFIG['phc_12345'] = { + config: {"token": "phc_12345", "supportedCompression": ["gzip", "gzip-js"], "hasFeatureFlags": false, "captureDeadClicks": false, "capturePerformance": {"network_timing": true, "web_vitals": false, "web_vitals_allowed_metrics": null}, "autocapture_opt_out": false, "autocaptureExceptions": false, "analytics": {"endpoint": "/i/v0/e/"}, "elementsChainAsString": true, "sessionRecording": {"endpoint": "/s/", "consoleLogRecordingEnabled": true, "recorderVersion": "v2", "sampleRate": null, "minimumDurationMilliseconds": null, "linkedFlag": null, "networkPayloadCapture": null, "urlTriggers": [], "urlBlocklist": [], "eventTriggers": [], "scriptConfig": null}, "heatmaps": false, "surveys": [], "defaultIdentifiedOnly": true}, + siteApps: [] + } })();\ """ ) @@ -330,8 +333,11 @@ def _assert_matches_config_array_js(self, data): [MOCKED_ARRAY_JS_CONTENT] (function() { - window._POSTHOG_CONFIG = {"token": "phc_12345", "supportedCompression": ["gzip", "gzip-js"], "hasFeatureFlags": false, "captureDeadClicks": false, "capturePerformance": {"network_timing": true, "web_vitals": false, "web_vitals_allowed_metrics": null}, "autocapture_opt_out": false, "autocaptureExceptions": false, "analytics": {"endpoint": "/i/v0/e/"}, "elementsChainAsString": true, "sessionRecording": {"endpoint": "/s/", "consoleLogRecordingEnabled": true, "recorderVersion": "v2", "sampleRate": null, "minimumDurationMilliseconds": null, "linkedFlag": null, "networkPayloadCapture": null, "urlTriggers": [], "urlBlocklist": [], "eventTriggers": [], "scriptConfig": null}, "heatmaps": false, "surveys": [], "defaultIdentifiedOnly": true}; - window._POSTHOG_JS_APPS = []; + window._POSTHOG_REMOTE_CONFIG = window._POSTHOG_REMOTE_CONFIG || {}; + window._POSTHOG_REMOTE_CONFIG['phc_12345'] = { + config: {"token": "phc_12345", "supportedCompression": ["gzip", "gzip-js"], "hasFeatureFlags": false, "captureDeadClicks": false, "capturePerformance": {"network_timing": true, "web_vitals": false, "web_vitals_allowed_metrics": null}, "autocapture_opt_out": false, "autocaptureExceptions": false, "analytics": {"endpoint": "/i/v0/e/"}, "elementsChainAsString": true, "sessionRecording": {"endpoint": "/s/", "consoleLogRecordingEnabled": true, "recorderVersion": "v2", "sampleRate": null, "minimumDurationMilliseconds": null, "linkedFlag": null, "networkPayloadCapture": null, "urlTriggers": [], "urlBlocklist": [], "eventTriggers": [], "scriptConfig": null}, "heatmaps": false, "surveys": [], "defaultIdentifiedOnly": true}, + siteApps: [] + } })();\ """ ) @@ -474,8 +480,11 @@ def test_renders_js_including_config(self): assert js == snapshot( """\ (function() { - window._POSTHOG_CONFIG = {"token": "phc_12345", "supportedCompression": ["gzip", "gzip-js"], "hasFeatureFlags": false, "captureDeadClicks": false, "capturePerformance": {"network_timing": true, "web_vitals": false, "web_vitals_allowed_metrics": null}, "autocapture_opt_out": false, "autocaptureExceptions": false, "analytics": {"endpoint": "/i/v0/e/"}, "elementsChainAsString": true, "sessionRecording": {"endpoint": "/s/", "consoleLogRecordingEnabled": true, "recorderVersion": "v2", "sampleRate": null, "minimumDurationMilliseconds": null, "linkedFlag": null, "networkPayloadCapture": null, "urlTriggers": [], "urlBlocklist": [], "eventTriggers": [], "scriptConfig": null}, "heatmaps": false, "surveys": [], "defaultIdentifiedOnly": true}; - window._POSTHOG_JS_APPS = []; + window._POSTHOG_REMOTE_CONFIG = window._POSTHOG_REMOTE_CONFIG || {}; + window._POSTHOG_REMOTE_CONFIG['phc_12345'] = { + config: {"token": "phc_12345", "supportedCompression": ["gzip", "gzip-js"], "hasFeatureFlags": false, "captureDeadClicks": false, "capturePerformance": {"network_timing": true, "web_vitals": false, "web_vitals_allowed_metrics": null}, "autocapture_opt_out": false, "autocaptureExceptions": false, "analytics": {"endpoint": "/i/v0/e/"}, "elementsChainAsString": true, "sessionRecording": {"endpoint": "/s/", "consoleLogRecordingEnabled": true, "recorderVersion": "v2", "sampleRate": null, "minimumDurationMilliseconds": null, "linkedFlag": null, "networkPayloadCapture": null, "urlTriggers": [], "urlBlocklist": [], "eventTriggers": [], "scriptConfig": null}, "heatmaps": false, "surveys": [], "defaultIdentifiedOnly": true}, + siteApps: [] + } })();\ """ ) @@ -517,8 +526,10 @@ def test_renders_js_including_site_apps(self): assert js == snapshot( """\ (function() { - window._POSTHOG_CONFIG = {"token": "phc_12345", "supportedCompression": ["gzip", "gzip-js"], "hasFeatureFlags": false, "captureDeadClicks": false, "capturePerformance": {"network_timing": true, "web_vitals": false, "web_vitals_allowed_metrics": null}, "autocapture_opt_out": false, "autocaptureExceptions": false, "analytics": {"endpoint": "/i/v0/e/"}, "elementsChainAsString": true, "sessionRecording": {"endpoint": "/s/", "consoleLogRecordingEnabled": true, "recorderVersion": "v2", "sampleRate": null, "minimumDurationMilliseconds": null, "linkedFlag": null, "networkPayloadCapture": null, "urlTriggers": [], "urlBlocklist": [], "eventTriggers": [], "scriptConfig": null}, "heatmaps": false, "surveys": [], "defaultIdentifiedOnly": true}; - window._POSTHOG_JS_APPS = [ + window._POSTHOG_REMOTE_CONFIG = window._POSTHOG_REMOTE_CONFIG || {}; + window._POSTHOG_REMOTE_CONFIG['phc_12345'] = { + config: {"token": "phc_12345", "supportedCompression": ["gzip", "gzip-js"], "hasFeatureFlags": false, "captureDeadClicks": false, "capturePerformance": {"network_timing": true, "web_vitals": false, "web_vitals_allowed_metrics": null}, "autocapture_opt_out": false, "autocaptureExceptions": false, "analytics": {"endpoint": "/i/v0/e/"}, "elementsChainAsString": true, "sessionRecording": {"endpoint": "/s/", "consoleLogRecordingEnabled": true, "recorderVersion": "v2", "sampleRate": null, "minimumDurationMilliseconds": null, "linkedFlag": null, "networkPayloadCapture": null, "urlTriggers": [], "urlBlocklist": [], "eventTriggers": [], "scriptConfig": null}, "heatmaps": false, "surveys": [], "defaultIdentifiedOnly": true}, + siteApps: [ { id: 'tokentoken', init: function(config) { @@ -539,7 +550,8 @@ def test_renders_js_including_site_apps(self): (function () { return { inject: (data) => console.log('injected but disabled!', data)}; })().inject({ config:{}, posthog:config.posthog }); config.callback(); } - }]; + }] + } })();\ """ # noqa: W291, W293 ) @@ -587,8 +599,10 @@ def test_renders_js_including_site_functions(self): assert js == snapshot( """\ (function() { - window._POSTHOG_CONFIG = {"token": "phc_12345", "supportedCompression": ["gzip", "gzip-js"], "hasFeatureFlags": false, "captureDeadClicks": false, "capturePerformance": {"network_timing": true, "web_vitals": false, "web_vitals_allowed_metrics": null}, "autocapture_opt_out": false, "autocaptureExceptions": false, "analytics": {"endpoint": "/i/v0/e/"}, "elementsChainAsString": true, "sessionRecording": {"endpoint": "/s/", "consoleLogRecordingEnabled": true, "recorderVersion": "v2", "sampleRate": null, "minimumDurationMilliseconds": null, "linkedFlag": null, "networkPayloadCapture": null, "urlTriggers": [], "urlBlocklist": [], "eventTriggers": [], "scriptConfig": null}, "heatmaps": false, "surveys": [], "defaultIdentifiedOnly": true}; - window._POSTHOG_JS_APPS = [ + window._POSTHOG_REMOTE_CONFIG = window._POSTHOG_REMOTE_CONFIG || {}; + window._POSTHOG_REMOTE_CONFIG['phc_12345'] = { + config: {"token": "phc_12345", "supportedCompression": ["gzip", "gzip-js"], "hasFeatureFlags": false, "captureDeadClicks": false, "capturePerformance": {"network_timing": true, "web_vitals": false, "web_vitals_allowed_metrics": null}, "autocapture_opt_out": false, "autocaptureExceptions": false, "analytics": {"endpoint": "/i/v0/e/"}, "elementsChainAsString": true, "sessionRecording": {"endpoint": "/s/", "consoleLogRecordingEnabled": true, "recorderVersion": "v2", "sampleRate": null, "minimumDurationMilliseconds": null, "linkedFlag": null, "networkPayloadCapture": null, "urlTriggers": [], "urlBlocklist": [], "eventTriggers": [], "scriptConfig": null}, "heatmaps": false, "surveys": [], "defaultIdentifiedOnly": true}, + siteApps: [ { id: 'SITE_DESTINATION_ID', init: function(config) { return (function() { @@ -779,7 +793,8 @@ def test_renders_js_including_site_functions(self): return { init: init }; })().init(config) } - }]; + }] + } })();\ """ # noqa: W291, W293 ) From c94e23a6db9d84300b9553811af2ad900cdb5c2d Mon Sep 17 00:00:00 2001 From: Paul D'Ambra Date: Tue, 17 Dec 2024 11:54:18 +0100 Subject: [PATCH 7/9] feat: remove the filter based queries (#26944) Co-authored-by: github-actions <41898282+github-actions[bot]@users.noreply.github.com> --- ee/clickhouse/queries/event_query.py | 2 - ...t_session_recording_list_from_filters.ambr | 1649 ----- ...est_session_recording_list_from_filters.py | 353 - .../session_recording_playlist.py | 24 +- posthog/constants.py | 2 - .../test/__snapshots__/test_in_cohort.ambr | 4 +- posthog/models/filters/__init__.py | 6 +- .../filters/mixins/session_recordings.py | 66 - .../filters/session_recordings_filter.py | 6 - posthog/queries/event_query/event_query.py | 2 - .../session_recording_list_from_filters.py | 651 -- .../queries/session_recording_properties.py | 106 - ...t_session_recording_list_from_filters.ambr | 5812 ----------------- ...est_session_recording_list_from_filters.py | 4207 ------------ .../test/test_session_recording_properties.py | 104 - .../session_recording_api.py | 209 +- .../test/test_session_recordings.py | 60 +- posthog/test/base.py | 2 +- 18 files changed, 45 insertions(+), 13220 deletions(-) delete mode 100644 ee/session_recordings/queries/test/__snapshots__/test_session_recording_list_from_filters.ambr delete mode 100644 ee/session_recordings/queries/test/test_session_recording_list_from_filters.py delete mode 100644 posthog/models/filters/mixins/session_recordings.py delete mode 100644 posthog/models/filters/session_recordings_filter.py delete mode 100644 posthog/session_recordings/queries/session_recording_list_from_filters.py delete mode 100644 posthog/session_recordings/queries/session_recording_properties.py delete mode 100644 posthog/session_recordings/queries/test/__snapshots__/test_session_recording_list_from_filters.ambr delete mode 100644 posthog/session_recordings/queries/test/test_session_recording_list_from_filters.py delete mode 100644 posthog/session_recordings/queries/test/test_session_recording_properties.py diff --git a/ee/clickhouse/queries/event_query.py b/ee/clickhouse/queries/event_query.py index 3f7857fd9a970..64f08da69d6bf 100644 --- a/ee/clickhouse/queries/event_query.py +++ b/ee/clickhouse/queries/event_query.py @@ -7,7 +7,6 @@ from posthog.models.filters.path_filter import PathFilter from posthog.models.filters.properties_timeline_filter import PropertiesTimelineFilter from posthog.models.filters.retention_filter import RetentionFilter -from posthog.models.filters.session_recordings_filter import SessionRecordingsFilter from posthog.models.filters.stickiness_filter import StickinessFilter from posthog.models.property import PropertyName from posthog.models.team import Team @@ -25,7 +24,6 @@ def __init__( PathFilter, RetentionFilter, StickinessFilter, - SessionRecordingsFilter, PropertiesTimelineFilter, ], team: Team, diff --git a/ee/session_recordings/queries/test/__snapshots__/test_session_recording_list_from_filters.ambr b/ee/session_recordings/queries/test/__snapshots__/test_session_recording_list_from_filters.ambr deleted file mode 100644 index 389933177e2ca..0000000000000 --- a/ee/session_recordings/queries/test/__snapshots__/test_session_recording_list_from_filters.ambr +++ /dev/null @@ -1,1649 +0,0 @@ -# serializer version: 1 -# name: TestClickhouseSessionRecordingsListFromFilters.test_effect_of_poe_settings_on_query_generated_0_test_poe_v1_still_falls_back_to_person_subquery - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, %(hogql_val_0)s)) AS start_time, - max(toTimeZone(s.max_last_timestamp, %(hogql_val_1)s)) AS end_time, - dateDiff(%(hogql_val_2)s, start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, %(hogql_val_3)s)), toDateTime64('2021-01-01 13:41:23.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff(%(hogql_val_4)s, start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, %(hogql_val_5)s), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, %(hogql_val_6)s), toDateTime64('2020-12-25 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, %(hogql_val_7)s), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), 0), in(s.session_id, - (SELECT events.`$session_id` AS session_id - FROM events - WHERE and(equals(events.team_id, 99999), notEmpty(events.`$session_id`), greaterOrEquals(toTimeZone(events.timestamp, %(hogql_val_8)s), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, %(hogql_val_9)s), now64(6, %(hogql_val_10)s)), greaterOrEquals(toTimeZone(events.timestamp, %(hogql_val_11)s), toDateTime64('2020-12-24 23:58:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, %(hogql_val_12)s), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), ifNull(equals(nullIf(nullIf(mat_pp_rgInternal, ''), 'null'), %(hogql_val_13)s), 0)) - GROUP BY events.`$session_id` - HAVING 1))) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 50000 - ''' -# --- -# name: TestClickhouseSessionRecordingsListFromFilters.test_effect_of_poe_settings_on_query_generated_1_test_poe_being_unavailable_we_fall_back_to_person_id_overrides - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, %(hogql_val_0)s)) AS start_time, - max(toTimeZone(s.max_last_timestamp, %(hogql_val_1)s)) AS end_time, - dateDiff(%(hogql_val_2)s, start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, %(hogql_val_3)s)), toDateTime64('2021-01-01 13:41:23.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff(%(hogql_val_4)s, start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, %(hogql_val_5)s), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, %(hogql_val_6)s), toDateTime64('2020-12-25 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, %(hogql_val_7)s), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), 0), in(s.session_id, - (SELECT events.`$session_id` AS session_id - FROM events - LEFT OUTER JOIN - (SELECT argMax(person_distinct_id_overrides.person_id, person_distinct_id_overrides.version) AS person_id, person_distinct_id_overrides.distinct_id AS distinct_id - FROM person_distinct_id_overrides - WHERE equals(person_distinct_id_overrides.team_id, 99999) - GROUP BY person_distinct_id_overrides.distinct_id - HAVING ifNull(equals(argMax(person_distinct_id_overrides.is_deleted, person_distinct_id_overrides.version), 0), 0) SETTINGS optimize_aggregation_in_order=1) AS events__override ON equals(events.distinct_id, events__override.distinct_id) - LEFT JOIN - (SELECT argMax(replaceRegexpAll(nullIf(nullIf(JSONExtractRaw(person.properties, %(hogql_val_8)s), ''), 'null'), '^"|"$', ''), person.version) AS properties___rgInternal, person.id AS id - FROM person - WHERE equals(person.team_id, 99999) - GROUP BY person.id - HAVING and(ifNull(equals(argMax(person.is_deleted, person.version), 0), 0), ifNull(less(argMax(toTimeZone(person.created_at, %(hogql_val_9)s), person.version), plus(now64(6, %(hogql_val_10)s), toIntervalDay(1))), 0)) SETTINGS optimize_aggregation_in_order=1) AS events__person ON equals(if(not(empty(events__override.distinct_id)), events__override.person_id, events.person_id), events__person.id) - WHERE and(equals(events.team_id, 99999), notEmpty(events.`$session_id`), greaterOrEquals(toTimeZone(events.timestamp, %(hogql_val_11)s), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, %(hogql_val_12)s), now64(6, %(hogql_val_13)s)), greaterOrEquals(toTimeZone(events.timestamp, %(hogql_val_14)s), toDateTime64('2020-12-24 23:58:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, %(hogql_val_15)s), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), ifNull(equals(events__person.properties___rgInternal, %(hogql_val_16)s), 0)) - GROUP BY events.`$session_id` - HAVING 1))) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 50000 - ''' -# --- -# name: TestClickhouseSessionRecordingsListFromFilters.test_effect_of_poe_settings_on_query_generated_2_test_poe_being_unavailable_we_fall_back_to_person_subquery_but_still_use_mat_props - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, %(hogql_val_0)s)) AS start_time, - max(toTimeZone(s.max_last_timestamp, %(hogql_val_1)s)) AS end_time, - dateDiff(%(hogql_val_2)s, start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, %(hogql_val_3)s)), toDateTime64('2021-01-01 13:41:23.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff(%(hogql_val_4)s, start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, %(hogql_val_5)s), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, %(hogql_val_6)s), toDateTime64('2020-12-25 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, %(hogql_val_7)s), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), 0), in(s.session_id, - (SELECT events.`$session_id` AS session_id - FROM events - LEFT OUTER JOIN - (SELECT argMax(person_distinct_id_overrides.person_id, person_distinct_id_overrides.version) AS person_id, person_distinct_id_overrides.distinct_id AS distinct_id - FROM person_distinct_id_overrides - WHERE equals(person_distinct_id_overrides.team_id, 99999) - GROUP BY person_distinct_id_overrides.distinct_id - HAVING ifNull(equals(argMax(person_distinct_id_overrides.is_deleted, person_distinct_id_overrides.version), 0), 0) SETTINGS optimize_aggregation_in_order=1) AS events__override ON equals(events.distinct_id, events__override.distinct_id) - LEFT JOIN - (SELECT argMax(replaceRegexpAll(nullIf(nullIf(JSONExtractRaw(person.properties, %(hogql_val_8)s), ''), 'null'), '^"|"$', ''), person.version) AS properties___rgInternal, person.id AS id - FROM person - WHERE equals(person.team_id, 99999) - GROUP BY person.id - HAVING and(ifNull(equals(argMax(person.is_deleted, person.version), 0), 0), ifNull(less(argMax(toTimeZone(person.created_at, %(hogql_val_9)s), person.version), plus(now64(6, %(hogql_val_10)s), toIntervalDay(1))), 0)) SETTINGS optimize_aggregation_in_order=1) AS events__person ON equals(if(not(empty(events__override.distinct_id)), events__override.person_id, events.person_id), events__person.id) - WHERE and(equals(events.team_id, 99999), notEmpty(events.`$session_id`), greaterOrEquals(toTimeZone(events.timestamp, %(hogql_val_11)s), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, %(hogql_val_12)s), now64(6, %(hogql_val_13)s)), greaterOrEquals(toTimeZone(events.timestamp, %(hogql_val_14)s), toDateTime64('2020-12-24 23:58:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, %(hogql_val_15)s), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), ifNull(equals(events__person.properties___rgInternal, %(hogql_val_16)s), 0)) - GROUP BY events.`$session_id` - HAVING 1))) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 50000 - ''' -# --- -# name: TestClickhouseSessionRecordingsListFromFilters.test_effect_of_poe_settings_on_query_generated_3_test_allow_denormalised_props_fix_does_not_stop_all_poe_processing - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, %(hogql_val_0)s)) AS start_time, - max(toTimeZone(s.max_last_timestamp, %(hogql_val_1)s)) AS end_time, - dateDiff(%(hogql_val_2)s, start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, %(hogql_val_3)s)), toDateTime64('2021-01-01 13:41:23.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff(%(hogql_val_4)s, start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, %(hogql_val_5)s), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, %(hogql_val_6)s), toDateTime64('2020-12-25 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, %(hogql_val_7)s), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), 0), in(s.session_id, - (SELECT events.`$session_id` AS session_id - FROM events - WHERE and(equals(events.team_id, 99999), notEmpty(events.`$session_id`), greaterOrEquals(toTimeZone(events.timestamp, %(hogql_val_8)s), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, %(hogql_val_9)s), now64(6, %(hogql_val_10)s)), greaterOrEquals(toTimeZone(events.timestamp, %(hogql_val_11)s), toDateTime64('2020-12-24 23:58:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, %(hogql_val_12)s), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), ifNull(equals(nullIf(nullIf(mat_pp_rgInternal, ''), 'null'), %(hogql_val_13)s), 0)) - GROUP BY events.`$session_id` - HAVING 1))) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 50000 - ''' -# --- -# name: TestClickhouseSessionRecordingsListFromFilters.test_effect_of_poe_settings_on_query_generated_4_test_poe_v2_available_person_properties_are_used_in_replay_listing - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, %(hogql_val_0)s)) AS start_time, - max(toTimeZone(s.max_last_timestamp, %(hogql_val_1)s)) AS end_time, - dateDiff(%(hogql_val_2)s, start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, %(hogql_val_3)s)), toDateTime64('2021-01-01 13:41:23.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff(%(hogql_val_4)s, start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, %(hogql_val_5)s), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, %(hogql_val_6)s), toDateTime64('2020-12-25 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, %(hogql_val_7)s), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), 0), in(s.session_id, - (SELECT events.`$session_id` AS session_id - FROM events - WHERE and(equals(events.team_id, 99999), notEmpty(events.`$session_id`), greaterOrEquals(toTimeZone(events.timestamp, %(hogql_val_8)s), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, %(hogql_val_9)s), now64(6, %(hogql_val_10)s)), greaterOrEquals(toTimeZone(events.timestamp, %(hogql_val_11)s), toDateTime64('2020-12-24 23:58:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, %(hogql_val_12)s), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), ifNull(equals(nullIf(nullIf(mat_pp_rgInternal, ''), 'null'), %(hogql_val_13)s), 0)) - GROUP BY events.`$session_id` - HAVING 1))) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 50000 - ''' -# --- -# name: TestClickhouseSessionRecordingsListFromFilters.test_event_filter_with_person_properties_materialized_00_poe_v2_and_materialized_columns_allowed_with_materialization - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-01 13:41:23.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-25 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), 0)) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestClickhouseSessionRecordingsListFromFilters.test_event_filter_with_person_properties_materialized_00_poe_v2_and_materialized_columns_allowed_with_materialization.1 - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-01 13:41:23.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-25 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), 0), in(s.session_id, - (SELECT events.`$session_id` AS session_id - FROM events - WHERE and(equals(events.team_id, 99999), notEmpty(events.`$session_id`), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), now64(6, 'UTC')), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-24 23:58:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), ifNull(equals(nullIf(nullIf(mat_pp_email, ''), 'null'), 'bla'), 0)) - GROUP BY events.`$session_id` - HAVING 1))) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestClickhouseSessionRecordingsListFromFilters.test_event_filter_with_person_properties_materialized_01_poe_v2_and_materialized_columns_allowed_without_materialization - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-01 13:41:23.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-25 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), 0)) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestClickhouseSessionRecordingsListFromFilters.test_event_filter_with_person_properties_materialized_01_poe_v2_and_materialized_columns_allowed_without_materialization.1 - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-01 13:41:23.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-25 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), 0), in(s.session_id, - (SELECT events.`$session_id` AS session_id - FROM events - WHERE and(equals(events.team_id, 99999), notEmpty(events.`$session_id`), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), now64(6, 'UTC')), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-24 23:58:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), ifNull(equals(nullIf(nullIf(mat_pp_email, ''), 'null'), 'bla'), 0)) - GROUP BY events.`$session_id` - HAVING 1))) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestClickhouseSessionRecordingsListFromFilters.test_event_filter_with_person_properties_materialized_02_poe_v2_and_materialized_columns_off_with_materialization - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-01 13:41:23.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-25 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), 0)) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestClickhouseSessionRecordingsListFromFilters.test_event_filter_with_person_properties_materialized_02_poe_v2_and_materialized_columns_off_with_materialization.1 - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-01 13:41:23.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-25 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), 0), in(s.session_id, - (SELECT events.`$session_id` AS session_id - FROM events - WHERE and(equals(events.team_id, 99999), notEmpty(events.`$session_id`), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), now64(6, 'UTC')), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-24 23:58:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), ifNull(equals(nullIf(nullIf(mat_pp_email, ''), 'null'), 'bla'), 0)) - GROUP BY events.`$session_id` - HAVING 1))) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestClickhouseSessionRecordingsListFromFilters.test_event_filter_with_person_properties_materialized_03_poe_v2_and_materialized_columns_off_without_materialization - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-01 13:41:23.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-25 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), 0)) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestClickhouseSessionRecordingsListFromFilters.test_event_filter_with_person_properties_materialized_03_poe_v2_and_materialized_columns_off_without_materialization.1 - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-01 13:41:23.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-25 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), 0), in(s.session_id, - (SELECT events.`$session_id` AS session_id - FROM events - WHERE and(equals(events.team_id, 99999), notEmpty(events.`$session_id`), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), now64(6, 'UTC')), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-24 23:58:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), ifNull(equals(nullIf(nullIf(mat_pp_email, ''), 'null'), 'bla'), 0)) - GROUP BY events.`$session_id` - HAVING 1))) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestClickhouseSessionRecordingsListFromFilters.test_event_filter_with_person_properties_materialized_04_poe_off_and_materialized_columns_allowed_with_materialization - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-01 13:41:23.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-25 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), 0)) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestClickhouseSessionRecordingsListFromFilters.test_event_filter_with_person_properties_materialized_04_poe_off_and_materialized_columns_allowed_with_materialization.1 - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-01 13:41:23.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-25 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), 0), in(s.session_id, - (SELECT events.`$session_id` AS session_id - FROM events - LEFT OUTER JOIN - (SELECT argMax(person_distinct_id_overrides.person_id, person_distinct_id_overrides.version) AS person_id, person_distinct_id_overrides.distinct_id AS distinct_id - FROM person_distinct_id_overrides - WHERE equals(person_distinct_id_overrides.team_id, 99999) - GROUP BY person_distinct_id_overrides.distinct_id - HAVING ifNull(equals(argMax(person_distinct_id_overrides.is_deleted, person_distinct_id_overrides.version), 0), 0) SETTINGS optimize_aggregation_in_order=1) AS events__override ON equals(events.distinct_id, events__override.distinct_id) - LEFT JOIN - (SELECT person.id AS id, nullIf(nullIf(person.pmat_email, ''), 'null') AS properties___email - FROM person - WHERE and(equals(person.team_id, 99999), ifNull(in(tuple(person.id, person.version), - (SELECT person.id AS id, max(person.version) AS version - FROM person - WHERE equals(person.team_id, 99999) - GROUP BY person.id - HAVING and(ifNull(equals(argMax(person.is_deleted, person.version), 0), 0), ifNull(less(argMax(toTimeZone(person.created_at, 'UTC'), person.version), plus(now64(6, 'UTC'), toIntervalDay(1))), 0)))), 0)) SETTINGS optimize_aggregation_in_order=1) AS events__person ON equals(if(not(empty(events__override.distinct_id)), events__override.person_id, events.person_id), events__person.id) - WHERE and(equals(events.team_id, 99999), notEmpty(events.`$session_id`), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), now64(6, 'UTC')), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-24 23:58:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), ifNull(equals(events__person.properties___email, 'bla'), 0)) - GROUP BY events.`$session_id` - HAVING 1))) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestClickhouseSessionRecordingsListFromFilters.test_event_filter_with_person_properties_materialized_05_poe_off_and_materialized_columns_allowed_without_materialization - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-01 13:41:23.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-25 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), 0)) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestClickhouseSessionRecordingsListFromFilters.test_event_filter_with_person_properties_materialized_05_poe_off_and_materialized_columns_allowed_without_materialization.1 - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-01 13:41:23.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-25 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), 0), in(s.session_id, - (SELECT events.`$session_id` AS session_id - FROM events - LEFT OUTER JOIN - (SELECT argMax(person_distinct_id_overrides.person_id, person_distinct_id_overrides.version) AS person_id, person_distinct_id_overrides.distinct_id AS distinct_id - FROM person_distinct_id_overrides - WHERE equals(person_distinct_id_overrides.team_id, 99999) - GROUP BY person_distinct_id_overrides.distinct_id - HAVING ifNull(equals(argMax(person_distinct_id_overrides.is_deleted, person_distinct_id_overrides.version), 0), 0) SETTINGS optimize_aggregation_in_order=1) AS events__override ON equals(events.distinct_id, events__override.distinct_id) - LEFT JOIN - (SELECT person.id AS id, nullIf(nullIf(person.pmat_email, ''), 'null') AS properties___email - FROM person - WHERE and(equals(person.team_id, 99999), ifNull(in(tuple(person.id, person.version), - (SELECT person.id AS id, max(person.version) AS version - FROM person - WHERE equals(person.team_id, 99999) - GROUP BY person.id - HAVING and(ifNull(equals(argMax(person.is_deleted, person.version), 0), 0), ifNull(less(argMax(toTimeZone(person.created_at, 'UTC'), person.version), plus(now64(6, 'UTC'), toIntervalDay(1))), 0)))), 0)) SETTINGS optimize_aggregation_in_order=1) AS events__person ON equals(if(not(empty(events__override.distinct_id)), events__override.person_id, events.person_id), events__person.id) - WHERE and(equals(events.team_id, 99999), notEmpty(events.`$session_id`), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), now64(6, 'UTC')), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-24 23:58:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), ifNull(equals(events__person.properties___email, 'bla'), 0)) - GROUP BY events.`$session_id` - HAVING 1))) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestClickhouseSessionRecordingsListFromFilters.test_event_filter_with_person_properties_materialized_06_poe_off_and_materialized_columns_not_allowed_with_materialization - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-01 13:41:23.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-25 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), 0)) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestClickhouseSessionRecordingsListFromFilters.test_event_filter_with_person_properties_materialized_06_poe_off_and_materialized_columns_not_allowed_with_materialization.1 - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-01 13:41:23.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-25 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), 0), in(s.session_id, - (SELECT events.`$session_id` AS session_id - FROM events - LEFT OUTER JOIN - (SELECT argMax(person_distinct_id_overrides.person_id, person_distinct_id_overrides.version) AS person_id, person_distinct_id_overrides.distinct_id AS distinct_id - FROM person_distinct_id_overrides - WHERE equals(person_distinct_id_overrides.team_id, 99999) - GROUP BY person_distinct_id_overrides.distinct_id - HAVING ifNull(equals(argMax(person_distinct_id_overrides.is_deleted, person_distinct_id_overrides.version), 0), 0) SETTINGS optimize_aggregation_in_order=1) AS events__override ON equals(events.distinct_id, events__override.distinct_id) - LEFT JOIN - (SELECT person.id AS id, nullIf(nullIf(person.pmat_email, ''), 'null') AS properties___email - FROM person - WHERE and(equals(person.team_id, 99999), ifNull(in(tuple(person.id, person.version), - (SELECT person.id AS id, max(person.version) AS version - FROM person - WHERE equals(person.team_id, 99999) - GROUP BY person.id - HAVING and(ifNull(equals(argMax(person.is_deleted, person.version), 0), 0), ifNull(less(argMax(toTimeZone(person.created_at, 'UTC'), person.version), plus(now64(6, 'UTC'), toIntervalDay(1))), 0)))), 0)) SETTINGS optimize_aggregation_in_order=1) AS events__person ON equals(if(not(empty(events__override.distinct_id)), events__override.person_id, events.person_id), events__person.id) - WHERE and(equals(events.team_id, 99999), notEmpty(events.`$session_id`), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), now64(6, 'UTC')), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-24 23:58:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), ifNull(equals(events__person.properties___email, 'bla'), 0)) - GROUP BY events.`$session_id` - HAVING 1))) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestClickhouseSessionRecordingsListFromFilters.test_event_filter_with_person_properties_materialized_07_poe_off_and_materialized_columns_not_allowed_without_materialization - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-01 13:41:23.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-25 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), 0)) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestClickhouseSessionRecordingsListFromFilters.test_event_filter_with_person_properties_materialized_07_poe_off_and_materialized_columns_not_allowed_without_materialization.1 - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-01 13:41:23.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-25 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), 0), in(s.session_id, - (SELECT events.`$session_id` AS session_id - FROM events - LEFT OUTER JOIN - (SELECT argMax(person_distinct_id_overrides.person_id, person_distinct_id_overrides.version) AS person_id, person_distinct_id_overrides.distinct_id AS distinct_id - FROM person_distinct_id_overrides - WHERE equals(person_distinct_id_overrides.team_id, 99999) - GROUP BY person_distinct_id_overrides.distinct_id - HAVING ifNull(equals(argMax(person_distinct_id_overrides.is_deleted, person_distinct_id_overrides.version), 0), 0) SETTINGS optimize_aggregation_in_order=1) AS events__override ON equals(events.distinct_id, events__override.distinct_id) - LEFT JOIN - (SELECT person.id AS id, nullIf(nullIf(person.pmat_email, ''), 'null') AS properties___email - FROM person - WHERE and(equals(person.team_id, 99999), ifNull(in(tuple(person.id, person.version), - (SELECT person.id AS id, max(person.version) AS version - FROM person - WHERE equals(person.team_id, 99999) - GROUP BY person.id - HAVING and(ifNull(equals(argMax(person.is_deleted, person.version), 0), 0), ifNull(less(argMax(toTimeZone(person.created_at, 'UTC'), person.version), plus(now64(6, 'UTC'), toIntervalDay(1))), 0)))), 0)) SETTINGS optimize_aggregation_in_order=1) AS events__person ON equals(if(not(empty(events__override.distinct_id)), events__override.person_id, events.person_id), events__person.id) - WHERE and(equals(events.team_id, 99999), notEmpty(events.`$session_id`), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), now64(6, 'UTC')), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-24 23:58:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), ifNull(equals(events__person.properties___email, 'bla'), 0)) - GROUP BY events.`$session_id` - HAVING 1))) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestClickhouseSessionRecordingsListFromFilters.test_event_filter_with_person_properties_materialized_08_poe_v1_and_materialized_columns_allowed_with_materialization - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-01 13:41:23.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-25 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), 0)) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestClickhouseSessionRecordingsListFromFilters.test_event_filter_with_person_properties_materialized_08_poe_v1_and_materialized_columns_allowed_with_materialization.1 - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-01 13:41:23.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-25 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), 0), in(s.session_id, - (SELECT events.`$session_id` AS session_id - FROM events - WHERE and(equals(events.team_id, 99999), notEmpty(events.`$session_id`), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), now64(6, 'UTC')), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-24 23:58:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), ifNull(equals(nullIf(nullIf(mat_pp_email, ''), 'null'), 'bla'), 0)) - GROUP BY events.`$session_id` - HAVING 1))) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestClickhouseSessionRecordingsListFromFilters.test_event_filter_with_person_properties_materialized_09_poe_v1_and_materialized_columns_allowed_without_materialization - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-01 13:41:23.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-25 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), 0)) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestClickhouseSessionRecordingsListFromFilters.test_event_filter_with_person_properties_materialized_09_poe_v1_and_materialized_columns_allowed_without_materialization.1 - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-01 13:41:23.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-25 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), 0), in(s.session_id, - (SELECT events.`$session_id` AS session_id - FROM events - WHERE and(equals(events.team_id, 99999), notEmpty(events.`$session_id`), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), now64(6, 'UTC')), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-24 23:58:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), ifNull(equals(nullIf(nullIf(mat_pp_email, ''), 'null'), 'bla'), 0)) - GROUP BY events.`$session_id` - HAVING 1))) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestClickhouseSessionRecordingsListFromFilters.test_event_filter_with_person_properties_materialized_10_poe_v1_and_not_materialized_columns_not_allowed_with_materialization - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-01 13:41:23.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-25 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), 0)) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestClickhouseSessionRecordingsListFromFilters.test_event_filter_with_person_properties_materialized_10_poe_v1_and_not_materialized_columns_not_allowed_with_materialization.1 - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-01 13:41:23.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-25 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), 0), in(s.session_id, - (SELECT events.`$session_id` AS session_id - FROM events - WHERE and(equals(events.team_id, 99999), notEmpty(events.`$session_id`), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), now64(6, 'UTC')), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-24 23:58:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), ifNull(equals(nullIf(nullIf(mat_pp_email, ''), 'null'), 'bla'), 0)) - GROUP BY events.`$session_id` - HAVING 1))) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestClickhouseSessionRecordingsListFromFilters.test_event_filter_with_person_properties_materialized_11_poe_v1_and_not_materialized_columns_not_allowed_without_materialization - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-01 13:41:23.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-25 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), 0)) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestClickhouseSessionRecordingsListFromFilters.test_event_filter_with_person_properties_materialized_11_poe_v1_and_not_materialized_columns_not_allowed_without_materialization.1 - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-01 13:41:23.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-25 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), 0), in(s.session_id, - (SELECT events.`$session_id` AS session_id - FROM events - WHERE and(equals(events.team_id, 99999), notEmpty(events.`$session_id`), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), now64(6, 'UTC')), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-24 23:58:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), ifNull(equals(nullIf(nullIf(mat_pp_email, ''), 'null'), 'bla'), 0)) - GROUP BY events.`$session_id` - HAVING 1))) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestClickhouseSessionRecordingsListFromFilters.test_person_id_filter_00_poe_v2_and_materialized_columns_allowed_with_materialization - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-01 13:41:23.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), 0), in(s.session_id, - (SELECT DISTINCT events.`$session_id` AS `$session_id` - FROM events - WHERE and(equals(events.team_id, 99999), equals(events.person_id, '00000000-0000-0000-0000-000000000000'), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-25 00:00:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), notEmpty(events.`$session_id`)))), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-25 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), 0)) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestClickhouseSessionRecordingsListFromFilters.test_person_id_filter_01_poe_v2_and_materialized_columns_allowed_without_materialization - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-01 13:41:23.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), 0), in(s.session_id, - (SELECT DISTINCT events.`$session_id` AS `$session_id` - FROM events - WHERE and(equals(events.team_id, 99999), equals(events.person_id, '00000000-0000-0000-0000-000000000000'), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-25 00:00:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), notEmpty(events.`$session_id`)))), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-25 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), 0)) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestClickhouseSessionRecordingsListFromFilters.test_person_id_filter_02_poe_v2_and_materialized_columns_off_with_materialization - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-01 13:41:23.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), 0), in(s.session_id, - (SELECT DISTINCT events.`$session_id` AS `$session_id` - FROM events - WHERE and(equals(events.team_id, 99999), equals(events.person_id, '00000000-0000-0000-0000-000000000000'), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-25 00:00:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), notEmpty(events.`$session_id`)))), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-25 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), 0)) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestClickhouseSessionRecordingsListFromFilters.test_person_id_filter_03_poe_v2_and_materialized_columns_off_without_materialization - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-01 13:41:23.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), 0), in(s.session_id, - (SELECT DISTINCT events.`$session_id` AS `$session_id` - FROM events - WHERE and(equals(events.team_id, 99999), equals(events.person_id, '00000000-0000-0000-0000-000000000000'), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-25 00:00:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), notEmpty(events.`$session_id`)))), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-25 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), 0)) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestClickhouseSessionRecordingsListFromFilters.test_person_id_filter_04_poe_off_and_materialized_columns_allowed_with_materialization - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-01 13:41:23.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), 0), in(s.session_id, - (SELECT DISTINCT events.`$session_id` AS `$session_id` - FROM events - LEFT OUTER JOIN - (SELECT argMax(person_distinct_id_overrides.person_id, person_distinct_id_overrides.version) AS person_id, person_distinct_id_overrides.distinct_id AS distinct_id - FROM person_distinct_id_overrides - WHERE equals(person_distinct_id_overrides.team_id, 99999) - GROUP BY person_distinct_id_overrides.distinct_id - HAVING ifNull(equals(argMax(person_distinct_id_overrides.is_deleted, person_distinct_id_overrides.version), 0), 0) SETTINGS optimize_aggregation_in_order=1) AS events__override ON equals(events.distinct_id, events__override.distinct_id) - WHERE and(equals(events.team_id, 99999), ifNull(equals(if(not(empty(events__override.distinct_id)), events__override.person_id, events.person_id), '00000000-0000-0000-0000-000000000000'), 0), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-25 00:00:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), notEmpty(events.`$session_id`)))), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-25 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), 0)) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestClickhouseSessionRecordingsListFromFilters.test_person_id_filter_05_poe_off_and_materialized_columns_allowed_without_materialization - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-01 13:41:23.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), 0), in(s.session_id, - (SELECT DISTINCT events.`$session_id` AS `$session_id` - FROM events - LEFT OUTER JOIN - (SELECT argMax(person_distinct_id_overrides.person_id, person_distinct_id_overrides.version) AS person_id, person_distinct_id_overrides.distinct_id AS distinct_id - FROM person_distinct_id_overrides - WHERE equals(person_distinct_id_overrides.team_id, 99999) - GROUP BY person_distinct_id_overrides.distinct_id - HAVING ifNull(equals(argMax(person_distinct_id_overrides.is_deleted, person_distinct_id_overrides.version), 0), 0) SETTINGS optimize_aggregation_in_order=1) AS events__override ON equals(events.distinct_id, events__override.distinct_id) - WHERE and(equals(events.team_id, 99999), ifNull(equals(if(not(empty(events__override.distinct_id)), events__override.person_id, events.person_id), '00000000-0000-0000-0000-000000000000'), 0), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-25 00:00:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), notEmpty(events.`$session_id`)))), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-25 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), 0)) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestClickhouseSessionRecordingsListFromFilters.test_person_id_filter_06_poe_off_and_materialized_columns_not_allowed_with_materialization - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-01 13:41:23.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), 0), in(s.session_id, - (SELECT DISTINCT events.`$session_id` AS `$session_id` - FROM events - LEFT OUTER JOIN - (SELECT argMax(person_distinct_id_overrides.person_id, person_distinct_id_overrides.version) AS person_id, person_distinct_id_overrides.distinct_id AS distinct_id - FROM person_distinct_id_overrides - WHERE equals(person_distinct_id_overrides.team_id, 99999) - GROUP BY person_distinct_id_overrides.distinct_id - HAVING ifNull(equals(argMax(person_distinct_id_overrides.is_deleted, person_distinct_id_overrides.version), 0), 0) SETTINGS optimize_aggregation_in_order=1) AS events__override ON equals(events.distinct_id, events__override.distinct_id) - WHERE and(equals(events.team_id, 99999), ifNull(equals(if(not(empty(events__override.distinct_id)), events__override.person_id, events.person_id), '00000000-0000-0000-0000-000000000000'), 0), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-25 00:00:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), notEmpty(events.`$session_id`)))), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-25 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), 0)) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestClickhouseSessionRecordingsListFromFilters.test_person_id_filter_07_poe_off_and_materialized_columns_not_allowed_without_materialization - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-01 13:41:23.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), 0), in(s.session_id, - (SELECT DISTINCT events.`$session_id` AS `$session_id` - FROM events - LEFT OUTER JOIN - (SELECT argMax(person_distinct_id_overrides.person_id, person_distinct_id_overrides.version) AS person_id, person_distinct_id_overrides.distinct_id AS distinct_id - FROM person_distinct_id_overrides - WHERE equals(person_distinct_id_overrides.team_id, 99999) - GROUP BY person_distinct_id_overrides.distinct_id - HAVING ifNull(equals(argMax(person_distinct_id_overrides.is_deleted, person_distinct_id_overrides.version), 0), 0) SETTINGS optimize_aggregation_in_order=1) AS events__override ON equals(events.distinct_id, events__override.distinct_id) - WHERE and(equals(events.team_id, 99999), ifNull(equals(if(not(empty(events__override.distinct_id)), events__override.person_id, events.person_id), '00000000-0000-0000-0000-000000000000'), 0), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-25 00:00:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), notEmpty(events.`$session_id`)))), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-25 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), 0)) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestClickhouseSessionRecordingsListFromFilters.test_person_id_filter_08_poe_v1_and_materialized_columns_allowed_with_materialization - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-01 13:41:23.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), 0), in(s.session_id, - (SELECT DISTINCT events.`$session_id` AS `$session_id` - FROM events - LEFT OUTER JOIN - (SELECT argMax(person_distinct_id_overrides.person_id, person_distinct_id_overrides.version) AS person_id, person_distinct_id_overrides.distinct_id AS distinct_id - FROM person_distinct_id_overrides - WHERE equals(person_distinct_id_overrides.team_id, 99999) - GROUP BY person_distinct_id_overrides.distinct_id - HAVING ifNull(equals(argMax(person_distinct_id_overrides.is_deleted, person_distinct_id_overrides.version), 0), 0) SETTINGS optimize_aggregation_in_order=1) AS events__override ON equals(events.distinct_id, events__override.distinct_id) - WHERE and(equals(events.team_id, 99999), ifNull(equals(if(not(empty(events__override.distinct_id)), events__override.person_id, events.person_id), '00000000-0000-0000-0000-000000000000'), 0), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-25 00:00:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), notEmpty(events.`$session_id`)))), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-25 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), 0)) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestClickhouseSessionRecordingsListFromFilters.test_person_id_filter_09_poe_v1_and_materialized_columns_allowed_without_materialization - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-01 13:41:23.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), 0), in(s.session_id, - (SELECT DISTINCT events.`$session_id` AS `$session_id` - FROM events - LEFT OUTER JOIN - (SELECT argMax(person_distinct_id_overrides.person_id, person_distinct_id_overrides.version) AS person_id, person_distinct_id_overrides.distinct_id AS distinct_id - FROM person_distinct_id_overrides - WHERE equals(person_distinct_id_overrides.team_id, 99999) - GROUP BY person_distinct_id_overrides.distinct_id - HAVING ifNull(equals(argMax(person_distinct_id_overrides.is_deleted, person_distinct_id_overrides.version), 0), 0) SETTINGS optimize_aggregation_in_order=1) AS events__override ON equals(events.distinct_id, events__override.distinct_id) - WHERE and(equals(events.team_id, 99999), ifNull(equals(if(not(empty(events__override.distinct_id)), events__override.person_id, events.person_id), '00000000-0000-0000-0000-000000000000'), 0), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-25 00:00:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), notEmpty(events.`$session_id`)))), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-25 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), 0)) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestClickhouseSessionRecordingsListFromFilters.test_person_id_filter_10_poe_v1_and_not_materialized_columns_not_allowed_with_materialization - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-01 13:41:23.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), 0), in(s.session_id, - (SELECT DISTINCT events.`$session_id` AS `$session_id` - FROM events - LEFT OUTER JOIN - (SELECT argMax(person_distinct_id_overrides.person_id, person_distinct_id_overrides.version) AS person_id, person_distinct_id_overrides.distinct_id AS distinct_id - FROM person_distinct_id_overrides - WHERE equals(person_distinct_id_overrides.team_id, 99999) - GROUP BY person_distinct_id_overrides.distinct_id - HAVING ifNull(equals(argMax(person_distinct_id_overrides.is_deleted, person_distinct_id_overrides.version), 0), 0) SETTINGS optimize_aggregation_in_order=1) AS events__override ON equals(events.distinct_id, events__override.distinct_id) - WHERE and(equals(events.team_id, 99999), ifNull(equals(if(not(empty(events__override.distinct_id)), events__override.person_id, events.person_id), '00000000-0000-0000-0000-000000000000'), 0), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-25 00:00:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), notEmpty(events.`$session_id`)))), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-25 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), 0)) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestClickhouseSessionRecordingsListFromFilters.test_person_id_filter_11_poe_v1_and_not_materialized_columns_not_allowed_without_materialization - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-01 13:41:23.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), 0), in(s.session_id, - (SELECT DISTINCT events.`$session_id` AS `$session_id` - FROM events - LEFT OUTER JOIN - (SELECT argMax(person_distinct_id_overrides.person_id, person_distinct_id_overrides.version) AS person_id, person_distinct_id_overrides.distinct_id AS distinct_id - FROM person_distinct_id_overrides - WHERE equals(person_distinct_id_overrides.team_id, 99999) - GROUP BY person_distinct_id_overrides.distinct_id - HAVING ifNull(equals(argMax(person_distinct_id_overrides.is_deleted, person_distinct_id_overrides.version), 0), 0) SETTINGS optimize_aggregation_in_order=1) AS events__override ON equals(events.distinct_id, events__override.distinct_id) - WHERE and(equals(events.team_id, 99999), ifNull(equals(if(not(empty(events__override.distinct_id)), events__override.person_id, events.person_id), '00000000-0000-0000-0000-000000000000'), 0), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-25 00:00:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), notEmpty(events.`$session_id`)))), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-25 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), 0)) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- diff --git a/ee/session_recordings/queries/test/test_session_recording_list_from_filters.py b/ee/session_recordings/queries/test/test_session_recording_list_from_filters.py deleted file mode 100644 index 391d12071966f..0000000000000 --- a/ee/session_recordings/queries/test/test_session_recording_list_from_filters.py +++ /dev/null @@ -1,353 +0,0 @@ -import re -from itertools import product -from uuid import uuid4 - -from dateutil.relativedelta import relativedelta -from django.utils.timezone import now -from freezegun import freeze_time -from parameterized import parameterized - -from ee.clickhouse.materialized_columns.columns import materialize -from posthog.clickhouse.client import sync_execute -from posthog.hogql.ast import CompareOperation, And, SelectQuery -from posthog.hogql.base import Expr -from posthog.hogql.context import HogQLContext -from posthog.hogql.printer import print_ast -from posthog.models import Person -from posthog.models.filters import SessionRecordingsFilter -from posthog.schema import PersonsOnEventsMode -from posthog.session_recordings.queries.session_recording_list_from_filters import ( - SessionRecordingListFromFilters, -) -from posthog.session_recordings.queries.test.session_replay_sql import produce_replay_summary -from posthog.session_recordings.sql.session_replay_event_sql import TRUNCATE_SESSION_REPLAY_EVENTS_TABLE_SQL -from posthog.test.base import ( - APIBaseTest, - ClickhouseTestMixin, - QueryMatchingTest, - snapshot_clickhouse_queries, - _create_event, -) - - -# The HogQL pair of TestClickhouseSessionRecordingsListFromSessionReplay can be renamed when delete the old one -@freeze_time("2021-01-01T13:46:23") -class TestClickhouseSessionRecordingsListFromFilters(ClickhouseTestMixin, APIBaseTest, QueryMatchingTest): - def _print_query(self, query: SelectQuery) -> str: - return print_ast( - query, - HogQLContext(team_id=self.team.pk, enable_select_queries=True), - "clickhouse", - pretty=True, - ) - - def tearDown(self) -> None: - sync_execute(TRUNCATE_SESSION_REPLAY_EVENTS_TABLE_SQL()) - - @property - def base_time(self): - return (now() - relativedelta(hours=1)).replace(microsecond=0, second=0) - - def create_event( - self, - distinct_id, - timestamp, - team=None, - event_name="$pageview", - properties=None, - ): - if team is None: - team = self.team - if properties is None: - properties = {"$os": "Windows 95", "$current_url": "aloha.com/2"} - return _create_event( - team=team, - event=event_name, - timestamp=timestamp, - distinct_id=distinct_id, - properties=properties, - ) - - @parameterized.expand( - [ - [ - "test_poe_v1_still_falls_back_to_person_subquery", - True, - False, - False, - PersonsOnEventsMode.PERSON_ID_NO_OVERRIDE_PROPERTIES_ON_EVENTS, - ], - [ - "test_poe_being_unavailable_we_fall_back_to_person_id_overrides", - False, - False, - False, - PersonsOnEventsMode.PERSON_ID_OVERRIDE_PROPERTIES_JOINED, - ], - [ - "test_poe_being_unavailable_we_fall_back_to_person_subquery_but_still_use_mat_props", - False, - False, - False, - PersonsOnEventsMode.PERSON_ID_OVERRIDE_PROPERTIES_JOINED, - ], - [ - "test_allow_denormalised_props_fix_does_not_stop_all_poe_processing", - False, - True, - False, - PersonsOnEventsMode.PERSON_ID_OVERRIDE_PROPERTIES_ON_EVENTS, - ], - [ - "test_poe_v2_available_person_properties_are_used_in_replay_listing", - False, - True, - True, - PersonsOnEventsMode.PERSON_ID_OVERRIDE_PROPERTIES_ON_EVENTS, - ], - ] - ) - def test_effect_of_poe_settings_on_query_generated( - self, - _name: str, - poe_v1: bool, - poe_v2: bool, - allow_denormalized_props: bool, - expected_poe_mode: PersonsOnEventsMode, - ) -> None: - with self.settings( - PERSON_ON_EVENTS_OVERRIDE=poe_v1, - PERSON_ON_EVENTS_V2_OVERRIDE=poe_v2, - ALLOW_DENORMALIZED_PROPS_IN_LISTING=allow_denormalized_props, - ): - assert self.team.person_on_events_mode == expected_poe_mode - materialize("events", "rgInternal", table_column="person_properties") - - filter = SessionRecordingsFilter( - team=self.team, - data={ - "properties": [ - { - "key": "rgInternal", - "value": ["false"], - "operator": "exact", - "type": "person", - } - ] - }, - ) - session_recording_list_instance = SessionRecordingListFromFilters( - filter=filter, team=self.team, hogql_query_modifiers=None - ) - - hogql_parsed_select = session_recording_list_instance.get_query() - printed_query = self._print_query(hogql_parsed_select) - - person_filtering_expr = self._matching_person_filter_expr_from(hogql_parsed_select) - - self._assert_is_events_person_filter(person_filtering_expr) - - if poe_v1 or poe_v2: - # Property used directly from event (from materialized column) - assert "ifNull(equals(nullIf(nullIf(mat_pp_rgInternal, ''), 'null')" in printed_query - else: - # We get the person property value from the persons JOIN - assert re.search( - r"argMax\(replaceRegexpAll\(nullIf\(nullIf\(JSONExtractRaw\(person\.properties, %\(hogql_val_\d+\)s\), ''\), 'null'\), '^\"|\"\$', ''\), person\.version\) AS properties___rgInternal", - printed_query, - ) - # Then we actually filter on that property value - assert re.search( - r"ifNull\(equals\(events__person\.properties___rgInternal, %\(hogql_val_\d+\)s\), 0\)", - printed_query, - ) - self.assertQueryMatchesSnapshot(printed_query) - - def _assert_is_pdi_filter(self, person_filtering_expr: list[Expr]) -> None: - assert person_filtering_expr[0].right.select_from.table.chain == ["person_distinct_ids"] - assert person_filtering_expr[0].right.where.left.chain == ["person", "properties", "rgInternal"] - - def _assert_is_events_person_filter(self, person_filtering_expr: list[Expr]) -> None: - assert person_filtering_expr[0].right.select_from.table.chain == ["events"] - event_person_condition = [ - x - for x in person_filtering_expr[0].right.where.exprs - if isinstance(x, CompareOperation) and x.left.chain == ["person", "properties", "rgInternal"] - ] - assert len(event_person_condition) == 1 - - def _matching_person_filter_expr_from(self, hogql_parsed_select: SelectQuery) -> list[Expr]: - where_conditions: list[Expr] = hogql_parsed_select.where.exprs - ands = [x for x in where_conditions if isinstance(x, And)] - assert len(ands) == 1 - and_comparisons = [x for x in ands[0].exprs if isinstance(x, CompareOperation)] - assert len(and_comparisons) == 1 - assert isinstance(and_comparisons[0].right, SelectQuery) - return and_comparisons - - settings_combinations = [ - ["poe v2 and materialized columns allowed", False, True, True], - ["poe v2 and materialized columns off", False, True, False], - ["poe off and materialized columns allowed", False, False, True], - ["poe off and materialized columns not allowed", False, False, False], - ["poe v1 and materialized columns allowed", True, False, True], - ["poe v1 and not materialized columns not allowed", True, False, False], - ] - - # Options for "materialize person columns" - materialization_options = [ - [" with materialization", True], - [" without materialization", False], - ] - - # Expand the parameter list to the product of all combinations with "materialize person columns" - # e.g. [a, b] x [c, d] = [a, c], [a, d], [b, c], [b, d] - test_case_combinations = [ - [f"{name}{mat_option}", poe_v1, poe, mat_columns, mat_person] - for (name, poe_v1, poe, mat_columns), (mat_option, mat_person) in product( - settings_combinations, materialization_options - ) - ] - - @parameterized.expand(test_case_combinations) - @snapshot_clickhouse_queries - def test_event_filter_with_person_properties_materialized( - self, - _name: str, - poe1_enabled: bool, - poe2_enabled: bool, - allow_denormalised_props: bool, - materialize_person_props: bool, - ) -> None: - # KLUDGE: I couldn't figure out how to use @also_test_with_materialized_columns(person_properties=["email"]) - # KLUDGE: and the parameterized.expand decorator at the same time, so we generate test case combos - # KLUDGE: for materialization on and off to test both sides the way the decorator would have - if materialize_person_props: - materialize("events", "email", table_column="person_properties") - materialize("person", "email") - - with self.settings( - PERSON_ON_EVENTS_OVERRIDE=poe1_enabled, - PERSON_ON_EVENTS_V2_OVERRIDE=poe2_enabled, - ALLOW_DENORMALIZED_PROPS_IN_LISTING=allow_denormalised_props, - ): - user_one = "test_event_filter_with_person_properties-user" - user_two = "test_event_filter_with_person_properties-user2" - session_id_one = f"test_event_filter_with_person_properties-1-{str(uuid4())}" - session_id_two = f"test_event_filter_with_person_properties-2-{str(uuid4())}" - - Person.objects.create(team=self.team, distinct_ids=[user_one], properties={"email": "bla"}) - Person.objects.create(team=self.team, distinct_ids=[user_two], properties={"email": "bla2"}) - - self._add_replay_with_pageview(session_id_one, user_one) - produce_replay_summary( - distinct_id=user_one, - session_id=session_id_one, - first_timestamp=(self.base_time + relativedelta(seconds=30)), - team_id=self.team.id, - ) - self._add_replay_with_pageview(session_id_two, user_two) - produce_replay_summary( - distinct_id=user_two, - session_id=session_id_two, - first_timestamp=(self.base_time + relativedelta(seconds=30)), - team_id=self.team.id, - ) - - match_everyone_filter = SessionRecordingsFilter( - team=self.team, - data={"properties": []}, - ) - - session_recording_list_instance = SessionRecordingListFromFilters( - filter=match_everyone_filter, team=self.team, hogql_query_modifiers=None - ) - (session_recordings, _, _) = session_recording_list_instance.run() - - assert sorted([x["session_id"] for x in session_recordings]) == sorted([session_id_one, session_id_two]) - - match_bla_filter = SessionRecordingsFilter( - team=self.team, - data={ - "properties": [ - { - "key": "email", - "value": ["bla"], - "operator": "exact", - "type": "person", - } - ] - }, - ) - - session_recording_list_instance = SessionRecordingListFromFilters( - filter=match_bla_filter, team=self.team, hogql_query_modifiers=None - ) - (session_recordings, _, _) = session_recording_list_instance.run() - - assert len(session_recordings) == 1 - assert session_recordings[0]["session_id"] == session_id_one - - def _add_replay_with_pageview(self, session_id: str, user: str) -> None: - self.create_event( - user, - self.base_time, - properties={"$session_id": session_id, "$window_id": str(uuid4())}, - ) - produce_replay_summary( - distinct_id=user, - session_id=session_id, - first_timestamp=self.base_time, - team_id=self.team.id, - ) - - @parameterized.expand(test_case_combinations) - @snapshot_clickhouse_queries - def test_person_id_filter( - self, - _name: str, - poe2_enabled: bool, - poe1_enabled: bool, - allow_denormalised_props: bool, - materialize_person_props: bool, - ) -> None: - # KLUDGE: I couldn't figure out how to use @also_test_with_materialized_columns(person_properties=["email"]) - # KLUDGE: and the parameterized.expand decorator at the same time, so we generate test case combos - # KLUDGE: for materialization on and off to test both sides the way the decorator would have - if materialize_person_props: - # it shouldn't matter to this test whether any column is materialized - # but let's keep the tests in this file similar so we flush out any unexpected interactions - materialize("events", "email", table_column="person_properties") - materialize("person", "email") - - with self.settings( - PERSON_ON_EVENTS_OVERRIDE=poe1_enabled, - PERSON_ON_EVENTS_V2_OVERRIDE=poe2_enabled, - ALLOW_DENORMALIZED_PROPS_IN_LISTING=allow_denormalised_props, - ): - three_user_ids = ["person-1-distinct-1", "person-1-distinct-2", "person-2"] - session_id_one = f"test_person_id_filter-session-one" - session_id_two = f"test_person_id_filter-session-two" - session_id_three = f"test_person_id_filter-session-three" - - p = Person.objects.create( - team=self.team, - distinct_ids=[three_user_ids[0], three_user_ids[1]], - properties={"email": "bla"}, - ) - Person.objects.create( - team=self.team, - distinct_ids=[three_user_ids[2]], - properties={"email": "bla2"}, - ) - - self._add_replay_with_pageview(session_id_one, three_user_ids[0]) - self._add_replay_with_pageview(session_id_two, three_user_ids[1]) - self._add_replay_with_pageview(session_id_three, three_user_ids[2]) - - filter = SessionRecordingsFilter(team=self.team, data={"person_uuid": str(p.uuid)}) - session_recording_list_instance = SessionRecordingListFromFilters( - filter=filter, team=self.team, hogql_query_modifiers=None - ) - (session_recordings, _, _) = session_recording_list_instance.run() - assert sorted([r["session_id"] for r in session_recordings]) == sorted([session_id_two, session_id_one]) diff --git a/ee/session_recordings/session_recording_playlist.py b/ee/session_recordings/session_recording_playlist.py index a3dc50c1228f5..7b1a962b187b7 100644 --- a/ee/session_recordings/session_recording_playlist.py +++ b/ee/session_recordings/session_recording_playlist.py @@ -1,4 +1,3 @@ -import json from typing import Any, Optional import structlog @@ -13,7 +12,7 @@ from posthog.api.forbid_destroy_model import ForbidDestroyModel from posthog.api.routing import TeamAndOrgViewSetMixin from posthog.api.shared import UserBasicSerializer -from posthog.constants import SESSION_RECORDINGS_FILTER_IDS, AvailableFeature +from posthog.constants import AvailableFeature from posthog.models import ( SessionRecording, SessionRecordingPlaylist, @@ -27,7 +26,6 @@ changes_between, log_activity, ) -from posthog.models.filters.session_recordings_filter import SessionRecordingsFilter from posthog.models.team.team import check_is_feature_available_for_team from posthog.models.utils import UUIDT from posthog.rate_limit import ( @@ -37,7 +35,6 @@ from posthog.schema import RecordingsQuery from posthog.session_recordings.session_recording_api import ( list_recordings_response, - list_recordings, query_as_params_to_dict, list_recordings_from_query, ) @@ -230,19 +227,12 @@ def recordings(self, request: request.Request, *args: Any, **kwargs: Any) -> res .values_list("recording_id", flat=True) ) - use_query_type = (request.GET.get("as_query", "False")).lower() == "true" - - if use_query_type: - data_dict = query_as_params_to_dict(request.GET.dict()) - query = RecordingsQuery.model_validate(data_dict) - query.session_ids = playlist_items - return list_recordings_response( - list_recordings_from_query(query, request, context=self.get_serializer_context()) - ) - else: - filter = SessionRecordingsFilter(request=request, team=self.team) - filter = filter.shallow_clone({SESSION_RECORDINGS_FILTER_IDS: json.dumps(playlist_items)}) - return list_recordings_response(list_recordings(filter, request, context=self.get_serializer_context())) + data_dict = query_as_params_to_dict(request.GET.dict()) + query = RecordingsQuery.model_validate(data_dict) + query.session_ids = playlist_items + return list_recordings_response( + list_recordings_from_query(query, request, context=self.get_serializer_context()) + ) # As of now, you can only "update" a session recording by adding or removing a recording from a static playlist @action( diff --git a/posthog/constants.py b/posthog/constants.py index 063d2894a3c12..af2ede6c13e68 100644 --- a/posthog/constants.py +++ b/posthog/constants.py @@ -44,8 +44,6 @@ class AvailableFeature(StrEnum): TREND_FILTER_TYPE_EVENTS = "events" TREND_FILTER_TYPE_DATA_WAREHOUSE = "data_warehouse" -SESSION_RECORDINGS_FILTER_IDS = "session_ids" - TRENDS_CUMULATIVE = "ActionsLineGraphCumulative" TRENDS_LINEAR = "ActionsLineGraph" TRENDS_TABLE = "ActionsTable" diff --git a/posthog/hogql/transforms/test/__snapshots__/test_in_cohort.ambr b/posthog/hogql/transforms/test/__snapshots__/test_in_cohort.ambr index 804183be31efe..0bccd58cfb853 100644 --- a/posthog/hogql/transforms/test/__snapshots__/test_in_cohort.ambr +++ b/posthog/hogql/transforms/test/__snapshots__/test_in_cohort.ambr @@ -42,7 +42,7 @@ FROM events LEFT JOIN ( SELECT person_id AS cohort_person_id, 1 AS matched, cohort_id FROM static_cohort_people - WHERE in(cohort_id, [11])) AS __in_cohort ON equals(__in_cohort.cohort_person_id, person_id) + WHERE in(cohort_id, [1, 2, 3, 4, 5 /* ... */])) AS __in_cohort ON equals(__in_cohort.cohort_person_id, person_id) WHERE and(1, equals(__in_cohort.matched, 1)) LIMIT 100 ''' @@ -66,7 +66,7 @@ FROM events LEFT JOIN ( SELECT person_id AS cohort_person_id, 1 AS matched, cohort_id FROM static_cohort_people - WHERE in(cohort_id, [12])) AS __in_cohort ON equals(__in_cohort.cohort_person_id, person_id) + WHERE in(cohort_id, [1, 2, 3, 4, 5 /* ... */])) AS __in_cohort ON equals(__in_cohort.cohort_person_id, person_id) WHERE and(1, equals(__in_cohort.matched, 1)) LIMIT 100 ''' diff --git a/posthog/models/filters/__init__.py b/posthog/models/filters/__init__.py index fa75a96fc7308..89b22e15cb06f 100644 --- a/posthog/models/filters/__init__.py +++ b/posthog/models/filters/__init__.py @@ -3,7 +3,6 @@ from .path_filter import PathFilter from .retention_filter import RetentionFilter from .stickiness_filter import StickinessFilter -from .session_recordings_filter import SessionRecordingsFilter from .properties_timeline_filter import PropertiesTimelineFilter __all__ = [ @@ -11,13 +10,10 @@ "PathFilter", "RetentionFilter", "StickinessFilter", - "SessionRecordingsFilter", "PropertiesTimelineFilter", "AnyFilter", ] -AnyFilter: TypeAlias = ( - Filter | PathFilter | RetentionFilter | StickinessFilter | SessionRecordingsFilter | PropertiesTimelineFilter -) +AnyFilter: TypeAlias = Filter | PathFilter | RetentionFilter | StickinessFilter | PropertiesTimelineFilter AnyInsightFilter: TypeAlias = Filter | PathFilter | RetentionFilter | StickinessFilter diff --git a/posthog/models/filters/mixins/session_recordings.py b/posthog/models/filters/mixins/session_recordings.py deleted file mode 100644 index e2df650f4a082..0000000000000 --- a/posthog/models/filters/mixins/session_recordings.py +++ /dev/null @@ -1,66 +0,0 @@ -import json -from typing import Optional, Literal, Union - -from posthog.hogql import ast -from posthog.constants import PERSON_UUID_FILTER, SESSION_RECORDINGS_FILTER_IDS, PropertyOperatorType -from posthog.models.filters.mixins.common import BaseParamMixin -from posthog.models.filters.mixins.property import PropertyMixin -from posthog.models.filters.mixins.utils import cached_property -from posthog.models.property import PropertyGroup - - -class PersonUUIDMixin(BaseParamMixin): - @cached_property - def person_uuid(self) -> Optional[str]: - return self._data.get(PERSON_UUID_FILTER, None) - - -class SessionRecordingsMixin(PropertyMixin, BaseParamMixin): - @cached_property - def order(self) -> str: - return self._data.get("order", "start_time") - - @cached_property - def console_log_filters(self) -> PropertyGroup: - property_group = self._parse_data(key="console_log_filters") - property_group.type = self.property_operand - return property_group - - @cached_property - def property_operand(self) -> PropertyOperatorType: - return PropertyOperatorType.AND if self._operand == "AND" else PropertyOperatorType.OR - - @cached_property - def ast_operand(self) -> type[Union[ast.And, ast.Or]]: - return ast.And if self._operand == "AND" else ast.Or - - @cached_property - def _operand(self) -> Literal["AND"] | Literal["OR"]: - return self._data.get("operand", "AND") - - @cached_property - def session_ids(self) -> Optional[list[str]]: - # Can be ['a', 'b'] or "['a', 'b']" or "a,b" - session_ids_str = self._data.get(SESSION_RECORDINGS_FILTER_IDS, None) - - if session_ids_str is None: - return None - - if isinstance(session_ids_str, list): - recordings_ids = session_ids_str - elif isinstance(session_ids_str, str): - if session_ids_str.startswith("["): - recordings_ids = json.loads(session_ids_str) - else: - recordings_ids = session_ids_str.split(",") - - if all(isinstance(recording_id, str) for recording_id in recordings_ids): - # Sort for stable queries - return sorted(recordings_ids) - - # If the property is at all present, we assume that the user wants to filter by it - return [] - - @cached_property - def having_predicates(self) -> PropertyGroup: - return self._parse_data(key="having_predicates") diff --git a/posthog/models/filters/session_recordings_filter.py b/posthog/models/filters/session_recordings_filter.py deleted file mode 100644 index f246e14ba0850..0000000000000 --- a/posthog/models/filters/session_recordings_filter.py +++ /dev/null @@ -1,6 +0,0 @@ -from .filter import Filter -from .mixins.session_recordings import PersonUUIDMixin, SessionRecordingsMixin - - -class SessionRecordingsFilter(SessionRecordingsMixin, PersonUUIDMixin, Filter): - pass diff --git a/posthog/queries/event_query/event_query.py b/posthog/queries/event_query/event_query.py index 8bb19d73dbcd7..7bab914cd0cda 100644 --- a/posthog/queries/event_query/event_query.py +++ b/posthog/queries/event_query/event_query.py @@ -10,7 +10,6 @@ from posthog.models.filters.path_filter import PathFilter from posthog.models.filters.properties_timeline_filter import PropertiesTimelineFilter from posthog.models.filters.retention_filter import RetentionFilter -from posthog.models.filters.session_recordings_filter import SessionRecordingsFilter from posthog.models.filters.stickiness_filter import StickinessFilter from posthog.models.property import PropertyGroup, PropertyName from posthog.models.property.util import parse_prop_grouped_clauses @@ -52,7 +51,6 @@ def __init__( PathFilter, RetentionFilter, StickinessFilter, - SessionRecordingsFilter, PropertiesTimelineFilter, ], team: Team, diff --git a/posthog/session_recordings/queries/session_recording_list_from_filters.py b/posthog/session_recordings/queries/session_recording_list_from_filters.py deleted file mode 100644 index 38522ae4d18ce..0000000000000 --- a/posthog/session_recordings/queries/session_recording_list_from_filters.py +++ /dev/null @@ -1,651 +0,0 @@ -import re -from typing import Any, NamedTuple, cast, Optional, Union -from datetime import datetime, timedelta - -import posthoganalytics - -from posthog.hogql import ast -from posthog.hogql.ast import CompareOperation -from posthog.hogql.constants import HogQLGlobalSettings -from posthog.hogql.parser import parse_select -from posthog.hogql.property import entity_to_expr, property_to_expr -from posthog.hogql.query import execute_hogql_query -from posthog.hogql_queries.insights.paginators import HogQLHasMorePaginator -from posthog.models import Team, Property -from posthog.models.filters.session_recordings_filter import SessionRecordingsFilter -from posthog.models.filters.mixins.utils import cached_property -from posthog.models.property import PropertyGroup -from posthog.schema import QueryTiming, HogQLQueryModifiers, PersonsOnEventsMode -from posthog.session_recordings.queries.session_replay_events import ttl_days -from posthog.constants import TREND_FILTER_TYPE_ACTIONS - -import structlog - -logger = structlog.get_logger(__name__) - - -def is_event_property(p: Property) -> bool: - return p.type == "event" or (p.type == "hogql" and bool(re.search(r"(? bool: - return p.type == "person" or (p.type == "hogql" and "person.properties" in p.key) - - -def is_group_property(p: Property) -> bool: - return p.type == "group" - - -def is_cohort_property(p: Property) -> bool: - return "cohort" in p.type - - -class SessionRecordingQueryResult(NamedTuple): - results: list - has_more_recording: bool - timings: list[QueryTiming] | None = None - - -class UnexpectedQueryProperties(Exception): - def __init__(self, remaining_properties: PropertyGroup | None): - self.remaining_properties = remaining_properties - super().__init__(f"Unexpected properties in query: {remaining_properties}") - - -class SessionRecordingListFromFilters: - SESSION_RECORDINGS_DEFAULT_LIMIT = 50 - - _team: Team - _filter: SessionRecordingsFilter - - BASE_QUERY: str = """ - SELECT s.session_id, - any(s.team_id), - any(s.distinct_id), - min(s.min_first_timestamp) as start_time, - max(s.max_last_timestamp) as end_time, - dateDiff('SECOND', start_time, end_time) as duration, - argMinMerge(s.first_url) as first_url, - sum(s.click_count) as click_count, - sum(s.keypress_count) as keypress_count, - sum(s.mouse_activity_count) as mouse_activity_count, - sum(s.active_milliseconds)/1000 as active_seconds, - (duration - active_seconds) as inactive_seconds, - sum(s.console_log_count) as console_log_count, - sum(s.console_warn_count) as console_warn_count, - sum(s.console_error_count) as console_error_count, - {ongoing_selection}, - round(( - ((sum(s.active_milliseconds) / 1000 + sum(s.click_count) + sum(s.keypress_count) + sum(s.console_error_count))) -- intent - / - ((sum(s.mouse_activity_count) + dateDiff('SECOND', start_time, end_time) + sum(s.console_error_count) + sum(s.console_log_count) + sum(s.console_warn_count))) - * 100 - ), 2) as activity_score - FROM raw_session_replay_events s - WHERE {where_predicates} - GROUP BY session_id - HAVING {having_predicates} - ORDER BY {order_by} DESC - """ - - @staticmethod - def _data_to_return(results: list[Any] | None) -> list[dict[str, Any]]: - default_columns = [ - "session_id", - "team_id", - "distinct_id", - "start_time", - "end_time", - "duration", - "first_url", - "click_count", - "keypress_count", - "mouse_activity_count", - "active_seconds", - "inactive_seconds", - "console_log_count", - "console_warn_count", - "console_error_count", - "ongoing", - "activity_score", - ] - - return [ - { - **dict(zip(default_columns, row[: len(default_columns)])), - } - for row in results or [] - ] - - def __init__( - self, - team: Team, - filter: SessionRecordingsFilter, - hogql_query_modifiers: Optional[HogQLQueryModifiers], - **_, - ): - self._team = team - self._filter = filter - self._paginator = HogQLHasMorePaginator( - limit=filter.limit or self.SESSION_RECORDINGS_DEFAULT_LIMIT, offset=filter.offset or 0 - ) - self._hogql_query_modifiers = hogql_query_modifiers - - @property - def ttl_days(self): - return ttl_days(self._team) - - def run(self) -> SessionRecordingQueryResult: - query = self.get_query() - - paginated_response = self._paginator.execute_hogql_query( - # TODO I guess the paginator needs to know how to handle union queries or all callers are supposed to collapse them or .... 🤷 - query=cast(ast.SelectQuery, query), - team=self._team, - query_type="SessionRecordingListQuery", - modifiers=self._hogql_query_modifiers, - settings=HogQLGlobalSettings(allow_experimental_analyzer=False), # This needs to be turned on eventually - ) - - return SessionRecordingQueryResult( - results=(self._data_to_return(self._paginator.results)), - has_more_recording=self._paginator.has_more(), - timings=paginated_response.timings, - ) - - def get_query(self): - return parse_select( - self.BASE_QUERY, - { - # Check if the most recent _timestamp is within five minutes of the current time - # proxy for a live session - "ongoing_selection": ast.Alias( - alias="ongoing", - expr=ast.CompareOperation( - left=ast.Call(name="max", args=[ast.Field(chain=["s", "_timestamp"])]), - right=ast.Constant( - # provided in a placeholder, so we can pass now from python to make tests easier 🙈 - value=datetime.utcnow() - timedelta(minutes=5), - ), - op=ast.CompareOperationOp.GtEq, - ), - ), - "order_by": self._order_by_clause(), - "where_predicates": self._where_predicates(), - "having_predicates": self._having_predicates(), - }, - ) - - def _order_by_clause(self) -> ast.Field: - return ast.Field(chain=[self._filter.order]) - - def _where_predicates(self) -> Union[ast.And, ast.Or]: - exprs: list[ast.Expr] = [ - ast.CompareOperation( - op=ast.CompareOperationOp.GtEq, - left=ast.Field(chain=["s", "min_first_timestamp"]), - right=ast.Constant(value=datetime.utcnow() - timedelta(days=self.ttl_days)), - ) - ] - - person_id_compare_operation = PersonsIdCompareOperation(self._team, self._filter, self.ttl_days).get_operation() - if person_id_compare_operation: - exprs.append(person_id_compare_operation) - - # we check for session_ids type not for truthiness since we want to allow empty lists - if isinstance(self._filter.session_ids, list): - exprs.append( - ast.CompareOperation( - op=ast.CompareOperationOp.In, - left=ast.Field(chain=["session_id"]), - right=ast.Constant(value=self._filter.session_ids), - ) - ) - - if self._filter.date_from: - exprs.append( - ast.CompareOperation( - op=ast.CompareOperationOp.GtEq, - left=ast.Field(chain=["s", "min_first_timestamp"]), - right=ast.Constant(value=self._filter.date_from), - ) - ) - if self._filter.date_to: - exprs.append( - ast.CompareOperation( - op=ast.CompareOperationOp.LtEq, - left=ast.Field(chain=["s", "min_first_timestamp"]), - right=ast.Constant(value=self._filter.date_to), - ) - ) - - optional_exprs: list[ast.Expr] = [] - - # if in PoE mode then we should be pushing person property queries into here - events_sub_query = ReplayFiltersEventsSubQuery(self._team, self._filter).get_query_for_session_id_matching() - if events_sub_query: - optional_exprs.append( - ast.CompareOperation( - op=ast.CompareOperationOp.In, - left=ast.Field(chain=["s", "session_id"]), - right=events_sub_query, - ) - ) - - # we want to avoid a join to persons since we don't ever need to select from them, - # so we create our own persons sub query here - # if PoE mode is on then this will be handled in the events subquery, and we don't need to do anything here - person_subquery = PersonsPropertiesSubQuery(self._team, self._filter, self.ttl_days).get_query() - if person_subquery: - optional_exprs.append( - ast.CompareOperation( - op=ast.CompareOperationOp.In, - left=ast.Field(chain=["s", "distinct_id"]), - right=person_subquery, - ) - ) - - cohort_subquery = CohortPropertyGroupsSubQuery(self._team, self._filter, self.ttl_days).get_query() - if cohort_subquery: - optional_exprs.append( - ast.CompareOperation( - op=ast.CompareOperationOp.In, - left=ast.Field(chain=["s", "distinct_id"]), - right=cohort_subquery, - ) - ) - - remaining_properties = self._strip_person_and_event_and_cohort_properties(self._filter.property_groups) - if remaining_properties: - posthoganalytics.capture_exception(UnexpectedQueryProperties(remaining_properties)) - optional_exprs.append(property_to_expr(remaining_properties, team=self._team, scope="replay")) - - if self._filter.console_log_filters.values: - console_logs_subquery = ast.SelectQuery( - select=[ast.Field(chain=["log_source_id"])], - select_from=ast.JoinExpr(table=ast.Field(chain=["console_logs_log_entries"])), - where=self._filter.ast_operand( - exprs=[ - property_to_expr(self._filter.console_log_filters, team=self._team), - ] - ), - ) - - optional_exprs.append( - ast.CompareOperation( - op=ast.CompareOperationOp.In, - left=ast.Field(chain=["session_id"]), - right=console_logs_subquery, - ) - ) - - if optional_exprs: - exprs.append(self._filter.ast_operand(exprs=optional_exprs)) - - return ast.And(exprs=exprs) - - def _having_predicates(self) -> ast.Expr: - return property_to_expr(self._filter.having_predicates, team=self._team, scope="replay") - - def _strip_person_and_event_and_cohort_properties(self, property_group: PropertyGroup) -> PropertyGroup | None: - property_groups_to_keep = [ - g - for g in property_group.flat - if not is_event_property(g) - and not is_person_property(g) - and not is_group_property(g) - and not is_cohort_property(g) - ] - - return ( - PropertyGroup( - type=self._filter.property_operand, - values=property_groups_to_keep, - ) - if property_groups_to_keep - else None - ) - - -def poe_is_active(team: Team) -> bool: - return team.person_on_events_mode is not None and team.person_on_events_mode != PersonsOnEventsMode.DISABLED - - -class PersonsPropertiesSubQuery: - _team: Team - _filter: SessionRecordingsFilter - _ttl_days: int - - def __init__(self, team: Team, filter: SessionRecordingsFilter, ttl_days: int): - self._team = team - self._filter = filter - self._ttl_days = ttl_days - - def get_query(self) -> ast.SelectQuery | ast.SelectSetQuery | None: - if self.person_properties and not poe_is_active(self._team): - return parse_select( - """ - SELECT distinct_id - FROM person_distinct_ids - WHERE {where_predicates} - """, - { - "where_predicates": self._where_predicates, - }, - ) - else: - return None - - @cached_property - def person_properties(self) -> PropertyGroup | None: - person_property_groups = [g for g in self._filter.property_groups.flat if is_person_property(g)] - return ( - PropertyGroup( - type=self._filter.property_operand, - values=person_property_groups, - ) - if person_property_groups - else None - ) - - @cached_property - def _where_predicates(self) -> ast.Expr: - return ( - property_to_expr(self.person_properties, team=self._team) - if self.person_properties - else ast.Constant(value=True) - ) - - -class CohortPropertyGroupsSubQuery: - _team: Team - _filter: SessionRecordingsFilter - _ttl_days: int - - raw_cohort_to_distinct_id = """ - SELECT - distinct_id -FROM raw_person_distinct_ids -WHERE distinct_id in (SELECT distinct_id FROM raw_person_distinct_ids WHERE 1=1 AND {cohort_predicate}) -GROUP BY distinct_id -HAVING argMax(is_deleted, version) = 0 AND {cohort_predicate} - """ - - def __init__(self, team: Team, filter: SessionRecordingsFilter, ttl_days: int): - self._team = team - self._filter = filter - self._ttl_days = ttl_days - - def get_query(self) -> ast.SelectQuery | ast.SelectSetQuery | None: - if self.cohort_properties: - return parse_select( - self.raw_cohort_to_distinct_id, - {"cohort_predicate": property_to_expr(self.cohort_properties, team=self._team, scope="replay")}, - ) - - return None - - @cached_property - def cohort_properties(self) -> PropertyGroup | None: - cohort_property_groups = [g for g in self._filter.property_groups.flat if is_cohort_property(g)] - return ( - PropertyGroup( - type=self._filter.property_operand, - values=cohort_property_groups, - ) - if cohort_property_groups - else None - ) - - -class PersonsIdCompareOperation: - _team: Team - _filter: SessionRecordingsFilter - _ttl_days: int - - def __init__(self, team: Team, filter: SessionRecordingsFilter, ttl_days: int): - self._team = team - self._filter = filter - self._ttl_days = ttl_days - - def get_operation(self) -> CompareOperation | None: - q = self.get_query() - if not q: - return None - - if poe_is_active(self._team): - return ast.CompareOperation( - op=ast.CompareOperationOp.In, - left=ast.Field(chain=["session_id"]), - right=q, - ) - else: - return ast.CompareOperation( - op=ast.CompareOperationOp.In, - left=ast.Field(chain=["distinct_id"]), - right=q, - ) - - def get_query(self) -> ast.SelectQuery | ast.SelectSetQuery | None: - if not self._filter.person_uuid: - return None - - # anchor to python now so that tests can freeze time - now = datetime.utcnow() - - if poe_is_active(self._team): - return parse_select( - """ - select - distinct `$session_id` - from - events - where - person_id = {person_id} - and timestamp <= {now} - and timestamp >= {ttl_date} - and timestamp >= {date_from} - and timestamp <= {date_to} - and notEmpty(`$session_id`) - """, - { - "person_id": ast.Constant(value=self._filter.person_uuid), - "ttl_days": ast.Constant(value=self._ttl_days), - "date_from": ast.Constant(value=self._filter.date_from), - "date_to": ast.Constant(value=self._filter.date_to), - "now": ast.Constant(value=now), - "ttl_date": ast.Constant(value=now - timedelta(days=self._ttl_days)), - }, - ) - else: - return parse_select( - """ - SELECT distinct_id - FROM person_distinct_ids - WHERE person_id = {person_id} - """, - { - "person_id": ast.Constant(value=self._filter.person_uuid), - }, - ) - - -class ReplayFiltersEventsSubQuery: - _team: Team - _filter: SessionRecordingsFilter - - @property - def ttl_days(self): - return ttl_days(self._team) - - def __init__( - self, - team: Team, - filter: SessionRecordingsFilter, - hogql_query_modifiers: Optional[HogQLQueryModifiers] = None, - ): - self._team = team - self._filter = filter - self._hogql_query_modifiers = hogql_query_modifiers - - @cached_property - def _event_predicates(self): - event_exprs: list[ast.Expr] = [] - event_names: set[int | str] = set() - - for entity in self._filter.entities: - if entity.type == TREND_FILTER_TYPE_ACTIONS: - action = entity.get_action() - event_names.update([ae for ae in action.get_step_events() if ae and ae not in event_names]) - else: - if entity.id and entity.id not in event_names: - event_names.add(entity.id) - - # TODO: we're not passing the "right" type in here - should we change the signature or do something else? - entity_exprs = [entity_to_expr(entity=entity)] # type: ignore - - if entity.property_groups: - entity_exprs.append(property_to_expr(entity.property_groups, team=self._team, scope="replay_entity")) - - event_exprs.append(ast.And(exprs=entity_exprs)) - - return event_exprs, list(event_names) - - def _select_from_events(self, select_expr: ast.Expr) -> ast.SelectQuery: - return ast.SelectQuery( - select=[select_expr], - select_from=ast.JoinExpr( - table=ast.Field(chain=["events"]), - ), - where=self._where_predicates(), - having=self._having_predicates(), - group_by=[ast.Field(chain=["$session_id"])], - ) - - def get_query_for_session_id_matching(self) -> ast.SelectQuery | ast.SelectSetQuery | None: - use_poe = poe_is_active(self._team) and self.person_properties - if self._filter.entities or self.event_properties or self.group_properties or use_poe: - return self._select_from_events(ast.Alias(alias="session_id", expr=ast.Field(chain=["$session_id"]))) - else: - return None - - def get_query_for_event_id_matching(self) -> ast.SelectQuery | ast.SelectSetQuery: - return self._select_from_events(ast.Call(name="groupUniqArray", args=[ast.Field(chain=["uuid"])])) - - def get_event_ids_for_session(self) -> SessionRecordingQueryResult: - query = self.get_query_for_event_id_matching() - - hogql_query_response = execute_hogql_query( - query=query, - team=self._team, - query_type="SessionRecordingMatchingEventsForSessionQuery", - modifiers=self._hogql_query_modifiers, - ) - - flattened_results = [str(uuid) for row in hogql_query_response.results for uuid in row[0]] - - return SessionRecordingQueryResult( - results=flattened_results, - has_more_recording=False, - timings=hogql_query_response.timings, - ) - - def _where_predicates(self) -> ast.Expr: - exprs: list[ast.Expr] = [ - ast.Call( - name="notEmpty", - args=[ast.Field(chain=["$session_id"])], - ), - # regardless of any other filters limit between TTL and current time - ast.CompareOperation( - op=ast.CompareOperationOp.GtEq, - left=ast.Field(chain=["timestamp"]), - right=ast.Constant(value=datetime.now() - timedelta(days=self.ttl_days)), - ), - ast.CompareOperation( - op=ast.CompareOperationOp.LtEq, - left=ast.Field(chain=["timestamp"]), - right=ast.Call(name="now", args=[]), - ), - ] - - # TRICKY: we're adding a buffer to the date range to ensure we get all the events - # you can start sending us events before the session starts - if self._filter.date_from: - exprs.append( - ast.CompareOperation( - op=ast.CompareOperationOp.GtEq, - left=ast.Field(chain=["timestamp"]), - right=ast.Constant(value=self._filter.date_from - timedelta(minutes=2)), - ) - ) - - # but we don't want to include events after date_to if provided - if self._filter.date_to: - exprs.append( - ast.CompareOperation( - op=ast.CompareOperationOp.LtEq, - left=ast.Field(chain=["timestamp"]), - right=ast.Constant(value=self._filter.date_to), - ) - ) - - (event_where_exprs, _) = self._event_predicates - if event_where_exprs: - # we OR all events in the where and use hasAll / hasAny in the HAVING clause - exprs.append(ast.Or(exprs=event_where_exprs)) - - if self.event_properties: - exprs.append(property_to_expr(self.event_properties, team=self._team, scope="replay")) - - if self.group_properties: - exprs.append(property_to_expr(self.group_properties, team=self._team)) - - if self._team.person_on_events_mode and self.person_properties: - exprs.append(property_to_expr(self.person_properties, team=self._team, scope="event")) - - if self._filter.session_ids: - exprs.append( - ast.CompareOperation( - op=ast.CompareOperationOp.In, - left=ast.Field(chain=["$session_id"]), - right=ast.Constant(value=self._filter.session_ids), - ) - ) - - return ast.And(exprs=exprs) - - def _having_predicates(self) -> ast.Expr: - (_, event_names) = self._event_predicates - - if event_names: - return ast.Call( - name="hasAll" if self._filter._operand == "AND" else "hasAny", - args=[ - ast.Call(name="groupUniqArray", args=[ast.Field(chain=["event"])]), - # KLUDGE: sorting only so that snapshot tests are consistent - ast.Constant(value=sorted(event_names)), - ], - ) - - return ast.Constant(value=True) - - @cached_property - def event_properties(self): - return [g for g in self._filter.property_groups.flat if is_event_property(g)] - - @cached_property - def group_properties(self): - return [g for g in self._filter.property_groups.flat if is_group_property(g)] - - @cached_property - def person_properties(self) -> PropertyGroup | None: - person_property_groups = [g for g in self._filter.property_groups.flat if is_person_property(g)] - return ( - PropertyGroup( - type=self._filter.property_operand, - values=person_property_groups, - ) - if person_property_groups - else None - ) diff --git a/posthog/session_recordings/queries/session_recording_properties.py b/posthog/session_recordings/queries/session_recording_properties.py deleted file mode 100644 index 2d2ef187c0407..0000000000000 --- a/posthog/session_recordings/queries/session_recording_properties.py +++ /dev/null @@ -1,106 +0,0 @@ -from datetime import timedelta -from typing import TYPE_CHECKING, Any, NamedTuple - -from posthog.client import sync_execute -from posthog.models.event.util import parse_properties -from posthog.models.filters.session_recordings_filter import SessionRecordingsFilter -from posthog.queries.event_query import EventQuery - -if TYPE_CHECKING: - from posthog.models import Team - - -class EventFiltersSQL(NamedTuple): - aggregate_select_clause: str - aggregate_having_clause: str - where_conditions: str - params: dict[str, Any] - - -class SessionRecordingProperties(EventQuery): - _filter: SessionRecordingsFilter - _session_ids: list[str] - - SESSION_RECORDING_PROPERTIES_ALLOWLIST = { - "$os", - "$browser", - "$device_type", - "$current_url", - "$host", - "$pathname", - "$geoip_country_code", - "$geoip_country_name", - } - - # First $pageview event in a recording is used to extract metadata (brower, location, etc.) without - # having to return all events. - _core_single_pageview_event_query = """ - SELECT - "$session_id" AS session_id, - any(properties) AS properties - FROM events - PREWHERE - team_id = %(team_id)s - AND event IN ['$pageview', '$autocapture'] - {session_ids_clause} - {events_timestamp_clause} - GROUP BY session_id - """ - - def __init__(self, team: "Team", session_ids: list[str], filter: SessionRecordingsFilter): - super().__init__(team=team, filter=filter) - self._session_ids = sorted(session_ids) # Sort for stable queries - - def _determine_should_join_distinct_ids(self) -> None: - self._should_join_distinct_ids = False - - # We want to select events beyond the range of the recording to handle the case where - # a recording spans the time boundaries - def _get_events_timestamp_clause(self) -> tuple[str, dict[str, Any]]: - timestamp_clause = "" - timestamp_params = {} - if self._filter.date_from: - timestamp_clause += "\nAND timestamp >= %(event_start_time)s" - timestamp_params["event_start_time"] = self._filter.date_from - timedelta(hours=12) - if self._filter.date_to: - timestamp_clause += "\nAND timestamp <= %(event_end_time)s" - timestamp_params["event_end_time"] = self._filter.date_to + timedelta(hours=12) - return timestamp_clause, timestamp_params - - def format_session_recording_id_filters(self) -> tuple[str, dict]: - where_conditions = "AND session_id IN %(session_ids)s" - return where_conditions, {"session_ids": self._session_ids} - - def get_query(self) -> tuple[str, dict[str, Any]]: - base_params = {"team_id": self._team_id} - ( - events_timestamp_clause, - events_timestamp_params, - ) = self._get_events_timestamp_clause() - ( - session_ids_clause, - session_ids_params, - ) = self.format_session_recording_id_filters() - - return ( - self._core_single_pageview_event_query.format( - events_timestamp_clause=events_timestamp_clause, - session_ids_clause=session_ids_clause, - ), - {**base_params, **events_timestamp_params, **session_ids_params}, - ) - - def _data_to_return(self, results: list[Any]) -> list[dict[str, Any]]: - return [ - { - "session_id": row[0], - "properties": parse_properties(row[1], self.SESSION_RECORDING_PROPERTIES_ALLOWLIST), - } - for row in results - ] - - def run(self) -> list: - query, query_params = self.get_query() - query_results = sync_execute(query, query_params) - session_recording_properties = self._data_to_return(query_results) - return session_recording_properties diff --git a/posthog/session_recordings/queries/test/__snapshots__/test_session_recording_list_from_filters.ambr b/posthog/session_recordings/queries/test/__snapshots__/test_session_recording_list_from_filters.ambr deleted file mode 100644 index 82e70f3063a2b..0000000000000 --- a/posthog/session_recordings/queries/test/__snapshots__/test_session_recording_list_from_filters.ambr +++ /dev/null @@ -1,5812 +0,0 @@ -# serializer version: 1 -# name: TestSessionRecordingsListFromFilters.test_action_filter - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2023-01-03 23:55:00.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2022-12-14 00:00:00.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2022-12-28 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2023-01-04 00:00:00.000000', 6, 'UTC')), 0), in(s.session_id, - (SELECT events.`$session_id` AS session_id - FROM events - WHERE and(equals(events.team_id, 99999), notEmpty(events.`$session_id`), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2022-12-14 00:00:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), now64(6, 'UTC')), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2022-12-27 23:58:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2023-01-04 00:00:00.000000', 6, 'UTC')), and(and(equals(events.event, 'custom-event'), and(ifNull(equals(replaceRegexpAll(nullIf(nullIf(JSONExtractRaw(events.properties, '$browser'), ''), 'null'), '^"|"$', ''), 'Firefox'), 0), ifNull(equals(nullIf(nullIf(events.`$session_id`, ''), 'null'), 'test_action_filter-session-one'), 0), ifNull(equals(nullIf(nullIf(events.`$window_id`, ''), 'null'), 'test_action_filter-window-id'), 0))), 1)) - GROUP BY events.`$session_id` - HAVING hasAll(groupUniqArray(events.event), ['custom-event'])))) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestSessionRecordingsListFromFilters.test_action_filter.1 - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2023-01-03 23:55:00.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2022-12-14 00:00:00.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2022-12-28 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2023-01-04 00:00:00.000000', 6, 'UTC')), 0), in(s.session_id, - (SELECT events.`$session_id` AS session_id - FROM events - WHERE and(equals(events.team_id, 99999), notEmpty(events.`$session_id`), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2022-12-14 00:00:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), now64(6, 'UTC')), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2022-12-27 23:58:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2023-01-04 00:00:00.000000', 6, 'UTC')), and(and(equals(events.event, 'custom-event'), and(ifNull(equals(nullIf(nullIf(events.`$session_id`, ''), 'null'), 'test_action_filter-session-one'), 0), ifNull(equals(nullIf(nullIf(events.`$window_id`, ''), 'null'), 'test_action_filter-window-id'), 0))), 1)) - GROUP BY events.`$session_id` - HAVING hasAll(groupUniqArray(events.event), ['custom-event'])))) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestSessionRecordingsListFromFilters.test_action_filter.2 - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2023-01-03 23:55:00.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2022-12-14 00:00:00.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2022-12-28 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2023-01-04 00:00:00.000000', 6, 'UTC')), 0), in(s.session_id, - (SELECT events.`$session_id` AS session_id - FROM events - WHERE and(equals(events.team_id, 99999), notEmpty(events.`$session_id`), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2022-12-14 00:00:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), now64(6, 'UTC')), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2022-12-27 23:58:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2023-01-04 00:00:00.000000', 6, 'UTC')), and(and(equals(events.event, 'custom-event'), and(ifNull(equals(nullIf(nullIf(events.`$session_id`, ''), 'null'), 'test_action_filter-session-one'), 0), ifNull(equals(nullIf(nullIf(events.`$window_id`, ''), 'null'), 'test_action_filter-window-id'), 0))), ifNull(equals(replaceRegexpAll(nullIf(nullIf(JSONExtractRaw(events.properties, '$browser'), ''), 'null'), '^"|"$', ''), 'Firefox'), 0))) - GROUP BY events.`$session_id` - HAVING hasAll(groupUniqArray(events.event), ['custom-event'])))) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestSessionRecordingsListFromFilters.test_action_filter.3 - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2023-01-03 23:55:00.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2022-12-14 00:00:00.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2022-12-28 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2023-01-04 00:00:00.000000', 6, 'UTC')), 0), in(s.session_id, - (SELECT events.`$session_id` AS session_id - FROM events - WHERE and(equals(events.team_id, 99999), notEmpty(events.`$session_id`), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2022-12-14 00:00:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), now64(6, 'UTC')), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2022-12-27 23:58:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2023-01-04 00:00:00.000000', 6, 'UTC')), and(and(equals(events.event, 'custom-event'), and(ifNull(equals(nullIf(nullIf(events.`$session_id`, ''), 'null'), 'test_action_filter-session-one'), 0), ifNull(equals(nullIf(nullIf(events.`$window_id`, ''), 'null'), 'test_action_filter-window-id'), 0))), ifNull(equals(replaceRegexpAll(nullIf(nullIf(JSONExtractRaw(events.properties, '$browser'), ''), 'null'), '^"|"$', ''), 'Chrome'), 0))) - GROUP BY events.`$session_id` - HAVING hasAll(groupUniqArray(events.event), ['custom-event'])))) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestSessionRecordingsListFromFilters.test_all_filters_at_once - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-01 13:41:23.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), 0), in(s.session_id, - (SELECT DISTINCT events.`$session_id` AS `$session_id` - FROM events - LEFT OUTER JOIN - (SELECT argMax(person_distinct_id_overrides.person_id, person_distinct_id_overrides.version) AS person_id, person_distinct_id_overrides.distinct_id AS distinct_id - FROM person_distinct_id_overrides - WHERE equals(person_distinct_id_overrides.team_id, 99999) - GROUP BY person_distinct_id_overrides.distinct_id - HAVING ifNull(equals(argMax(person_distinct_id_overrides.is_deleted, person_distinct_id_overrides.version), 0), 0) SETTINGS optimize_aggregation_in_order=1) AS events__override ON equals(events.distinct_id, events__override.distinct_id) - WHERE and(equals(events.team_id, 99999), ifNull(equals(if(not(empty(events__override.distinct_id)), events__override.person_id, events.person_id), '00000000-0000-0000-0000-000000000000'), 0), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-22 00:00:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-04 23:59:59.999999', 6, 'UTC')), notEmpty(events.`$session_id`)))), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-22 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-04 23:59:59.999999', 6, 'UTC')), 0), in(s.session_id, - (SELECT events.`$session_id` AS session_id - FROM events - WHERE and(equals(events.team_id, 99999), notEmpty(events.`$session_id`), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), now64(6, 'UTC')), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-21 23:58:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-04 23:59:59.999999', 6, 'UTC')), or(and(equals(events.event, '$pageview'), 1), and(equals(events.event, 'custom-event'), 1))) - GROUP BY events.`$session_id` - HAVING hasAll(groupUniqArray(events.event), ['$pageview', 'custom-event'])))) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestSessionRecordingsListFromFilters.test_any_event_filter_with_properties - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-21 19:55:00.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-31 20:00:00.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-14 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-21 20:00:00.000000', 6, 'UTC')), 0), in(s.session_id, - (SELECT events.`$session_id` AS session_id - FROM events - WHERE and(equals(events.team_id, 99999), notEmpty(events.`$session_id`), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-31 20:00:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), now64(6, 'UTC')), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-13 23:58:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-21 20:00:00.000000', 6, 'UTC')), and(1, 1)) - GROUP BY events.`$session_id` - HAVING 1))) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestSessionRecordingsListFromFilters.test_any_event_filter_with_properties.1 - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-21 19:55:00.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-31 20:00:00.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-14 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-21 20:00:00.000000', 6, 'UTC')), 0), in(s.session_id, - (SELECT events.`$session_id` AS session_id - FROM events - WHERE and(equals(events.team_id, 99999), notEmpty(events.`$session_id`), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-31 20:00:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), now64(6, 'UTC')), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-13 23:58:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-21 20:00:00.000000', 6, 'UTC')), and(1, ifNull(equals(replaceRegexpAll(nullIf(nullIf(JSONExtractRaw(events.properties, '$browser'), ''), 'null'), '^"|"$', ''), 'Chrome'), 0))) - GROUP BY events.`$session_id` - HAVING 1))) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestSessionRecordingsListFromFilters.test_any_event_filter_with_properties.2 - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-21 19:55:00.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-31 20:00:00.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-14 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-21 20:00:00.000000', 6, 'UTC')), 0), in(s.session_id, - (SELECT events.`$session_id` AS session_id - FROM events - WHERE and(equals(events.team_id, 99999), notEmpty(events.`$session_id`), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-31 20:00:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), now64(6, 'UTC')), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-13 23:58:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-21 20:00:00.000000', 6, 'UTC')), and(1, ifNull(equals(replaceRegexpAll(nullIf(nullIf(JSONExtractRaw(events.properties, '$browser'), ''), 'null'), '^"|"$', ''), 'Firefox'), 0))) - GROUP BY events.`$session_id` - HAVING 1))) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestSessionRecordingsListFromFilters.test_any_event_filter_with_properties_materialized - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-21 19:55:00.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-31 20:00:00.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-14 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-21 20:00:00.000000', 6, 'UTC')), 0), in(s.session_id, - (SELECT events.`$session_id` AS session_id - FROM events - WHERE and(equals(events.team_id, 99999), notEmpty(events.`$session_id`), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-31 20:00:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), now64(6, 'UTC')), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-13 23:58:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-21 20:00:00.000000', 6, 'UTC')), and(1, 1)) - GROUP BY events.`$session_id` - HAVING 1))) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestSessionRecordingsListFromFilters.test_any_event_filter_with_properties_materialized.1 - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-21 19:55:00.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-31 20:00:00.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-14 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-21 20:00:00.000000', 6, 'UTC')), 0), in(s.session_id, - (SELECT events.`$session_id` AS session_id - FROM events - WHERE and(equals(events.team_id, 99999), notEmpty(events.`$session_id`), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-31 20:00:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), now64(6, 'UTC')), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-13 23:58:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-21 20:00:00.000000', 6, 'UTC')), and(1, ifNull(equals(nullIf(nullIf(events.`mat_$browser`, ''), 'null'), 'Chrome'), 0))) - GROUP BY events.`$session_id` - HAVING 1))) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestSessionRecordingsListFromFilters.test_any_event_filter_with_properties_materialized.2 - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-21 19:55:00.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-31 20:00:00.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-14 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-21 20:00:00.000000', 6, 'UTC')), 0), in(s.session_id, - (SELECT events.`$session_id` AS session_id - FROM events - WHERE and(equals(events.team_id, 99999), notEmpty(events.`$session_id`), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-31 20:00:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), now64(6, 'UTC')), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-13 23:58:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-21 20:00:00.000000', 6, 'UTC')), and(1, ifNull(equals(nullIf(nullIf(events.`mat_$browser`, ''), 'null'), 'Firefox'), 0))) - GROUP BY events.`$session_id` - HAVING 1))) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestSessionRecordingsListFromFilters.test_basic_query - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-01 13:41:23.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-25 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), 0)) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestSessionRecordingsListFromFilters.test_basic_query_active_sessions - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-01 13:41:23.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-25 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), 0)) - GROUP BY s.session_id - HAVING ifNull(greater(duration, 60), 0) - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestSessionRecordingsListFromFilters.test_basic_query_active_sessions.1 - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-01 13:41:23.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-25 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), 0)) - GROUP BY s.session_id - HAVING ifNull(greater(active_seconds, '60'), 0) - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestSessionRecordingsListFromFilters.test_basic_query_active_sessions.2 - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-01 13:41:23.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-25 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), 0)) - GROUP BY s.session_id - HAVING ifNull(greater(inactive_seconds, '60'), 0) - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestSessionRecordingsListFromFilters.test_basic_query_with_ordering - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-01 13:41:23.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-25 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), 0)) - GROUP BY s.session_id - HAVING 1 - ORDER BY active_seconds DESC - LIMIT 4 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestSessionRecordingsListFromFilters.test_basic_query_with_ordering.1 - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-01 13:41:23.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-25 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), 0)) - GROUP BY s.session_id - HAVING 1 - ORDER BY console_error_count DESC - LIMIT 4 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestSessionRecordingsListFromFilters.test_basic_query_with_ordering.2 - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-01 13:41:23.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-25 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), 0)) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 4 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestSessionRecordingsListFromFilters.test_basic_query_with_paging - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-01 13:41:23.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-25 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), 0)) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 2 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestSessionRecordingsListFromFilters.test_basic_query_with_paging.1 - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-01 13:41:23.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-25 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), 0)) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 2 - OFFSET 1 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestSessionRecordingsListFromFilters.test_basic_query_with_paging.2 - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-01 13:41:23.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-25 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), 0)) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 2 - OFFSET 2 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestSessionRecordingsListFromFilters.test_date_from_filter - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-01 13:41:23.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-01 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), 0)) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestSessionRecordingsListFromFilters.test_date_from_filter.1 - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-01 13:41:23.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-30 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), 0)) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestSessionRecordingsListFromFilters.test_date_from_filter_cannot_search_before_ttl - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-01 12:41:00.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-11 12:46:00.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-12 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-01 12:46:00.000000', 6, 'UTC')), 0)) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestSessionRecordingsListFromFilters.test_date_from_filter_cannot_search_before_ttl.1 - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-01 12:41:00.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-11 12:46:00.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-11 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-01 12:46:00.000000', 6, 'UTC')), 0)) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestSessionRecordingsListFromFilters.test_date_from_filter_cannot_search_before_ttl.2 - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-01 12:41:00.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-11 12:46:00.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-10 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-01 12:46:00.000000', 6, 'UTC')), 0)) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestSessionRecordingsListFromFilters.test_date_to_filter - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-01 13:41:23.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-25 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-28 23:59:59.999999', 6, 'UTC')), 0)) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestSessionRecordingsListFromFilters.test_date_to_filter.1 - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-01 13:41:23.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-25 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-29 23:59:59.999999', 6, 'UTC')), 0)) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestSessionRecordingsListFromFilters.test_duration_filter - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-01 13:41:23.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-25 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), 0)) - GROUP BY s.session_id - HAVING ifNull(greater(duration, 60), 0) - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestSessionRecordingsListFromFilters.test_duration_filter.1 - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-01 13:41:23.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-25 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), 0)) - GROUP BY s.session_id - HAVING ifNull(less(duration, 60), 0) - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestSessionRecordingsListFromFilters.test_event_filter - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-01 13:41:23.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-25 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), 0), in(s.session_id, - (SELECT events.`$session_id` AS session_id - FROM events - WHERE and(equals(events.team_id, 99999), notEmpty(events.`$session_id`), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), now64(6, 'UTC')), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-24 23:58:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), and(equals(events.event, '$pageview'), 1)) - GROUP BY events.`$session_id` - HAVING hasAll(groupUniqArray(events.event), ['$pageview'])))) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestSessionRecordingsListFromFilters.test_event_filter.1 - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-01 13:41:23.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-25 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), 0), in(s.session_id, - (SELECT events.`$session_id` AS session_id - FROM events - WHERE and(equals(events.team_id, 99999), notEmpty(events.`$session_id`), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), now64(6, 'UTC')), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-24 23:58:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), and(equals(events.event, '$autocapture'), 1)) - GROUP BY events.`$session_id` - HAVING hasAll(groupUniqArray(events.event), ['$autocapture'])))) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestSessionRecordingsListFromFilters.test_event_filter_has_ttl_applied_too - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-01 13:41:23.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-25 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), 0), in(s.session_id, - (SELECT events.`$session_id` AS session_id - FROM events - WHERE and(equals(events.team_id, 99999), notEmpty(events.`$session_id`), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), now64(6, 'UTC')), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-24 23:58:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), and(equals(events.event, '$pageview'), 1)) - GROUP BY events.`$session_id` - HAVING hasAll(groupUniqArray(events.event), ['$pageview'])))) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestSessionRecordingsListFromFilters.test_event_filter_has_ttl_applied_too.1 - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-01 13:41:23.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-25 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), 0)) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestSessionRecordingsListFromFilters.test_event_filter_with_active_sessions - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-01 13:41:23.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-25 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), 0), in(s.session_id, - (SELECT events.`$session_id` AS session_id - FROM events - WHERE and(equals(events.team_id, 99999), notEmpty(events.`$session_id`), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), now64(6, 'UTC')), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-24 23:58:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), and(equals(events.event, '$pageview'), 1)) - GROUP BY events.`$session_id` - HAVING hasAll(groupUniqArray(events.event), ['$pageview'])))) - GROUP BY s.session_id - HAVING ifNull(greater(duration, 60), 0) - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestSessionRecordingsListFromFilters.test_event_filter_with_active_sessions.1 - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-01 13:41:23.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-25 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), 0), in(s.session_id, - (SELECT events.`$session_id` AS session_id - FROM events - WHERE and(equals(events.team_id, 99999), notEmpty(events.`$session_id`), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), now64(6, 'UTC')), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-24 23:58:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), and(equals(events.event, '$pageview'), 1)) - GROUP BY events.`$session_id` - HAVING hasAll(groupUniqArray(events.event), ['$pageview'])))) - GROUP BY s.session_id - HAVING ifNull(greater(active_seconds, 60), 0) - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestSessionRecordingsListFromFilters.test_event_filter_with_group_filter - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-21 19:55:00.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-31 20:00:00.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-14 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-21 20:00:00.000000', 6, 'UTC')), 0), in(s.session_id, - (SELECT events.`$session_id` AS session_id - FROM events - LEFT JOIN - (SELECT argMax(replaceRegexpAll(nullIf(nullIf(JSONExtractRaw(groups.group_properties, 'name'), ''), 'null'), '^"|"$', ''), toTimeZone(groups._timestamp, 'UTC')) AS properties___name, groups.group_type_index AS index, groups.group_key AS key - FROM groups - WHERE and(equals(groups.team_id, 99999), equals(index, 1)) - GROUP BY groups.group_type_index, groups.group_key) AS events__group_1 ON equals(events.`$group_1`, events__group_1.key) - WHERE and(equals(events.team_id, 99999), notEmpty(events.`$session_id`), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-31 20:00:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), now64(6, 'UTC')), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-13 23:58:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-21 20:00:00.000000', 6, 'UTC')), and(equals(events.event, '$pageview'), ifNull(equals(events__group_1.properties___name, 'org one'), 0))) - GROUP BY events.`$session_id` - HAVING hasAll(groupUniqArray(events.event), ['$pageview'])))) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestSessionRecordingsListFromFilters.test_event_filter_with_group_filter.1 - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-21 19:55:00.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-31 20:00:00.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-14 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-21 20:00:00.000000', 6, 'UTC')), 0), in(s.session_id, - (SELECT events.`$session_id` AS session_id - FROM events - LEFT JOIN - (SELECT argMax(replaceRegexpAll(nullIf(nullIf(JSONExtractRaw(groups.group_properties, 'name'), ''), 'null'), '^"|"$', ''), toTimeZone(groups._timestamp, 'UTC')) AS properties___name, groups.group_type_index AS index, groups.group_key AS key - FROM groups - WHERE and(equals(groups.team_id, 99999), equals(index, 1)) - GROUP BY groups.group_type_index, groups.group_key) AS events__group_1 ON equals(events.`$group_1`, events__group_1.key) - WHERE and(equals(events.team_id, 99999), notEmpty(events.`$session_id`), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-31 20:00:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), now64(6, 'UTC')), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-13 23:58:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-21 20:00:00.000000', 6, 'UTC')), ifNull(equals(events__group_1.properties___name, 'org one'), 0)) - GROUP BY events.`$session_id` - HAVING 1))) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestSessionRecordingsListFromFilters.test_event_filter_with_group_filter.2 - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-21 19:55:00.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-31 20:00:00.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-14 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-21 20:00:00.000000', 6, 'UTC')), 0), in(s.session_id, - (SELECT events.`$session_id` AS session_id - FROM events - LEFT JOIN - (SELECT argMax(replaceRegexpAll(nullIf(nullIf(JSONExtractRaw(groups.group_properties, 'name'), ''), 'null'), '^"|"$', ''), toTimeZone(groups._timestamp, 'UTC')) AS properties___name, groups.group_type_index AS index, groups.group_key AS key - FROM groups - WHERE and(equals(groups.team_id, 99999), equals(index, 2)) - GROUP BY groups.group_type_index, groups.group_key) AS events__group_2 ON equals(events.`$group_2`, events__group_2.key) - WHERE and(equals(events.team_id, 99999), notEmpty(events.`$session_id`), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-31 20:00:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), now64(6, 'UTC')), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-13 23:58:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-21 20:00:00.000000', 6, 'UTC')), ifNull(equals(events__group_2.properties___name, 'org one'), 0)) - GROUP BY events.`$session_id` - HAVING 1))) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestSessionRecordingsListFromFilters.test_event_filter_with_hogql_event_properties_test_accounts_excluded - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-21 19:55:00.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-31 20:00:00.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-14 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-21 20:00:00.000000', 6, 'UTC')), 0), in(s.session_id, - (SELECT events.`$session_id` AS session_id - FROM events - WHERE and(equals(events.team_id, 99999), notEmpty(events.`$session_id`), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-31 20:00:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), now64(6, 'UTC')), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-13 23:58:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-21 20:00:00.000000', 6, 'UTC')), and(equals(events.event, '$pageview'), 1)) - GROUP BY events.`$session_id` - HAVING hasAll(groupUniqArray(events.event), ['$pageview'])))) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestSessionRecordingsListFromFilters.test_event_filter_with_hogql_event_properties_test_accounts_excluded.1 - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-21 19:55:00.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-31 20:00:00.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-14 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-21 20:00:00.000000', 6, 'UTC')), 0), in(s.session_id, - (SELECT events.`$session_id` AS session_id - FROM events - LEFT OUTER JOIN - (SELECT argMax(person_distinct_id_overrides.person_id, person_distinct_id_overrides.version) AS person_id, person_distinct_id_overrides.distinct_id AS distinct_id - FROM person_distinct_id_overrides - WHERE equals(person_distinct_id_overrides.team_id, 99999) - GROUP BY person_distinct_id_overrides.distinct_id - HAVING ifNull(equals(argMax(person_distinct_id_overrides.is_deleted, person_distinct_id_overrides.version), 0), 0) SETTINGS optimize_aggregation_in_order=1) AS events__override ON equals(events.distinct_id, events__override.distinct_id) - LEFT JOIN - (SELECT person.id AS id, replaceRegexpAll(nullIf(nullIf(JSONExtractRaw(person.properties, 'email'), ''), 'null'), '^"|"$', '') AS properties___email - FROM person - WHERE and(equals(person.team_id, 99999), ifNull(in(tuple(person.id, person.version), - (SELECT person.id AS id, max(person.version) AS version - FROM person - WHERE equals(person.team_id, 99999) - GROUP BY person.id - HAVING and(ifNull(equals(argMax(person.is_deleted, person.version), 0), 0), ifNull(less(argMax(toTimeZone(person.created_at, 'UTC'), person.version), plus(now64(6, 'UTC'), toIntervalDay(1))), 0)))), 0)) SETTINGS optimize_aggregation_in_order=1) AS events__person ON equals(if(not(empty(events__override.distinct_id)), events__override.person_id, events.person_id), events__person.id) - WHERE and(equals(events.team_id, 99999), notEmpty(events.`$session_id`), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-31 20:00:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), now64(6, 'UTC')), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-13 23:58:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-21 20:00:00.000000', 6, 'UTC')), and(equals(events.event, '$pageview'), 1), ifNull(equals(events__person.properties___email, 'bla'), 0)) - GROUP BY events.`$session_id` - HAVING hasAll(groupUniqArray(events.event), ['$pageview'])))) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestSessionRecordingsListFromFilters.test_event_filter_with_hogql_event_properties_test_accounts_excluded.2 - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-21 19:55:00.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-31 20:00:00.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-14 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-21 20:00:00.000000', 6, 'UTC')), 0), in(s.session_id, - (SELECT events.`$session_id` AS session_id - FROM events - LEFT OUTER JOIN - (SELECT argMax(person_distinct_id_overrides.person_id, person_distinct_id_overrides.version) AS person_id, person_distinct_id_overrides.distinct_id AS distinct_id - FROM person_distinct_id_overrides - WHERE equals(person_distinct_id_overrides.team_id, 99999) - GROUP BY person_distinct_id_overrides.distinct_id - HAVING ifNull(equals(argMax(person_distinct_id_overrides.is_deleted, person_distinct_id_overrides.version), 0), 0) SETTINGS optimize_aggregation_in_order=1) AS events__override ON equals(events.distinct_id, events__override.distinct_id) - LEFT JOIN - (SELECT person.id AS id, replaceRegexpAll(nullIf(nullIf(JSONExtractRaw(person.properties, 'email'), ''), 'null'), '^"|"$', '') AS properties___email - FROM person - WHERE and(equals(person.team_id, 99999), ifNull(in(tuple(person.id, person.version), - (SELECT person.id AS id, max(person.version) AS version - FROM person - WHERE equals(person.team_id, 99999) - GROUP BY person.id - HAVING and(ifNull(equals(argMax(person.is_deleted, person.version), 0), 0), ifNull(less(argMax(toTimeZone(person.created_at, 'UTC'), person.version), plus(now64(6, 'UTC'), toIntervalDay(1))), 0)))), 0)) SETTINGS optimize_aggregation_in_order=1) AS events__person ON equals(if(not(empty(events__override.distinct_id)), events__override.person_id, events.person_id), events__person.id) - WHERE and(equals(events.team_id, 99999), notEmpty(events.`$session_id`), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-31 20:00:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), now64(6, 'UTC')), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-13 23:58:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-21 20:00:00.000000', 6, 'UTC')), ifNull(equals(replaceRegexpAll(nullIf(nullIf(JSONExtractRaw(events.properties, '$browser'), ''), 'null'), '^"|"$', ''), 'Chrome'), 0), ifNull(equals(events__person.properties___email, 'bla'), 0)) - GROUP BY events.`$session_id` - HAVING 1))) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestSessionRecordingsListFromFilters.test_event_filter_with_hogql_event_properties_test_accounts_excluded_materialized - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-21 19:55:00.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-31 20:00:00.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-14 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-21 20:00:00.000000', 6, 'UTC')), 0), in(s.session_id, - (SELECT events.`$session_id` AS session_id - FROM events - WHERE and(equals(events.team_id, 99999), notEmpty(events.`$session_id`), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-31 20:00:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), now64(6, 'UTC')), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-13 23:58:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-21 20:00:00.000000', 6, 'UTC')), and(equals(events.event, '$pageview'), 1)) - GROUP BY events.`$session_id` - HAVING hasAll(groupUniqArray(events.event), ['$pageview'])))) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestSessionRecordingsListFromFilters.test_event_filter_with_hogql_event_properties_test_accounts_excluded_materialized.1 - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-21 19:55:00.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-31 20:00:00.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-14 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-21 20:00:00.000000', 6, 'UTC')), 0), in(s.session_id, - (SELECT events.`$session_id` AS session_id - FROM events - LEFT OUTER JOIN - (SELECT argMax(person_distinct_id_overrides.person_id, person_distinct_id_overrides.version) AS person_id, person_distinct_id_overrides.distinct_id AS distinct_id - FROM person_distinct_id_overrides - WHERE equals(person_distinct_id_overrides.team_id, 99999) - GROUP BY person_distinct_id_overrides.distinct_id - HAVING ifNull(equals(argMax(person_distinct_id_overrides.is_deleted, person_distinct_id_overrides.version), 0), 0) SETTINGS optimize_aggregation_in_order=1) AS events__override ON equals(events.distinct_id, events__override.distinct_id) - LEFT JOIN - (SELECT person.id AS id, nullIf(nullIf(person.pmat_email, ''), 'null') AS properties___email - FROM person - WHERE and(equals(person.team_id, 99999), ifNull(in(tuple(person.id, person.version), - (SELECT person.id AS id, max(person.version) AS version - FROM person - WHERE equals(person.team_id, 99999) - GROUP BY person.id - HAVING and(ifNull(equals(argMax(person.is_deleted, person.version), 0), 0), ifNull(less(argMax(toTimeZone(person.created_at, 'UTC'), person.version), plus(now64(6, 'UTC'), toIntervalDay(1))), 0)))), 0)) SETTINGS optimize_aggregation_in_order=1) AS events__person ON equals(if(not(empty(events__override.distinct_id)), events__override.person_id, events.person_id), events__person.id) - WHERE and(equals(events.team_id, 99999), notEmpty(events.`$session_id`), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-31 20:00:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), now64(6, 'UTC')), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-13 23:58:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-21 20:00:00.000000', 6, 'UTC')), and(equals(events.event, '$pageview'), 1), ifNull(equals(events__person.properties___email, 'bla'), 0)) - GROUP BY events.`$session_id` - HAVING hasAll(groupUniqArray(events.event), ['$pageview'])))) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestSessionRecordingsListFromFilters.test_event_filter_with_hogql_event_properties_test_accounts_excluded_materialized.2 - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-21 19:55:00.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-31 20:00:00.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-14 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-21 20:00:00.000000', 6, 'UTC')), 0), in(s.session_id, - (SELECT events.`$session_id` AS session_id - FROM events - LEFT OUTER JOIN - (SELECT argMax(person_distinct_id_overrides.person_id, person_distinct_id_overrides.version) AS person_id, person_distinct_id_overrides.distinct_id AS distinct_id - FROM person_distinct_id_overrides - WHERE equals(person_distinct_id_overrides.team_id, 99999) - GROUP BY person_distinct_id_overrides.distinct_id - HAVING ifNull(equals(argMax(person_distinct_id_overrides.is_deleted, person_distinct_id_overrides.version), 0), 0) SETTINGS optimize_aggregation_in_order=1) AS events__override ON equals(events.distinct_id, events__override.distinct_id) - LEFT JOIN - (SELECT person.id AS id, nullIf(nullIf(person.pmat_email, ''), 'null') AS properties___email - FROM person - WHERE and(equals(person.team_id, 99999), ifNull(in(tuple(person.id, person.version), - (SELECT person.id AS id, max(person.version) AS version - FROM person - WHERE equals(person.team_id, 99999) - GROUP BY person.id - HAVING and(ifNull(equals(argMax(person.is_deleted, person.version), 0), 0), ifNull(less(argMax(toTimeZone(person.created_at, 'UTC'), person.version), plus(now64(6, 'UTC'), toIntervalDay(1))), 0)))), 0)) SETTINGS optimize_aggregation_in_order=1) AS events__person ON equals(if(not(empty(events__override.distinct_id)), events__override.person_id, events.person_id), events__person.id) - WHERE and(equals(events.team_id, 99999), notEmpty(events.`$session_id`), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-31 20:00:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), now64(6, 'UTC')), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-13 23:58:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-21 20:00:00.000000', 6, 'UTC')), ifNull(equals(nullIf(nullIf(events.`mat_$browser`, ''), 'null'), 'Chrome'), 0), ifNull(equals(events__person.properties___email, 'bla'), 0)) - GROUP BY events.`$session_id` - HAVING 1))) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestSessionRecordingsListFromFilters.test_event_filter_with_hogql_person_properties - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-01 13:41:23.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-25 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), 0), in(s.session_id, - (SELECT events.`$session_id` AS session_id - FROM events - LEFT OUTER JOIN - (SELECT argMax(person_distinct_id_overrides.person_id, person_distinct_id_overrides.version) AS person_id, person_distinct_id_overrides.distinct_id AS distinct_id - FROM person_distinct_id_overrides - WHERE equals(person_distinct_id_overrides.team_id, 99999) - GROUP BY person_distinct_id_overrides.distinct_id - HAVING ifNull(equals(argMax(person_distinct_id_overrides.is_deleted, person_distinct_id_overrides.version), 0), 0) SETTINGS optimize_aggregation_in_order=1) AS events__override ON equals(events.distinct_id, events__override.distinct_id) - LEFT JOIN - (SELECT person.id AS id, replaceRegexpAll(nullIf(nullIf(JSONExtractRaw(person.properties, 'email'), ''), 'null'), '^"|"$', '') AS properties___email - FROM person - WHERE and(equals(person.team_id, 99999), ifNull(in(tuple(person.id, person.version), - (SELECT person.id AS id, max(person.version) AS version - FROM person - WHERE equals(person.team_id, 99999) - GROUP BY person.id - HAVING and(ifNull(equals(argMax(person.is_deleted, person.version), 0), 0), ifNull(less(argMax(toTimeZone(person.created_at, 'UTC'), person.version), plus(now64(6, 'UTC'), toIntervalDay(1))), 0)))), 0)) SETTINGS optimize_aggregation_in_order=1) AS events__person ON equals(if(not(empty(events__override.distinct_id)), events__override.person_id, events.person_id), events__person.id) - WHERE and(equals(events.team_id, 99999), notEmpty(events.`$session_id`), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), now64(6, 'UTC')), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-24 23:58:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), and(equals(events.event, '$pageview'), ifNull(equals(events__person.properties___email, 'bla'), 0))) - GROUP BY events.`$session_id` - HAVING hasAll(groupUniqArray(events.event), ['$pageview'])))) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestSessionRecordingsListFromFilters.test_event_filter_with_hogql_person_properties.1 - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-01 13:41:23.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-25 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), 0), in(s.session_id, - (SELECT events.`$session_id` AS session_id - FROM events - LEFT OUTER JOIN - (SELECT argMax(person_distinct_id_overrides.person_id, person_distinct_id_overrides.version) AS person_id, person_distinct_id_overrides.distinct_id AS distinct_id - FROM person_distinct_id_overrides - WHERE equals(person_distinct_id_overrides.team_id, 99999) - GROUP BY person_distinct_id_overrides.distinct_id - HAVING ifNull(equals(argMax(person_distinct_id_overrides.is_deleted, person_distinct_id_overrides.version), 0), 0) SETTINGS optimize_aggregation_in_order=1) AS events__override ON equals(events.distinct_id, events__override.distinct_id) - LEFT JOIN - (SELECT person.id AS id, replaceRegexpAll(nullIf(nullIf(JSONExtractRaw(person.properties, 'email'), ''), 'null'), '^"|"$', '') AS properties___email - FROM person - WHERE and(equals(person.team_id, 99999), ifNull(in(tuple(person.id, person.version), - (SELECT person.id AS id, max(person.version) AS version - FROM person - WHERE equals(person.team_id, 99999) - GROUP BY person.id - HAVING and(ifNull(equals(argMax(person.is_deleted, person.version), 0), 0), ifNull(less(argMax(toTimeZone(person.created_at, 'UTC'), person.version), plus(now64(6, 'UTC'), toIntervalDay(1))), 0)))), 0)) SETTINGS optimize_aggregation_in_order=1) AS events__person ON equals(if(not(empty(events__override.distinct_id)), events__override.person_id, events.person_id), events__person.id) - WHERE and(equals(events.team_id, 99999), notEmpty(events.`$session_id`), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), now64(6, 'UTC')), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-24 23:58:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), and(equals(events.event, '$pageview'), ifNull(equals(events__person.properties___email, 'something else'), 0))) - GROUP BY events.`$session_id` - HAVING hasAll(groupUniqArray(events.event), ['$pageview'])))) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestSessionRecordingsListFromFilters.test_event_filter_with_hogql_properties - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-01 13:41:23.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-25 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), 0), in(s.session_id, - (SELECT events.`$session_id` AS session_id - FROM events - WHERE and(equals(events.team_id, 99999), notEmpty(events.`$session_id`), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), now64(6, 'UTC')), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-24 23:58:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), and(equals(events.event, '$pageview'), ifNull(equals(replaceRegexpAll(nullIf(nullIf(JSONExtractRaw(events.properties, '$browser'), ''), 'null'), '^"|"$', ''), 'Chrome'), 0))) - GROUP BY events.`$session_id` - HAVING hasAll(groupUniqArray(events.event), ['$pageview'])))) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestSessionRecordingsListFromFilters.test_event_filter_with_hogql_properties.1 - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-01 13:41:23.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-25 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), 0), in(s.session_id, - (SELECT events.`$session_id` AS session_id - FROM events - WHERE and(equals(events.team_id, 99999), notEmpty(events.`$session_id`), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), now64(6, 'UTC')), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-24 23:58:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), and(equals(events.event, '$pageview'), ifNull(equals(replaceRegexpAll(nullIf(nullIf(JSONExtractRaw(events.properties, '$browser'), ''), 'null'), '^"|"$', ''), 'Firefox'), 0))) - GROUP BY events.`$session_id` - HAVING hasAll(groupUniqArray(events.event), ['$pageview'])))) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestSessionRecordingsListFromFilters.test_event_filter_with_hogql_properties_materialized - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-01 13:41:23.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-25 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), 0), in(s.session_id, - (SELECT events.`$session_id` AS session_id - FROM events - WHERE and(equals(events.team_id, 99999), notEmpty(events.`$session_id`), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), now64(6, 'UTC')), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-24 23:58:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), and(equals(events.event, '$pageview'), ifNull(equals(nullIf(nullIf(events.`mat_$browser`, ''), 'null'), 'Chrome'), 0))) - GROUP BY events.`$session_id` - HAVING hasAll(groupUniqArray(events.event), ['$pageview'])))) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestSessionRecordingsListFromFilters.test_event_filter_with_hogql_properties_materialized.1 - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-01 13:41:23.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-25 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), 0), in(s.session_id, - (SELECT events.`$session_id` AS session_id - FROM events - WHERE and(equals(events.team_id, 99999), notEmpty(events.`$session_id`), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), now64(6, 'UTC')), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-24 23:58:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), and(equals(events.event, '$pageview'), ifNull(equals(nullIf(nullIf(events.`mat_$browser`, ''), 'null'), 'Firefox'), 0))) - GROUP BY events.`$session_id` - HAVING hasAll(groupUniqArray(events.event), ['$pageview'])))) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestSessionRecordingsListFromFilters.test_event_filter_with_matching_on_session_id - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-01 13:41:23.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-25 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), 0), in(s.session_id, - (SELECT events.`$session_id` AS session_id - FROM events - WHERE and(equals(events.team_id, 99999), notEmpty(events.`$session_id`), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), now64(6, 'UTC')), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-24 23:58:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), and(equals(events.event, '$pageview'), 1)) - GROUP BY events.`$session_id` - HAVING hasAll(groupUniqArray(events.event), ['$pageview'])))) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestSessionRecordingsListFromFilters.test_event_filter_with_matching_on_session_id.1 - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-01 13:41:23.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-25 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), 0), in(s.session_id, - (SELECT events.`$session_id` AS session_id - FROM events - WHERE and(equals(events.team_id, 99999), notEmpty(events.`$session_id`), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), now64(6, 'UTC')), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-24 23:58:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), and(equals(events.event, '$autocapture'), 1)) - GROUP BY events.`$session_id` - HAVING hasAll(groupUniqArray(events.event), ['$autocapture'])))) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestSessionRecordingsListFromFilters.test_event_filter_with_properties - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-01 13:41:23.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-25 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), 0), in(s.session_id, - (SELECT events.`$session_id` AS session_id - FROM events - WHERE and(equals(events.team_id, 99999), notEmpty(events.`$session_id`), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), now64(6, 'UTC')), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-24 23:58:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), and(equals(events.event, '$pageview'), ifNull(equals(replaceRegexpAll(nullIf(nullIf(JSONExtractRaw(events.properties, '$browser'), ''), 'null'), '^"|"$', ''), 'Chrome'), 0))) - GROUP BY events.`$session_id` - HAVING hasAll(groupUniqArray(events.event), ['$pageview'])))) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestSessionRecordingsListFromFilters.test_event_filter_with_properties.1 - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-01 13:41:23.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-25 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), 0), in(s.session_id, - (SELECT events.`$session_id` AS session_id - FROM events - WHERE and(equals(events.team_id, 99999), notEmpty(events.`$session_id`), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), now64(6, 'UTC')), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-24 23:58:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), and(equals(events.event, '$pageview'), ifNull(equals(replaceRegexpAll(nullIf(nullIf(JSONExtractRaw(events.properties, '$browser'), ''), 'null'), '^"|"$', ''), 'Firefox'), 0))) - GROUP BY events.`$session_id` - HAVING hasAll(groupUniqArray(events.event), ['$pageview'])))) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestSessionRecordingsListFromFilters.test_event_filter_with_properties.2 - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-01 13:41:23.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-25 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), 0), in(s.session_id, - (SELECT events.`$session_id` AS session_id - FROM events - WHERE and(equals(events.team_id, 99999), notEmpty(events.`$session_id`), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), now64(6, 'UTC')), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-24 23:58:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), and(equals(events.event, 'a_different_event'), ifNull(equals(replaceRegexpAll(nullIf(nullIf(JSONExtractRaw(events.properties, '$browser'), ''), 'null'), '^"|"$', ''), 'Chrome'), 0))) - GROUP BY events.`$session_id` - HAVING hasAll(groupUniqArray(events.event), ['a_different_event'])))) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestSessionRecordingsListFromFilters.test_event_filter_with_properties.3 - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-01 13:41:23.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-25 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), 0), in(s.session_id, - (SELECT events.`$session_id` AS session_id - FROM events - WHERE and(equals(events.team_id, 99999), notEmpty(events.`$session_id`), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), now64(6, 'UTC')), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-24 23:58:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), and(equals(events.event, 'a_different_event'), ifNull(equals(replaceRegexpAll(nullIf(nullIf(JSONExtractRaw(events.properties, '$browser'), ''), 'null'), '^"|"$', ''), 'Safari'), 0))) - GROUP BY events.`$session_id` - HAVING hasAll(groupUniqArray(events.event), ['a_different_event'])))) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestSessionRecordingsListFromFilters.test_event_filter_with_properties_materialized - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-01 13:41:23.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-25 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), 0), in(s.session_id, - (SELECT events.`$session_id` AS session_id - FROM events - WHERE and(equals(events.team_id, 99999), notEmpty(events.`$session_id`), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), now64(6, 'UTC')), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-24 23:58:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), and(equals(events.event, '$pageview'), ifNull(equals(nullIf(nullIf(events.`mat_$browser`, ''), 'null'), 'Chrome'), 0))) - GROUP BY events.`$session_id` - HAVING hasAll(groupUniqArray(events.event), ['$pageview'])))) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestSessionRecordingsListFromFilters.test_event_filter_with_properties_materialized.1 - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-01 13:41:23.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-25 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), 0), in(s.session_id, - (SELECT events.`$session_id` AS session_id - FROM events - WHERE and(equals(events.team_id, 99999), notEmpty(events.`$session_id`), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), now64(6, 'UTC')), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-24 23:58:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), and(equals(events.event, '$pageview'), ifNull(equals(nullIf(nullIf(events.`mat_$browser`, ''), 'null'), 'Firefox'), 0))) - GROUP BY events.`$session_id` - HAVING hasAll(groupUniqArray(events.event), ['$pageview'])))) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestSessionRecordingsListFromFilters.test_event_filter_with_properties_materialized.2 - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-01 13:41:23.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-25 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), 0), in(s.session_id, - (SELECT events.`$session_id` AS session_id - FROM events - WHERE and(equals(events.team_id, 99999), notEmpty(events.`$session_id`), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), now64(6, 'UTC')), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-24 23:58:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), and(equals(events.event, 'a_different_event'), ifNull(equals(nullIf(nullIf(events.`mat_$browser`, ''), 'null'), 'Chrome'), 0))) - GROUP BY events.`$session_id` - HAVING hasAll(groupUniqArray(events.event), ['a_different_event'])))) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestSessionRecordingsListFromFilters.test_event_filter_with_properties_materialized.3 - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-01 13:41:23.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-25 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), 0), in(s.session_id, - (SELECT events.`$session_id` AS session_id - FROM events - WHERE and(equals(events.team_id, 99999), notEmpty(events.`$session_id`), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), now64(6, 'UTC')), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-24 23:58:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), and(equals(events.event, 'a_different_event'), ifNull(equals(nullIf(nullIf(events.`mat_$browser`, ''), 'null'), 'Safari'), 0))) - GROUP BY events.`$session_id` - HAVING hasAll(groupUniqArray(events.event), ['a_different_event'])))) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestSessionRecordingsListFromFilters.test_event_filter_with_test_accounts_excluded - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-21 19:55:00.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-31 20:00:00.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-14 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-21 20:00:00.000000', 6, 'UTC')), 0), in(s.session_id, - (SELECT events.`$session_id` AS session_id - FROM events - LEFT OUTER JOIN - (SELECT argMax(person_distinct_id_overrides.person_id, person_distinct_id_overrides.version) AS person_id, person_distinct_id_overrides.distinct_id AS distinct_id - FROM person_distinct_id_overrides - WHERE equals(person_distinct_id_overrides.team_id, 99999) - GROUP BY person_distinct_id_overrides.distinct_id - HAVING ifNull(equals(argMax(person_distinct_id_overrides.is_deleted, person_distinct_id_overrides.version), 0), 0) SETTINGS optimize_aggregation_in_order=1) AS events__override ON equals(events.distinct_id, events__override.distinct_id) - LEFT JOIN - (SELECT person.id AS id, replaceRegexpAll(nullIf(nullIf(JSONExtractRaw(person.properties, 'email'), ''), 'null'), '^"|"$', '') AS properties___email - FROM person - WHERE and(equals(person.team_id, 99999), ifNull(in(tuple(person.id, person.version), - (SELECT person.id AS id, max(person.version) AS version - FROM person - WHERE equals(person.team_id, 99999) - GROUP BY person.id - HAVING and(ifNull(equals(argMax(person.is_deleted, person.version), 0), 0), ifNull(less(argMax(toTimeZone(person.created_at, 'UTC'), person.version), plus(now64(6, 'UTC'), toIntervalDay(1))), 0)))), 0)) SETTINGS optimize_aggregation_in_order=1) AS events__person ON equals(if(not(empty(events__override.distinct_id)), events__override.person_id, events.person_id), events__person.id) - WHERE and(equals(events.team_id, 99999), notEmpty(events.`$session_id`), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-31 20:00:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), now64(6, 'UTC')), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-13 23:58:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-21 20:00:00.000000', 6, 'UTC')), and(equals(events.event, '$pageview'), 1), and(ifNull(equals(replaceRegexpAll(nullIf(nullIf(JSONExtractRaw(events.properties, 'is_internal_user'), ''), 'null'), '^"|"$', ''), 'false'), 0), ifNull(equals(replaceRegexpAll(nullIf(nullIf(JSONExtractRaw(events.properties, '$browser'), ''), 'null'), '^"|"$', ''), 'Chrome'), 0)), ifNull(notILike(events__person.properties___email, '%@posthog.com%'), 1)) - GROUP BY events.`$session_id` - HAVING hasAll(groupUniqArray(events.event), ['$pageview'])))) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestSessionRecordingsListFromFilters.test_event_filter_with_test_accounts_excluded.1 - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-21 19:55:00.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-31 20:00:00.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-14 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-21 20:00:00.000000', 6, 'UTC')), 0), in(s.session_id, - (SELECT events.`$session_id` AS session_id - FROM events - WHERE and(equals(events.team_id, 99999), notEmpty(events.`$session_id`), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-31 20:00:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), now64(6, 'UTC')), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-13 23:58:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-21 20:00:00.000000', 6, 'UTC')), and(equals(events.event, '$pageview'), 1)) - GROUP BY events.`$session_id` - HAVING hasAll(groupUniqArray(events.event), ['$pageview'])))) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestSessionRecordingsListFromFilters.test_event_filter_with_test_accounts_excluded_materialized - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-21 19:55:00.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-31 20:00:00.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-14 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-21 20:00:00.000000', 6, 'UTC')), 0), in(s.session_id, - (SELECT events.`$session_id` AS session_id - FROM events - LEFT OUTER JOIN - (SELECT argMax(person_distinct_id_overrides.person_id, person_distinct_id_overrides.version) AS person_id, person_distinct_id_overrides.distinct_id AS distinct_id - FROM person_distinct_id_overrides - WHERE equals(person_distinct_id_overrides.team_id, 99999) - GROUP BY person_distinct_id_overrides.distinct_id - HAVING ifNull(equals(argMax(person_distinct_id_overrides.is_deleted, person_distinct_id_overrides.version), 0), 0) SETTINGS optimize_aggregation_in_order=1) AS events__override ON equals(events.distinct_id, events__override.distinct_id) - LEFT JOIN - (SELECT person.id AS id, nullIf(nullIf(person.pmat_email, ''), 'null') AS properties___email - FROM person - WHERE and(equals(person.team_id, 99999), ifNull(in(tuple(person.id, person.version), - (SELECT person.id AS id, max(person.version) AS version - FROM person - WHERE equals(person.team_id, 99999) - GROUP BY person.id - HAVING and(ifNull(equals(argMax(person.is_deleted, person.version), 0), 0), ifNull(less(argMax(toTimeZone(person.created_at, 'UTC'), person.version), plus(now64(6, 'UTC'), toIntervalDay(1))), 0)))), 0)) SETTINGS optimize_aggregation_in_order=1) AS events__person ON equals(if(not(empty(events__override.distinct_id)), events__override.person_id, events.person_id), events__person.id) - WHERE and(equals(events.team_id, 99999), notEmpty(events.`$session_id`), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-31 20:00:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), now64(6, 'UTC')), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-13 23:58:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-21 20:00:00.000000', 6, 'UTC')), and(equals(events.event, '$pageview'), 1), and(ifNull(equals(nullIf(nullIf(events.mat_is_internal_user, ''), 'null'), 'false'), 0), ifNull(equals(replaceRegexpAll(nullIf(nullIf(JSONExtractRaw(events.properties, '$browser'), ''), 'null'), '^"|"$', ''), 'Chrome'), 0)), ifNull(notILike(events__person.properties___email, '%@posthog.com%'), 1)) - GROUP BY events.`$session_id` - HAVING hasAll(groupUniqArray(events.event), ['$pageview'])))) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestSessionRecordingsListFromFilters.test_event_filter_with_test_accounts_excluded_materialized.1 - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-21 19:55:00.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-31 20:00:00.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-14 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-21 20:00:00.000000', 6, 'UTC')), 0), in(s.session_id, - (SELECT events.`$session_id` AS session_id - FROM events - WHERE and(equals(events.team_id, 99999), notEmpty(events.`$session_id`), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-31 20:00:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), now64(6, 'UTC')), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-13 23:58:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-21 20:00:00.000000', 6, 'UTC')), and(equals(events.event, '$pageview'), 1)) - GROUP BY events.`$session_id` - HAVING hasAll(groupUniqArray(events.event), ['$pageview'])))) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestSessionRecordingsListFromFilters.test_event_filter_with_two_events_and_multiple_teams - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-21 19:55:00.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-31 20:00:00.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-14 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-21 20:00:00.000000', 6, 'UTC')), 0), in(s.session_id, - (SELECT events.`$session_id` AS session_id - FROM events - WHERE and(equals(events.team_id, 99999), notEmpty(events.`$session_id`), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-31 20:00:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), now64(6, 'UTC')), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-13 23:58:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-21 20:00:00.000000', 6, 'UTC')), or(and(equals(events.event, '$pageview'), 1), and(equals(events.event, '$pageleave'), 1))) - GROUP BY events.`$session_id` - HAVING hasAll(groupUniqArray(events.event), ['$pageleave', '$pageview'])))) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestSessionRecordingsListFromFilters.test_filter_for_recordings_by_console_text_0__key_level_value_warn_error_operator_exact_type_log_entry_key_message_value_message_4_operator_icontains_type_log_entry_ - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-21 19:55:00.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-31 20:00:00.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-14 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-21 20:00:00.000000', 6, 'UTC')), 0), in(s.session_id, - (SELECT console_logs_log_entries.log_source_id AS log_source_id - FROM - (SELECT log_entries.log_source_id AS log_source_id, log_entries.level AS level, log_entries.message AS message - FROM log_entries - WHERE and(equals(log_entries.team_id, 99999), equals(log_entries.log_source, 'session_replay'))) AS console_logs_log_entries - WHERE or(or(ifNull(equals(console_logs_log_entries.level, 'warn'), 0), ifNull(equals(console_logs_log_entries.level, 'error'), 0)), ifNull(ilike(console_logs_log_entries.message, '%message 4%'), 0))))) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestSessionRecordingsListFromFilters.test_filter_for_recordings_by_console_text_1__key_level_value_warn_error_operator_exact_type_log_entry_key_message_value_message_5_operator_icontains_type_log_entry_ - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-21 19:55:00.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-31 20:00:00.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-14 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-21 20:00:00.000000', 6, 'UTC')), 0), in(s.session_id, - (SELECT console_logs_log_entries.log_source_id AS log_source_id - FROM - (SELECT log_entries.log_source_id AS log_source_id, log_entries.level AS level, log_entries.message AS message - FROM log_entries - WHERE and(equals(log_entries.team_id, 99999), equals(log_entries.log_source, 'session_replay'))) AS console_logs_log_entries - WHERE and(or(ifNull(equals(console_logs_log_entries.level, 'warn'), 0), ifNull(equals(console_logs_log_entries.level, 'error'), 0)), ifNull(ilike(console_logs_log_entries.message, '%message 5%'), 0))))) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestSessionRecordingsListFromFilters.test_filter_for_recordings_by_console_text_2__key_level_value_info_operator_exact_type_log_entry_key_message_value_message_5_operator_icontains_type_log_entry_ - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-21 19:55:00.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-31 20:00:00.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-14 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-21 20:00:00.000000', 6, 'UTC')), 0), in(s.session_id, - (SELECT console_logs_log_entries.log_source_id AS log_source_id - FROM - (SELECT log_entries.log_source_id AS log_source_id, log_entries.level AS level, log_entries.message AS message - FROM log_entries - WHERE and(equals(log_entries.team_id, 99999), equals(log_entries.log_source, 'session_replay'))) AS console_logs_log_entries - WHERE and(ifNull(equals(console_logs_log_entries.level, 'info'), 0), ifNull(ilike(console_logs_log_entries.message, '%message 5%'), 0))))) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestSessionRecordingsListFromFilters.test_filter_for_recordings_by_snapshot_source - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-01 13:41:23.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-25 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), 0)) - GROUP BY s.session_id - HAVING ifNull(equals(argMinMerge(s.snapshot_source), 'web'), 0) - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestSessionRecordingsListFromFilters.test_filter_for_recordings_by_snapshot_source.1 - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-01 13:41:23.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-25 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), 0)) - GROUP BY s.session_id - HAVING ifNull(equals(argMinMerge(s.snapshot_source), 'mobile'), 0) - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestSessionRecordingsListFromFilters.test_filter_for_recordings_with_console_errors - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-21 19:55:00.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-31 20:00:00.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-14 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-21 20:00:00.000000', 6, 'UTC')), 0), in(s.session_id, - (SELECT console_logs_log_entries.log_source_id AS log_source_id - FROM - (SELECT log_entries.log_source_id AS log_source_id, log_entries.level AS level - FROM log_entries - WHERE and(equals(log_entries.team_id, 99999), equals(log_entries.log_source, 'session_replay'))) AS console_logs_log_entries - WHERE ifNull(equals(console_logs_log_entries.level, 'error'), 0)))) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestSessionRecordingsListFromFilters.test_filter_for_recordings_with_console_errors.1 - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-21 19:55:00.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-31 20:00:00.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-14 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-21 20:00:00.000000', 6, 'UTC')), 0), in(s.session_id, - (SELECT console_logs_log_entries.log_source_id AS log_source_id - FROM - (SELECT log_entries.log_source_id AS log_source_id, log_entries.level AS level - FROM log_entries - WHERE and(equals(log_entries.team_id, 99999), equals(log_entries.log_source, 'session_replay'))) AS console_logs_log_entries - WHERE ifNull(equals(console_logs_log_entries.level, 'info'), 0)))) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestSessionRecordingsListFromFilters.test_filter_for_recordings_with_console_logs - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-21 19:55:00.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-31 20:00:00.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-14 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-21 20:00:00.000000', 6, 'UTC')), 0), in(s.session_id, - (SELECT console_logs_log_entries.log_source_id AS log_source_id - FROM - (SELECT log_entries.log_source_id AS log_source_id, log_entries.level AS level - FROM log_entries - WHERE and(equals(log_entries.team_id, 99999), equals(log_entries.log_source, 'session_replay'))) AS console_logs_log_entries - WHERE ifNull(equals(console_logs_log_entries.level, 'info'), 0)))) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestSessionRecordingsListFromFilters.test_filter_for_recordings_with_console_logs.1 - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-21 19:55:00.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-31 20:00:00.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-14 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-21 20:00:00.000000', 6, 'UTC')), 0), in(s.session_id, - (SELECT console_logs_log_entries.log_source_id AS log_source_id - FROM - (SELECT log_entries.log_source_id AS log_source_id, log_entries.level AS level - FROM log_entries - WHERE and(equals(log_entries.team_id, 99999), equals(log_entries.log_source, 'session_replay'))) AS console_logs_log_entries - WHERE ifNull(equals(console_logs_log_entries.level, 'warn'), 0)))) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestSessionRecordingsListFromFilters.test_filter_for_recordings_with_console_warns - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-21 19:55:00.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-31 20:00:00.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-14 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-21 20:00:00.000000', 6, 'UTC')), 0), in(s.session_id, - (SELECT console_logs_log_entries.log_source_id AS log_source_id - FROM - (SELECT log_entries.log_source_id AS log_source_id, log_entries.level AS level - FROM log_entries - WHERE and(equals(log_entries.team_id, 99999), equals(log_entries.log_source, 'session_replay'))) AS console_logs_log_entries - WHERE ifNull(equals(console_logs_log_entries.level, 'warn'), 0)))) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestSessionRecordingsListFromFilters.test_filter_for_recordings_with_console_warns.1 - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-21 19:55:00.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-31 20:00:00.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-14 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-21 20:00:00.000000', 6, 'UTC')), 0), in(s.session_id, - (SELECT console_logs_log_entries.log_source_id AS log_source_id - FROM - (SELECT log_entries.log_source_id AS log_source_id, log_entries.level AS level - FROM log_entries - WHERE and(equals(log_entries.team_id, 99999), equals(log_entries.log_source, 'session_replay'))) AS console_logs_log_entries - WHERE ifNull(equals(console_logs_log_entries.level, 'info'), 0)))) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestSessionRecordingsListFromFilters.test_filter_for_recordings_with_mixed_console_counts - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-21 19:55:00.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-31 20:00:00.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-14 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-21 20:00:00.000000', 6, 'UTC')), 0), in(s.session_id, - (SELECT console_logs_log_entries.log_source_id AS log_source_id - FROM - (SELECT log_entries.log_source_id AS log_source_id, log_entries.level AS level - FROM log_entries - WHERE and(equals(log_entries.team_id, 99999), equals(log_entries.log_source, 'session_replay'))) AS console_logs_log_entries - WHERE or(ifNull(equals(console_logs_log_entries.level, 'warn'), 0), ifNull(equals(console_logs_log_entries.level, 'error'), 0))))) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestSessionRecordingsListFromFilters.test_filter_for_recordings_with_mixed_console_counts.1 - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-21 19:55:00.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-31 20:00:00.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-14 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-21 20:00:00.000000', 6, 'UTC')), 0), in(s.session_id, - (SELECT console_logs_log_entries.log_source_id AS log_source_id - FROM - (SELECT log_entries.log_source_id AS log_source_id, log_entries.level AS level - FROM log_entries - WHERE and(equals(log_entries.team_id, 99999), equals(log_entries.log_source, 'session_replay'))) AS console_logs_log_entries - WHERE ifNull(equals(console_logs_log_entries.level, 'info'), 0)))) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestSessionRecordingsListFromFilters.test_filter_on_session_ids - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-01 13:41:23.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), - ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), 0), - in(s.session_id, - ['00000000-0000-0000-0000-000000000000', '00000000-0000-0000-0000-000000000001' /* ... */], - ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-25 00:00:00.000000', 6, 'UTC')), 0), - ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), 0)) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestSessionRecordingsListFromFilters.test_filter_on_session_ids.1 - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-01 13:41:23.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), - ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), 0), - in(s.session_id, - ['00000000-0000-0000-0000-000000000000', '00000000-0000-0000-0000-000000000001' /* ... */], - ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-25 00:00:00.000000', 6, 'UTC')), 0), - ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), 0)) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestSessionRecordingsListFromFilters.test_filter_with_cohort_properties - ''' - - SELECT count(DISTINCT person_id) - FROM cohortpeople - WHERE team_id = 99999 - AND cohort_id = 99999 - AND version = NULL - ''' -# --- -# name: TestSessionRecordingsListFromFilters.test_filter_with_cohort_properties.1 - ''' - /* cohort_calculation: */ - SELECT count(DISTINCT person_id) - FROM cohortpeople - WHERE team_id = 99999 - AND cohort_id = 99999 - AND version = 0 - ''' -# --- -# name: TestSessionRecordingsListFromFilters.test_filter_with_cohort_properties.2 - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-08-21 19:55:00.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-07-31 20:00:00.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-08-14 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-08-21 20:00:00.000000', 6, 'UTC')), 0), in(s.distinct_id, - (SELECT person_distinct_id2.distinct_id AS distinct_id - FROM person_distinct_id2 - WHERE and(equals(person_distinct_id2.team_id, 99999), in(person_distinct_id2.distinct_id, - (SELECT person_distinct_id2.distinct_id AS distinct_id - FROM person_distinct_id2 - WHERE and(equals(person_distinct_id2.team_id, 99999), 1, in(person_distinct_id2.person_id, - (SELECT cohortpeople.person_id AS person_id - FROM cohortpeople - WHERE and(equals(cohortpeople.team_id, 99999), equals(cohortpeople.cohort_id, 99999), equals(cohortpeople.version, 0)))))))) - GROUP BY person_distinct_id2.distinct_id - HAVING and(ifNull(equals(argMax(person_distinct_id2.is_deleted, person_distinct_id2.version), 0), 0), in(person_distinct_id2.person_id, - (SELECT cohortpeople.person_id AS person_id - FROM cohortpeople - WHERE and(equals(cohortpeople.team_id, 99999), equals(cohortpeople.cohort_id, 99999), equals(cohortpeople.version, 0)))))))) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestSessionRecordingsListFromFilters.test_filter_with_events_and_cohorts - ''' - - SELECT count(DISTINCT person_id) - FROM cohortpeople - WHERE team_id = 99999 - AND cohort_id = 99999 - AND version = NULL - ''' -# --- -# name: TestSessionRecordingsListFromFilters.test_filter_with_events_and_cohorts.1 - ''' - /* cohort_calculation: */ - SELECT count(DISTINCT person_id) - FROM cohortpeople - WHERE team_id = 99999 - AND cohort_id = 99999 - AND version = 0 - ''' -# --- -# name: TestSessionRecordingsListFromFilters.test_filter_with_events_and_cohorts.2 - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-08-21 19:55:00.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-07-31 20:00:00.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-08-14 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-08-21 20:00:00.000000', 6, 'UTC')), 0), and(in(s.session_id, - (SELECT events.`$session_id` AS session_id - FROM events - WHERE and(equals(events.team_id, 99999), notEmpty(events.`$session_id`), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-07-31 20:00:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), now64(6, 'UTC')), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-08-13 23:58:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-08-21 20:00:00.000000', 6, 'UTC')), and(equals(events.event, '$pageview'), 1)) - GROUP BY events.`$session_id` - HAVING hasAll(groupUniqArray(events.event), ['$pageview']))), in(s.distinct_id, - (SELECT person_distinct_id2.distinct_id AS distinct_id - FROM person_distinct_id2 - WHERE and(equals(person_distinct_id2.team_id, 99999), in(person_distinct_id2.distinct_id, - (SELECT person_distinct_id2.distinct_id AS distinct_id - FROM person_distinct_id2 - WHERE and(equals(person_distinct_id2.team_id, 99999), 1, in(person_distinct_id2.person_id, - (SELECT cohortpeople.person_id AS person_id - FROM cohortpeople - WHERE and(equals(cohortpeople.team_id, 99999), equals(cohortpeople.cohort_id, 99999), equals(cohortpeople.version, 0)))))))) - GROUP BY person_distinct_id2.distinct_id - HAVING and(ifNull(equals(argMax(person_distinct_id2.is_deleted, person_distinct_id2.version), 0), 0), in(person_distinct_id2.person_id, - (SELECT cohortpeople.person_id AS person_id - FROM cohortpeople - WHERE and(equals(cohortpeople.team_id, 99999), equals(cohortpeople.cohort_id, 99999), equals(cohortpeople.version, 0))))))))) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestSessionRecordingsListFromFilters.test_filter_with_events_and_cohorts.3 - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-08-21 19:55:00.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-07-31 20:00:00.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-08-14 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-08-21 20:00:00.000000', 6, 'UTC')), 0), and(in(s.session_id, - (SELECT events.`$session_id` AS session_id - FROM events - WHERE and(equals(events.team_id, 99999), notEmpty(events.`$session_id`), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-07-31 20:00:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), now64(6, 'UTC')), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-08-13 23:58:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-08-21 20:00:00.000000', 6, 'UTC')), and(equals(events.event, 'custom_event'), 1)) - GROUP BY events.`$session_id` - HAVING hasAll(groupUniqArray(events.event), ['custom_event']))), in(s.distinct_id, - (SELECT person_distinct_id2.distinct_id AS distinct_id - FROM person_distinct_id2 - WHERE and(equals(person_distinct_id2.team_id, 99999), in(person_distinct_id2.distinct_id, - (SELECT person_distinct_id2.distinct_id AS distinct_id - FROM person_distinct_id2 - WHERE and(equals(person_distinct_id2.team_id, 99999), 1, in(person_distinct_id2.person_id, - (SELECT cohortpeople.person_id AS person_id - FROM cohortpeople - WHERE and(equals(cohortpeople.team_id, 99999), equals(cohortpeople.cohort_id, 99999), equals(cohortpeople.version, 0)))))))) - GROUP BY person_distinct_id2.distinct_id - HAVING and(ifNull(equals(argMax(person_distinct_id2.is_deleted, person_distinct_id2.version), 0), 0), in(person_distinct_id2.person_id, - (SELECT cohortpeople.person_id AS person_id - FROM cohortpeople - WHERE and(equals(cohortpeople.team_id, 99999), equals(cohortpeople.cohort_id, 99999), equals(cohortpeople.version, 0))))))))) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestSessionRecordingsListFromFilters.test_filter_with_person_properties_exact - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-01 13:41:23.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-25 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), 0), in(s.session_id, - (SELECT events.`$session_id` AS session_id - FROM events - LEFT OUTER JOIN - (SELECT argMax(person_distinct_id_overrides.person_id, person_distinct_id_overrides.version) AS person_id, person_distinct_id_overrides.distinct_id AS distinct_id - FROM person_distinct_id_overrides - WHERE equals(person_distinct_id_overrides.team_id, 99999) - GROUP BY person_distinct_id_overrides.distinct_id - HAVING ifNull(equals(argMax(person_distinct_id_overrides.is_deleted, person_distinct_id_overrides.version), 0), 0) SETTINGS optimize_aggregation_in_order=1) AS events__override ON equals(events.distinct_id, events__override.distinct_id) - LEFT JOIN - (SELECT person.id AS id, replaceRegexpAll(nullIf(nullIf(JSONExtractRaw(person.properties, 'email'), ''), 'null'), '^"|"$', '') AS properties___email - FROM person - WHERE and(equals(person.team_id, 99999), ifNull(in(tuple(person.id, person.version), - (SELECT person.id AS id, max(person.version) AS version - FROM person - WHERE equals(person.team_id, 99999) - GROUP BY person.id - HAVING and(ifNull(equals(argMax(person.is_deleted, person.version), 0), 0), ifNull(less(argMax(toTimeZone(person.created_at, 'UTC'), person.version), plus(now64(6, 'UTC'), toIntervalDay(1))), 0)))), 0)) SETTINGS optimize_aggregation_in_order=1) AS events__person ON equals(if(not(empty(events__override.distinct_id)), events__override.person_id, events.person_id), events__person.id) - WHERE and(equals(events.team_id, 99999), notEmpty(events.`$session_id`), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), now64(6, 'UTC')), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-24 23:58:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), ifNull(equals(events__person.properties___email, 'bla@gmail.com'), 0)) - GROUP BY events.`$session_id` - HAVING 1))) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestSessionRecordingsListFromFilters.test_filter_with_person_properties_not_contains - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-01 13:41:23.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-25 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), 0), in(s.session_id, - (SELECT events.`$session_id` AS session_id - FROM events - LEFT OUTER JOIN - (SELECT argMax(person_distinct_id_overrides.person_id, person_distinct_id_overrides.version) AS person_id, person_distinct_id_overrides.distinct_id AS distinct_id - FROM person_distinct_id_overrides - WHERE equals(person_distinct_id_overrides.team_id, 99999) - GROUP BY person_distinct_id_overrides.distinct_id - HAVING ifNull(equals(argMax(person_distinct_id_overrides.is_deleted, person_distinct_id_overrides.version), 0), 0) SETTINGS optimize_aggregation_in_order=1) AS events__override ON equals(events.distinct_id, events__override.distinct_id) - LEFT JOIN - (SELECT person.id AS id, replaceRegexpAll(nullIf(nullIf(JSONExtractRaw(person.properties, 'email'), ''), 'null'), '^"|"$', '') AS properties___email - FROM person - WHERE and(equals(person.team_id, 99999), ifNull(in(tuple(person.id, person.version), - (SELECT person.id AS id, max(person.version) AS version - FROM person - WHERE equals(person.team_id, 99999) - GROUP BY person.id - HAVING and(ifNull(equals(argMax(person.is_deleted, person.version), 0), 0), ifNull(less(argMax(toTimeZone(person.created_at, 'UTC'), person.version), plus(now64(6, 'UTC'), toIntervalDay(1))), 0)))), 0)) SETTINGS optimize_aggregation_in_order=1) AS events__person ON equals(if(not(empty(events__override.distinct_id)), events__override.person_id, events.person_id), events__person.id) - WHERE and(equals(events.team_id, 99999), notEmpty(events.`$session_id`), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), now64(6, 'UTC')), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-24 23:58:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), ifNull(notILike(events__person.properties___email, '%gmail.com%'), 1)) - GROUP BY events.`$session_id` - HAVING 1))) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestSessionRecordingsListFromFilters.test_filter_with_static_and_dynamic_cohort_properties - ''' - - SELECT count(DISTINCT person_id) - FROM person_static_cohort - WHERE team_id = 99999 - AND cohort_id = 99999 - ''' -# --- -# name: TestSessionRecordingsListFromFilters.test_filter_with_static_and_dynamic_cohort_properties.1 - ''' - - SELECT count(DISTINCT person_id) - FROM cohortpeople - WHERE team_id = 99999 - AND cohort_id = 99999 - AND version = NULL - ''' -# --- -# name: TestSessionRecordingsListFromFilters.test_filter_with_static_and_dynamic_cohort_properties.2 - ''' - /* cohort_calculation: */ - SELECT count(DISTINCT person_id) - FROM cohortpeople - WHERE team_id = 99999 - AND cohort_id = 99999 - AND version = 0 - ''' -# --- -# name: TestSessionRecordingsListFromFilters.test_filter_with_static_and_dynamic_cohort_properties.3 - ''' - - SELECT count(DISTINCT person_id) - FROM cohortpeople - WHERE team_id = 99999 - AND cohort_id = 99999 - AND version = NULL - ''' -# --- -# name: TestSessionRecordingsListFromFilters.test_filter_with_static_and_dynamic_cohort_properties.4 - ''' - /* cohort_calculation: */ - SELECT count(DISTINCT person_id) - FROM cohortpeople - WHERE team_id = 99999 - AND cohort_id = 99999 - AND version = 0 - ''' -# --- -# name: TestSessionRecordingsListFromFilters.test_filter_with_static_and_dynamic_cohort_properties.5 - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-08-21 19:55:00.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-07-31 20:00:00.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-08-14 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-08-21 20:00:00.000000', 6, 'UTC')), 0), in(s.distinct_id, - (SELECT person_distinct_id2.distinct_id AS distinct_id - FROM person_distinct_id2 - WHERE and(equals(person_distinct_id2.team_id, 99999), in(person_distinct_id2.distinct_id, - (SELECT person_distinct_id2.distinct_id AS distinct_id - FROM person_distinct_id2 - WHERE and(equals(person_distinct_id2.team_id, 99999), 1, in(person_distinct_id2.person_id, - (SELECT person_static_cohort.person_id AS person_id - FROM person_static_cohort - WHERE and(equals(person_static_cohort.team_id, 99999), equals(person_static_cohort.cohort_id, 99999)))))))) - GROUP BY person_distinct_id2.distinct_id - HAVING and(ifNull(equals(argMax(person_distinct_id2.is_deleted, person_distinct_id2.version), 0), 0), in(person_distinct_id2.person_id, - (SELECT person_static_cohort.person_id AS person_id - FROM person_static_cohort - WHERE and(equals(person_static_cohort.team_id, 99999), equals(person_static_cohort.cohort_id, 99999)))))))) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestSessionRecordingsListFromFilters.test_filter_with_static_and_dynamic_cohort_properties.6 - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-08-21 19:55:00.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-07-31 20:00:00.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-08-14 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-08-21 20:00:00.000000', 6, 'UTC')), 0), in(s.distinct_id, - (SELECT person_distinct_id2.distinct_id AS distinct_id - FROM person_distinct_id2 - WHERE and(equals(person_distinct_id2.team_id, 99999), in(person_distinct_id2.distinct_id, - (SELECT person_distinct_id2.distinct_id AS distinct_id - FROM person_distinct_id2 - WHERE and(equals(person_distinct_id2.team_id, 99999), 1, in(person_distinct_id2.person_id, - (SELECT cohortpeople.person_id AS person_id - FROM cohortpeople - WHERE and(equals(cohortpeople.team_id, 99999), equals(cohortpeople.cohort_id, 99999), equals(cohortpeople.version, 0)))))))) - GROUP BY person_distinct_id2.distinct_id - HAVING and(ifNull(equals(argMax(person_distinct_id2.is_deleted, person_distinct_id2.version), 0), 0), in(person_distinct_id2.person_id, - (SELECT cohortpeople.person_id AS person_id - FROM cohortpeople - WHERE and(equals(cohortpeople.team_id, 99999), equals(cohortpeople.cohort_id, 99999), equals(cohortpeople.version, 0)))))))) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestSessionRecordingsListFromFilters.test_filter_with_static_and_dynamic_cohort_properties.7 - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-08-21 19:55:00.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-07-31 20:00:00.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-08-14 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-08-21 20:00:00.000000', 6, 'UTC')), 0), in(s.distinct_id, - (SELECT person_distinct_id2.distinct_id AS distinct_id - FROM person_distinct_id2 - WHERE and(equals(person_distinct_id2.team_id, 99999), in(person_distinct_id2.distinct_id, - (SELECT person_distinct_id2.distinct_id AS distinct_id - FROM person_distinct_id2 - WHERE and(equals(person_distinct_id2.team_id, 99999), 1, and(in(person_distinct_id2.person_id, - (SELECT cohortpeople.person_id AS person_id - FROM cohortpeople - WHERE and(equals(cohortpeople.team_id, 99999), equals(cohortpeople.cohort_id, 99999), equals(cohortpeople.version, 0)))), in(person_distinct_id2.person_id, - (SELECT person_static_cohort.person_id AS person_id - FROM person_static_cohort - WHERE and(equals(person_static_cohort.team_id, 99999), equals(person_static_cohort.cohort_id, 99999))))))))) - GROUP BY person_distinct_id2.distinct_id - HAVING and(ifNull(equals(argMax(person_distinct_id2.is_deleted, person_distinct_id2.version), 0), 0), and(in(person_distinct_id2.person_id, - (SELECT cohortpeople.person_id AS person_id - FROM cohortpeople - WHERE and(equals(cohortpeople.team_id, 99999), equals(cohortpeople.cohort_id, 99999), equals(cohortpeople.version, 0)))), in(person_distinct_id2.person_id, - (SELECT person_static_cohort.person_id AS person_id - FROM person_static_cohort - WHERE and(equals(person_static_cohort.team_id, 99999), equals(person_static_cohort.cohort_id, 99999))))))))) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestSessionRecordingsListFromFilters.test_listing_ignores_future_replays - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2023-08-30 11:55:01.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2023-08-09 12:00:01.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2023-08-23 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2023-08-30 12:00:01.000000', 6, 'UTC')), 0)) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestSessionRecordingsListFromFilters.test_multiple_event_filters - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-01 13:41:23.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-25 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), 0), in(s.session_id, - (SELECT events.`$session_id` AS session_id - FROM events - WHERE and(equals(events.team_id, 99999), notEmpty(events.`$session_id`), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), now64(6, 'UTC')), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-24 23:58:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), or(and(equals(events.event, '$pageview'), 1), and(equals(events.event, 'new-event'), 1))) - GROUP BY events.`$session_id` - HAVING hasAll(groupUniqArray(events.event), ['$pageview', 'new-event'])))) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestSessionRecordingsListFromFilters.test_multiple_event_filters.1 - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-01 13:41:23.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-25 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), 0), in(s.session_id, - (SELECT events.`$session_id` AS session_id - FROM events - WHERE and(equals(events.team_id, 99999), notEmpty(events.`$session_id`), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), now64(6, 'UTC')), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-24 23:58:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), or(and(equals(events.event, '$pageview'), 1), and(equals(events.event, 'new-event2'), 1))) - GROUP BY events.`$session_id` - HAVING hasAll(groupUniqArray(events.event), ['$pageview', 'new-event2'])))) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestSessionRecordingsListFromFilters.test_multiple_event_filters.2 - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-01 13:41:23.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-25 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), 0), in(s.session_id, - (SELECT events.`$session_id` AS session_id - FROM events - WHERE and(equals(events.team_id, 99999), notEmpty(events.`$session_id`), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), now64(6, 'UTC')), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-24 23:58:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), or(and(equals(events.event, '$pageview'), 1), and(equals(events.event, 'new-event2'), 1))) - GROUP BY events.`$session_id` - HAVING hasAny(groupUniqArray(events.event), ['$pageview', 'new-event2'])))) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestSessionRecordingsListFromFilters.test_multiple_event_filters.3 - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-01 13:41:23.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-25 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), 0), in(s.session_id, - (SELECT events.`$session_id` AS session_id - FROM events - WHERE and(equals(events.team_id, 99999), notEmpty(events.`$session_id`), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), now64(6, 'UTC')), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-24 23:58:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), or(and(equals(events.event, '$pageview'), ifNull(equals(replaceRegexpAll(nullIf(nullIf(JSONExtractRaw(events.properties, 'foo'), ''), 'null'), '^"|"$', ''), 'bar'), 0)), and(equals(events.event, '$pageview'), ifNull(equals(replaceRegexpAll(nullIf(nullIf(JSONExtractRaw(events.properties, 'bar'), ''), 'null'), '^"|"$', ''), 'foo'), 0)))) - GROUP BY events.`$session_id` - HAVING hasAll(groupUniqArray(events.event), ['$pageview'])))) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestSessionRecordingsListFromFilters.test_multiple_event_filters.4 - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-01 13:41:23.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-25 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), 0), in(s.session_id, - (SELECT events.`$session_id` AS session_id - FROM events - WHERE and(equals(events.team_id, 99999), notEmpty(events.`$session_id`), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), now64(6, 'UTC')), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-24 23:58:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), or(and(equals(events.event, '$pageview'), ifNull(equals(replaceRegexpAll(nullIf(nullIf(JSONExtractRaw(events.properties, 'foo'), ''), 'null'), '^"|"$', ''), 'bar'), 0)), and(equals(events.event, 'new-event'), ifNull(equals(replaceRegexpAll(nullIf(nullIf(JSONExtractRaw(events.properties, 'foo'), ''), 'null'), '^"|"$', ''), 'bar'), 0)))) - GROUP BY events.`$session_id` - HAVING hasAll(groupUniqArray(events.event), ['$pageview', 'new-event'])))) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestSessionRecordingsListFromFilters.test_multiple_event_filters.5 - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-01 13:41:23.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-25 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), 0), in(s.session_id, - (SELECT events.`$session_id` AS session_id - FROM events - WHERE and(equals(events.team_id, 99999), notEmpty(events.`$session_id`), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), now64(6, 'UTC')), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-24 23:58:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), or(and(equals(events.event, '$pageview'), ifNull(equals(replaceRegexpAll(nullIf(nullIf(JSONExtractRaw(events.properties, 'foo'), ''), 'null'), '^"|"$', ''), 'bar'), 0)), and(equals(events.event, 'new-event'), ifNull(equals(replaceRegexpAll(nullIf(nullIf(JSONExtractRaw(events.properties, 'foo'), ''), 'null'), '^"|"$', ''), 'bar'), 0)))) - GROUP BY events.`$session_id` - HAVING hasAny(groupUniqArray(events.event), ['$pageview', 'new-event'])))) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestSessionRecordingsListFromFilters.test_operand_or_event_filters - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-01 13:41:23.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-25 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), 0), in(s.session_id, - (SELECT events.`$session_id` AS session_id - FROM events - WHERE and(equals(events.team_id, 99999), notEmpty(events.`$session_id`), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), now64(6, 'UTC')), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-24 23:58:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), or(and(equals(events.event, '$pageview'), 1), and(equals(events.event, 'custom_event'), 1))) - GROUP BY events.`$session_id` - HAVING hasAll(groupUniqArray(events.event), ['$pageview', 'custom_event'])))) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestSessionRecordingsListFromFilters.test_operand_or_event_filters.1 - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-01 13:41:23.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-25 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), 0), in(s.session_id, - (SELECT events.`$session_id` AS session_id - FROM events - WHERE and(equals(events.team_id, 99999), notEmpty(events.`$session_id`), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), now64(6, 'UTC')), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-24 23:58:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), or(and(equals(events.event, '$pageview'), 1), and(equals(events.event, 'custom_event'), 1))) - GROUP BY events.`$session_id` - HAVING hasAny(groupUniqArray(events.event), ['$pageview', 'custom_event'])))) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestSessionRecordingsListFromFilters.test_operand_or_filters_0__key_level_value_warn_operator_exact_type_log_entry_key_message_value_random_operator_exact_type_log_entry_ - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-01 13:41:23.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-25 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), 0), in(s.session_id, - (SELECT console_logs_log_entries.log_source_id AS log_source_id - FROM - (SELECT log_entries.log_source_id AS log_source_id, log_entries.level AS level, log_entries.message AS message - FROM log_entries - WHERE and(equals(log_entries.team_id, 99999), equals(log_entries.log_source, 'session_replay'))) AS console_logs_log_entries - WHERE and(ifNull(equals(console_logs_log_entries.level, 'warn'), 0), ifNull(equals(console_logs_log_entries.message, 'random'), 0))))) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestSessionRecordingsListFromFilters.test_operand_or_filters_1__key_level_value_info_operator_exact_type_log_entry_key_message_value_random_operator_exact_type_log_entry_ - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-01 13:41:23.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-25 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), 0), in(s.session_id, - (SELECT console_logs_log_entries.log_source_id AS log_source_id - FROM - (SELECT log_entries.log_source_id AS log_source_id, log_entries.level AS level, log_entries.message AS message - FROM log_entries - WHERE and(equals(log_entries.team_id, 99999), equals(log_entries.log_source, 'session_replay'))) AS console_logs_log_entries - WHERE and(ifNull(equals(console_logs_log_entries.level, 'info'), 0), ifNull(equals(console_logs_log_entries.message, 'random'), 0))))) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestSessionRecordingsListFromFilters.test_operand_or_filters_2__key_level_value_warn_operator_exact_type_log_entry_ - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-01 13:41:23.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-25 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), 0), in(s.session_id, - (SELECT console_logs_log_entries.log_source_id AS log_source_id - FROM - (SELECT log_entries.log_source_id AS log_source_id, log_entries.level AS level - FROM log_entries - WHERE and(equals(log_entries.team_id, 99999), equals(log_entries.log_source, 'session_replay'))) AS console_logs_log_entries - WHERE ifNull(equals(console_logs_log_entries.level, 'warn'), 0)))) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestSessionRecordingsListFromFilters.test_operand_or_filters_3__key_message_value_random_operator_exact_type_log_entry_ - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-01 13:41:23.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-25 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), 0), in(s.session_id, - (SELECT console_logs_log_entries.log_source_id AS log_source_id - FROM - (SELECT log_entries.log_source_id AS log_source_id, log_entries.message AS message - FROM log_entries - WHERE and(equals(log_entries.team_id, 99999), equals(log_entries.log_source, 'session_replay'))) AS console_logs_log_entries - WHERE ifNull(equals(console_logs_log_entries.message, 'random'), 0)))) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestSessionRecordingsListFromFilters.test_operand_or_filters_4__key_level_value_warn_operator_exact_type_log_entry_key_message_value_random_operator_exact_type_log_entry_ - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-01 13:41:23.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-25 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), 0), in(s.session_id, - (SELECT console_logs_log_entries.log_source_id AS log_source_id - FROM - (SELECT log_entries.log_source_id AS log_source_id, log_entries.level AS level, log_entries.message AS message - FROM log_entries - WHERE and(equals(log_entries.team_id, 99999), equals(log_entries.log_source, 'session_replay'))) AS console_logs_log_entries - WHERE or(ifNull(equals(console_logs_log_entries.level, 'warn'), 0), ifNull(equals(console_logs_log_entries.message, 'random'), 0))))) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestSessionRecordingsListFromFilters.test_operand_or_mandatory_filters - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-01 13:41:23.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), 0), in(s.session_id, - (SELECT DISTINCT events.`$session_id` AS `$session_id` - FROM events - LEFT OUTER JOIN - (SELECT argMax(person_distinct_id_overrides.person_id, person_distinct_id_overrides.version) AS person_id, person_distinct_id_overrides.distinct_id AS distinct_id - FROM person_distinct_id_overrides - WHERE equals(person_distinct_id_overrides.team_id, 99999) - GROUP BY person_distinct_id_overrides.distinct_id - HAVING ifNull(equals(argMax(person_distinct_id_overrides.is_deleted, person_distinct_id_overrides.version), 0), 0) SETTINGS optimize_aggregation_in_order=1) AS events__override ON equals(events.distinct_id, events__override.distinct_id) - WHERE and(equals(events.team_id, 99999), ifNull(equals(if(not(empty(events__override.distinct_id)), events__override.person_id, events.person_id), '00000000-0000-0000-0000-000000000000'), 0), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-25 00:00:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), notEmpty(events.`$session_id`)))), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-25 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), 0), in(s.session_id, - (SELECT events.`$session_id` AS session_id - FROM events - WHERE and(equals(events.team_id, 99999), notEmpty(events.`$session_id`), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), now64(6, 'UTC')), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-24 23:58:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), and(equals(events.event, '$pageview'), 1)) - GROUP BY events.`$session_id` - HAVING hasAny(groupUniqArray(events.event), ['$pageview'])))) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestSessionRecordingsListFromFilters.test_operand_or_mandatory_filters.1 - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-01 13:41:23.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), 0), in(s.session_id, - (SELECT DISTINCT events.`$session_id` AS `$session_id` - FROM events - LEFT OUTER JOIN - (SELECT argMax(person_distinct_id_overrides.person_id, person_distinct_id_overrides.version) AS person_id, person_distinct_id_overrides.distinct_id AS distinct_id - FROM person_distinct_id_overrides - WHERE equals(person_distinct_id_overrides.team_id, 99999) - GROUP BY person_distinct_id_overrides.distinct_id - HAVING ifNull(equals(argMax(person_distinct_id_overrides.is_deleted, person_distinct_id_overrides.version), 0), 0) SETTINGS optimize_aggregation_in_order=1) AS events__override ON equals(events.distinct_id, events__override.distinct_id) - WHERE and(equals(events.team_id, 99999), ifNull(equals(if(not(empty(events__override.distinct_id)), events__override.person_id, events.person_id), '00000000-0000-0000-0000-000000000000'), 0), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-25 00:00:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), notEmpty(events.`$session_id`)))), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-25 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), 0), in(s.session_id, - (SELECT events.`$session_id` AS session_id - FROM events - WHERE and(equals(events.team_id, 99999), notEmpty(events.`$session_id`), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), now64(6, 'UTC')), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-24 23:58:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), and(equals(events.event, '$pageview'), 1)) - GROUP BY events.`$session_id` - HAVING hasAny(groupUniqArray(events.event), ['$pageview'])))) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestSessionRecordingsListFromFilters.test_operand_or_mandatory_filters.2 - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-01 13:41:23.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), 0), in(s.session_id, ['session_id_one']), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-25 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), 0), in(s.session_id, - (SELECT events.`$session_id` AS session_id - FROM events - WHERE and(equals(events.team_id, 99999), notEmpty(events.`$session_id`), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), now64(6, 'UTC')), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-24 23:58:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), and(equals(events.event, '$pageview'), 1), in(events.`$session_id`, ['session_id_one'])) - GROUP BY events.`$session_id` - HAVING hasAny(groupUniqArray(events.event), ['$pageview'])))) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestSessionRecordingsListFromFilters.test_operand_or_mandatory_filters.3 - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-01 13:41:23.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), 0), in(s.session_id, ['session_id_two']), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-25 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), 0), in(s.session_id, - (SELECT events.`$session_id` AS session_id - FROM events - WHERE and(equals(events.team_id, 99999), notEmpty(events.`$session_id`), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), now64(6, 'UTC')), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-24 23:58:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), and(equals(events.event, '$pageview'), 1), in(events.`$session_id`, ['session_id_two'])) - GROUP BY events.`$session_id` - HAVING hasAny(groupUniqArray(events.event), ['$pageview'])))) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestSessionRecordingsListFromFilters.test_operand_or_person_filters - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-01 13:41:23.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-25 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), 0), in(s.session_id, - (SELECT events.`$session_id` AS session_id - FROM events - LEFT OUTER JOIN - (SELECT argMax(person_distinct_id_overrides.person_id, person_distinct_id_overrides.version) AS person_id, person_distinct_id_overrides.distinct_id AS distinct_id - FROM person_distinct_id_overrides - WHERE equals(person_distinct_id_overrides.team_id, 99999) - GROUP BY person_distinct_id_overrides.distinct_id - HAVING ifNull(equals(argMax(person_distinct_id_overrides.is_deleted, person_distinct_id_overrides.version), 0), 0) SETTINGS optimize_aggregation_in_order=1) AS events__override ON equals(events.distinct_id, events__override.distinct_id) - LEFT JOIN - (SELECT person.id AS id, replaceRegexpAll(nullIf(nullIf(JSONExtractRaw(person.properties, 'email'), ''), 'null'), '^"|"$', '') AS properties___email - FROM person - WHERE and(equals(person.team_id, 99999), ifNull(in(tuple(person.id, person.version), - (SELECT person.id AS id, max(person.version) AS version - FROM person - WHERE equals(person.team_id, 99999) - GROUP BY person.id - HAVING and(ifNull(equals(argMax(person.is_deleted, person.version), 0), 0), ifNull(less(argMax(toTimeZone(person.created_at, 'UTC'), person.version), plus(now64(6, 'UTC'), toIntervalDay(1))), 0)))), 0)) SETTINGS optimize_aggregation_in_order=1) AS events__person ON equals(if(not(empty(events__override.distinct_id)), events__override.person_id, events.person_id), events__person.id) - WHERE and(equals(events.team_id, 99999), notEmpty(events.`$session_id`), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), now64(6, 'UTC')), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-24 23:58:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), and(ifNull(equals(events__person.properties___email, 'test@posthog.com'), 0), ifNull(equals(events__person.properties___email, 'david@posthog.com'), 0))) - GROUP BY events.`$session_id` - HAVING 1))) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestSessionRecordingsListFromFilters.test_operand_or_person_filters.1 - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-01 13:41:23.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-25 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), 0), in(s.session_id, - (SELECT events.`$session_id` AS session_id - FROM events - LEFT OUTER JOIN - (SELECT argMax(person_distinct_id_overrides.person_id, person_distinct_id_overrides.version) AS person_id, person_distinct_id_overrides.distinct_id AS distinct_id - FROM person_distinct_id_overrides - WHERE equals(person_distinct_id_overrides.team_id, 99999) - GROUP BY person_distinct_id_overrides.distinct_id - HAVING ifNull(equals(argMax(person_distinct_id_overrides.is_deleted, person_distinct_id_overrides.version), 0), 0) SETTINGS optimize_aggregation_in_order=1) AS events__override ON equals(events.distinct_id, events__override.distinct_id) - LEFT JOIN - (SELECT person.id AS id, replaceRegexpAll(nullIf(nullIf(JSONExtractRaw(person.properties, 'email'), ''), 'null'), '^"|"$', '') AS properties___email - FROM person - WHERE and(equals(person.team_id, 99999), ifNull(in(tuple(person.id, person.version), - (SELECT person.id AS id, max(person.version) AS version - FROM person - WHERE equals(person.team_id, 99999) - GROUP BY person.id - HAVING and(ifNull(equals(argMax(person.is_deleted, person.version), 0), 0), ifNull(less(argMax(toTimeZone(person.created_at, 'UTC'), person.version), plus(now64(6, 'UTC'), toIntervalDay(1))), 0)))), 0)) SETTINGS optimize_aggregation_in_order=1) AS events__person ON equals(if(not(empty(events__override.distinct_id)), events__override.person_id, events.person_id), events__person.id) - WHERE and(equals(events.team_id, 99999), notEmpty(events.`$session_id`), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), now64(6, 'UTC')), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-24 23:58:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), or(ifNull(equals(events__person.properties___email, 'test@posthog.com'), 0), ifNull(equals(events__person.properties___email, 'david@posthog.com'), 0))) - GROUP BY events.`$session_id` - HAVING 1))) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestSessionRecordingsListFromFilters.test_ordering - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-21 19:55:00.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-31 20:00:00.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-14 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-21 20:00:00.000000', 6, 'UTC')), 0)) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestSessionRecordingsListFromFilters.test_ordering.1 - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-21 19:55:00.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-31 20:00:00.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-14 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-21 20:00:00.000000', 6, 'UTC')), 0)) - GROUP BY s.session_id - HAVING 1 - ORDER BY mouse_activity_count DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestSessionRecordingsListFromFilters.test_person_id_filter - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-01 13:41:23.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), 0), in(s.session_id, - (SELECT DISTINCT events.`$session_id` AS `$session_id` - FROM events - LEFT OUTER JOIN - (SELECT argMax(person_distinct_id_overrides.person_id, person_distinct_id_overrides.version) AS person_id, person_distinct_id_overrides.distinct_id AS distinct_id - FROM person_distinct_id_overrides - WHERE equals(person_distinct_id_overrides.team_id, 99999) - GROUP BY person_distinct_id_overrides.distinct_id - HAVING ifNull(equals(argMax(person_distinct_id_overrides.is_deleted, person_distinct_id_overrides.version), 0), 0) SETTINGS optimize_aggregation_in_order=1) AS events__override ON equals(events.distinct_id, events__override.distinct_id) - WHERE and(equals(events.team_id, 99999), ifNull(equals(if(not(empty(events__override.distinct_id)), events__override.person_id, events.person_id), '00000000-0000-0000-0000-000000000000'), 0), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-25 00:00:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), notEmpty(events.`$session_id`)))), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-25 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), 0)) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestSessionRecordingsListFromFilters.test_sessions_with_current_data - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-01 13:41:23.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-25 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), 0)) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestSessionRecordingsListFromFilters.test_top_level_event_host_property_test_account_filter - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-21 19:55:00.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-31 20:00:00.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-14 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-21 20:00:00.000000', 6, 'UTC')), 0), in(s.session_id, - (SELECT events.`$session_id` AS session_id - FROM events - WHERE and(equals(events.team_id, 99999), notEmpty(events.`$session_id`), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-31 20:00:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), now64(6, 'UTC')), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-13 23:58:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-21 20:00:00.000000', 6, 'UTC')), and(equals(events.event, '$pageview'), 1)) - GROUP BY events.`$session_id` - HAVING hasAll(groupUniqArray(events.event), ['$pageview'])))) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestSessionRecordingsListFromFilters.test_top_level_event_host_property_test_account_filter.1 - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-21 19:55:00.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-31 20:00:00.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-14 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-21 20:00:00.000000', 6, 'UTC')), 0), in(s.session_id, - (SELECT events.`$session_id` AS session_id - FROM events - WHERE and(equals(events.team_id, 99999), notEmpty(events.`$session_id`), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-31 20:00:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), now64(6, 'UTC')), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-13 23:58:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-21 20:00:00.000000', 6, 'UTC')), ifNull(not(match(toString(replaceRegexpAll(nullIf(nullIf(JSONExtractRaw(events.properties, '$host'), ''), 'null'), '^"|"$', '')), '^(localhost|127\\.0\\.0\\.1)($|:)')), 1)) - GROUP BY events.`$session_id` - HAVING 1))) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestSessionRecordingsListFromFilters.test_top_level_event_host_property_test_account_filter_materialized - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-21 19:55:00.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-31 20:00:00.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-14 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-21 20:00:00.000000', 6, 'UTC')), 0), in(s.session_id, - (SELECT events.`$session_id` AS session_id - FROM events - WHERE and(equals(events.team_id, 99999), notEmpty(events.`$session_id`), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-31 20:00:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), now64(6, 'UTC')), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-13 23:58:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-21 20:00:00.000000', 6, 'UTC')), and(equals(events.event, '$pageview'), 1)) - GROUP BY events.`$session_id` - HAVING hasAll(groupUniqArray(events.event), ['$pageview'])))) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestSessionRecordingsListFromFilters.test_top_level_event_host_property_test_account_filter_materialized.1 - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-21 19:55:00.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-31 20:00:00.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-14 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-21 20:00:00.000000', 6, 'UTC')), 0), in(s.session_id, - (SELECT events.`$session_id` AS session_id - FROM events - WHERE and(equals(events.team_id, 99999), notEmpty(events.`$session_id`), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-31 20:00:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), now64(6, 'UTC')), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-13 23:58:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-21 20:00:00.000000', 6, 'UTC')), ifNull(not(match(toString(nullIf(nullIf(events.`mat_$host`, ''), 'null')), '^(localhost|127\\.0\\.0\\.1)($|:)')), 1)) - GROUP BY events.`$session_id` - HAVING 1))) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestSessionRecordingsListFromFilters.test_top_level_event_property_test_account_filter - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-21 19:55:00.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-31 20:00:00.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-14 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-21 20:00:00.000000', 6, 'UTC')), 0), in(s.session_id, - (SELECT events.`$session_id` AS session_id - FROM events - WHERE and(equals(events.team_id, 99999), notEmpty(events.`$session_id`), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-31 20:00:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), now64(6, 'UTC')), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-13 23:58:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-21 20:00:00.000000', 6, 'UTC')), and(equals(events.event, '$pageview'), 1)) - GROUP BY events.`$session_id` - HAVING hasAll(groupUniqArray(events.event), ['$pageview'])))) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestSessionRecordingsListFromFilters.test_top_level_event_property_test_account_filter.1 - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-21 19:55:00.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-31 20:00:00.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-14 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-21 20:00:00.000000', 6, 'UTC')), 0), in(s.session_id, - (SELECT events.`$session_id` AS session_id - FROM events - WHERE and(equals(events.team_id, 99999), notEmpty(events.`$session_id`), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-31 20:00:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), now64(6, 'UTC')), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-13 23:58:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-21 20:00:00.000000', 6, 'UTC')), ifNull(equals(replaceRegexpAll(nullIf(nullIf(JSONExtractRaw(events.properties, 'is_internal_user'), ''), 'null'), '^"|"$', ''), 'false'), 0)) - GROUP BY events.`$session_id` - HAVING 1))) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestSessionRecordingsListFromFilters.test_top_level_event_property_test_account_filter_allowing_denormalized_props - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-21 19:55:00.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-31 20:00:00.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-14 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-21 20:00:00.000000', 6, 'UTC')), 0), in(s.session_id, - (SELECT events.`$session_id` AS session_id - FROM events - WHERE and(equals(events.team_id, 99999), notEmpty(events.`$session_id`), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-31 20:00:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), now64(6, 'UTC')), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-13 23:58:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-21 20:00:00.000000', 6, 'UTC')), and(equals(events.event, '$pageview'), 1)) - GROUP BY events.`$session_id` - HAVING hasAll(groupUniqArray(events.event), ['$pageview'])))) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestSessionRecordingsListFromFilters.test_top_level_event_property_test_account_filter_allowing_denormalized_props.1 - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-21 19:55:00.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-31 20:00:00.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-14 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-21 20:00:00.000000', 6, 'UTC')), 0), in(s.session_id, - (SELECT events.`$session_id` AS session_id - FROM events - WHERE and(equals(events.team_id, 99999), notEmpty(events.`$session_id`), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-31 20:00:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), now64(6, 'UTC')), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-13 23:58:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-21 20:00:00.000000', 6, 'UTC')), ifNull(equals(replaceRegexpAll(nullIf(nullIf(JSONExtractRaw(events.properties, 'is_internal_user'), ''), 'null'), '^"|"$', ''), 'false'), 0)) - GROUP BY events.`$session_id` - HAVING 1))) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestSessionRecordingsListFromFilters.test_top_level_event_property_test_account_filter_allowing_denormalized_props_materialized - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-21 19:55:00.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-31 20:00:00.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-14 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-21 20:00:00.000000', 6, 'UTC')), 0), in(s.session_id, - (SELECT events.`$session_id` AS session_id - FROM events - WHERE and(equals(events.team_id, 99999), notEmpty(events.`$session_id`), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-31 20:00:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), now64(6, 'UTC')), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-13 23:58:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-21 20:00:00.000000', 6, 'UTC')), and(equals(events.event, '$pageview'), 1)) - GROUP BY events.`$session_id` - HAVING hasAll(groupUniqArray(events.event), ['$pageview'])))) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestSessionRecordingsListFromFilters.test_top_level_event_property_test_account_filter_allowing_denormalized_props_materialized.1 - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-21 19:55:00.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-31 20:00:00.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-14 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-21 20:00:00.000000', 6, 'UTC')), 0), in(s.session_id, - (SELECT events.`$session_id` AS session_id - FROM events - WHERE and(equals(events.team_id, 99999), notEmpty(events.`$session_id`), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-31 20:00:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), now64(6, 'UTC')), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-13 23:58:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-21 20:00:00.000000', 6, 'UTC')), ifNull(equals(nullIf(nullIf(events.mat_is_internal_user, ''), 'null'), 'false'), 0)) - GROUP BY events.`$session_id` - HAVING 1))) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestSessionRecordingsListFromFilters.test_top_level_event_property_test_account_filter_materialized - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-21 19:55:00.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-31 20:00:00.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-14 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-21 20:00:00.000000', 6, 'UTC')), 0), in(s.session_id, - (SELECT events.`$session_id` AS session_id - FROM events - WHERE and(equals(events.team_id, 99999), notEmpty(events.`$session_id`), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-31 20:00:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), now64(6, 'UTC')), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-13 23:58:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-21 20:00:00.000000', 6, 'UTC')), and(equals(events.event, '$pageview'), 1)) - GROUP BY events.`$session_id` - HAVING hasAll(groupUniqArray(events.event), ['$pageview'])))) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestSessionRecordingsListFromFilters.test_top_level_event_property_test_account_filter_materialized.1 - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-21 19:55:00.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-31 20:00:00.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-14 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-21 20:00:00.000000', 6, 'UTC')), 0), in(s.session_id, - (SELECT events.`$session_id` AS session_id - FROM events - WHERE and(equals(events.team_id, 99999), notEmpty(events.`$session_id`), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-31 20:00:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), now64(6, 'UTC')), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-13 23:58:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-21 20:00:00.000000', 6, 'UTC')), ifNull(equals(nullIf(nullIf(events.mat_is_internal_user, ''), 'null'), 'false'), 0)) - GROUP BY events.`$session_id` - HAVING 1))) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestSessionRecordingsListFromFilters.test_top_level_hogql_event_property_test_account_filter - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-21 19:55:00.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-31 20:00:00.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-14 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-21 20:00:00.000000', 6, 'UTC')), 0), in(s.session_id, - (SELECT events.`$session_id` AS session_id - FROM events - WHERE and(equals(events.team_id, 99999), notEmpty(events.`$session_id`), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-31 20:00:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), now64(6, 'UTC')), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-13 23:58:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-21 20:00:00.000000', 6, 'UTC')), and(equals(events.event, '$pageview'), 1)) - GROUP BY events.`$session_id` - HAVING hasAll(groupUniqArray(events.event), ['$pageview'])))) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestSessionRecordingsListFromFilters.test_top_level_hogql_event_property_test_account_filter.1 - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-21 19:55:00.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-31 20:00:00.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-14 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-21 20:00:00.000000', 6, 'UTC')), 0), in(s.session_id, - (SELECT events.`$session_id` AS session_id - FROM events - WHERE and(equals(events.team_id, 99999), notEmpty(events.`$session_id`), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-31 20:00:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), now64(6, 'UTC')), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-13 23:58:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-21 20:00:00.000000', 6, 'UTC')), ifNull(equals(replaceRegexpAll(nullIf(nullIf(JSONExtractRaw(events.properties, 'is_internal_user'), ''), 'null'), '^"|"$', ''), 'true'), 0)) - GROUP BY events.`$session_id` - HAVING 1))) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestSessionRecordingsListFromFilters.test_top_level_hogql_event_property_test_account_filter_materialized - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-21 19:55:00.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-31 20:00:00.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-14 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-21 20:00:00.000000', 6, 'UTC')), 0), in(s.session_id, - (SELECT events.`$session_id` AS session_id - FROM events - WHERE and(equals(events.team_id, 99999), notEmpty(events.`$session_id`), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-31 20:00:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), now64(6, 'UTC')), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-13 23:58:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-21 20:00:00.000000', 6, 'UTC')), and(equals(events.event, '$pageview'), 1)) - GROUP BY events.`$session_id` - HAVING hasAll(groupUniqArray(events.event), ['$pageview'])))) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestSessionRecordingsListFromFilters.test_top_level_hogql_event_property_test_account_filter_materialized.1 - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-21 19:55:00.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-31 20:00:00.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-14 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-21 20:00:00.000000', 6, 'UTC')), 0), in(s.session_id, - (SELECT events.`$session_id` AS session_id - FROM events - WHERE and(equals(events.team_id, 99999), notEmpty(events.`$session_id`), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-31 20:00:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), now64(6, 'UTC')), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-13 23:58:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-21 20:00:00.000000', 6, 'UTC')), ifNull(equals(nullIf(nullIf(events.mat_is_internal_user, ''), 'null'), 'true'), 0)) - GROUP BY events.`$session_id` - HAVING 1))) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestSessionRecordingsListFromFilters.test_top_level_hogql_person_property_test_account_filter - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-21 19:55:00.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-31 20:00:00.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-14 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-21 20:00:00.000000', 6, 'UTC')), 0), in(s.session_id, - (SELECT events.`$session_id` AS session_id - FROM events - WHERE and(equals(events.team_id, 99999), notEmpty(events.`$session_id`), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-31 20:00:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), now64(6, 'UTC')), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-13 23:58:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-21 20:00:00.000000', 6, 'UTC')), and(equals(events.event, '$pageview'), 1)) - GROUP BY events.`$session_id` - HAVING hasAll(groupUniqArray(events.event), ['$pageview'])))) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestSessionRecordingsListFromFilters.test_top_level_hogql_person_property_test_account_filter.1 - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-21 19:55:00.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-31 20:00:00.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-14 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-21 20:00:00.000000', 6, 'UTC')), 0), in(s.session_id, - (SELECT events.`$session_id` AS session_id - FROM events - LEFT OUTER JOIN - (SELECT argMax(person_distinct_id_overrides.person_id, person_distinct_id_overrides.version) AS person_id, person_distinct_id_overrides.distinct_id AS distinct_id - FROM person_distinct_id_overrides - WHERE equals(person_distinct_id_overrides.team_id, 99999) - GROUP BY person_distinct_id_overrides.distinct_id - HAVING ifNull(equals(argMax(person_distinct_id_overrides.is_deleted, person_distinct_id_overrides.version), 0), 0) SETTINGS optimize_aggregation_in_order=1) AS events__override ON equals(events.distinct_id, events__override.distinct_id) - LEFT JOIN - (SELECT person.id AS id, replaceRegexpAll(nullIf(nullIf(JSONExtractRaw(person.properties, 'email'), ''), 'null'), '^"|"$', '') AS properties___email - FROM person - WHERE and(equals(person.team_id, 99999), ifNull(in(tuple(person.id, person.version), - (SELECT person.id AS id, max(person.version) AS version - FROM person - WHERE equals(person.team_id, 99999) - GROUP BY person.id - HAVING and(ifNull(equals(argMax(person.is_deleted, person.version), 0), 0), ifNull(less(argMax(toTimeZone(person.created_at, 'UTC'), person.version), plus(now64(6, 'UTC'), toIntervalDay(1))), 0)))), 0)) SETTINGS optimize_aggregation_in_order=1) AS events__person ON equals(if(not(empty(events__override.distinct_id)), events__override.person_id, events.person_id), events__person.id) - WHERE and(equals(events.team_id, 99999), notEmpty(events.`$session_id`), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-31 20:00:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), now64(6, 'UTC')), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-13 23:58:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-21 20:00:00.000000', 6, 'UTC')), ifNull(equals(events__person.properties___email, 'bla'), 0)) - GROUP BY events.`$session_id` - HAVING 1))) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestSessionRecordingsListFromFilters.test_top_level_hogql_person_property_test_account_filter_materialized - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-21 19:55:00.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-31 20:00:00.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-14 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-21 20:00:00.000000', 6, 'UTC')), 0), in(s.session_id, - (SELECT events.`$session_id` AS session_id - FROM events - WHERE and(equals(events.team_id, 99999), notEmpty(events.`$session_id`), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-31 20:00:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), now64(6, 'UTC')), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-13 23:58:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-21 20:00:00.000000', 6, 'UTC')), and(equals(events.event, '$pageview'), 1)) - GROUP BY events.`$session_id` - HAVING hasAll(groupUniqArray(events.event), ['$pageview'])))) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestSessionRecordingsListFromFilters.test_top_level_hogql_person_property_test_account_filter_materialized.1 - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-21 19:55:00.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-31 20:00:00.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-14 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-21 20:00:00.000000', 6, 'UTC')), 0), in(s.session_id, - (SELECT events.`$session_id` AS session_id - FROM events - LEFT OUTER JOIN - (SELECT argMax(person_distinct_id_overrides.person_id, person_distinct_id_overrides.version) AS person_id, person_distinct_id_overrides.distinct_id AS distinct_id - FROM person_distinct_id_overrides - WHERE equals(person_distinct_id_overrides.team_id, 99999) - GROUP BY person_distinct_id_overrides.distinct_id - HAVING ifNull(equals(argMax(person_distinct_id_overrides.is_deleted, person_distinct_id_overrides.version), 0), 0) SETTINGS optimize_aggregation_in_order=1) AS events__override ON equals(events.distinct_id, events__override.distinct_id) - LEFT JOIN - (SELECT person.id AS id, nullIf(nullIf(person.pmat_email, ''), 'null') AS properties___email - FROM person - WHERE and(equals(person.team_id, 99999), ifNull(in(tuple(person.id, person.version), - (SELECT person.id AS id, max(person.version) AS version - FROM person - WHERE equals(person.team_id, 99999) - GROUP BY person.id - HAVING and(ifNull(equals(argMax(person.is_deleted, person.version), 0), 0), ifNull(less(argMax(toTimeZone(person.created_at, 'UTC'), person.version), plus(now64(6, 'UTC'), toIntervalDay(1))), 0)))), 0)) SETTINGS optimize_aggregation_in_order=1) AS events__person ON equals(if(not(empty(events__override.distinct_id)), events__override.person_id, events.person_id), events__person.id) - WHERE and(equals(events.team_id, 99999), notEmpty(events.`$session_id`), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-31 20:00:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), now64(6, 'UTC')), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-13 23:58:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-21 20:00:00.000000', 6, 'UTC')), ifNull(equals(events__person.properties___email, 'bla'), 0)) - GROUP BY events.`$session_id` - HAVING 1))) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestSessionRecordingsListFromFilters.test_top_level_person_property_test_account_filter - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-21 19:55:00.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-31 20:00:00.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-14 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-21 20:00:00.000000', 6, 'UTC')), 0), in(s.session_id, - (SELECT events.`$session_id` AS session_id - FROM events - WHERE and(equals(events.team_id, 99999), notEmpty(events.`$session_id`), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-31 20:00:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), now64(6, 'UTC')), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-13 23:58:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-21 20:00:00.000000', 6, 'UTC')), and(equals(events.event, '$pageview'), 1)) - GROUP BY events.`$session_id` - HAVING hasAll(groupUniqArray(events.event), ['$pageview'])))) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestSessionRecordingsListFromFilters.test_top_level_person_property_test_account_filter.1 - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-21 19:55:00.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-31 20:00:00.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-14 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-21 20:00:00.000000', 6, 'UTC')), 0), in(s.session_id, - (SELECT events.`$session_id` AS session_id - FROM events - LEFT OUTER JOIN - (SELECT argMax(person_distinct_id_overrides.person_id, person_distinct_id_overrides.version) AS person_id, person_distinct_id_overrides.distinct_id AS distinct_id - FROM person_distinct_id_overrides - WHERE equals(person_distinct_id_overrides.team_id, 99999) - GROUP BY person_distinct_id_overrides.distinct_id - HAVING ifNull(equals(argMax(person_distinct_id_overrides.is_deleted, person_distinct_id_overrides.version), 0), 0) SETTINGS optimize_aggregation_in_order=1) AS events__override ON equals(events.distinct_id, events__override.distinct_id) - LEFT JOIN - (SELECT person.id AS id, replaceRegexpAll(nullIf(nullIf(JSONExtractRaw(person.properties, 'email'), ''), 'null'), '^"|"$', '') AS properties___email - FROM person - WHERE and(equals(person.team_id, 99999), ifNull(in(tuple(person.id, person.version), - (SELECT person.id AS id, max(person.version) AS version - FROM person - WHERE equals(person.team_id, 99999) - GROUP BY person.id - HAVING and(ifNull(equals(argMax(person.is_deleted, person.version), 0), 0), ifNull(less(argMax(toTimeZone(person.created_at, 'UTC'), person.version), plus(now64(6, 'UTC'), toIntervalDay(1))), 0)))), 0)) SETTINGS optimize_aggregation_in_order=1) AS events__person ON equals(if(not(empty(events__override.distinct_id)), events__override.person_id, events.person_id), events__person.id) - WHERE and(equals(events.team_id, 99999), notEmpty(events.`$session_id`), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-31 20:00:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), now64(6, 'UTC')), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-13 23:58:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-21 20:00:00.000000', 6, 'UTC')), ifNull(equals(events__person.properties___email, 'bla'), 0)) - GROUP BY events.`$session_id` - HAVING 1))) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestSessionRecordingsListFromFilters.test_top_level_person_property_test_account_filter_materialized - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-21 19:55:00.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-31 20:00:00.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-14 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-21 20:00:00.000000', 6, 'UTC')), 0), in(s.session_id, - (SELECT events.`$session_id` AS session_id - FROM events - WHERE and(equals(events.team_id, 99999), notEmpty(events.`$session_id`), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-31 20:00:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), now64(6, 'UTC')), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-13 23:58:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-21 20:00:00.000000', 6, 'UTC')), and(equals(events.event, '$pageview'), 1)) - GROUP BY events.`$session_id` - HAVING hasAll(groupUniqArray(events.event), ['$pageview'])))) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- -# name: TestSessionRecordingsListFromFilters.test_top_level_person_property_test_account_filter_materialized.1 - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count) AS click_count, - sum(s.keypress_count) AS keypress_count, - sum(s.mouse_activity_count) AS mouse_activity_count, - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count, - ifNull(greaterOrEquals(max(toTimeZone(s._timestamp, 'UTC')), toDateTime64('2021-01-21 19:55:00.000000', 6, 'UTC')), 0) AS ongoing, - round(multiply(divide(plus(plus(plus(divide(sum(s.active_milliseconds), 1000), sum(s.click_count)), sum(s.keypress_count)), sum(s.console_error_count)), plus(plus(plus(plus(sum(s.mouse_activity_count), dateDiff('SECOND', start_time, end_time)), sum(s.console_error_count)), sum(s.console_log_count)), sum(s.console_warn_count))), 100), 2) AS activity_score - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 99999), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-31 20:00:00.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-14 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-21 20:00:00.000000', 6, 'UTC')), 0), in(s.session_id, - (SELECT events.`$session_id` AS session_id - FROM events - LEFT OUTER JOIN - (SELECT argMax(person_distinct_id_overrides.person_id, person_distinct_id_overrides.version) AS person_id, person_distinct_id_overrides.distinct_id AS distinct_id - FROM person_distinct_id_overrides - WHERE equals(person_distinct_id_overrides.team_id, 99999) - GROUP BY person_distinct_id_overrides.distinct_id - HAVING ifNull(equals(argMax(person_distinct_id_overrides.is_deleted, person_distinct_id_overrides.version), 0), 0) SETTINGS optimize_aggregation_in_order=1) AS events__override ON equals(events.distinct_id, events__override.distinct_id) - LEFT JOIN - (SELECT person.id AS id, nullIf(nullIf(person.pmat_email, ''), 'null') AS properties___email - FROM person - WHERE and(equals(person.team_id, 99999), ifNull(in(tuple(person.id, person.version), - (SELECT person.id AS id, max(person.version) AS version - FROM person - WHERE equals(person.team_id, 99999) - GROUP BY person.id - HAVING and(ifNull(equals(argMax(person.is_deleted, person.version), 0), 0), ifNull(less(argMax(toTimeZone(person.created_at, 'UTC'), person.version), plus(now64(6, 'UTC'), toIntervalDay(1))), 0)))), 0)) SETTINGS optimize_aggregation_in_order=1) AS events__person ON equals(if(not(empty(events__override.distinct_id)), events__override.person_id, events.person_id), events__person.id) - WHERE and(equals(events.team_id, 99999), notEmpty(events.`$session_id`), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2020-12-31 20:00:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), now64(6, 'UTC')), greaterOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-13 23:58:00.000000', 6, 'UTC')), lessOrEquals(toTimeZone(events.timestamp, 'UTC'), toDateTime64('2021-01-21 20:00:00.000000', 6, 'UTC')), ifNull(equals(events__person.properties___email, 'bla'), 0)) - GROUP BY events.`$session_id` - HAVING 1))) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0, - allow_experimental_analyzer=0 - ''' -# --- diff --git a/posthog/session_recordings/queries/test/test_session_recording_list_from_filters.py b/posthog/session_recordings/queries/test/test_session_recording_list_from_filters.py deleted file mode 100644 index 41f1226fff27b..0000000000000 --- a/posthog/session_recordings/queries/test/test_session_recording_list_from_filters.py +++ /dev/null @@ -1,4207 +0,0 @@ -from datetime import datetime, UTC -from typing import Literal -from unittest.mock import ANY -from uuid import uuid4 - -from dateutil.relativedelta import relativedelta -from django.utils.timezone import now -from freezegun import freeze_time -from parameterized import parameterized - -from posthog.clickhouse.client import sync_execute -from posthog.clickhouse.log_entries import TRUNCATE_LOG_ENTRIES_TABLE_SQL -from posthog.constants import AvailableFeature -from posthog.models import Cohort, GroupTypeMapping, Person -from posthog.models.action import Action -from posthog.models.filters.session_recordings_filter import SessionRecordingsFilter -from posthog.models.group.util import create_group -from posthog.models.team import Team -from posthog.session_recordings.queries.session_recording_list_from_filters import ( - SessionRecordingListFromFilters, - SessionRecordingQueryResult, -) -from posthog.session_recordings.queries.session_replay_events import ttl_days -from posthog.session_recordings.queries.test.session_replay_sql import ( - produce_replay_summary, -) -from posthog.session_recordings.sql.session_replay_event_sql import ( - TRUNCATE_SESSION_REPLAY_EVENTS_TABLE_SQL, -) -from posthog.test.base import ( - APIBaseTest, - ClickhouseTestMixin, - _create_event, - also_test_with_materialized_columns, - flush_persons_and_events, - snapshot_clickhouse_queries, -) - - -@freeze_time("2021-01-01T13:46:23") -class TestSessionRecordingFilterDateRange(APIBaseTest): - def test_with_relative_dates(self) -> None: - the_filter = SessionRecordingsFilter(team=self.team, data={"date_from": "-3d", "date_to": "-24h"}) - - assert the_filter.date_from == datetime(2020, 12, 29, 0, 0, 0, 0, UTC) - assert the_filter.date_to == datetime(year=2020, month=12, day=31, hour=13, minute=0, second=0, tzinfo=UTC) - - def test_with_string_dates(self) -> None: - the_filter = SessionRecordingsFilter(team=self.team, data={"date_from": "2020-12-29", "date_to": "2021-01-01"}) - - assert the_filter.date_from == datetime(2020, 12, 29, 0, 0, 0, 0, UTC) - assert the_filter.date_to == datetime( - year=2021, month=1, day=1, hour=23, minute=59, second=59, microsecond=999999, tzinfo=UTC - ) - - def test_with_string_date_times(self) -> None: - the_filter = SessionRecordingsFilter( - team=self.team, data={"date_from": "2020-12-29T12:23:45Z", "date_to": "2021-01-01T13:34:42Z"} - ) - - assert the_filter.date_from == datetime(2020, 12, 29, 12, 23, 45, tzinfo=UTC) - assert the_filter.date_to == datetime(year=2021, month=1, day=1, hour=13, minute=34, second=42, tzinfo=UTC) - - def test_with_no_date_from(self) -> None: - the_filter = SessionRecordingsFilter( - team=self.team, data={"date_from": None, "date_to": "2021-01-01T13:34:42Z"} - ) - - # defaults to start of 7 days ago - assert the_filter.date_from == datetime(2020, 12, 25, 0, 0, 0, 0, UTC) - assert the_filter.date_to == datetime(year=2021, month=1, day=1, hour=13, minute=34, second=42, tzinfo=UTC) - - def test_with_no_date_to(self) -> None: - the_filter = SessionRecordingsFilter( - team=self.team, data={"date_from": "2021-01-01T11:34:42Z", "date_to": None} - ) - - assert the_filter.date_from == datetime(2021, 1, 1, 11, 34, 42, tzinfo=UTC) - # defaults to now - assert the_filter.date_to == datetime(year=2021, month=1, day=1, hour=13, minute=46, second=23, tzinfo=UTC) - - -@freeze_time("2021-01-01T13:46:23") -class TestSessionRecordingsListFromFilters(ClickhouseTestMixin, APIBaseTest): - def setUp(self): - super().setUp() - sync_execute(TRUNCATE_SESSION_REPLAY_EVENTS_TABLE_SQL()) - sync_execute(TRUNCATE_LOG_ENTRIES_TABLE_SQL) - - def create_action(self, name, team_id=None, properties=None): - if team_id is None: - team_id = self.team.pk - if properties is None: - properties = [] - action = Action.objects.create( - team_id=team_id, - name=name, - steps_json=[ - { - "event": name, - "properties": properties, - } - ], - ) - return action - - def create_event( - self, - distinct_id, - timestamp, - team=None, - event_name="$pageview", - properties=None, - ): - if team is None: - team = self.team - if properties is None: - properties = {"$os": "Windows 95", "$current_url": "aloha.com/2"} - return _create_event( - team=team, - event=event_name, - timestamp=timestamp, - distinct_id=distinct_id, - properties=properties, - ) - - def _filter_recordings_by(self, recordings_filter: dict | None = None) -> SessionRecordingQueryResult: - the_filter = SessionRecordingsFilter(team=self.team, data=recordings_filter or {}) - session_recording_list_instance = SessionRecordingListFromFilters( - filter=the_filter, team=self.team, hogql_query_modifiers=None - ) - return session_recording_list_instance.run() - - @property - def an_hour_ago(self): - return (now() - relativedelta(hours=1)).replace(microsecond=0, second=0) - - @snapshot_clickhouse_queries - def test_basic_query(self): - user = "test_basic_query-user" - Person.objects.create(team=self.team, distinct_ids=[user], properties={"email": "bla"}) - - session_id_one = f"test_basic_query-{str(uuid4())}" - session_id_two = f"test_basic_query-{str(uuid4())}" - - produce_replay_summary( - session_id=session_id_one, - team_id=self.team.pk, - # can CH handle a timestamp with no T - first_timestamp=self.an_hour_ago.isoformat().replace("T", " "), - last_timestamp=(self.an_hour_ago + relativedelta(seconds=20)).isoformat().replace("T", " "), - distinct_id=user, - first_url="https://example.io/home", - click_count=2, - keypress_count=2, - mouse_activity_count=2, - active_milliseconds=50 * 1000 * 0.5, # 50% of the total expected duration - ) - - produce_replay_summary( - session_id=session_id_one, - team_id=self.team.pk, - # can CH handle a timestamp with no T - first_timestamp=(self.an_hour_ago + relativedelta(seconds=10)), - last_timestamp=(self.an_hour_ago + relativedelta(seconds=50)), - distinct_id=user, - first_url="https://a-different-url.com", - click_count=2, - keypress_count=2, - mouse_activity_count=2, - active_milliseconds=0, # 30% of the total expected duration - ) - - produce_replay_summary( - session_id=session_id_two, - team_id=self.team.pk, - # can CH handle a timestamp with no T - first_timestamp=(self.an_hour_ago + relativedelta(seconds=20)), - last_timestamp=(self.an_hour_ago + relativedelta(seconds=2000)), - distinct_id=user, - first_url="https://another-url.com", - click_count=2, - keypress_count=2, - mouse_activity_count=2, - active_milliseconds=1980 * 1000 * 0.4, # 40% of the total expected duration - ) - - session_recordings, more_recordings_available, _ = self._filter_recordings_by({"no_filter": None}) - - assert session_recordings == [ - { - "session_id": session_id_two, - "activity_score": 40.16, - "team_id": self.team.pk, - "distinct_id": user, - "click_count": 2, - "keypress_count": 2, - "mouse_activity_count": 2, - "duration": 1980, - "active_seconds": 792.0, - "inactive_seconds": 1188.0, - "start_time": self.an_hour_ago + relativedelta(seconds=20), - "end_time": self.an_hour_ago + relativedelta(seconds=2000), - "first_url": "https://another-url.com", - "console_log_count": 0, - "console_warn_count": 0, - "console_error_count": 0, - "ongoing": 1, - }, - { - "session_id": session_id_one, - "activity_score": 61.11, - "team_id": self.team.pk, - "distinct_id": user, - "click_count": 4, - "keypress_count": 4, - "mouse_activity_count": 4, - "duration": 50, - "active_seconds": 25.0, - "inactive_seconds": 25.0, - "start_time": self.an_hour_ago, - "end_time": self.an_hour_ago + relativedelta(seconds=50), - "first_url": "https://example.io/home", - "console_log_count": 0, - "console_warn_count": 0, - "console_error_count": 0, - "ongoing": 1, - }, - ] - - assert more_recordings_available is False - - @snapshot_clickhouse_queries - def test_basic_query_active_sessions( - self, - ): - user = "test_basic_query-user" - Person.objects.create(team=self.team, distinct_ids=[user], properties={"email": "bla"}) - - session_id_total_is_61 = f"test_basic_query_active_sessions-total-{str(uuid4())}" - session_id_active_is_61 = f"test_basic_query_active_sessions-active-{str(uuid4())}" - session_id_inactive_is_61 = f"test_basic_query_active_sessions-inactive-{str(uuid4())}" - - produce_replay_summary( - session_id=session_id_total_is_61, - team_id=self.team.pk, - # can CH handle a timestamp with no T - first_timestamp=self.an_hour_ago.isoformat().replace("T", " "), - last_timestamp=(self.an_hour_ago + relativedelta(seconds=61)).isoformat().replace("T", " "), - distinct_id=user, - first_url="https://example.io/home", - click_count=2, - keypress_count=2, - mouse_activity_count=2, - active_milliseconds=59000, - ) - - produce_replay_summary( - session_id=session_id_active_is_61, - team_id=self.team.pk, - # can CH handle a timestamp with no T - first_timestamp=self.an_hour_ago, - last_timestamp=(self.an_hour_ago + relativedelta(seconds=59)), - distinct_id=user, - first_url="https://a-different-url.com", - click_count=2, - keypress_count=2, - mouse_activity_count=2, - active_milliseconds=61000, - ) - - produce_replay_summary( - session_id=session_id_inactive_is_61, - team_id=self.team.pk, - # can CH handle a timestamp with no T - first_timestamp=self.an_hour_ago, - last_timestamp=(self.an_hour_ago + relativedelta(seconds=61)), - distinct_id=user, - first_url="https://a-different-url.com", - click_count=0, - keypress_count=0, - mouse_activity_count=0, - active_milliseconds=0, - ) - - (session_recordings, _, _) = self._filter_recordings_by( - { - "having_predicates": '[{"type":"recording","key":"duration","value":60,"operator":"gt"}]', - } - ) - - assert sorted( - [(s["session_id"], s["duration"], s["active_seconds"]) for s in session_recordings], - key=lambda x: x[0], - ) == [ - (session_id_inactive_is_61, 61, 0.0), - (session_id_total_is_61, 61, 59.0), - ] - - (session_recordings, _, _) = self._filter_recordings_by( - { - "having_predicates": '[{"type":"recording","key":"active_seconds","value":"60","operator":"gt"}]', - } - ) - - assert [(s["session_id"], s["duration"], s["active_seconds"]) for s in session_recordings] == [ - (session_id_active_is_61, 59, 61.0) - ] - - (session_recordings, _, _) = self._filter_recordings_by( - { - "having_predicates": '[{"type":"recording","key":"inactive_seconds","value":"60","operator":"gt"}]', - } - ) - - assert [(s["session_id"], s["duration"], s["inactive_seconds"]) for s in session_recordings] == [ - (session_id_inactive_is_61, 61, 61.0) - ] - - @snapshot_clickhouse_queries - def test_sessions_with_current_data( - self, - ): - user = "test_sessions_with_current_data-user" - Person.objects.create(team=self.team, distinct_ids=[user], properties={"email": "bla"}) - - session_id_inactive = f"test_sessions_with_current_data-inactive-{str(uuid4())}" - session_id_active = f"test_sessions_with_current_data-active-{str(uuid4())}" - - produce_replay_summary( - session_id=session_id_inactive, - team_id=self.team.pk, - first_timestamp=self.an_hour_ago, - last_timestamp=self.an_hour_ago + relativedelta(seconds=60), - distinct_id=user, - first_url="https://example.io/home", - click_count=2, - keypress_count=2, - mouse_activity_count=2, - active_milliseconds=59000, - kafka_timestamp=(datetime.utcnow() - relativedelta(minutes=6)), - ) - - produce_replay_summary( - session_id=session_id_active, - team_id=self.team.pk, - first_timestamp=self.an_hour_ago, - last_timestamp=self.an_hour_ago + relativedelta(seconds=60), - distinct_id=user, - first_url="https://a-different-url.com", - click_count=2, - keypress_count=2, - mouse_activity_count=2, - active_milliseconds=61000, - kafka_timestamp=(datetime.utcnow() - relativedelta(minutes=3)), - ) - - (session_recordings, _, _) = self._filter_recordings_by({}) - assert sorted( - [(s["session_id"], s["ongoing"]) for s in session_recordings], - key=lambda x: x[0], - ) == [ - (session_id_active, 1), - (session_id_inactive, 0), - ] - - @snapshot_clickhouse_queries - def test_basic_query_with_paging(self): - user = "test_basic_query_with_paging-user" - Person.objects.create(team=self.team, distinct_ids=[user], properties={"email": "bla"}) - - session_id_one = f"id_one_test_basic_query_with_paging-{str(uuid4())}" - session_id_two = f"id_two_test_basic_query_with_paging-{str(uuid4())}" - - produce_replay_summary( - session_id=session_id_one, - team_id=self.team.pk, - # can CH handle a timestamp with no T - first_timestamp=self.an_hour_ago.isoformat().replace("T", " "), - last_timestamp=(self.an_hour_ago + relativedelta(seconds=20)).isoformat().replace("T", " "), - distinct_id=user, - first_url="https://example.io/home", - click_count=2, - keypress_count=2, - mouse_activity_count=2, - active_milliseconds=50 * 1000 * 0.5, # 50% of the total expected duration - ) - - produce_replay_summary( - session_id=session_id_one, - team_id=self.team.pk, - # can CH handle a timestamp with no T - first_timestamp=(self.an_hour_ago + relativedelta(seconds=10)), - last_timestamp=(self.an_hour_ago + relativedelta(seconds=50)), - distinct_id=user, - first_url="https://a-different-url.com", - click_count=2, - keypress_count=2, - mouse_activity_count=2, - active_milliseconds=0, # 30% of the total expected duration - ) - - produce_replay_summary( - session_id=session_id_two, - team_id=self.team.pk, - # can CH handle a timestamp with no T - first_timestamp=(self.an_hour_ago + relativedelta(seconds=20)), - last_timestamp=(self.an_hour_ago + relativedelta(seconds=2000)), - distinct_id=user, - first_url="https://another-url.com", - click_count=2, - keypress_count=2, - mouse_activity_count=2, - active_milliseconds=1980 * 1000 * 0.4, # 40% of the total expected duration - ) - - (session_recordings, more_recordings_available, _) = self._filter_recordings_by( - {"no_filter": None, "limit": 1, "offset": 0} - ) - - assert session_recordings == [ - { - "activity_score": 40.16, - "session_id": session_id_two, - "team_id": self.team.pk, - "distinct_id": user, - "click_count": 2, - "keypress_count": 2, - "mouse_activity_count": 2, - "duration": 1980, - "active_seconds": 792.0, - "inactive_seconds": 1188.0, - "start_time": self.an_hour_ago + relativedelta(seconds=20), - "end_time": self.an_hour_ago + relativedelta(seconds=2000), - "first_url": "https://another-url.com", - "console_log_count": 0, - "console_warn_count": 0, - "console_error_count": 0, - "ongoing": 1, - } - ] - - assert more_recordings_available is True - - (session_recordings, more_recordings_available, _) = self._filter_recordings_by( - {"no_filter": None, "limit": 1, "offset": 1} - ) - - assert session_recordings == [ - { - "session_id": session_id_one, - "activity_score": 61.11, - "team_id": self.team.pk, - "distinct_id": user, - "click_count": 4, - "keypress_count": 4, - "mouse_activity_count": 4, - "duration": 50, - "active_seconds": 25.0, - "inactive_seconds": 25.0, - "start_time": self.an_hour_ago, - "end_time": self.an_hour_ago + relativedelta(seconds=50), - "first_url": "https://example.io/home", - "console_log_count": 0, - "console_warn_count": 0, - "console_error_count": 0, - "ongoing": 1, - }, - ] - - assert more_recordings_available is False - - (session_recordings, more_recordings_available, _) = self._filter_recordings_by( - {"no_filter": None, "limit": 1, "offset": 2} - ) - - assert session_recordings == [] - - assert more_recordings_available is False - - @snapshot_clickhouse_queries - def test_basic_query_with_ordering(self): - user = "test_basic_query_with_ordering-user" - Person.objects.create(team=self.team, distinct_ids=[user], properties={"email": "bla"}) - - session_id_one = f"test_basic_query_with_ordering-session-1-{str(uuid4())}" - session_id_two = f"test_basic_query_with_ordering-session-2-{str(uuid4())}" - - session_one_start = self.an_hour_ago + relativedelta(seconds=10) - produce_replay_summary( - session_id=session_id_one, - team_id=self.team.pk, - # can CH handle a timestamp with no T - first_timestamp=session_one_start, - last_timestamp=(self.an_hour_ago + relativedelta(seconds=50)), - distinct_id=user, - console_error_count=1000, - active_milliseconds=1, # most errors, but the least activity - ) - - produce_replay_summary( - session_id=session_id_one, - team_id=self.team.pk, - # can CH handle a timestamp with no T - first_timestamp=session_one_start, - last_timestamp=(self.an_hour_ago + relativedelta(seconds=50)), - distinct_id=user, - console_error_count=12, - active_milliseconds=1, # most errors, but the least activity - ) - - session_two_start = self.an_hour_ago - produce_replay_summary( - session_id=session_id_two, - team_id=self.team.pk, - # starts before session one - first_timestamp=session_two_start, - last_timestamp=(self.an_hour_ago + relativedelta(seconds=50)), - distinct_id=user, - console_error_count=430, - active_milliseconds=1000, # most activity, but the least errors - ) - - (session_recordings) = self._filter_recordings_by( - {"no_filter": None, "limit": 3, "offset": 0, "order": "active_seconds"} - ) - - ordered_by_activity = [(r["session_id"], r["active_seconds"]) for r in session_recordings.results] - assert ordered_by_activity == [(session_id_two, 1.0), (session_id_one, 0.002)] - - (session_recordings) = self._filter_recordings_by( - {"no_filter": None, "limit": 3, "offset": 0, "order": "console_error_count"} - ) - - ordered_by_errors = [(r["session_id"], r["console_error_count"]) for r in session_recordings.results] - assert ordered_by_errors == [(session_id_one, 1012), (session_id_two, 430)] - - (session_recordings) = self._filter_recordings_by( - {"no_filter": None, "limit": 3, "offset": 0, "order": "start_time"} - ) - - ordered_by_default = [(r["session_id"], r["start_time"]) for r in session_recordings.results] - assert ordered_by_default == [(session_id_one, session_one_start), (session_id_two, session_two_start)] - - def test_first_url_selection(self): - user = "test_first_url_selection-user" - Person.objects.create(team=self.team, distinct_ids=[user], properties={"email": "bla"}) - - session_id_one = f"first-url-on-first-event-{str(uuid4())}" - session_id_two = f"first-url-not-on-first-event-{str(uuid4())}" - session_id_three = f"no-url-from-many-{str(uuid4())}" - session_id_four = f"events-inserted-out-of-order-{str(uuid4())}" - - # session one has the first url on the first event - produce_replay_summary( - session_id=session_id_one, - team_id=self.team.pk, - # can CH handle a timestamp with no T - first_timestamp=self.an_hour_ago, - last_timestamp=self.an_hour_ago + relativedelta(seconds=20), - first_url="https://on-first-event.com", - ) - - produce_replay_summary( - session_id=session_id_one, - team_id=self.team.pk, - first_timestamp=self.an_hour_ago + relativedelta(seconds=10), - last_timestamp=self.an_hour_ago + relativedelta(seconds=20), - first_url="https://on-second-event.com", - ) - - produce_replay_summary( - session_id=session_id_one, - team_id=self.team.pk, - first_timestamp=self.an_hour_ago + relativedelta(seconds=20), - last_timestamp=self.an_hour_ago + relativedelta(seconds=40), - first_url="https://on-third-event.com", - ) - - # session two has no URL on the first event - produce_replay_summary( - session_id=session_id_two, - team_id=self.team.pk, - first_timestamp=(self.an_hour_ago + relativedelta(seconds=10)), - last_timestamp=(self.an_hour_ago + relativedelta(seconds=50)), - first_url=None, - ) - - produce_replay_summary( - session_id=session_id_two, - team_id=self.team.pk, - first_timestamp=(self.an_hour_ago + relativedelta(seconds=20)), - last_timestamp=(self.an_hour_ago + relativedelta(seconds=30)), - first_url="https://first-is-on-second-event.com", - ) - - produce_replay_summary( - session_id=session_id_two, - team_id=self.team.pk, - first_timestamp=(self.an_hour_ago + relativedelta(seconds=25)), - last_timestamp=(self.an_hour_ago + relativedelta(seconds=30)), - first_url="https://another-on-the-session.com", - ) - - # session three has no URLs - produce_replay_summary( - session_id=session_id_three, - team_id=self.team.pk, - first_timestamp=self.an_hour_ago, - last_timestamp=self.an_hour_ago + relativedelta(seconds=50), - distinct_id=user, - first_url=None, - ) - - produce_replay_summary( - session_id=session_id_three, - team_id=self.team.pk, - first_timestamp=(self.an_hour_ago + relativedelta(seconds=10)), - last_timestamp=self.an_hour_ago + relativedelta(seconds=50), - distinct_id=user, - first_url=None, - ) - - produce_replay_summary( - session_id=session_id_three, - team_id=self.team.pk, - first_timestamp=(self.an_hour_ago + relativedelta(seconds=20)), - last_timestamp=self.an_hour_ago + relativedelta(seconds=60), - distinct_id=user, - first_url=None, - ) - - # session four events are received out of order - produce_replay_summary( - session_id=session_id_four, - team_id=self.team.pk, - first_timestamp=self.an_hour_ago + relativedelta(seconds=20), - last_timestamp=self.an_hour_ago + relativedelta(seconds=25), - first_url="https://on-first-received-event.com", - ) - produce_replay_summary( - session_id=session_id_four, - team_id=self.team.pk, - first_timestamp=self.an_hour_ago + relativedelta(seconds=10), - last_timestamp=self.an_hour_ago + relativedelta(seconds=25), - first_url="https://on-second-received-event-but-actually-first.com", - ) - - session_recordings, more_recordings_available, _ = self._filter_recordings_by({"no_filter": None}) - - assert sorted( - [{"session_id": r["session_id"], "first_url": r["first_url"]} for r in session_recordings], - key=lambda x: x["session_id"], - ) == sorted( - [ - { - "session_id": session_id_one, - "first_url": "https://on-first-event.com", - }, - { - "session_id": session_id_two, - "first_url": "https://first-is-on-second-event.com", - }, - { - "session_id": session_id_three, - "first_url": None, - }, - { - "session_id": session_id_four, - "first_url": "https://on-second-received-event-but-actually-first.com", - }, - ], - # mypy unhappy about this lambda when first_url can be None 🤷️ - key=lambda x: x["session_id"], # type: ignore - ) - - def test_recordings_dont_leak_data_between_teams(self): - another_team = Team.objects.create(organization=self.organization) - user = "test_recordings_dont_leak_data_between_teams-user" - Person.objects.create(team=self.team, distinct_ids=[user], properties={"email": "bla"}) - Person.objects.create(team=another_team, distinct_ids=[user], properties={"email": "bla"}) - - session_id_one = f"test_recordings_dont_leak_data_between_teams-1-{str(uuid4())}" - session_id_two = f"test_recordings_dont_leak_data_between_teams-2-{str(uuid4())}" - - produce_replay_summary( - session_id=session_id_one, - team_id=another_team.pk, - distinct_id=user, - first_timestamp=self.an_hour_ago, - last_timestamp=self.an_hour_ago + relativedelta(seconds=20), - click_count=2, - keypress_count=2, - mouse_activity_count=2, - active_milliseconds=20 * 1000 * 0.5, # 50% of the total expected duration - ) - - produce_replay_summary( - session_id=session_id_two, - team_id=self.team.pk, - distinct_id=user, - first_timestamp=self.an_hour_ago, - last_timestamp=self.an_hour_ago + relativedelta(seconds=20), - click_count=2, - keypress_count=2, - mouse_activity_count=2, - active_milliseconds=20 * 1000 * 0.5, # 50% of the total expected duration - ) - - (session_recordings, _, _) = self._filter_recordings_by({"no_filter": None}) - - assert [{"session": r["session_id"], "user": r["distinct_id"]} for r in session_recordings] == [ - {"session": session_id_two, "user": user} - ] - - @snapshot_clickhouse_queries - def test_event_filter(self): - user = "test_event_filter-user" - Person.objects.create(team=self.team, distinct_ids=[user], properties={"email": "bla"}) - session_id_one = f"test_event_filter-{str(uuid4())}" - produce_replay_summary( - distinct_id=user, - session_id=session_id_one, - first_timestamp=self.an_hour_ago, - team_id=self.team.id, - ) - self.create_event( - user, - self.an_hour_ago, - properties={"$session_id": session_id_one, "$window_id": str(uuid4())}, - ) - produce_replay_summary( - distinct_id=user, - session_id=session_id_one, - first_timestamp=self.an_hour_ago, - team_id=self.team.id, - ) - - (session_recordings, _, _) = self._filter_recordings_by( - { - "events": [ - { - "id": "$pageview", - "type": "events", - "order": 0, - "name": "$pageview", - } - ] - } - ) - - assert len(session_recordings) == 1 - assert session_recordings[0]["session_id"] == session_id_one - - (session_recordings, _, _) = self._filter_recordings_by( - { - "events": [ - { - "id": "$autocapture", - "type": "events", - "order": 0, - "name": "$autocapture", - } - ] - } - ) - assert session_recordings == [] - - @snapshot_clickhouse_queries - def test_event_filter_has_ttl_applied_too(self): - user = "test_event_filter_has_ttl_applied_too-user" - Person.objects.create(team=self.team, distinct_ids=[user], properties={"email": "bla"}) - session_id_one = f"test_event_filter_has_ttl_applied_too-{str(uuid4())}" - - # this is artificially incorrect data, the session events are within TTL - produce_replay_summary( - distinct_id=user, - session_id=session_id_one, - first_timestamp=self.an_hour_ago, - team_id=self.team.id, - ) - # but the page view event is outside TTL - self.create_event( - user, - self.an_hour_ago - relativedelta(days=SessionRecordingListFromFilters.SESSION_RECORDINGS_DEFAULT_LIMIT + 1), - properties={"$session_id": session_id_one, "$window_id": str(uuid4())}, - ) - - (session_recordings, _, _) = self._filter_recordings_by( - { - "events": [ - { - "id": "$pageview", - "type": "events", - "order": 0, - "name": "$pageview", - } - ] - } - ) - assert len(session_recordings) == 0 - - (session_recordings, _, _) = self._filter_recordings_by({}) - # without an event filter the recording is present, showing that the TTL was applied to the events table too - # we want this to limit the amount of event data we query - assert len(session_recordings) == 1 - assert session_recordings[0]["session_id"] == session_id_one - - @snapshot_clickhouse_queries - def test_ttl_days(self): - assert ttl_days(self.team) == 21 - - with self.is_cloud(True): - # Far enough in the future from `days_since_blob_ingestion` but not paid - with freeze_time("2023-09-01T12:00:01Z"): - assert ttl_days(self.team) == 30 - - self.team.organization.available_product_features = [ - {"key": AvailableFeature.RECORDINGS_PLAYLISTS, "name": AvailableFeature.RECORDINGS_PLAYLISTS} - ] - - # Far enough in the future from `days_since_blob_ingestion` but paid - with freeze_time("2023-12-01T12:00:01Z"): - assert ttl_days(self.team) == 90 - - # Not far enough in the future from `days_since_blob_ingestion` - with freeze_time("2023-09-05T12:00:01Z"): - assert ttl_days(self.team) == 35 - - @snapshot_clickhouse_queries - def test_listing_ignores_future_replays(self): - with freeze_time("2023-08-29T12:00:01Z"): - produce_replay_summary(team_id=self.team.id, session_id="29th Aug") - - with freeze_time("2023-09-01T12:00:01Z"): - produce_replay_summary(team_id=self.team.id, session_id="1st-sep") - - with freeze_time("2023-09-02T12:00:01Z"): - produce_replay_summary(team_id=self.team.id, session_id="2nd-sep") - - with freeze_time("2023-09-03T12:00:01Z"): - produce_replay_summary(team_id=self.team.id, session_id="3rd-sep") - - with freeze_time("2023-08-30T12:00:01Z"): - recordings = self._filter_recordings_by() - - # recordings in the future don't show - assert [s["session_id"] for s in recordings.results] == ["29th Aug"] - - @snapshot_clickhouse_queries - def test_filter_on_session_ids(self): - user = "test_session_ids-user" - Person.objects.create(team=self.team, distinct_ids=[user], properties={"email": "bla"}) - - first_session_id = str(uuid4()) - second_session_id = str(uuid4()) - third_session_id = str(uuid4()) - - produce_replay_summary( - session_id=first_session_id, - team_id=self.team.pk, - first_timestamp=self.an_hour_ago, - last_timestamp=(self.an_hour_ago + relativedelta(minutes=5)), - distinct_id=user, - first_url="https://example.io/home", - click_count=2, - keypress_count=2, - mouse_activity_count=2, - active_milliseconds=59000, - ) - - produce_replay_summary( - session_id=second_session_id, - team_id=self.team.pk, - first_timestamp=self.an_hour_ago, - last_timestamp=(self.an_hour_ago + relativedelta(minutes=1)), - distinct_id=user, - first_url="https://example.io/home", - click_count=2, - keypress_count=2, - mouse_activity_count=2, - active_milliseconds=61000, - ) - - produce_replay_summary( - session_id=third_session_id, - team_id=self.team.pk, - first_timestamp=self.an_hour_ago, - last_timestamp=(self.an_hour_ago + relativedelta(minutes=10)), - distinct_id=user, - first_url="https://example.io/home", - click_count=0, - keypress_count=0, - mouse_activity_count=0, - active_milliseconds=0, - ) - - (session_recordings, _, _) = self._filter_recordings_by( - { - "session_ids": [first_session_id], - } - ) - - assert len(session_recordings) == 1 - assert session_recordings[0]["session_id"] == first_session_id - - (session_recordings, _, _) = self._filter_recordings_by( - { - "session_ids": [first_session_id, second_session_id], - } - ) - - assert sorted([s["session_id"] for s in session_recordings]) == sorted( - [ - first_session_id, - second_session_id, - ] - ) - - @snapshot_clickhouse_queries - def test_event_filter_with_active_sessions( - self, - ): - user = "test_basic_query-user" - Person.objects.create(team=self.team, distinct_ids=[user], properties={"email": "bla"}) - - session_id_total_is_61 = f"test_basic_query_active_sessions-total-{str(uuid4())}" - session_id_active_is_61 = f"test_basic_query_active_sessions-active-{str(uuid4())}" - - self.create_event( - user, - self.an_hour_ago, - properties={ - "$session_id": session_id_total_is_61, - "$window_id": str(uuid4()), - }, - ) - produce_replay_summary( - session_id=session_id_total_is_61, - team_id=self.team.pk, - # can CH handle a timestamp with no T - first_timestamp=self.an_hour_ago.isoformat().replace("T", " "), - last_timestamp=(self.an_hour_ago + relativedelta(seconds=61)).isoformat().replace("T", " "), - distinct_id=user, - first_url="https://example.io/home", - click_count=2, - keypress_count=2, - mouse_activity_count=2, - active_milliseconds=59000, - ) - - self.create_event( - user, - self.an_hour_ago, - properties={ - "$session_id": session_id_active_is_61, - "$window_id": str(uuid4()), - }, - ) - produce_replay_summary( - session_id=session_id_active_is_61, - team_id=self.team.pk, - # can CH handle a timestamp with no T - first_timestamp=self.an_hour_ago, - last_timestamp=(self.an_hour_ago + relativedelta(seconds=59)), - distinct_id=user, - first_url="https://a-different-url.com", - click_count=2, - keypress_count=2, - mouse_activity_count=2, - active_milliseconds=61000, - ) - - (session_recordings, _, _) = self._filter_recordings_by( - { - "events": [ - { - "id": "$pageview", - "type": "events", - "order": 0, - "name": "$pageview", - } - ], - "having_predicates": '[{"type":"recording","key":"duration","value":60,"operator":"gt"}]', - } - ) - - assert [(s["session_id"], s["duration"], s["active_seconds"]) for s in session_recordings] == [ - (session_id_total_is_61, 61, 59.0) - ] - - (session_recordings, _, _) = self._filter_recordings_by( - { - "events": [ - { - "id": "$pageview", - "type": "events", - "order": 0, - "name": "$pageview", - } - ], - "having_predicates": '[{"type":"recording","key":"active_seconds","value":60,"operator":"gt"}]', - } - ) - - assert [(s["session_id"], s["duration"], s["active_seconds"]) for s in session_recordings] == [ - (session_id_active_is_61, 59, 61.0) - ] - - @also_test_with_materialized_columns(["$current_url", "$browser"]) - @snapshot_clickhouse_queries - def test_event_filter_with_properties(self): - user = "test_event_filter_with_properties-user" - Person.objects.create(team=self.team, distinct_ids=[user], properties={"email": "bla"}) - session_id_one = f"test_event_filter_with_properties-{str(uuid4())}" - produce_replay_summary( - distinct_id=user, - session_id=session_id_one, - first_timestamp=self.an_hour_ago, - team_id=self.team.id, - ) - self.create_event( - user, - self.an_hour_ago, - properties={ - "$browser": "Chrome", - "$session_id": session_id_one, - "$window_id": str(uuid4()), - }, - ) - self.create_event( - user, - self.an_hour_ago, - event_name="a_different_event", - properties={ - "$browser": "Safari", - "$session_id": session_id_one, - "$window_id": str(uuid4()), - }, - ) - produce_replay_summary( - distinct_id=user, - session_id=session_id_one, - first_timestamp=(self.an_hour_ago + relativedelta(seconds=30)), - team_id=self.team.id, - ) - - (session_recordings, _, _) = self._filter_recordings_by( - { - "events": [ - { - "id": "$pageview", - "type": "events", - "order": 0, - "name": "$pageview", - "properties": [ - { - "key": "$browser", - "value": ["Chrome"], - "operator": "exact", - "type": "event", - } - ], - } - ] - } - ) - assert len(session_recordings) == 1 - assert session_recordings[0]["session_id"] == session_id_one - - (session_recordings, _, _) = self._filter_recordings_by( - { - "events": [ - { - "id": "$pageview", - "type": "events", - "order": 0, - "name": "$pageview", - "properties": [ - { - "key": "$browser", - "value": ["Firefox"], - "operator": "exact", - "type": "event", - } - ], - } - ] - } - ) - assert session_recordings == [] - - (session_recordings, _, _) = self._filter_recordings_by( - { - "events": [ - { - "id": "a_different_event", - "type": "events", - "order": 0, - "name": "a_different_event", - "properties": [ - { - "key": "$browser", - "value": ["Chrome"], - "operator": "exact", - "type": "event", - } - ], - } - ] - } - ) - assert len(session_recordings) == 0 - - (session_recordings, _, _) = self._filter_recordings_by( - { - "events": [ - { - "id": "a_different_event", - "type": "events", - "order": 0, - "name": "a_different_event", - "properties": [ - { - "key": "$browser", - "value": ["Safari"], - "operator": "exact", - "type": "event", - } - ], - } - ] - } - ) - assert len(session_recordings) == 1 - assert session_recordings[0]["session_id"] == session_id_one - - @snapshot_clickhouse_queries - def test_multiple_event_filters(self): - session_id = f"test_multiple_event_filters-{str(uuid4())}" - user = "test_multiple_event_filters-user" - Person.objects.create(team=self.team, distinct_ids=[user], properties={"email": "bla"}) - produce_replay_summary( - distinct_id=user, - session_id=session_id, - first_timestamp=self.an_hour_ago, - team_id=self.team.id, - ) - - self.create_event( - user, - self.an_hour_ago, - properties={"$session_id": session_id, "$window_id": "1", "foo": "bar"}, - ) - self.create_event( - user, - self.an_hour_ago, - properties={"$session_id": session_id, "$window_id": "1", "bar": "foo"}, - ) - self.create_event( - user, - self.an_hour_ago, - properties={"$session_id": session_id, "$window_id": "1", "bar": "foo"}, - event_name="new-event", - ) - produce_replay_summary( - distinct_id=user, - session_id=session_id, - first_timestamp=(self.an_hour_ago + relativedelta(seconds=30)), - team_id=self.team.id, - ) - - (session_recordings, _, _) = self._filter_recordings_by( - { - "events": [ - { - "id": "$pageview", - "type": "events", - "order": 0, - "name": "$pageview", - }, - { - "id": "new-event", - "type": "events", - "order": 0, - "name": "new-event", - }, - ] - } - ) - - assert len(session_recordings) == 1 - assert session_recordings[0]["session_id"] == session_id - - (session_recordings, _, _) = self._filter_recordings_by( - { - "events": [ - { - "id": "$pageview", - "type": "events", - "order": 0, - "name": "$pageview", - }, - { - "id": "new-event2", - "type": "events", - "order": 0, - "name": "new-event2", - }, - ] - } - ) - assert session_recordings == [] - - # it uses hasAny instead of hasAll because of the OR filter - (session_recordings, _, _) = self._filter_recordings_by( - { - "events": [ - { - "id": "$pageview", - "type": "events", - "order": 0, - "name": "$pageview", - }, - { - "id": "new-event2", - "type": "events", - "order": 0, - "name": "new-event2", - }, - ], - "operand": "OR", - } - ) - assert len(session_recordings) == 1 - - # two events with the same name - (session_recordings, _, _) = self._filter_recordings_by( - { - "events": [ - { - "id": "$pageview", - "type": "events", - "name": "$pageview", - "properties": [{"key": "foo", "value": ["bar"], "operator": "exact", "type": "event"}], - }, - { - "id": "$pageview", - "type": "events", - "name": "$pageview", - "properties": [{"key": "bar", "value": ["foo"], "operator": "exact", "type": "event"}], - }, - ], - "operand": "AND", - } - ) - assert len(session_recordings) == 1 - - # two events with different names - (session_recordings, _, _) = self._filter_recordings_by( - { - "events": [ - { - "id": "$pageview", - "type": "events", - "name": "$pageview", - "properties": [{"key": "foo", "value": ["bar"], "operator": "exact", "type": "event"}], - }, - { - "id": "new-event", - "type": "events", - "name": "new-event", - "properties": [{"key": "foo", "value": ["bar"], "operator": "exact", "type": "event"}], - }, - ], - "operand": "AND", - } - ) - assert len(session_recordings) == 0 - - # two events with different names - (session_recordings, _, _) = self._filter_recordings_by( - { - "events": [ - { - "id": "$pageview", - "type": "events", - "name": "$pageview", - "properties": [{"key": "foo", "value": ["bar"], "operator": "exact", "type": "event"}], - }, - { - "id": "new-event", - "type": "events", - "name": "new-event", - "properties": [{"key": "foo", "value": ["bar"], "operator": "exact", "type": "event"}], - }, - ], - "operand": "OR", - } - ) - assert len(session_recordings) == 1 - - @snapshot_clickhouse_queries - @also_test_with_materialized_columns(["$session_id", "$browser"], person_properties=["email"]) - @freeze_time("2023-01-04") - def test_action_filter(self): - user = "test_action_filter-user" - Person.objects.create(team=self.team, distinct_ids=[user], properties={"email": "bla"}) - session_id_one = f"test_action_filter-session-one" - window_id = "test_action_filter-window-id" - action_with_properties = self.create_action( - "custom-event", - properties=[ - {"key": "$browser", "value": "Firefox"}, - {"key": "$session_id", "value": session_id_one}, - {"key": "$window_id", "value": window_id}, - ], - ) - action_without_properties = self.create_action( - name="custom-event", - properties=[ - {"key": "$session_id", "value": session_id_one}, - {"key": "$window_id", "value": window_id}, - ], - ) - - produce_replay_summary( - distinct_id=user, - session_id=session_id_one, - first_timestamp=self.an_hour_ago, - team_id=self.team.id, - ) - self.create_event( - user, - self.an_hour_ago, - event_name="custom-event", - properties={ - "$browser": "Chrome", - "$session_id": session_id_one, - "$window_id": window_id, - }, - ) - produce_replay_summary( - distinct_id=user, - session_id=session_id_one, - first_timestamp=(self.an_hour_ago + relativedelta(seconds=30)), - team_id=self.team.id, - ) - - (session_recordings, _, _) = self._filter_recordings_by( - { - "actions": [ - { - "id": action_with_properties.id, - "type": "actions", - "order": 1, - "name": "custom-event", - } - ] - } - ) - assert session_recordings == [] - - (session_recordings, _, _) = self._filter_recordings_by( - { - "actions": [ - { - "id": action_without_properties.id, - "type": "actions", - "order": 1, - "name": "custom-event", - } - ] - } - ) - - assert len(session_recordings) == 1 - assert session_recordings[0]["session_id"] == session_id_one - - # Adding properties to an action - (session_recordings, _, _) = self._filter_recordings_by( - { - "actions": [ - { - "id": action_without_properties.id, - "type": "actions", - "order": 1, - "name": "custom-event", - "properties": [ - { - "key": "$browser", - "value": ["Firefox"], - "operator": "exact", - "type": "event", - } - ], - } - ] - } - ) - assert session_recordings == [] - - # Adding matching properties to an action - (session_recordings, _, _) = self._filter_recordings_by( - { - "actions": [ - { - "id": action_without_properties.id, - "type": "actions", - "order": 1, - "name": "custom-event", - "properties": [ - { - "key": "$browser", - "value": ["Chrome"], - "operator": "exact", - "type": "event", - } - ], - } - ] - } - ) - - assert len(session_recordings) == 1 - assert session_recordings[0]["session_id"] == session_id_one - - def test_all_sessions_recording_object_keys_with_entity_filter(self): - user = "test_all_sessions_recording_object_keys_with_entity_filter-user" - Person.objects.create(team=self.team, distinct_ids=[user], properties={"email": "bla"}) - session_id = f"test_all_sessions_recording_object_keys_with_entity_filter-{str(uuid4())}" - window_id = str(uuid4()) - - produce_replay_summary( - distinct_id=user, - session_id=session_id, - first_timestamp=self.an_hour_ago, - last_timestamp=(self.an_hour_ago + relativedelta(seconds=60)), - team_id=self.team.id, - first_url="https://recieved-out-of-order.com/second", - ) - self.create_event( - user, - self.an_hour_ago, - properties={"$session_id": session_id, "$window_id": window_id}, - ) - produce_replay_summary( - distinct_id=user, - session_id=session_id, - first_timestamp=self.an_hour_ago, - last_timestamp=(self.an_hour_ago + relativedelta(seconds=30)), - team_id=self.team.id, - first_url="https://recieved-out-of-order.com/first", - ) - - (session_recordings, _, _) = self._filter_recordings_by( - { - "events": [ - { - "id": "$pageview", - "type": "events", - "order": 0, - "name": "$pageview", - } - ] - } - ) - - assert session_recordings == [ - { - "activity_score": 0, - "session_id": session_id, - "distinct_id": user, - "duration": 60, - "start_time": self.an_hour_ago, - "end_time": self.an_hour_ago + relativedelta(seconds=60), - "active_seconds": 0.0, - "click_count": 0, - "first_url": "https://recieved-out-of-order.com/first", - "inactive_seconds": 60.0, - "keypress_count": 0, - "mouse_activity_count": 0, - "team_id": self.team.id, - "console_log_count": 0, - "console_warn_count": 0, - "console_error_count": 0, - "ongoing": 1, - } - ] - - @snapshot_clickhouse_queries - def test_duration_filter(self): - user = "test_duration_filter-user" - Person.objects.create(team=self.team, distinct_ids=[user], properties={"email": "bla"}) - - session_id_one = "session one is 29 seconds long" - produce_replay_summary( - distinct_id=user, - session_id=session_id_one, - first_timestamp=self.an_hour_ago, - last_timestamp=(self.an_hour_ago + relativedelta(seconds=29)), - team_id=self.team.id, - ) - - session_id_two = "session two is 61 seconds long" - produce_replay_summary( - distinct_id=user, - session_id=session_id_two, - first_timestamp=self.an_hour_ago, - last_timestamp=(self.an_hour_ago + relativedelta(seconds=61)), - team_id=self.team.id, - ) - - (session_recordings, _, _) = self._filter_recordings_by( - {"having_predicates": '[{"type":"recording","key":"duration","value":60,"operator":"gt"}]'} - ) - assert [r["session_id"] for r in session_recordings] == [session_id_two] - - (session_recordings, _, _) = self._filter_recordings_by( - {"having_predicates": '[{"type":"recording","key":"duration","value":60,"operator":"lt"}]'} - ) - assert [r["session_id"] for r in session_recordings] == [session_id_one] - - @snapshot_clickhouse_queries - def test_operand_or_person_filters(self): - user = "test_operand_or_filter-user" - Person.objects.create(team=self.team, distinct_ids=[user], properties={"email": "test@posthog.com"}) - - second_user = "test_operand_or_filter-second_user" - Person.objects.create(team=self.team, distinct_ids=[second_user], properties={"email": "david@posthog.com"}) - - session_id_one = "session_id_one" - produce_replay_summary( - distinct_id=user, - session_id=session_id_one, - first_timestamp=self.an_hour_ago, - last_timestamp=(self.an_hour_ago + relativedelta(seconds=30)), - team_id=self.team.id, - ) - - session_id_two = "session_id_two" - produce_replay_summary( - distinct_id=second_user, - session_id=session_id_two, - first_timestamp=self.an_hour_ago, - last_timestamp=(self.an_hour_ago + relativedelta(seconds=30)), - team_id=self.team.id, - ) - - (session_recordings, _, _) = self._filter_recordings_by( - { - "properties": [ - { - "key": "email", - "value": ["test@posthog.com"], - "operator": "exact", - "type": "person", - }, - { - "key": "email", - "value": ["david@posthog.com"], - "operator": "exact", - "type": "person", - }, - ], - "operand": "AND", - } - ) - assert len(session_recordings) == 0 - - (session_recordings, _, _) = self._filter_recordings_by( - { - "properties": [ - { - "key": "email", - "value": ["test@posthog.com"], - "operator": "exact", - "type": "person", - }, - { - "key": "email", - "value": ["david@posthog.com"], - "operator": "exact", - "type": "person", - }, - ], - "operand": "OR", - } - ) - assert len(session_recordings) == 2 - assert sorted([r["session_id"] for r in session_recordings]) == sorted([session_id_one, session_id_two]) - - @snapshot_clickhouse_queries - def test_operand_or_event_filters(self): - user = "test_operand_or_filter-user" - Person.objects.create(team=self.team, distinct_ids=[user], properties={"email": "test@posthog.com"}) - - second_user = "test_operand_or_filter-second_user" - Person.objects.create(team=self.team, distinct_ids=[second_user], properties={"email": "david@posthog.com"}) - - session_id_one = "session_id_one" - produce_replay_summary( - distinct_id=user, - session_id=session_id_one, - first_timestamp=self.an_hour_ago, - last_timestamp=(self.an_hour_ago + relativedelta(seconds=30)), - team_id=self.team.id, - ) - self.create_event( - user, - self.an_hour_ago + relativedelta(seconds=10), - properties={"$session_id": session_id_one}, - ) - - session_id_two = "session_id_two" - produce_replay_summary( - distinct_id=second_user, - session_id=session_id_two, - first_timestamp=self.an_hour_ago, - last_timestamp=(self.an_hour_ago + relativedelta(seconds=30)), - team_id=self.team.id, - ) - self.create_event( - user, - self.an_hour_ago + relativedelta(seconds=10), - event_name="custom_event", - properties={"$session_id": session_id_two}, - ) - - session_id_three = "session_id_three" - produce_replay_summary( - distinct_id=second_user, - session_id=session_id_three, - first_timestamp=self.an_hour_ago, - last_timestamp=(self.an_hour_ago + relativedelta(seconds=30)), - team_id=self.team.id, - ) - - (session_recordings, _, _) = self._filter_recordings_by( - { - "events": [ - { - "id": "$pageview", - "type": "events", - "order": 0, - "name": "$pageview", - }, - { - "id": "custom_event", - "type": "events", - "order": 0, - "name": "custom_event", - }, - ], - "operand": "AND", - } - ) - assert len(session_recordings) == 0 - - (session_recordings, _, _) = self._filter_recordings_by( - { - "events": [ - { - "id": "$pageview", - "type": "events", - "order": 0, - "name": "$pageview", - }, - { - "id": "custom_event", - "type": "events", - "order": 0, - "name": "custom_event", - }, - ], - "operand": "OR", - } - ) - assert len(session_recordings) == 2 - assert sorted([r["session_id"] for r in session_recordings]) == sorted([session_id_two, session_id_one]) - - @parameterized.expand( - [ - # Case 1: Neither has WARN and message "random" - ( - '[{"key": "level", "value": ["warn"], "operator": "exact", "type": "log_entry"}, {"key": "message", "value": "random", "operator": "exact", "type": "log_entry"}]', - "AND", - 0, - [], - ), - # Case 2: AND only matches one recording - ( - '[{"key": "level", "value": ["info"], "operator": "exact", "type": "log_entry"}, {"key": "message", "value": "random", "operator": "exact", "type": "log_entry"}]', - "AND", - 1, - ["both_log_filters"], - ), - # Case 3: Only one is WARN level - ( - '[{"key": "level", "value": ["warn"], "operator": "exact", "type": "log_entry"}]', - "AND", - 1, - ["one_log_filter"], - ), - # Case 4: Only one has message "random" - ( - '[{"key": "message", "value": "random", "operator": "exact", "type": "log_entry"}]', - "AND", - 1, - ["both_log_filters"], - ), - # Case 5: OR matches both - ( - '[{"key": "level", "value": ["warn"], "operator": "exact", "type": "log_entry"}, {"key": "message", "value": "random", "operator": "exact", "type": "log_entry"}]', - "OR", - 2, - ["both_log_filters", "one_log_filter"], - ), - ] - ) - @snapshot_clickhouse_queries - def test_operand_or_filters( - self, - console_log_filters: str, - operand: Literal["AND", "OR"], - expected_count: int, - expected_session_ids: list[str], - ) -> None: - user = "test_operand_or_filter-user" - Person.objects.create(team=self.team, distinct_ids=[user], properties={"email": "bla"}) - - session_with_both_log_filters = "both_log_filters" - produce_replay_summary( - distinct_id="user", - session_id=session_with_both_log_filters, - first_timestamp=self.an_hour_ago, - team_id=self.team.id, - console_log_count=1, - log_messages={"info": ["random"]}, - ) - - session_with_one_log_filter = "one_log_filter" - produce_replay_summary( - distinct_id="user", - session_id=session_with_one_log_filter, - first_timestamp=self.an_hour_ago, - team_id=self.team.id, - console_warn_count=1, - log_messages={"warn": ["warn"]}, - ) - - session_recordings, _, _ = self._filter_recordings_by( - {"console_log_filters": console_log_filters, "operand": operand} - ) - - assert len(session_recordings) == expected_count - assert sorted([rec["session_id"] for rec in session_recordings]) == sorted(expected_session_ids) - - @snapshot_clickhouse_queries - def test_operand_or_mandatory_filters(self): - user = "test_operand_or_filter-user" - person = Person.objects.create(team=self.team, distinct_ids=[user], properties={"email": "bla"}) - - second_user = "test_operand_or_filter-second_user" - second_person = Person.objects.create(team=self.team, distinct_ids=[second_user], properties={"email": "bla"}) - - session_id_one = "session_id_one" - produce_replay_summary( - distinct_id=user, - session_id=session_id_one, - first_timestamp=self.an_hour_ago, - last_timestamp=(self.an_hour_ago + relativedelta(seconds=30)), - team_id=self.team.id, - ) - self.create_event( - user, - self.an_hour_ago + relativedelta(seconds=10), - properties={"$session_id": session_id_one}, - ) - - session_id_two = "session_id_two" - produce_replay_summary( - distinct_id=second_user, - session_id=session_id_two, - first_timestamp=self.an_hour_ago, - last_timestamp=(self.an_hour_ago + relativedelta(seconds=30)), - team_id=self.team.id, - ) - - # person or event filter -> person matches, event matches -> returns session - (session_recordings, _, _) = self._filter_recordings_by( - { - "person_uuid": str(person.uuid), - "events": [ - { - "id": "$pageview", - "type": "events", - "order": 0, - "name": "$pageview", - } - ], - "operand": "OR", - } - ) - assert len(session_recordings) == 1 - assert session_recordings[0]["session_id"] == session_id_one - - # person or event filter -> person does not match, event matches -> does not return session - (session_recordings, _, _) = self._filter_recordings_by( - { - "person_uuid": str(second_person.uuid), - "events": [ - { - "id": "$pageview", - "type": "events", - "order": 0, - "name": "$pageview", - } - ], - "operand": "OR", - } - ) - assert len(session_recordings) == 0 - - # session_id or event filter -> person matches, event matches -> returns session - (session_recordings, _, _) = self._filter_recordings_by( - { - "session_ids": [session_id_one], - "events": [ - { - "id": "$pageview", - "type": "events", - "order": 0, - "name": "$pageview", - } - ], - "operand": "OR", - } - ) - assert len(session_recordings) == 1 - assert session_recordings[0]["session_id"] == session_id_one - - # session_id or event filter -> person does not match, event matches -> does not return session - (session_recordings, _, _) = self._filter_recordings_by( - { - "session_ids": [session_id_two], - "events": [ - { - "id": "$pageview", - "type": "events", - "order": 0, - "name": "$pageview", - } - ], - "operand": "OR", - } - ) - assert len(session_recordings) == 0 - - @snapshot_clickhouse_queries - def test_date_from_filter(self): - user = "test_date_from_filter-user" - Person.objects.create(team=self.team, distinct_ids=[user], properties={"email": "bla"}) - - produce_replay_summary( - distinct_id=user, - session_id="three days before base time", - first_timestamp=(self.an_hour_ago - relativedelta(days=3, seconds=100)), - last_timestamp=(self.an_hour_ago - relativedelta(days=3)), - team_id=self.team.id, - ) - produce_replay_summary( - distinct_id=user, - session_id="two days before base time", - first_timestamp=(self.an_hour_ago - relativedelta(days=2, seconds=100)), - last_timestamp=(self.an_hour_ago - relativedelta(days=2)), - team_id=self.team.id, - ) - - (session_recordings, _, _) = self._filter_recordings_by({"date_from": self.an_hour_ago.strftime("%Y-%m-%d")}) - assert session_recordings == [] - - (session_recordings, _, _) = self._filter_recordings_by( - {"date_from": (self.an_hour_ago - relativedelta(days=2)).strftime("%Y-%m-%d")} - ) - assert len(session_recordings) == 1 - assert session_recordings[0]["session_id"] == "two days before base time" - - @snapshot_clickhouse_queries - def test_date_from_filter_cannot_search_before_ttl(self): - with freeze_time(self.an_hour_ago): - user = "test_date_from_filter_cannot_search_before_ttl-user" - Person.objects.create(team=self.team, distinct_ids=[user], properties={"email": "bla"}) - - produce_replay_summary( - distinct_id=user, - session_id="storage is past ttl", - first_timestamp=(self.an_hour_ago - relativedelta(days=22)), - # an illegally long session but it started 22 days ago - last_timestamp=(self.an_hour_ago - relativedelta(days=3)), - team_id=self.team.id, - ) - produce_replay_summary( - distinct_id=user, - session_id="storage is not past ttl", - first_timestamp=(self.an_hour_ago - relativedelta(days=19)), - last_timestamp=(self.an_hour_ago - relativedelta(days=2)), - team_id=self.team.id, - ) - - (session_recordings, _, _) = self._filter_recordings_by( - {"date_from": (self.an_hour_ago - relativedelta(days=20)).strftime("%Y-%m-%d")} - ) - assert len(session_recordings) == 1 - assert session_recordings[0]["session_id"] == "storage is not past ttl" - - (session_recordings, _, _) = self._filter_recordings_by( - {"date_from": (self.an_hour_ago - relativedelta(days=21)).strftime("%Y-%m-%d")} - ) - assert len(session_recordings) == 1 - assert session_recordings[0]["session_id"] == "storage is not past ttl" - - (session_recordings, _, _) = self._filter_recordings_by( - {"date_from": (self.an_hour_ago - relativedelta(days=22)).strftime("%Y-%m-%d")} - ) - assert len(session_recordings) == 1 - assert session_recordings[0]["session_id"] == "storage is not past ttl" - - @snapshot_clickhouse_queries - def test_date_to_filter(self): - user = "test_date_to_filter-user" - Person.objects.create(team=self.team, distinct_ids=[user], properties={"email": "bla"}) - produce_replay_summary( - distinct_id=user, - session_id="three days before base time", - first_timestamp=(self.an_hour_ago - relativedelta(days=3, seconds=100)), - last_timestamp=(self.an_hour_ago - relativedelta(days=3)), - team_id=self.team.id, - ) - produce_replay_summary( - distinct_id=user, - session_id="two days before base time", - first_timestamp=(self.an_hour_ago - relativedelta(days=2, seconds=100)), - last_timestamp=(self.an_hour_ago - relativedelta(days=2)), - team_id=self.team.id, - ) - - (session_recordings, _, _) = self._filter_recordings_by( - {"date_to": (self.an_hour_ago - relativedelta(days=4)).strftime("%Y-%m-%d")} - ) - assert session_recordings == [] - - (session_recordings, _, _) = self._filter_recordings_by( - {"date_to": (self.an_hour_ago - relativedelta(days=3)).strftime("%Y-%m-%d")} - ) - - assert len(session_recordings) == 1 - assert session_recordings[0]["session_id"] == "three days before base time" - - def test_recording_that_spans_time_bounds(self): - user = "test_recording_that_spans_time_bounds-user" - Person.objects.create(team=self.team, distinct_ids=[user], properties={"email": "bla"}) - day_line = datetime(2021, 11, 5) - session_id = f"session-one-{user}" - produce_replay_summary( - distinct_id=user, - session_id=session_id, - first_timestamp=(day_line - relativedelta(hours=3)), - last_timestamp=(day_line + relativedelta(hours=3)), - team_id=self.team.id, - ) - - (session_recordings, _, _) = self._filter_recordings_by( - { - "date_to": day_line.strftime("%Y-%m-%d"), - "date_from": (day_line - relativedelta(days=10)).strftime("%Y-%m-%d"), - } - ) - - assert len(session_recordings) == 1 - assert session_recordings[0]["session_id"] == session_id - assert session_recordings[0]["duration"] == 6 * 60 * 60 - - @snapshot_clickhouse_queries - def test_person_id_filter(self): - three_user_ids = [str(uuid4()) for _ in range(3)] - session_id_one = f"test_person_id_filter-{str(uuid4())}" - session_id_two = f"test_person_id_filter-{str(uuid4())}" - p = Person.objects.create( - team=self.team, - distinct_ids=[three_user_ids[0], three_user_ids[1]], - properties={"email": "bla"}, - ) - produce_replay_summary( - distinct_id=three_user_ids[0], - session_id=session_id_one, - team_id=self.team.id, - ) - produce_replay_summary( - distinct_id=three_user_ids[1], - session_id=session_id_two, - team_id=self.team.id, - ) - produce_replay_summary( - distinct_id=three_user_ids[2], - session_id=str(uuid4()), - team_id=self.team.id, - ) - - (session_recordings, _, _) = self._filter_recordings_by({"person_uuid": str(p.uuid)}) - assert sorted([r["session_id"] for r in session_recordings]) == sorted([session_id_two, session_id_one]) - - @snapshot_clickhouse_queries - def test_all_filters_at_once(self): - three_user_ids = [str(uuid4()) for _ in range(3)] - target_session_id = f"test_all_filters_at_once-{str(uuid4())}" - - p = Person.objects.create( - team=self.team, - distinct_ids=[three_user_ids[0], three_user_ids[1]], - properties={"email": "bla"}, - ) - custom_event_action = self.create_action(name="custom-event") - - produce_replay_summary( - distinct_id=three_user_ids[0], - session_id=target_session_id, - first_timestamp=(self.an_hour_ago - relativedelta(days=3)), - team_id=self.team.id, - ) - produce_replay_summary( - # does not match because of user distinct id - distinct_id=three_user_ids[2], - session_id=target_session_id, - first_timestamp=(self.an_hour_ago - relativedelta(days=3)), - team_id=self.team.id, - ) - self.create_event( - three_user_ids[0], - self.an_hour_ago - relativedelta(days=3), - properties={"$session_id": target_session_id}, - ) - self.create_event( - three_user_ids[0], - self.an_hour_ago - relativedelta(days=3), - event_name="custom-event", - properties={"$browser": "Chrome", "$session_id": target_session_id}, - ) - produce_replay_summary( - distinct_id=three_user_ids[1], - session_id=target_session_id, - first_timestamp=(self.an_hour_ago - relativedelta(days=3) + relativedelta(hours=6)), - team_id=self.team.id, - ) - produce_replay_summary( - distinct_id=three_user_ids[1], - # does not match because of session id - session_id=str(uuid4()), - first_timestamp=(self.an_hour_ago - relativedelta(days=3) + relativedelta(hours=6)), - team_id=self.team.id, - ) - - flush_persons_and_events() - - (session_recordings, _, _) = self._filter_recordings_by( - { - "person_uuid": str(p.uuid), - "date_to": (self.an_hour_ago + relativedelta(days=3)).strftime("%Y-%m-%d"), - "date_from": (self.an_hour_ago - relativedelta(days=10)).strftime("%Y-%m-%d"), - "session_recording_duration": '{"type":"recording","key":"duration","value":60,"operator":"gt"}', - "events": [ - { - "id": "$pageview", - "type": "events", - "order": 0, - "name": "$pageview", - } - ], - "actions": [ - { - "id": custom_event_action.id, - "type": "actions", - "order": 1, - "name": "custom-event", - } - ], - } - ) - - assert len(session_recordings) == 1 - assert session_recordings[0]["session_id"] == target_session_id - - def test_teams_dont_leak_event_filter(self): - user = "test_teams_dont_leak_event_filter-user" - Person.objects.create(team=self.team, distinct_ids=[user], properties={"email": "bla"}) - another_team = Team.objects.create(organization=self.organization) - - session_id = f"test_teams_dont_leak_event_filter-{str(uuid4())}" - produce_replay_summary( - distinct_id=user, - session_id=session_id, - first_timestamp=self.an_hour_ago, - team_id=self.team.id, - ) - self.create_event(1, self.an_hour_ago + relativedelta(seconds=15), team=another_team) - produce_replay_summary( - distinct_id=user, - session_id=session_id, - first_timestamp=(self.an_hour_ago + relativedelta(seconds=30)), - team_id=self.team.id, - ) - - (session_recordings, _, _) = self._filter_recordings_by( - { - "events": [ - { - "id": "$pageview", - "type": "events", - "order": 0, - "name": "$pageview", - } - ] - } - ) - assert session_recordings == [] - - @snapshot_clickhouse_queries - @also_test_with_materialized_columns(person_properties=["email"]) - def test_filter_with_person_properties_exact(self): - session_id_one, session_id_two = self._two_sessions_two_persons( - "test_filter_with_person_properties_exact", - session_one_person_properties={"email": "bla@gmail.com"}, - session_two_person_properties={"email": "bla2@hotmail.com"}, - ) - - query_results: SessionRecordingQueryResult = self._filter_recordings_by( - { - "properties": [ - { - "key": "email", - "value": ["bla@gmail.com"], - "operator": "exact", - "type": "person", - } - ] - } - ) - - assert [x["session_id"] for x in query_results.results] == [session_id_one] - - @snapshot_clickhouse_queries - @also_test_with_materialized_columns(person_properties=["email"]) - def test_filter_with_person_properties_not_contains(self): - session_id_one, session_id_two = self._two_sessions_two_persons( - "test_filter_with_person_properties_not_contains", - session_one_person_properties={"email": "bla@gmail.com"}, - session_two_person_properties={"email": "bla2@hotmail.com"}, - ) - - query_results: SessionRecordingQueryResult = self._filter_recordings_by( - {"properties": [{"key": "email", "value": "gmail.com", "operator": "not_icontains", "type": "person"}]} - ) - - assert [x["session_id"] for x in query_results.results] == [session_id_two] - - def _two_sessions_two_persons( - self, label: str, session_one_person_properties: dict, session_two_person_properties: dict - ) -> tuple[str, str]: - sessions = [] - - for i in range(2): - user = f"{label}-user-{i}" - session = f"{label}-session-{i}" - sessions.append(session) - - Person.objects.create( - team=self.team, - distinct_ids=[user], - properties=session_one_person_properties if i == 0 else session_two_person_properties, - ) - - produce_replay_summary( - distinct_id=user, - session_id=session, - first_timestamp=self.an_hour_ago, - team_id=self.team.id, - ) - produce_replay_summary( - distinct_id=user, - session_id=session, - first_timestamp=(self.an_hour_ago + relativedelta(seconds=30)), - team_id=self.team.id, - ) - - return sessions[0], sessions[1] - - @snapshot_clickhouse_queries - @also_test_with_materialized_columns(person_properties=["$some_prop"]) - def test_filter_with_cohort_properties(self): - with self.settings(USE_PRECALCULATED_CH_COHORT_PEOPLE=True): - with freeze_time("2021-08-21T20:00:00.000Z"): - user_one = "test_filter_with_cohort_properties-user" - user_two = "test_filter_with_cohort_properties-user2" - session_id_one = f"test_filter_with_cohort_properties-1-{str(uuid4())}" - session_id_two = f"test_filter_with_cohort_properties-2-{str(uuid4())}" - - Person.objects.create(team=self.team, distinct_ids=[user_one], properties={"email": "bla"}) - Person.objects.create( - team=self.team, - distinct_ids=[user_two], - properties={"email": "bla2", "$some_prop": "some_val"}, - ) - cohort = Cohort.objects.create( - team=self.team, - name="cohort1", - groups=[ - { - "properties": [ - { - "key": "$some_prop", - "value": "some_val", - "type": "person", - } - ] - } - ], - ) - cohort.calculate_people_ch(pending_version=0) - - produce_replay_summary( - distinct_id=user_one, - session_id=session_id_one, - first_timestamp=self.an_hour_ago, - team_id=self.team.id, - ) - # self.create_event(user_one, self.base_time, team=self.team) - produce_replay_summary( - distinct_id=user_one, - session_id=session_id_one, - first_timestamp=self.an_hour_ago + relativedelta(seconds=30), - team_id=self.team.id, - ) - produce_replay_summary( - distinct_id=user_two, - session_id=session_id_two, - first_timestamp=self.an_hour_ago, - team_id=self.team.id, - ) - # self.create_event(user_two, self.base_time, team=self.team) - produce_replay_summary( - distinct_id=user_two, - session_id=session_id_two, - first_timestamp=self.an_hour_ago + relativedelta(seconds=30), - team_id=self.team.id, - ) - - (session_recordings, _, _) = self._filter_recordings_by( - { - "properties": [ - { - "key": "id", - "value": cohort.pk, - "operator": "in", - "type": "cohort", - } - ] - } - ) - - assert [x["session_id"] for x in session_recordings] == [session_id_two] - - @snapshot_clickhouse_queries - @also_test_with_materialized_columns(person_properties=["$some_prop"]) - def test_filter_with_static_and_dynamic_cohort_properties(self): - with self.settings(USE_PRECALCULATED_CH_COHORT_PEOPLE=True): - with freeze_time("2021-08-21T20:00:00.000Z"): - user_one = "test_filter_with_cohort_properties-user-in-static-cohort" - user_two = "test_filter_with_cohort_properties-user2-in-dynamic-cohort" - user_three = "test_filter_with_cohort_properties-user3-in-both-cohort" - - session_id_one = ( - f"in-static-cohort-test_filter_with_static_and_dynamic_cohort_properties-1-{str(uuid4())}" - ) - session_id_two = ( - f"in-dynamic-cohort-test_filter_with_static_and_dynamic_cohort_properties-2-{str(uuid4())}" - ) - session_id_three = ( - f"in-both-cohort-test_filter_with_static_and_dynamic_cohort_properties-3-{str(uuid4())}" - ) - - Person.objects.create(team=self.team, distinct_ids=[user_one], properties={"email": "in@static.cohort"}) - Person.objects.create( - team=self.team, - distinct_ids=[user_two], - properties={"email": "in@dynamic.cohort", "$some_prop": "some_val"}, - ) - Person.objects.create( - team=self.team, - distinct_ids=[user_three], - properties={"email": "in@both.cohorts", "$some_prop": "some_val"}, - ) - - dynamic_cohort = Cohort.objects.create( - team=self.team, - name="cohort1", - groups=[ - { - "properties": [ - { - "key": "$some_prop", - "value": "some_val", - "type": "person", - } - ] - } - ], - ) - - static_cohort = Cohort.objects.create(team=self.team, name="a static cohort", groups=[], is_static=True) - static_cohort.insert_users_by_list([user_one, user_three]) - - dynamic_cohort.calculate_people_ch(pending_version=0) - static_cohort.calculate_people_ch(pending_version=0) - - replay_summaries = [ - (user_one, session_id_one), - (user_two, session_id_two), - (user_three, session_id_three), - ] - for distinct_id, session_id in replay_summaries: - produce_replay_summary( - distinct_id=distinct_id, - session_id=session_id, - first_timestamp=self.an_hour_ago, - team_id=self.team.id, - ) - produce_replay_summary( - distinct_id=distinct_id, - session_id=session_id, - first_timestamp=self.an_hour_ago + relativedelta(seconds=30), - team_id=self.team.id, - ) - - (session_recordings, _, _) = self._filter_recordings_by( - { - "properties": [ - { - "key": "id", - "value": static_cohort.pk, - "operator": "in", - "type": "cohort", - }, - ] - } - ) - - assert sorted([x["session_id"] for x in session_recordings]) == sorted( - [session_id_one, session_id_three] - ) - - (session_recordings, _, _) = self._filter_recordings_by( - { - "properties": [ - { - "key": "id", - "value": dynamic_cohort.pk, - "operator": "in", - "type": "cohort", - }, - ] - } - ) - - assert sorted([x["session_id"] for x in session_recordings]) == sorted( - [session_id_two, session_id_three] - ) - - (session_recordings, _, _) = self._filter_recordings_by( - { - "properties": [ - { - "key": "id", - "value": dynamic_cohort.pk, - "operator": "in", - "type": "cohort", - }, - { - "key": "id", - "value": static_cohort.pk, - "operator": "in", - "type": "cohort", - }, - ] - } - ) - - assert sorted([x["session_id"] for x in session_recordings]) == [session_id_three] - - @snapshot_clickhouse_queries - @also_test_with_materialized_columns(person_properties=["$some_prop"]) - def test_filter_with_events_and_cohorts(self): - with self.settings(USE_PRECALCULATED_CH_COHORT_PEOPLE=True): - with freeze_time("2021-08-21T20:00:00.000Z"): - user_one = "test_filter_with_events_and_cohorts-user" - user_two = "test_filter_with_events_and_cohorts-user2" - session_id_one = f"test_filter_with_events_and_cohorts-1-{str(uuid4())}" - session_id_two = f"test_filter_with_events_and_cohorts-2-{str(uuid4())}" - - Person.objects.create(team=self.team, distinct_ids=[user_one], properties={"email": "bla"}) - Person.objects.create( - team=self.team, - distinct_ids=[user_two], - properties={"email": "bla2", "$some_prop": "some_val"}, - ) - cohort = Cohort.objects.create( - team=self.team, - name="cohort1", - groups=[ - { - "properties": [ - { - "key": "$some_prop", - "value": "some_val", - "type": "person", - } - ] - } - ], - ) - cohort.calculate_people_ch(pending_version=0) - - produce_replay_summary( - distinct_id=user_one, - session_id=session_id_one, - first_timestamp=self.an_hour_ago, - team_id=self.team.id, - ) - self.create_event( - user_one, - self.an_hour_ago, - team=self.team, - event_name="custom_event", - properties={"$session_id": session_id_one}, - ) - produce_replay_summary( - distinct_id=user_one, - session_id=session_id_one, - first_timestamp=self.an_hour_ago + relativedelta(seconds=30), - team_id=self.team.id, - ) - produce_replay_summary( - distinct_id=user_two, - session_id=session_id_two, - first_timestamp=self.an_hour_ago, - team_id=self.team.id, - ) - self.create_event( - user_two, - self.an_hour_ago, - team=self.team, - event_name="custom_event", - properties={"$session_id": session_id_two}, - ) - produce_replay_summary( - distinct_id=user_two, - session_id=session_id_two, - first_timestamp=self.an_hour_ago + relativedelta(seconds=30), - team_id=self.team.id, - ) - - (session_recordings, _, _) = self._filter_recordings_by( - { - # has to be in the cohort and pageview has to be in the events - # test data has one user in the cohort but no pageviews - "properties": [ - { - "key": "id", - "value": cohort.pk, - "operator": "in", - "type": "cohort", - } - ], - "events": [ - { - "id": "$pageview", - "type": "events", - "order": 0, - "name": "$pageview", - } - ], - } - ) - - assert len(session_recordings) == 0 - - (session_recordings, _, _) = self._filter_recordings_by( - { - "properties": [ - { - "key": "id", - "value": cohort.pk, - "operator": "in", - "type": "cohort", - } - ], - "events": [ - { - "id": "custom_event", - "type": "events", - "order": 0, - "name": "custom_event", - } - ], - } - ) - - assert [x["session_id"] for x in session_recordings] == [session_id_two] - - @snapshot_clickhouse_queries - @also_test_with_materialized_columns(["$current_url"]) - def test_event_filter_with_matching_on_session_id(self): - user_distinct_id = "test_event_filter_with_matching_on_session_id-user" - Person.objects.create(team=self.team, distinct_ids=[user_distinct_id], properties={"email": "bla"}) - session_id = f"test_event_filter_with_matching_on_session_id-1-{str(uuid4())}" - - self.create_event( - user_distinct_id, - self.an_hour_ago, - event_name="$pageview", - properties={"$session_id": session_id}, - ) - self.create_event( - user_distinct_id, - self.an_hour_ago, - event_name="$autocapture", - properties={"$session_id": str(uuid4())}, - ) - - produce_replay_summary( - distinct_id=user_distinct_id, - session_id=session_id, - first_timestamp=self.an_hour_ago, - team_id=self.team.id, - ) - produce_replay_summary( - distinct_id=user_distinct_id, - session_id=session_id, - first_timestamp=self.an_hour_ago + relativedelta(seconds=30), - team_id=self.team.id, - ) - - (session_recordings, _, _) = self._filter_recordings_by( - { - "events": [ - { - "id": "$pageview", - "type": "events", - "order": 0, - "name": "$pageview", - } - ] - } - ) - - assert len(session_recordings) == 1 - assert session_recordings[0]["session_id"] == session_id - - (session_recordings, _, _) = self._filter_recordings_by( - { - "events": [ - { - "id": "$autocapture", - "type": "events", - "order": 0, - "name": "$autocapture", - } - ] - } - ) - assert session_recordings == [] - - @also_test_with_materialized_columns(event_properties=["$current_url", "$browser"], person_properties=["email"]) - @snapshot_clickhouse_queries - def test_event_filter_with_hogql_properties(self): - user = "test_event_filter_with_hogql_properties-user" - - Person.objects.create(team=self.team, distinct_ids=[user], properties={"email": "bla"}) - - session_id = f"test_event_filter_with_hogql_properties-1-{str(uuid4())}" - self.create_event( - user, - self.an_hour_ago, - properties={ - "$browser": "Chrome", - "$session_id": session_id, - "$window_id": str(uuid4()), - }, - ) - - produce_replay_summary( - distinct_id=user, - session_id=session_id, - first_timestamp=self.an_hour_ago, - team_id=self.team.id, - ) - produce_replay_summary( - distinct_id=user, - session_id=session_id, - first_timestamp=self.an_hour_ago + relativedelta(seconds=30), - team_id=self.team.id, - ) - - (session_recordings, _, _) = self._filter_recordings_by( - { - "events": [ - { - "id": "$pageview", - "type": "events", - "order": 0, - "name": "$pageview", - "properties": [ - {"key": "properties.$browser == 'Chrome'", "type": "hogql"}, - ], - } - ] - } - ) - - assert len(session_recordings) == 1 - assert session_recordings[0]["session_id"] == session_id - - (session_recordings, _, _) = self._filter_recordings_by( - { - "events": [ - { - "id": "$pageview", - "type": "events", - "order": 0, - "name": "$pageview", - "properties": [{"key": "properties.$browser == 'Firefox'", "type": "hogql"}], - } - ] - } - ) - - assert session_recordings == [] - - @snapshot_clickhouse_queries - def test_event_filter_with_hogql_person_properties(self): - user = "test_event_filter_with_hogql_properties-user" - - Person.objects.create(team=self.team, distinct_ids=[user], properties={"email": "bla"}) - - session_id = f"test_event_filter_with_hogql_properties-1-{str(uuid4())}" - self.create_event( - user, - self.an_hour_ago, - properties={ - "$browser": "Chrome", - "$session_id": session_id, - "$window_id": str(uuid4()), - }, - ) - - produce_replay_summary( - distinct_id=user, - session_id=session_id, - first_timestamp=self.an_hour_ago, - team_id=self.team.id, - ) - produce_replay_summary( - distinct_id=user, - session_id=session_id, - first_timestamp=self.an_hour_ago + relativedelta(seconds=30), - team_id=self.team.id, - ) - - (session_recordings, _, _) = self._filter_recordings_by( - { - "events": [ - { - "id": "$pageview", - "type": "events", - "order": 0, - "name": "$pageview", - "properties": [ - { - "key": "person.properties.email == 'bla'", - "type": "hogql", - }, - ], - } - ] - } - ) - - assert len(session_recordings) == 1 - assert session_recordings[0]["session_id"] == session_id - - (session_recordings, _, _) = self._filter_recordings_by( - { - "events": [ - { - "id": "$pageview", - "type": "events", - "order": 0, - "name": "$pageview", - "properties": [ - { - "key": "person.properties.email == 'something else'", - "type": "hogql", - }, - ], - } - ] - } - ) - - assert session_recordings == [] - - @also_test_with_materialized_columns(["$current_url", "$browser"]) - @snapshot_clickhouse_queries - @freeze_time("2021-01-21T20:00:00.000Z") - def test_any_event_filter_with_properties(self): - Person.objects.create(team=self.team, distinct_ids=["user"], properties={"email": "bla"}) - - page_view_session_id = f"pageview-session-{str(uuid4())}" - my_custom_event_session_id = f"my-custom-event-session-{str(uuid4())}" - non_matching__event_session_id = f"non-matching-event-session-{str(uuid4())}" - - self.create_event( - "user", - self.an_hour_ago, - properties={ - "$browser": "Chrome", - "$session_id": page_view_session_id, - "$window_id": "1", - }, - event_name="$pageview", - ) - - self.create_event( - "user", - self.an_hour_ago, - properties={ - "$browser": "Chrome", - "$session_id": my_custom_event_session_id, - "$window_id": "1", - }, - event_name="my-custom-event", - ) - - self.create_event( - "user", - self.an_hour_ago, - properties={ - "$browser": "Safari", - "$session_id": non_matching__event_session_id, - "$window_id": "1", - }, - event_name="my-non-matching-event", - ) - - produce_replay_summary( - distinct_id="user", - session_id=page_view_session_id, - first_timestamp=self.an_hour_ago, - team_id=self.team.id, - ) - produce_replay_summary( - distinct_id="user", - session_id=my_custom_event_session_id, - first_timestamp=self.an_hour_ago + relativedelta(seconds=30), - team_id=self.team.id, - ) - produce_replay_summary( - distinct_id="user", - session_id=non_matching__event_session_id, - first_timestamp=self.an_hour_ago + relativedelta(seconds=30), - team_id=self.team.id, - ) - - (session_recordings, _, _) = self._filter_recordings_by( - { - "events": [ - { - # an id of null means "match any event" - "id": None, - "type": "events", - "order": 0, - "name": "All events", - "properties": [], - } - ] - } - ) - - assert sorted( - [sr["session_id"] for sr in session_recordings], - ) == [ - my_custom_event_session_id, - non_matching__event_session_id, - page_view_session_id, - ] - - (session_recordings, _, _) = self._filter_recordings_by( - { - "events": [ - { - # an id of null means "match any event" - "id": None, - "type": "events", - "order": 0, - "name": "All events", - "properties": [ - { - "key": "$browser", - "value": ["Chrome"], - "operator": "exact", - "type": "event", - } - ], - } - ] - } - ) - - assert sorted( - [sr["session_id"] for sr in session_recordings], - ) == [ - my_custom_event_session_id, - page_view_session_id, - ] - - (session_recordings, _, _) = self._filter_recordings_by( - { - "events": [ - { - "id": None, - "type": "events", - "order": 0, - "name": "All events", - "properties": [ - { - "key": "$browser", - "value": ["Firefox"], - "operator": "exact", - "type": "event", - } - ], - } - ] - } - ) - assert session_recordings == [] - - @snapshot_clickhouse_queries - @freeze_time("2021-01-21T20:00:00.000Z") - def test_filter_for_recordings_with_console_logs(self): - Person.objects.create(team=self.team, distinct_ids=["user"], properties={"email": "bla"}) - - with_logs_session_id = f"with-logs-session-{str(uuid4())}" - without_logs_session_id = f"no-logs-session-{str(uuid4())}" - - produce_replay_summary( - distinct_id="user", - session_id=with_logs_session_id, - first_timestamp=self.an_hour_ago, - team_id=self.team.id, - console_log_count=4, - log_messages={ - "info": [ - "info", - "info", - "info", - ], - }, - ) - - produce_replay_summary( - distinct_id="user", - session_id=without_logs_session_id, - first_timestamp=self.an_hour_ago, - team_id=self.team.id, - ) - - # (session_recordings, _, _) = self._filter_recordings_by({"console_logs": ["info"]}) - - (session_recordings, _, _) = self._filter_recordings_by( - { - "console_log_filters": '[{"key": "level", "value": ["info"], "operator": "exact", "type": "log_entry"}]', - "operand": "AND", - } - ) - - actual = sorted( - [(sr["session_id"], sr["console_log_count"]) for sr in session_recordings], - key=lambda x: x[0], - ) - - assert actual == [ - (with_logs_session_id, 4), - ] - - (session_recordings, _, _) = self._filter_recordings_by( - { - "console_log_filters": '[{"key": "level", "value": ["warn"], "operator": "exact", "type": "log_entry"}]', - "operand": "AND", - } - ) - assert session_recordings == [] - - @snapshot_clickhouse_queries - @freeze_time("2021-01-21T20:00:00.000Z") - def test_filter_for_recordings_with_console_warns(self): - Person.objects.create(team=self.team, distinct_ids=["user"], properties={"email": "bla"}) - - with_logs_session_id = f"with-logs-session-{str(uuid4())}" - without_logs_session_id = f"no-logs-session-{str(uuid4())}" - - produce_replay_summary( - distinct_id="user", - session_id=with_logs_session_id, - first_timestamp=self.an_hour_ago, - team_id=self.team.id, - console_warn_count=4, - log_messages={ - "warn": [ - "warn", - "warn", - "warn", - "warn", - ], - }, - ) - produce_replay_summary( - distinct_id="user", - session_id=without_logs_session_id, - first_timestamp=self.an_hour_ago, - team_id=self.team.id, - ) - - (session_recordings, _, _) = self._filter_recordings_by( - { - "console_log_filters": '[{"key": "level", "value": ["warn"], "operator": "exact", "type": "log_entry"}]', - "operand": "AND", - } - ) - - assert sorted( - [(sr["session_id"], sr["console_warn_count"]) for sr in session_recordings], - key=lambda x: x[0], - ) == [ - (with_logs_session_id, 4), - ] - - (session_recordings, _, _) = self._filter_recordings_by( - { - "console_log_filters": '[{"key": "level", "value": ["info"], "operator": "exact", "type": "log_entry"}]', - "operand": "AND", - } - ) - - assert session_recordings == [] - - @snapshot_clickhouse_queries - @freeze_time("2021-01-21T20:00:00.000Z") - def test_filter_for_recordings_with_console_errors(self): - Person.objects.create(team=self.team, distinct_ids=["user"], properties={"email": "bla"}) - - with_logs_session_id = f"with-logs-session-{str(uuid4())}" - without_logs_session_id = f"no-logs-session-{str(uuid4())}" - - produce_replay_summary( - distinct_id="user", - session_id=with_logs_session_id, - first_timestamp=self.an_hour_ago, - team_id=self.team.id, - console_error_count=4, - log_messages={ - "error": [ - "error", - "error", - "error", - "error", - ], - }, - ) - produce_replay_summary( - distinct_id="user", - session_id=without_logs_session_id, - first_timestamp=self.an_hour_ago, - team_id=self.team.id, - ) - - (session_recordings, _, _) = self._filter_recordings_by( - { - "console_log_filters": '[{"key": "level", "value": ["error"], "operator": "exact", "type": "log_entry"}]', - "operand": "AND", - } - ) - - assert sorted( - [(sr["session_id"], sr["console_error_count"]) for sr in session_recordings], - key=lambda x: x[0], - ) == [ - (with_logs_session_id, 4), - ] - - (session_recordings, _, _) = self._filter_recordings_by( - { - "console_log_filters": '[{"key": "level", "value": ["info"], "operator": "exact", "type": "log_entry"}]', - "operand": "AND", - } - ) - - assert session_recordings == [] - - @snapshot_clickhouse_queries - @freeze_time("2021-01-21T20:00:00.000Z") - def test_filter_for_recordings_with_mixed_console_counts(self): - Person.objects.create(team=self.team, distinct_ids=["user"], properties={"email": "bla"}) - - with_logs_session_id = f"with-logs-session-{str(uuid4())}" - with_warns_session_id = f"with-warns-session-{str(uuid4())}" - with_errors_session_id = f"with-errors-session-{str(uuid4())}" - with_two_session_id = f"with-two-session-{str(uuid4())}" - - produce_replay_summary( - distinct_id="user", - session_id=with_logs_session_id, - first_timestamp=self.an_hour_ago, - team_id=self.team.id, - console_log_count=4, - log_messages={ - "info": [ - "info", - "info", - "info", - "info", - ], - }, - ) - produce_replay_summary( - distinct_id="user", - session_id=with_warns_session_id, - first_timestamp=self.an_hour_ago, - team_id=self.team.id, - console_warn_count=4, - log_messages={ - "warn": [ - "warn", - "warn", - "warn", - "warn", - ], - }, - ) - produce_replay_summary( - distinct_id="user", - session_id=with_errors_session_id, - first_timestamp=self.an_hour_ago, - team_id=self.team.id, - console_error_count=4, - log_messages={ - "error": [ - "error", - "error", - "error", - "error", - ], - }, - ) - produce_replay_summary( - distinct_id="user", - session_id=with_two_session_id, - first_timestamp=self.an_hour_ago, - team_id=self.team.id, - console_error_count=4, - console_log_count=3, - log_messages={ - "error": [ - "error", - "error", - "error", - "error", - ], - "info": [ - "info", - "info", - "info", - ], - }, - ) - - (session_recordings, _, _) = self._filter_recordings_by( - { - "console_log_filters": '[{"key": "level", "value": ["warn", "error"], "operator": "exact", "type": "log_entry"}]', - "operand": "AND", - } - ) - - assert sorted([sr["session_id"] for sr in session_recordings]) == sorted( - [ - with_errors_session_id, - with_two_session_id, - with_warns_session_id, - ] - ) - - (session_recordings, _, _) = self._filter_recordings_by( - { - "console_log_filters": '[{"key": "level", "value": ["info"], "operator": "exact", "type": "log_entry"}]', - "operand": "AND", - } - ) - - assert sorted([sr["session_id"] for sr in session_recordings]) == sorted( - [ - with_two_session_id, - with_logs_session_id, - ] - ) - - @parameterized.expand( - [ - # Case 1: OR operand, message 4 matches in warn and error - ( - '[{"key": "level", "value": ["warn", "error"], "operator": "exact", "type": "log_entry"}, {"key": "message", "value": "message 4", "operator": "icontains", "type": "log_entry"}]', - "OR", - ["with-errors-session", "with-two-session", "with-warns-session", "with-logs-session"], - ), - # Case 2: AND operand, message 5 matches only in warn - ( - '[{"key": "level", "value": ["warn", "error"], "operator": "exact", "type": "log_entry"}, {"key": "message", "value": "message 5", "operator": "icontains", "type": "log_entry"}]', - "AND", - ["with-warns-session"], - ), - # Case 3: AND operand, message 5 does not match log level "info" - ( - '[{"key": "level", "value": ["info"], "operator": "exact", "type": "log_entry"}, {"key": "message", "value": "message 5", "operator": "icontains", "type": "log_entry"}]', - "AND", - [], - ), - ] - ) - @snapshot_clickhouse_queries - @freeze_time("2021-01-21T20:00:00.000Z") - def test_filter_for_recordings_by_console_text( - self, - console_log_filters: str, - operand: Literal["AND", "OR"], - expected_session_ids: list[str], - ) -> None: - Person.objects.create(team=self.team, distinct_ids=["user"], properties={"email": "bla"}) - - # Create sessions - produce_replay_summary( - distinct_id="user", - session_id="with-logs-session", - first_timestamp=self.an_hour_ago, - team_id=self.team.id, - console_log_count=4, - log_messages={ - "info": [ - "log message 1", - "log message 2", - "log message 3", - "log message 4", - ] - }, - ) - produce_replay_summary( - distinct_id="user", - session_id="with-warns-session", - first_timestamp=self.an_hour_ago, - team_id=self.team.id, - console_warn_count=5, - log_messages={ - "warn": [ - "warn message 1", - "warn message 2", - "warn message 3", - "warn message 4", - "warn message 5", - ] - }, - ) - produce_replay_summary( - distinct_id="user", - session_id="with-errors-session", - first_timestamp=self.an_hour_ago, - team_id=self.team.id, - console_error_count=4, - log_messages={ - "error": [ - "error message 1", - "error message 2", - "error message 3", - "error message 4", - ] - }, - ) - produce_replay_summary( - distinct_id="user", - session_id="with-two-session", - first_timestamp=self.an_hour_ago, - team_id=self.team.id, - console_error_count=4, - console_log_count=3, - log_messages={ - "error": [ - "error message 1", - "error message 2", - "error message 3", - "error message 4", - ], - "info": ["log message 1", "log message 2", "log message 3"], - }, - ) - produce_replay_summary( - distinct_id="user", - session_id="with-no-matches-session", - first_timestamp=self.an_hour_ago, - team_id=self.team.id, - console_error_count=4, - console_log_count=3, - log_messages={ - "info": ["log message 1", "log message 2", "log message 3"], - }, - ) - - # Perform the filtering and validate results - session_recordings, _, _ = self._filter_recordings_by( - {"console_log_filters": console_log_filters, "operand": operand} - ) - - assert sorted([sr["session_id"] for sr in session_recordings]) == sorted(expected_session_ids) - - @snapshot_clickhouse_queries - def test_filter_for_recordings_by_snapshot_source(self): - user = "test_duration_filter-user" - Person.objects.create(team=self.team, distinct_ids=[user], properties={"email": "bla"}) - - session_id_one = "session one id" - produce_replay_summary( - distinct_id=user, - session_id=session_id_one, - team_id=self.team.id, - snapshot_source="web", - ) - - session_id_two = "session two id" - produce_replay_summary( - distinct_id=user, - session_id=session_id_two, - team_id=self.team.id, - snapshot_source="mobile", - ) - - (session_recordings, _, _) = self._filter_recordings_by( - { - "having_predicates": '[{"key": "snapshot_source", "value": ["web"], "operator": "exact", "type": "recording"}]' - } - ) - assert [r["session_id"] for r in session_recordings] == [session_id_one] - - (session_recordings, _, _) = self._filter_recordings_by( - { - "having_predicates": '[{"key": "snapshot_source", "value": ["mobile"], "operator": "exact", "type": "recording"}]' - } - ) - assert [r["session_id"] for r in session_recordings] == [session_id_two] - - @also_test_with_materialized_columns( - event_properties=["is_internal_user"], - person_properties=["email"], - verify_no_jsonextract=False, - ) - @freeze_time("2021-01-21T20:00:00.000Z") - @snapshot_clickhouse_queries - def test_event_filter_with_test_accounts_excluded(self): - self.team.test_account_filters = [ - { - "key": "email", - "value": "@posthog.com", - "operator": "not_icontains", - "type": "person", - }, - { - "key": "is_internal_user", - "value": ["false"], - "operator": "exact", - "type": "event", - }, - {"key": "properties.$browser == 'Chrome'", "type": "hogql"}, - ] - self.team.save() - - Person.objects.create(team=self.team, distinct_ids=["user"], properties={"email": "bla"}) - - produce_replay_summary( - distinct_id="user", - session_id="1", - first_timestamp=self.an_hour_ago, - team_id=self.team.id, - ) - self.create_event( - "user", - self.an_hour_ago, - properties={ - "$session_id": "1", - "$window_id": "1", - "is_internal_user": "true", - }, - ) - produce_replay_summary( - distinct_id="user", - session_id="1", - first_timestamp=self.an_hour_ago + relativedelta(seconds=30), - team_id=self.team.id, - ) - - (session_recordings, _, _) = self._filter_recordings_by( - { - "events": [ - { - "id": "$pageview", - "type": "events", - "order": 0, - "name": "$pageview", - } - ], - "filter_test_accounts": True, - } - ) - self.assertEqual(len(session_recordings), 0) - - (session_recordings, _, _) = self._filter_recordings_by( - { - "events": [ - { - "id": "$pageview", - "type": "events", - "order": 0, - "name": "$pageview", - } - ], - "filter_test_accounts": False, - } - ) - self.assertEqual(len(session_recordings), 1) - - @also_test_with_materialized_columns( - event_properties=["$browser"], - person_properties=["email"], - verify_no_jsonextract=False, - ) - @freeze_time("2021-01-21T20:00:00.000Z") - @snapshot_clickhouse_queries - def test_event_filter_with_hogql_event_properties_test_accounts_excluded(self): - self.team.test_account_filters = [ - {"key": "properties.$browser == 'Chrome'", "type": "hogql"}, - ] - self.team.save() - - Person.objects.create(team=self.team, distinct_ids=["user"], properties={"email": "bla"}) - Person.objects.create( - team=self.team, - distinct_ids=["user2"], - properties={"email": "not-the-other-one"}, - ) - - produce_replay_summary( - distinct_id="user", - session_id="1", - first_timestamp=self.an_hour_ago, - team_id=self.team.id, - ) - self.create_event( - "user", - self.an_hour_ago, - properties={"$session_id": "1", "$window_id": "1", "$browser": "Chrome"}, - ) - produce_replay_summary( - distinct_id="user", - session_id="1", - first_timestamp=self.an_hour_ago + relativedelta(seconds=30), - team_id=self.team.id, - ) - - produce_replay_summary( - distinct_id="user2", - session_id="2", - first_timestamp=self.an_hour_ago, - team_id=self.team.id, - ) - self.create_event( - "user2", - self.an_hour_ago, - properties={"$session_id": "2", "$window_id": "1", "$browser": "Firefox"}, - ) - - # there are 2 pageviews - (session_recordings, _, _) = self._filter_recordings_by( - { - # pageview that matches the hogql test_accounts filter - "events": [ - { - "id": "$pageview", - "type": "events", - "order": 0, - "name": "$pageview", - } - ], - "filter_test_accounts": False, - } - ) - self.assertEqual(len(session_recordings), 2) - - self.team.test_account_filters = [ - {"key": "person.properties.email == 'bla'", "type": "hogql"}, - ] - self.team.save() - - (session_recordings, _, _) = self._filter_recordings_by( - { - # only 1 pageview that matches the hogql test_accounts filter - "events": [ - { - "id": "$pageview", - "type": "events", - "order": 0, - "name": "$pageview", - } - ], - "filter_test_accounts": True, - } - ) - self.assertEqual(len(session_recordings), 1) - - self.team.test_account_filters = [ - {"key": "properties.$browser == 'Chrome'", "type": "hogql"}, - {"key": "person.properties.email == 'bla'", "type": "hogql"}, - ] - self.team.save() - - # one user sessions matches the person + event test_account filter - (session_recordings, _, _) = self._filter_recordings_by( - { - "filter_test_accounts": True, - } - ) - self.assertEqual(len(session_recordings), 1) - - # TRICKY: we had to disable use of materialized columns for part of the query generation - # due to RAM usage issues on the EU cluster - @also_test_with_materialized_columns(event_properties=["is_internal_user"], verify_no_jsonextract=False) - @freeze_time("2021-01-21T20:00:00.000Z") - @snapshot_clickhouse_queries - def test_top_level_event_property_test_account_filter(self): - """ - This is a regression test. A user with an $ip test account filter - reported the filtering wasn't working. - - The filter wasn't triggering the "should join events check", and so we didn't apply the filter at all - """ - self.team.test_account_filters = [ - { - "key": "is_internal_user", - "value": ["false"], - "operator": "exact", - "type": "event", - }, - ] - self.team.save() - - Person.objects.create(team=self.team, distinct_ids=["user"], properties={"email": "bla"}) - Person.objects.create( - team=self.team, - distinct_ids=["user2"], - properties={"email": "not-the-other-one"}, - ) - - produce_replay_summary( - distinct_id="user", - session_id="1", - first_timestamp=self.an_hour_ago, - team_id=self.team.id, - ) - self.create_event( - "user", - self.an_hour_ago, - properties={ - "$session_id": "1", - "$window_id": "1", - "is_internal_user": False, - }, - ) - produce_replay_summary( - distinct_id="user", - session_id="1", - first_timestamp=self.an_hour_ago + relativedelta(seconds=30), - team_id=self.team.id, - ) - - produce_replay_summary( - distinct_id="user2", - session_id="2", - first_timestamp=self.an_hour_ago, - team_id=self.team.id, - ) - self.create_event( - "user2", - self.an_hour_ago, - properties={ - "$session_id": "2", - "$window_id": "1", - "is_internal_user": True, - }, - ) - - # there are 2 pageviews - (session_recordings, _, _) = self._filter_recordings_by( - { - # pageview that matches the hogql test_accounts filter - "events": [ - { - "id": "$pageview", - "type": "events", - "order": 0, - "name": "$pageview", - } - ], - "filter_test_accounts": False, - } - ) - self.assertEqual(len(session_recordings), 2) - - (session_recordings, _, _) = self._filter_recordings_by( - { - # only 1 pageview that matches the test_accounts filter - "filter_test_accounts": True, - } - ) - self.assertEqual(len(session_recordings), 1) - - # TRICKY: we had to disable use of materialized columns for part of the query generation - # due to RAM usage issues on the EU cluster - @also_test_with_materialized_columns(event_properties=["is_internal_user"], verify_no_jsonextract=True) - @freeze_time("2021-01-21T20:00:00.000Z") - @snapshot_clickhouse_queries - def test_top_level_event_property_test_account_filter_allowing_denormalized_props(self): - """ - This is a duplicate of the test test_top_level_event_property_test_account_filter - but with denormalized props allowed - """ - - with self.settings(ALLOW_DENORMALIZED_PROPS_IN_LISTING=True): - self.team.test_account_filters = [ - { - "key": "is_internal_user", - "value": ["false"], - "operator": "exact", - "type": "event", - }, - ] - self.team.save() - - Person.objects.create(team=self.team, distinct_ids=["user"], properties={"email": "bla"}) - Person.objects.create( - team=self.team, - distinct_ids=["user2"], - properties={"email": "not-the-other-one"}, - ) - - produce_replay_summary( - distinct_id="user", - session_id="1", - first_timestamp=self.an_hour_ago, - team_id=self.team.id, - ) - self.create_event( - "user", - self.an_hour_ago, - properties={ - "$session_id": "1", - "$window_id": "1", - "is_internal_user": False, - }, - ) - produce_replay_summary( - distinct_id="user", - session_id="1", - first_timestamp=self.an_hour_ago + relativedelta(seconds=30), - team_id=self.team.id, - ) - - produce_replay_summary( - distinct_id="user2", - session_id="2", - first_timestamp=self.an_hour_ago, - team_id=self.team.id, - ) - self.create_event( - "user2", - self.an_hour_ago, - properties={ - "$session_id": "2", - "$window_id": "1", - "is_internal_user": True, - }, - ) - - # there are 2 pageviews - (session_recordings, _, _) = self._filter_recordings_by( - { - # pageview that matches the hogql test_accounts filter - "events": [ - { - "id": "$pageview", - "type": "events", - "order": 0, - "name": "$pageview", - } - ], - "filter_test_accounts": False, - } - ) - self.assertEqual(len(session_recordings), 2) - - (session_recordings, _, _) = self._filter_recordings_by( - { - # only 1 pageview that matches the test_accounts filter - "filter_test_accounts": True, - } - ) - self.assertEqual(len(session_recordings), 1) - - @also_test_with_materialized_columns(event_properties=["is_internal_user"]) - @freeze_time("2021-01-21T20:00:00.000Z") - @snapshot_clickhouse_queries - def test_top_level_hogql_event_property_test_account_filter(self): - """ - This is a regression test. A user with an $ip test account filter - reported the filtering wasn't working. - - The filter wasn't triggering the "should join events" check, and so we didn't apply the filter at all - """ - self.team.test_account_filters = [ - {"key": "properties.is_internal_user == 'true'", "type": "hogql"}, - ] - self.team.save() - - Person.objects.create(team=self.team, distinct_ids=["user"], properties={"email": "bla"}) - Person.objects.create( - team=self.team, - distinct_ids=["user2"], - properties={"email": "not-the-other-one"}, - ) - - produce_replay_summary( - distinct_id="user", - session_id="1", - first_timestamp=self.an_hour_ago, - team_id=self.team.id, - ) - self.create_event( - "user", - self.an_hour_ago, - properties={ - "$session_id": "1", - "$window_id": "1", - "is_internal_user": False, - }, - ) - produce_replay_summary( - distinct_id="user", - session_id="1", - first_timestamp=self.an_hour_ago + relativedelta(seconds=30), - team_id=self.team.id, - ) - - produce_replay_summary( - distinct_id="user2", - session_id="2", - first_timestamp=self.an_hour_ago, - team_id=self.team.id, - ) - self.create_event( - "user2", - self.an_hour_ago, - properties={ - "$session_id": "2", - "$window_id": "1", - "is_internal_user": True, - }, - ) - - # there are 2 pageviews - (session_recordings, _, _) = self._filter_recordings_by( - { - # pageview that matches the hogql test_accounts filter - "events": [ - { - "id": "$pageview", - "type": "events", - "order": 0, - "name": "$pageview", - } - ], - "filter_test_accounts": False, - } - ) - self.assertEqual(len(session_recordings), 2) - - (session_recordings, _, _) = self._filter_recordings_by( - { - # only 1 pageview that matches the test_accounts filter - "filter_test_accounts": True, - } - ) - self.assertEqual(len(session_recordings), 1) - - @also_test_with_materialized_columns(person_properties=["email"], verify_no_jsonextract=False) - @freeze_time("2021-01-21T20:00:00.000Z") - @snapshot_clickhouse_queries - def test_top_level_hogql_person_property_test_account_filter(self): - """ - This is a regression test. A user with an $ip test account filter - reported the filtering wasn't working. - - The filter wasn't triggering the "should join events" check, and so we didn't apply the filter at all - """ - self.team.test_account_filters = [ - {"key": "person.properties.email == 'bla'", "type": "hogql"}, - ] - self.team.save() - - Person.objects.create(team=self.team, distinct_ids=["user"], properties={"email": "bla"}) - Person.objects.create( - team=self.team, - distinct_ids=["user2"], - properties={"email": "not-the-other-one"}, - ) - - produce_replay_summary( - distinct_id="user", - session_id="1", - first_timestamp=self.an_hour_ago, - team_id=self.team.id, - ) - self.create_event( - "user", - self.an_hour_ago, - properties={ - "$session_id": "1", - "$window_id": "1", - "is_internal_user": False, - }, - ) - produce_replay_summary( - distinct_id="user", - session_id="1", - first_timestamp=self.an_hour_ago + relativedelta(seconds=30), - team_id=self.team.id, - ) - - produce_replay_summary( - distinct_id="user2", - session_id="2", - first_timestamp=self.an_hour_ago, - team_id=self.team.id, - ) - self.create_event( - "user2", - self.an_hour_ago, - properties={ - "$session_id": "2", - "$window_id": "1", - "is_internal_user": True, - }, - ) - - # there are 2 pageviews - (session_recordings, _, _) = self._filter_recordings_by( - { - # pageview that matches the hogql test_accounts filter - "events": [ - { - "id": "$pageview", - "type": "events", - "order": 0, - "name": "$pageview", - } - ], - "filter_test_accounts": False, - } - ) - self.assertEqual(len(session_recordings), 2) - - (session_recordings, _, _) = self._filter_recordings_by( - { - # only 1 pageview that matches the test_accounts filter - "filter_test_accounts": True, - } - ) - self.assertEqual(len(session_recordings), 1) - - @also_test_with_materialized_columns(person_properties=["email"], verify_no_jsonextract=False) - @freeze_time("2021-01-21T20:00:00.000Z") - @snapshot_clickhouse_queries - def test_top_level_person_property_test_account_filter(self): - """ - This is a regression test. A user with an $ip test account filter - reported the filtering wasn't working. - - The filter wasn't triggering the "should join events" check, and so we didn't apply the filter at all - """ - self.team.test_account_filters = [{"key": "email", "value": ["bla"], "operator": "exact", "type": "person"}] - self.team.save() - - Person.objects.create(team=self.team, distinct_ids=["user"], properties={"email": "bla"}) - Person.objects.create( - team=self.team, - distinct_ids=["user2"], - properties={"email": "not-the-other-one"}, - ) - - produce_replay_summary( - distinct_id="user", - session_id="1", - first_timestamp=self.an_hour_ago, - team_id=self.team.id, - ) - self.create_event( - "user", - self.an_hour_ago, - properties={ - "$session_id": "1", - "$window_id": "1", - "is_internal_user": False, - }, - ) - produce_replay_summary( - distinct_id="user", - session_id="1", - first_timestamp=self.an_hour_ago + relativedelta(seconds=30), - team_id=self.team.id, - ) - - produce_replay_summary( - distinct_id="user2", - session_id="2", - first_timestamp=self.an_hour_ago, - team_id=self.team.id, - ) - self.create_event( - "user2", - self.an_hour_ago, - properties={ - "$session_id": "2", - "$window_id": "1", - "is_internal_user": True, - }, - ) - - # there are 2 pageviews - (session_recordings, _, _) = self._filter_recordings_by( - { - # pageview that matches the hogql test_accounts filter - "events": [ - { - "id": "$pageview", - "type": "events", - "order": 0, - "name": "$pageview", - } - ], - "filter_test_accounts": False, - } - ) - self.assertEqual(len(session_recordings), 2) - - (session_recordings, _, _) = self._filter_recordings_by( - { - # only 1 pageview that matches the test_accounts filter - "filter_test_accounts": True, - } - ) - self.assertEqual(len(session_recordings), 1) - - @freeze_time("2021-01-21T20:00:00.000Z") - @snapshot_clickhouse_queries - def test_event_filter_with_two_events_and_multiple_teams(self): - another_team = Team.objects.create(organization=self.organization) - - # two teams, user with the same properties - Person.objects.create(team=self.team, distinct_ids=["user"], properties={"email": "bla"}) - Person.objects.create(team=another_team, distinct_ids=["user"], properties={"email": "bla"}) - - # a recording session with a pageview and a pageleave - self._a_session_with_two_events(self.team, "1") - self._a_session_with_two_events(another_team, "2") - - (session_recordings, _, _) = self._filter_recordings_by( - { - "events": [ - { - "id": "$pageview", - "type": "events", - "order": 0, - "name": "$pageview", - }, - { - "id": "$pageleave", - "type": "events", - "order": 0, - "name": "$pageleave", - }, - ], - } - ) - - self.assertEqual([sr["session_id"] for sr in session_recordings], ["1"]) - - def _a_session_with_two_events(self, team: Team, session_id: str) -> None: - produce_replay_summary( - distinct_id="user", - session_id=session_id, - first_timestamp=self.an_hour_ago, - team_id=team.pk, - ) - self.create_event( - "user", - self.an_hour_ago, - team=team, - event_name="$pageview", - properties={"$session_id": session_id, "$window_id": "1"}, - ) - self.create_event( - "user", - self.an_hour_ago, - team=team, - event_name="$pageleave", - properties={"$session_id": session_id, "$window_id": "1"}, - ) - - @freeze_time("2021-01-21T20:00:00.000Z") - @snapshot_clickhouse_queries - def test_event_filter_with_group_filter(self): - Person.objects.create(team=self.team, distinct_ids=["user"], properties={"email": "bla"}) - session_id = f"test_event_filter_with_group_filter-ONE-{uuid4()}" - different_group_session = f"test_event_filter_with_group_filter-TWO-{uuid4()}" - - produce_replay_summary( - distinct_id="user", - session_id=session_id, - first_timestamp=self.an_hour_ago, - team_id=self.team.pk, - ) - produce_replay_summary( - distinct_id="user", - session_id=different_group_session, - first_timestamp=self.an_hour_ago, - team_id=self.team.pk, - ) - - GroupTypeMapping.objects.create( - team=self.team, project_id=self.team.project_id, group_type="project", group_type_index=0 - ) - create_group( - team_id=self.team.pk, - group_type_index=0, - group_key="project:1", - properties={"name": "project one"}, - ) - - GroupTypeMapping.objects.create( - team=self.team, project_id=self.team.project_id, group_type="organization", group_type_index=1 - ) - create_group( - team_id=self.team.pk, - group_type_index=1, - group_key="org:1", - properties={"name": "org one"}, - ) - - self.create_event( - "user", - self.an_hour_ago, - team=self.team, - event_name="$pageview", - properties={ - "$session_id": session_id, - "$window_id": "1", - "$group_1": "org:1", - }, - ) - self.create_event( - "user", - self.an_hour_ago, - team=self.team, - event_name="$pageview", - properties={ - "$session_id": different_group_session, - "$window_id": "1", - "$group_0": "project:1", - }, - ) - - (session_recordings, _, _) = self._filter_recordings_by( - { - "events": [ - { - "id": "$pageview", - "type": "events", - "order": 0, - "name": "$pageview", - "properties": [ - { - "key": "name", - "value": ["org one"], - "operator": "exact", - "type": "group", - "group_type_index": 1, - } - ], - } - ], - } - ) - - self.assertEqual([sr["session_id"] for sr in session_recordings], [session_id]) - - (session_recordings, _, _) = self._filter_recordings_by( - { - "properties": [ - { - "key": "name", - "value": ["org one"], - "operator": "exact", - "type": "group", - "group_type_index": 1, - } - ], - } - ) - self.assertEqual([sr["session_id"] for sr in session_recordings], [session_id]) - - (session_recordings, _, _) = self._filter_recordings_by( - { - "properties": [ - { - "key": "name", - "value": ["org one"], - "operator": "exact", - "type": "group", - "group_type_index": 2, - } - ], - } - ) - self.assertEqual(session_recordings, []) - - @freeze_time("2021-01-21T20:00:00.000Z") - @snapshot_clickhouse_queries - def test_ordering(self): - session_id_one = f"test_ordering-one" - session_id_two = f"test_ordering-two" - session_id_three = f"test_ordering-three" - - produce_replay_summary( - session_id=session_id_one, - team_id=self.team.id, - mouse_activity_count=50, - first_timestamp=(self.an_hour_ago + relativedelta(seconds=60)), - ) - produce_replay_summary( - session_id=session_id_two, - team_id=self.team.id, - mouse_activity_count=100, - first_timestamp=(self.an_hour_ago), - ) - produce_replay_summary( - session_id=session_id_three, - team_id=self.team.id, - mouse_activity_count=10, - first_timestamp=(self.an_hour_ago + relativedelta(minutes=10)), - ) - - (session_recordings, _, _) = self._filter_recordings_by({"order": "start_time"}) - assert [r["session_id"] for r in session_recordings] == [session_id_three, session_id_one, session_id_two] - - (session_recordings, _, _) = self._filter_recordings_by({"order": "mouse_activity_count"}) - assert [r["session_id"] for r in session_recordings] == [session_id_two, session_id_one, session_id_three] - - @also_test_with_materialized_columns(event_properties=["$host"], verify_no_jsonextract=False) - @freeze_time("2021-01-21T20:00:00.000Z") - @snapshot_clickhouse_queries - def test_top_level_event_host_property_test_account_filter(self): - """ - This is a regression test. See: https://posthoghelp.zendesk.com/agent/tickets/18059 - """ - self.team.test_account_filters = [ - {"key": "$host", "type": "event", "value": "^(localhost|127\\.0\\.0\\.1)($|:)", "operator": "not_regex"}, - ] - self.team.save() - - Person.objects.create(team=self.team, distinct_ids=["user"], properties={"email": "bla"}) - Person.objects.create( - team=self.team, - distinct_ids=["user2"], - properties={"email": "not-the-other-one"}, - ) - - produce_replay_summary( - distinct_id="user", - session_id="1", - first_timestamp=self.an_hour_ago, - team_id=self.team.id, - ensure_analytics_event_in_session=False, - ) - # the session needs to have multiple matching or not matching events - for _ in range(10): - self.create_event( - "user", - self.an_hour_ago, - properties={ - "$session_id": "1", - "$window_id": "1", - "$host": "localhost", - }, - ) - - produce_replay_summary( - distinct_id="user", - session_id="1", - first_timestamp=self.an_hour_ago + relativedelta(seconds=30), - team_id=self.team.id, - click_count=10, - ensure_analytics_event_in_session=False, - ) - - for _ in range(10): - self.create_event( - "user2", - self.an_hour_ago, - properties={ - "$session_id": "2", - "$window_id": "1", - "$host": "example.com", - }, - ) - produce_replay_summary( - distinct_id="user2", - session_id="2", - first_timestamp=self.an_hour_ago, - team_id=self.team.id, - click_count=10, - ensure_analytics_event_in_session=False, - ) - - # there are 2 pageviews - (session_recordings, _, _) = self._filter_recordings_by( - { - # pageview that matches the hogql test_accounts filter - "events": [ - { - "id": "$pageview", - "type": "events", - "order": 0, - "name": "$pageview", - } - ], - "filter_test_accounts": False, - } - ) - self.assertEqual(len(session_recordings), 2) - - (session_recordings, _, _) = self._filter_recordings_by( - { - # only 1 pageview that matches the test_accounts filter - "filter_test_accounts": True, - } - ) - assert session_recordings == [ - { - "active_seconds": 0.0, - "activity_score": 0.28, - "click_count": 10, # in the bug this value was 10 X number of events in the session - "console_error_count": 0, - "console_log_count": 0, - "console_warn_count": 0, - "distinct_id": "user2", - "duration": 3600, - "end_time": ANY, - "first_url": "https://not-provided-by-test.com", - "inactive_seconds": 3600.0, - "keypress_count": 0, - "mouse_activity_count": 0, - "session_id": "2", - "start_time": ANY, - "team_id": self.team.id, - "ongoing": 1, - } - ] diff --git a/posthog/session_recordings/queries/test/test_session_recording_properties.py b/posthog/session_recordings/queries/test/test_session_recording_properties.py deleted file mode 100644 index 7972eb742abb0..0000000000000 --- a/posthog/session_recordings/queries/test/test_session_recording_properties.py +++ /dev/null @@ -1,104 +0,0 @@ -from dateutil.relativedelta import relativedelta -from django.utils.timezone import now -from freezegun.api import freeze_time - -from posthog.models import Person -from posthog.models.filters.session_recordings_filter import SessionRecordingsFilter -from posthog.session_recordings.queries.session_recording_properties import ( - SessionRecordingProperties, -) -from posthog.session_recordings.queries.test.session_replay_sql import ( - produce_replay_summary, -) -from posthog.test.base import ( - BaseTest, - ClickhouseTestMixin, - _create_event, - snapshot_clickhouse_queries, -) - - -class TestSessionRecordingProperties(BaseTest, ClickhouseTestMixin): - def create_event( - self, - distinct_id, - timestamp, - team=None, - event_name="$pageview", - properties=None, - ): - if properties is None: - properties = {"$os": "Windows 95", "$current_url": "aloha.com/2"} - if team is None: - team = self.team - _create_event( - team=team, - event=event_name, - timestamp=timestamp, - distinct_id=distinct_id, - properties=properties, - ) - - @property - def base_time(self): - return now() - relativedelta(hours=1) - - @freeze_time("2021-01-21T20:00:00.000Z") - @snapshot_clickhouse_queries - def test_properties_list(self): - Person.objects.create(team=self.team, distinct_ids=["user"], properties={"email": "bla"}) - produce_replay_summary( - team_id=self.team.id, - session_id="1", - distinct_id="user", - first_timestamp=self.base_time, - last_timestamp=self.base_time, - ) - produce_replay_summary( - team_id=self.team.id, - session_id="2", - distinct_id="user", - first_timestamp=self.base_time, - last_timestamp=self.base_time, - ) - - event_props = { - "$session_id": "1", - "$window_id": "1", - "should_not_be_included": "1", - "$browser": "Chrome", - "$os": "Mac OS X", - "$device_type": "Desktop", - "$current_url": "https://blah.com/blah", - "$host": "blah.com", - "$pathname": "/blah", - "$geoip_country_code": "KR", - } - self.create_event( - "user", - self.base_time, - properties=event_props, - ) - self.create_event( - "user", - self.base_time, - properties=event_props, - ) - - filter = SessionRecordingsFilter(team=self.team, data={"no_filter": None}) - session_recording_properties_instance = SessionRecordingProperties( - filter=filter, team=self.team, session_ids=["1"] - ) - session_recordings_properties = session_recording_properties_instance.run() - self.assertEqual(len(session_recordings_properties), 1) - self.assertEqual(session_recordings_properties[0]["properties"]["$browser"], "Chrome") - self.assertEqual(session_recordings_properties[0]["properties"]["$os"], "Mac OS X") - self.assertEqual(session_recordings_properties[0]["properties"]["$device_type"], "Desktop") - self.assertEqual( - session_recordings_properties[0]["properties"]["$current_url"], - "https://blah.com/blah", - ) - self.assertEqual(session_recordings_properties[0]["properties"]["$host"], "blah.com") - self.assertEqual(session_recordings_properties[0]["properties"]["$pathname"], "/blah") - self.assertEqual(session_recordings_properties[0]["properties"]["$geoip_country_code"], "KR") - self.assertNotIn("should_not_be_included", session_recordings_properties[0]["properties"]) diff --git a/posthog/session_recordings/session_recording_api.py b/posthog/session_recordings/session_recording_api.py index ae58b14213ed2..979a4b1983967 100644 --- a/posthog/session_recordings/session_recording_api.py +++ b/posthog/session_recordings/session_recording_api.py @@ -28,10 +28,8 @@ from posthog.api.utils import action, safe_clickhouse_string from posthog.auth import PersonalAPIKeyAuthentication, SharingAccessTokenAuthentication from posthog.cloud_utils import is_cloud -from posthog.constants import SESSION_RECORDINGS_FILTER_IDS from posthog.event_usage import report_user_action from posthog.models import Team, User -from posthog.models.filters.session_recordings_filter import SessionRecordingsFilter from posthog.models.person.person import PersonDistinctId from posthog.rate_limit import ( ClickHouseBurstRateThrottle, @@ -43,14 +41,7 @@ from posthog.session_recordings.models.session_recording_event import ( SessionRecordingViewed, ) -from posthog.session_recordings.queries.session_recording_list_from_filters import ( - ReplayFiltersEventsSubQuery, - SessionRecordingListFromFilters, -) from posthog.session_recordings.queries.session_recording_list_from_query import SessionRecordingListFromQuery -from posthog.session_recordings.queries.session_recording_properties import ( - SessionRecordingProperties, -) from posthog.session_recordings.queries.session_replay_events import SessionReplayEvents from posthog.session_recordings.realtime_snapshots import ( get_realtime_snapshots, @@ -337,19 +328,15 @@ def safely_get_object(self, queryset) -> SessionRecording: return recording def list(self, request: request.Request, *args: Any, **kwargs: Any) -> Response: - use_query_type = (request.GET.get("as_query", "False")).lower() == "true" - if use_query_type: - data_dict = query_as_params_to_dict(request.GET.dict()) - query = RecordingsQuery.model_validate(data_dict) - # a little duplication for now - self._maybe_report_recording_list_filters_changed(request, team=self.team) - return list_recordings_response( - list_recordings_from_query(query, request, context=self.get_serializer_context()) - ) - else: - filter = SessionRecordingsFilter(request=request, team=self.team) - self._maybe_report_recording_list_filters_changed(request, team=self.team) - return list_recordings_response(list_recordings(filter, request, context=self.get_serializer_context())) + data_dict = query_as_params_to_dict(request.GET.dict()) + # we used to send `version` and it's not part of query, so we pop to make sure + data_dict.pop("version", None) + query = RecordingsQuery.model_validate(data_dict) + + self._maybe_report_recording_list_filters_changed(request, team=self.team) + return list_recordings_response( + list_recordings_from_query(query, request, context=self.get_serializer_context()) + ) @extend_schema( exclude=True, @@ -361,48 +348,26 @@ def list(self, request: request.Request, *args: Any, **kwargs: Any) -> Response: ) @action(methods=["GET"], detail=False) def matching_events(self, request: request.Request, *args: Any, **kwargs: Any) -> JsonResponse: - use_query_type = (request.GET.get("as_query", "False")).lower() == "true" - - if use_query_type: - data_dict = query_as_params_to_dict(request.GET.dict()) - query = RecordingsQuery.model_validate(data_dict) - - # a little duplication for now - if not query.session_ids or len(query.session_ids) != 1: - raise exceptions.ValidationError( - "Must specify exactly one session_id", - ) + data_dict = query_as_params_to_dict(request.GET.dict()) + query = RecordingsQuery.model_validate(data_dict) - if not query.events and not query.actions: - raise exceptions.ValidationError( - "Must specify at least one event or action filter", - ) - - distinct_id = str(cast(User, request.user).distinct_id) - modifiers = safely_read_modifiers_overrides(distinct_id, self.team) - results, _, timings = ( - posthog.session_recordings.queries.session_recording_list_from_query.ReplayFiltersEventsSubQuery( - query=query, team=self.team, hogql_query_modifiers=modifiers - ).get_event_ids_for_session() + if not query.session_ids or len(query.session_ids) != 1: + raise exceptions.ValidationError( + "Must specify exactly one session_id", ) - else: - filter = SessionRecordingsFilter(request=request, team=self.team) - if not filter.session_ids or len(filter.session_ids) != 1: - raise exceptions.ValidationError( - "Must specify exactly one session_id", - ) - - if not filter.events and not filter.actions: - raise exceptions.ValidationError( - "Must specify at least one event or action filter", - ) + if not query.events and not query.actions: + raise exceptions.ValidationError( + "Must specify at least one event or action filter", + ) - distinct_id = str(cast(User, request.user).distinct_id) - modifiers = safely_read_modifiers_overrides(distinct_id, self.team) - results, _, timings = ReplayFiltersEventsSubQuery( - filter=filter, team=self.team, hogql_query_modifiers=modifiers + distinct_id = str(cast(User, request.user).distinct_id) + modifiers = safely_read_modifiers_overrides(distinct_id, self.team) + results, _, timings = ( + posthog.session_recordings.queries.session_recording_list_from_query.ReplayFiltersEventsSubQuery( + query=query, team=self.team, hogql_query_modifiers=modifiers ).get_event_ids_for_session() + ) response = JsonResponse(data={"results": results}) @@ -666,32 +631,6 @@ def _distinct_id_from_request(request): else: return "anonymous" - # Returns properties given a list of session recording ids - @extend_schema(exclude=True) - @action(methods=["GET"], detail=False) - def properties(self, request: request.Request, **kwargs): - filter = SessionRecordingsFilter(request=request, team=self.team) - session_ids = [ - recording_id for recording_id in json.loads(self.request.GET.get("session_ids", "[]")) if recording_id - ] - for session_id in session_ids: - if not isinstance(session_id, str): - raise exceptions.ValidationError(f"Invalid session_id: {session_id} - not a string") - session_recordings_properties = SessionRecordingProperties( - team=self.team, filter=filter, session_ids=session_ids - ).run() - - if not request.user.is_authenticated: # for mypy - raise exceptions.NotAuthenticated() - - session_recording_serializer = SessionRecordingPropertiesSerializer( - data=session_recordings_properties, many=True - ) - - session_recording_serializer.is_valid(raise_exception=True) - - return Response({"results": session_recording_serializer.data}) - @extend_schema(exclude=True) @action(methods=["POST"], detail=True) def summarize(self, request: request.Request, **kwargs): @@ -945,106 +884,6 @@ def list_recordings_from_query( ) -def list_recordings( - filter: SessionRecordingsFilter, request: request.Request, context: dict[str, Any] -) -> tuple[dict, dict]: - """ - As we can store recordings in S3 or in Clickhouse we need to do a few things here - - A. If filter.session_ids is specified: - 1. We first try to load them directly from Postgres if they have been persisted to S3 (they might have fell out of CH) - 2. Any that couldn't be found are then loaded from Clickhouse - B. Otherwise we just load all values from Clickhouse - 2. Once loaded we convert them to SessionRecording objects in case we have any other persisted data - """ - - all_session_ids = filter.session_ids - - recordings: list[SessionRecording] = [] - more_recordings_available = False - team = context["get_team"]() - hogql_timings: list[QueryTiming] | None = None - - timer = ServerTimingsGathered() - - if all_session_ids: - with timer("load_persisted_recordings"): - # If we specify the session ids (like from pinned recordings) we can optimise by only going to Postgres - sorted_session_ids = sorted(all_session_ids) - - persisted_recordings_queryset = SessionRecording.objects.filter( - team=team, session_id__in=sorted_session_ids - ).exclude(object_storage_path=None) - - persisted_recordings = persisted_recordings_queryset.all() - - recordings = recordings + list(persisted_recordings) - - remaining_session_ids = list(set(all_session_ids) - {x.session_id for x in persisted_recordings}) - filter = filter.shallow_clone({SESSION_RECORDINGS_FILTER_IDS: remaining_session_ids}) - - if (all_session_ids and filter.session_ids) or not all_session_ids: - distinct_id = str(cast(User, request.user).distinct_id) - modifiers = safely_read_modifiers_overrides(distinct_id, team) - - with timer("load_recordings_from_hogql"): - (ch_session_recordings, more_recordings_available, hogql_timings) = SessionRecordingListFromFilters( - filter=filter, team=team, hogql_query_modifiers=modifiers - ).run() - - with timer("build_recordings"): - recordings_from_clickhouse = SessionRecording.get_or_build_from_clickhouse(team, ch_session_recordings) - recordings = recordings + recordings_from_clickhouse - - recordings = [x for x in recordings if not x.deleted] - - # If we have specified session_ids we need to sort them by the order they were specified - if all_session_ids: - recordings = sorted( - recordings, - key=lambda x: cast(list[str], all_session_ids).index(x.session_id), - ) - - if not request.user.is_authenticated: # for mypy - raise exceptions.NotAuthenticated() - - # Update the viewed status for all loaded recordings - with timer("load_viewed_recordings"): - viewed_session_recordings = set( - SessionRecordingViewed.objects.filter(team=team, user=request.user).values_list("session_id", flat=True) - ) - - with timer("load_persons"): - # Get the related persons for all the recordings - distinct_ids = sorted([x.distinct_id for x in recordings if x.distinct_id]) - person_distinct_ids = PersonDistinctId.objects.filter(distinct_id__in=distinct_ids, team=team).select_related( - "person" - ) - - with timer("process_persons"): - distinct_id_to_person = {} - for person_distinct_id in person_distinct_ids: - person_distinct_id.person._distinct_ids = [ - person_distinct_id.distinct_id - ] # Stop the person from loading all distinct ids - distinct_id_to_person[person_distinct_id.distinct_id] = person_distinct_id.person - - for recording in recordings: - recording.viewed = recording.session_id in viewed_session_recordings - person = distinct_id_to_person.get(recording.distinct_id) if recording.distinct_id else None - if person: - recording.person = person - - session_recording_serializer = SessionRecordingSerializer(recordings, context=context, many=True) - results = session_recording_serializer.data - - all_timings = _generate_timings(hogql_timings, timer) - return ( - {"results": results, "has_next": more_recordings_available, "version": 3}, - all_timings, - ) - - def safely_read_modifiers_overrides(distinct_id: str, team: Team) -> HogQLQueryModifiers: modifiers = HogQLQueryModifiers() diff --git a/posthog/session_recordings/test/test_session_recordings.py b/posthog/session_recordings/test/test_session_recordings.py index 5a474ade819be..561d30ea83b02 100644 --- a/posthog/session_recordings/test/test_session_recordings.py +++ b/posthog/session_recordings/test/test_session_recordings.py @@ -14,10 +14,7 @@ from rest_framework import status from posthog.api.test.test_team import create_team -from posthog.constants import SESSION_RECORDINGS_FILTER_IDS from posthog.models import Organization, Person, SessionRecording -from posthog.models.filters.session_recordings_filter import SessionRecordingsFilter -from posthog.models.property import Property from posthog.models.team import Team from posthog.schema import RecordingsQuery, LogEntryPropertyFilter from posthog.session_recordings.models.session_recording_event import ( @@ -90,7 +87,8 @@ def test_get_session_recordings(self): session_id_two = f"test_get_session_recordings-2" self.produce_replay_summary("user2", session_id_two, base_time + relativedelta(seconds=20)) - response = self.client.get(f"/api/projects/{self.team.id}/session_recordings") + # include `as_query` since we don't want to break while deploying the code that no longer needs it + response = self.client.get(f"/api/projects/{self.team.id}/session_recordings?as_query=true") self.assertEqual(response.status_code, status.HTTP_200_OK) response_data = response.json() @@ -171,46 +169,8 @@ def test_can_list_recordings_even_when_the_person_has_multiple_distinct_ids(self assert results_[1]["distinct_id"] in twelve_distinct_ids @patch("posthoganalytics.capture") - @patch("posthog.session_recordings.session_recording_api.SessionRecordingListFromFilters") - @patch("posthog.session_recordings.session_recording_api.list_recordings_from_query") - def test_console_log_filters_are_correctly_passed_to_listing_when_filters_are_used( - self, mock_query_lister, mock_summary_lister, mock_capture - ): - mock_summary_lister.return_value.run.return_value = ([], False) - mock_query_lister.return_value.run.return_value = ([], False) - - params_string = urlencode( - { - "console_log_filters": '[{"key": "console_log_level", "value": ["warn", "error"], "operator": "exact", "type": "log_entry"}]', - "user_modified_filters": '{"my_filter": "something"}', - "as_query": False, - } - ) - self.client.get(f"/api/projects/{self.team.id}/session_recordings?{params_string}") - - assert len(mock_summary_lister.call_args_list) == 1 - assert len(mock_query_lister.call_args_list) == 0 - filter_passed_to_mock: SessionRecordingsFilter = mock_summary_lister.call_args_list[0].kwargs["filter"] - console_filter = cast(Property, filter_passed_to_mock.console_log_filters.values[0]) - assert console_filter.value == ["warn", "error"] - assert mock_capture.call_args_list[0] == call( - self.user.distinct_id, - "recording list filters changed", - properties={ - "$current_url": ANY, - "$session_id": ANY, - "partial_filter_chosen_my_filter": "something", - }, - groups=ANY, - ) - - @patch("posthoganalytics.capture") - @patch("posthog.session_recordings.session_recording_api.SessionRecordingListFromFilters") @patch("posthog.session_recordings.session_recording_api.list_recordings_from_query") - def test_console_log_filters_are_correctly_passed_to_listing_when_query_is_used( - self, mock_query_lister, mock_summary_lister, mock_capture - ): - mock_summary_lister.return_value.run.return_value = ([], False) + def test_console_log_filters_are_correctly_passed_to_listing(self, mock_query_lister, mock_capture): mock_query_lister.return_value = ([], False) params_string = urlencode( @@ -222,7 +182,6 @@ def test_console_log_filters_are_correctly_passed_to_listing_when_query_is_used( ) self.client.get(f"/api/projects/{self.team.id}/session_recordings?{params_string}") - assert len(mock_summary_lister.call_args_list) == 0 assert len(mock_query_lister.call_args_list) == 1 query_passed_to_mock: RecordingsQuery = mock_query_lister.call_args_list[0][0][0] maybe_the_filter = ( @@ -1104,7 +1063,7 @@ def test_get_via_sharing_token(self, mock_copy_objects: MagicMock) -> None: def test_get_matching_events_for_must_not_send_multiple_session_ids(self) -> None: query_params = [ - f'{SESSION_RECORDINGS_FILTER_IDS}=["{str(uuid.uuid4())}", "{str(uuid.uuid4())}"]', + f'session_ids=["{str(uuid.uuid4())}", "{str(uuid.uuid4())}"]', ] response = self.client.get( f"/api/projects/{self.team.id}/session_recordings/matching_events?{'&'.join(query_params)}" @@ -1129,7 +1088,7 @@ def test_get_matching_events_for_must_send_a_single_session_id_filter(self) -> N def test_get_matching_events_for_must_send_at_least_an_event_filter(self) -> None: query_params = [ - f'{SESSION_RECORDINGS_FILTER_IDS}=["{str(uuid.uuid4())}"]', + f'session_ids=["{str(uuid.uuid4())}"]', ] response = self.client.get( @@ -1146,7 +1105,7 @@ def test_get_matching_events_for_must_send_at_least_an_event_filter(self) -> Non def test_get_matching_events_for_unknown_session(self) -> None: session_id = str(uuid.uuid4()) query_params = [ - f'{SESSION_RECORDINGS_FILTER_IDS}=["{session_id}"]', + f'session_ids=["{session_id}"]', 'events=[{"id": "$pageview", "type": "events", "order": 0, "name": "$pageview"}]', ] response = self.client.get( @@ -1183,7 +1142,7 @@ def test_get_matching_events_with_query(self) -> None: time.sleep(1) query_params = [ - f'{SESSION_RECORDINGS_FILTER_IDS}=["{session_id}"]', + f'session_ids=["{session_id}"]', 'events=[{"id": "$pageview", "type": "events", "order": 0, "name": "$pageview"}]', ] @@ -1222,12 +1181,13 @@ def test_get_matching_events(self) -> None: time.sleep(1) query_params = [ - f'{SESSION_RECORDINGS_FILTER_IDS}=["{session_id}"]', + f'session_ids=["{session_id}"]', 'events=[{"id": "$pageview", "type": "events", "order": 0, "name": "$pageview"}]', ] response = self.client.get( - f"/api/projects/{self.team.id}/session_recordings/matching_events?{'&'.join(query_params)}" + # include `as_query` since we don't want to break while deploying the code that no longer needs it + f"/api/projects/{self.team.id}/session_recordings/matching_events?{'&'.join(query_params)}&as_query=true" ) assert response.status_code == status.HTTP_200_OK diff --git a/posthog/test/base.py b/posthog/test/base.py index a8a06dcf3e480..dc0760f4a3b5b 100644 --- a/posthog/test/base.py +++ b/posthog/test/base.py @@ -157,7 +157,7 @@ def clean_varying_query_parts(query, replace_all_numbers): #### Cohort replacements # replace cohort id lists in queries too query = re.sub( - r"in\(([^,]+\.?cohort_id), \[(\d+(, ?\d+)*)]\)", + r"in\(([^,]*cohort_id),\s*\[(\d+(?:,\s*\d+)*)]\)", r"in(\1, [1, 2, 3, 4, 5 /* ... */])", query, ) From 9cbcf09f7032498874614086f2a0bd4c23bbe815 Mon Sep 17 00:00:00 2001 From: Anders <754494+andehen@users.noreply.github.com> Date: Tue, 17 Dec 2024 12:29:00 +0100 Subject: [PATCH 8/9] chore(experiments): hogql migration cleanup - secondary metrics (#26962) --- .../ExperimentView/SecondaryMetricsTable.tsx | 44 ++++--------------- 1 file changed, 8 insertions(+), 36 deletions(-) diff --git a/frontend/src/scenes/experiments/ExperimentView/SecondaryMetricsTable.tsx b/frontend/src/scenes/experiments/ExperimentView/SecondaryMetricsTable.tsx index 52a189c4c324a..5474962ec738b 100644 --- a/frontend/src/scenes/experiments/ExperimentView/SecondaryMetricsTable.tsx +++ b/frontend/src/scenes/experiments/ExperimentView/SecondaryMetricsTable.tsx @@ -2,19 +2,13 @@ import { IconInfo, IconPencil, IconPlus } from '@posthog/icons' import { LemonButton, LemonTable, LemonTableColumns, Tooltip } from '@posthog/lemon-ui' import { useActions, useValues } from 'kea' import { EntityFilterInfo } from 'lib/components/EntityFilterInfo' -import { FEATURE_FLAGS } from 'lib/constants' import { IconAreaChart } from 'lib/lemon-ui/icons' import { capitalizeFirstLetter, humanFriendlyNumber } from 'lib/utils' import { useState } from 'react' import { Experiment, InsightType } from '~/types' -import { - experimentLogic, - getDefaultFilters, - getDefaultFunnelsMetric, - TabularSecondaryMetricResults, -} from '../experimentLogic' +import { experimentLogic, getDefaultFunnelsMetric, TabularSecondaryMetricResults } from '../experimentLogic' import { SecondaryMetricChartModal } from '../Metrics/SecondaryMetricChartModal' import { SecondaryMetricModal } from '../Metrics/SecondaryMetricModal' import { VariantTag } from './components' @@ -39,7 +33,6 @@ export function SecondaryMetricsTable({ experimentId }: { experimentId: Experime credibleIntervalForVariant, experimentMathAggregationForTrends, getHighestProbabilityVariant, - featureFlags, } = useValues(experimentLogic({ experimentId })) const { loadExperiment } = useActions(experimentLogic({ experimentId })) @@ -64,13 +57,7 @@ export function SecondaryMetricsTable({ experimentId }: { experimentId: Experime setModalMetricIdx(null) } - // :FLAG: CLEAN UP AFTER MIGRATION - let metrics - if (featureFlags[FEATURE_FLAGS.EXPERIMENTS_HOGQL]) { - metrics = experiment.metrics_secondary - } else { - metrics = experiment.secondary_metrics - } + const metrics = experiment.metrics_secondary const columns: LemonTableColumns = [ { @@ -339,7 +326,7 @@ const AddSecondaryMetricButton = ({ metrics: any openEditModal: (metricIdx: number) => void }): JSX.Element => { - const { experiment, featureFlags } = useValues(experimentLogic({ experimentId })) + const { experiment } = useValues(experimentLogic({ experimentId })) const { setExperiment } = useActions(experimentLogic({ experimentId })) return ( { - // :FLAG: CLEAN UP AFTER MIGRATION - if (featureFlags[FEATURE_FLAGS.EXPERIMENTS_HOGQL]) { - const newMetricsSecondary = [...experiment.metrics_secondary, getDefaultFunnelsMetric()] - setExperiment({ - metrics_secondary: newMetricsSecondary, - }) - openEditModal(newMetricsSecondary.length - 1) - } else { - const newSecondaryMetrics = [ - ...experiment.secondary_metrics, - { - name: '', - filters: getDefaultFilters(InsightType.FUNNELS, undefined), - }, - ] - setExperiment({ - secondary_metrics: newSecondaryMetrics, - }) - openEditModal(newSecondaryMetrics.length - 1) - } + const newMetricsSecondary = [...experiment.metrics_secondary, getDefaultFunnelsMetric()] + setExperiment({ + metrics_secondary: newMetricsSecondary, + }) + openEditModal(newMetricsSecondary.length - 1) }} disabledReason={ metrics.length >= MAX_SECONDARY_METRICS From 22a9101552d96de0b1742ae8bfc4dd7e1f5d2e20 Mon Sep 17 00:00:00 2001 From: Daniel Bachhuber Date: Tue, 17 Dec 2024 04:43:01 -0800 Subject: [PATCH 9/9] chore(experiments): Add assertions for property math operations (#26952) Co-authored-by: Juraj Majerik --- .../test_experiment_trends_query_runner.py | 215 ++++++++++++++++++ 1 file changed, 215 insertions(+) diff --git a/posthog/hogql_queries/experiments/test/test_experiment_trends_query_runner.py b/posthog/hogql_queries/experiments/test/test_experiment_trends_query_runner.py index bba1366c33418..9ed6636bc2324 100644 --- a/posthog/hogql_queries/experiments/test/test_experiment_trends_query_runner.py +++ b/posthog/hogql_queries/experiments/test/test_experiment_trends_query_runner.py @@ -1522,6 +1522,221 @@ def test_query_runner_standard_flow_v2_stats(self): self.assertEqual(test_variant.count, 5.0) self.assertEqual(test_variant.exposure, 1.0) + @freeze_time("2020-01-01T12:00:00Z") + def test_query_runner_property_math_sum(self): + self._test_query_runner_property_math( + math="sum", + expected_control={ + "count": 10, + "absolute_exposure": 5, + "data": [0.0, 0.0, 1.0, 3.0, 6.0, 10.0, 10.0, 10.0, 10.0, 10.0, 10.0, 10.0, 10.0, 10.0, 10.0], + }, + expected_test={ + "count": 90, + "absolute_exposure": 10, + "data": [0.0, 0.0, 2.0, 6.0, 12.0, 20.0, 30.0, 42.0, 56.0, 72.0, 90.0, 90.0, 90.0, 90.0, 90.0], + }, + ) + + @freeze_time("2020-01-01T12:00:00Z") + def test_query_runner_property_math_avg(self): + self._test_query_runner_property_math( + math="avg", + expected_control={ + "count": 10, + "absolute_exposure": 5, + "data": [0.0, 0.0, 1.0, 3.0, 6.0, 10.0, 10.0, 10.0, 10.0, 10.0, 10.0, 10.0, 10.0, 10.0, 10.0], + }, + expected_test={ + "count": 90, + "absolute_exposure": 10, + "data": [0.0, 0.0, 2.0, 6.0, 12.0, 20.0, 30.0, 42.0, 56.0, 72.0, 90.0, 90.0, 90.0, 90.0, 90.0], + }, + ) + + @freeze_time("2020-01-01T12:00:00Z") + def test_query_runner_property_math_min(self): + self._test_query_runner_property_math( + math="min", + expected_control={ + "count": 5, + "absolute_exposure": 5, + "data": [0.0, 1.0, 2.0, 3.0, 4.0, 5.0, 5.0, 5.0, 5.0, 5.0, 5.0, 5.0, 5.0, 5.0, 5.0], + }, + expected_test={ + "count": 10, + "absolute_exposure": 10, + "data": [0.0, 1.0, 2.0, 3.0, 4.0, 5.0, 6.0, 7.0, 8.0, 9.0, 10.0, 10.0, 10.0, 10.0, 10.0], + }, + ) + + @freeze_time("2020-01-01T12:00:00Z") + def test_query_runner_property_math_max(self): + self._test_query_runner_property_math( + math="max", + expected_control={ + "count": 5, + "absolute_exposure": 5, + "data": [0.0, 1.0, 2.0, 3.0, 4.0, 5.0, 5.0, 5.0, 5.0, 5.0, 5.0, 5.0, 5.0, 5.0, 5.0], + }, + expected_test={ + "count": 10, + "absolute_exposure": 10, + "data": [0.0, 1.0, 2.0, 3.0, 4.0, 5.0, 6.0, 7.0, 8.0, 9.0, 10.0, 10.0, 10.0, 10.0, 10.0], + }, + ) + + @freeze_time("2020-01-01T12:00:00Z") + def test_query_runner_property_math_median(self): + self._test_query_runner_property_math( + math="median", + expected_control={ + "count": 5, + "absolute_exposure": 5, + "data": [0.0, 1.0, 2.0, 3.0, 4.0, 5.0, 5.0, 5.0, 5.0, 5.0, 5.0, 5.0, 5.0, 5.0, 5.0], + }, + expected_test={ + "count": 10, + "absolute_exposure": 10, + "data": [0.0, 1.0, 2.0, 3.0, 4.0, 5.0, 6.0, 7.0, 8.0, 9.0, 10.0, 10.0, 10.0, 10.0, 10.0], + }, + ) + + @freeze_time("2020-01-01T12:00:00Z") + def test_query_runner_property_math_p90(self): + self._test_query_runner_property_math( + math="p90", + expected_control={ + "count": 5, + "absolute_exposure": 5, + "data": [0.0, 1.0, 2.0, 3.0, 4.0, 5.0, 5.0, 5.0, 5.0, 5.0, 5.0, 5.0, 5.0, 5.0, 5.0], + }, + expected_test={ + "count": 10, + "absolute_exposure": 10, + "data": [0.0, 1.0, 2.0, 3.0, 4.0, 5.0, 6.0, 7.0, 8.0, 9.0, 10.0, 10.0, 10.0, 10.0, 10.0], + }, + ) + + @freeze_time("2020-01-01T12:00:00Z") + def test_query_runner_property_math_p95(self): + self._test_query_runner_property_math( + math="p95", + expected_control={ + "count": 5, + "absolute_exposure": 5, + "data": [0.0, 1.0, 2.0, 3.0, 4.0, 5.0, 5.0, 5.0, 5.0, 5.0, 5.0, 5.0, 5.0, 5.0, 5.0], + }, + expected_test={ + "count": 10, + "absolute_exposure": 10, + "data": [0.0, 1.0, 2.0, 3.0, 4.0, 5.0, 6.0, 7.0, 8.0, 9.0, 10.0, 10.0, 10.0, 10.0, 10.0], + }, + ) + + @freeze_time("2020-01-01T12:00:00Z") + def test_query_runner_property_math_p99(self): + self._test_query_runner_property_math( + math="p99", + expected_control={ + "count": 5, + "absolute_exposure": 5, + "data": [0.0, 1.0, 2.0, 3.0, 4.0, 5.0, 5.0, 5.0, 5.0, 5.0, 5.0, 5.0, 5.0, 5.0, 5.0], + }, + expected_test={ + "count": 10, + "absolute_exposure": 10, + "data": [0.0, 1.0, 2.0, 3.0, 4.0, 5.0, 6.0, 7.0, 8.0, 9.0, 10.0, 10.0, 10.0, 10.0, 10.0], + }, + ) + + def _test_query_runner_property_math(self, math, expected_control, expected_test): + feature_flag = self.create_feature_flag() + experiment = self.create_experiment(feature_flag=feature_flag, start_date=datetime(2020, 1, 1)) + + feature_flag_property = f"$feature/{feature_flag.key}" + + # control values are 0, 1, 2, 3, 4 + # test values are 0, 2, 4, 6, 8, 10, 12, 14, 16, 18 + + # Populate metric + exposure events + for variant, count in [("control", 5), ("test", 10)]: + for i in range(count): + _create_event( + team=self.team, + event="$feature_flag_called", + distinct_id=f"user_{variant}_{i}", + properties={ + "$feature_flag_response": variant, + feature_flag_property: variant, + "$feature_flag": feature_flag.key, + }, + timestamp=datetime(2020, 1, i + 1), + ) + _create_event( + team=self.team, + event="purchase", + distinct_id=f"user_{variant}_{i}", + properties={ + feature_flag_property: variant, + "amount": i * (1 if variant == "control" else 2), + }, + timestamp=datetime(2020, 1, i + 2), + ) + + count_query = TrendsQuery( + series=[ + EventsNode( + event="purchase", + math=math, + math_property="amount", + math_property_type="event_properties", + ) + ] + ) + exposure_query = TrendsQuery(series=[EventsNode(event="$feature_flag_called")]) + experiment_query = ExperimentTrendsQuery( + experiment_id=experiment.id, + kind="ExperimentTrendsQuery", + count_query=count_query, + exposure_query=exposure_query, + ) + + experiment.metrics = [{"type": "primary", "query": experiment_query.model_dump()}] + experiment.save() + + query_runner = ExperimentTrendsQueryRunner( + query=ExperimentTrendsQuery(**experiment.metrics[0]["query"]), team=self.team + ) + + flush_persons_and_events() + + result = query_runner.calculate() + + trend_result = cast(ExperimentTrendsQueryResponse, result) + + self.assertEqual(len(result.variants), 2) + + control_result = next(variant for variant in trend_result.variants if variant.key == "control") + test_result = next(variant for variant in trend_result.variants if variant.key == "test") + + control_insight = next(variant for variant in trend_result.insight if variant["breakdown_value"] == "control") + test_insight = next(variant for variant in trend_result.insight if variant["breakdown_value"] == "test") + + self.assertEqual(control_result.count, expected_control["count"]) + self.assertEqual(test_result.count, expected_test["count"]) + self.assertEqual(control_result.absolute_exposure, expected_control["absolute_exposure"]) + self.assertEqual(test_result.absolute_exposure, expected_test["absolute_exposure"]) + + self.assertEqual( + control_insight["data"], + expected_control["data"], + ) + self.assertEqual( + test_insight["data"], + expected_test["data"], + ) + @freeze_time("2020-01-01T12:00:00Z") def test_validate_event_variants_no_events(self): feature_flag = self.create_feature_flag()