diff --git a/frontend/src/lib/utils.tsx b/frontend/src/lib/utils.tsx
index f512aa8e5dd85d..15b90eeb5473a5 100644
--- a/frontend/src/lib/utils.tsx
+++ b/frontend/src/lib/utils.tsx
@@ -124,6 +124,10 @@ export function percentage(
maximumFractionDigits: number = 2,
fixedPrecision: boolean = false
): string {
+ if (division === Infinity) {
+ return '∞%'
+ }
+
return division.toLocaleString('en-US', {
style: 'percent',
maximumFractionDigits,
diff --git a/frontend/src/queries/types.ts b/frontend/src/queries/types.ts
index afbae272868160..849ebf6c33e36b 100644
--- a/frontend/src/queries/types.ts
+++ b/frontend/src/queries/types.ts
@@ -49,7 +49,7 @@ export type QueryContextColumnComponent = ComponentType<{
}>
interface QueryContextColumn {
- title?: string
+ title?: JSX.Element | string
renderTitle?: QueryContextColumnTitleComponent
render?: QueryContextColumnComponent
align?: 'left' | 'right' | 'center' // default is left
diff --git a/frontend/src/scenes/session-recordings/playlist/sessionRecordingsListPropertiesLogic.ts b/frontend/src/scenes/session-recordings/playlist/sessionRecordingsListPropertiesLogic.ts
index 578b9a5be65236..3f579f64ff8570 100644
--- a/frontend/src/scenes/session-recordings/playlist/sessionRecordingsListPropertiesLogic.ts
+++ b/frontend/src/scenes/session-recordings/playlist/sessionRecordingsListPropertiesLogic.ts
@@ -2,6 +2,7 @@ import { actions, connect, kea, listeners, path, reducers } from 'kea'
import { loaders } from 'kea-loaders'
import api from 'lib/api'
import { dayjs } from 'lib/dayjs'
+import { CORE_FILTER_DEFINITIONS_BY_GROUP } from 'lib/taxonomy'
import { eventUsageLogic } from 'lib/utils/eventUsageLogic'
import { HogQLQuery, NodeKind } from '~/queries/schema'
@@ -40,7 +41,7 @@ export const sessionRecordingsListPropertiesLogic = kea
}
+ // We don't yet allow deleting individual environments, as we still use `team` fields with `on_delete=CASCADE`
+ // on many models that conceptually are project-level (such as insights or feature flags). That `on_delete=CASCADE`
+ // means currently deleting an environment would also delete resources a user wouldn't expect to disappear.
+ // TODO: Remove once point 15 ("Denormalize models") of https://github.com/PostHog/posthog/issues/13418#issuecomment-2180883524 is resolved
+ return Deletion of individual environments is coming soon.
+
return (
<>
diff --git a/frontend/src/scenes/web-analytics/tiles/WebAnalyticsTile.tsx b/frontend/src/scenes/web-analytics/tiles/WebAnalyticsTile.tsx
index e42d6c2de89cb9..35079d1241faa2 100644
--- a/frontend/src/scenes/web-analytics/tiles/WebAnalyticsTile.tsx
+++ b/frontend/src/scenes/web-analytics/tiles/WebAnalyticsTile.tsx
@@ -1,11 +1,14 @@
-import { IconGear } from '@posthog/icons'
+import { IconGear, IconTrending } from '@posthog/icons'
+import { Tooltip } from '@posthog/lemon-ui'
+import clsx from 'clsx'
import { useActions, useValues } from 'kea'
+import { getColorVar } from 'lib/colors'
import { IntervalFilterStandalone } from 'lib/components/IntervalFilter'
import { ProductIntroduction } from 'lib/components/ProductIntroduction/ProductIntroduction'
-import { IconOpenInNew } from 'lib/lemon-ui/icons'
+import { IconOpenInNew, IconTrendingDown, IconTrendingFlat } from 'lib/lemon-ui/icons'
import { LemonButton } from 'lib/lemon-ui/LemonButton'
import { LemonSwitch } from 'lib/lemon-ui/LemonSwitch'
-import { UnexpectedNeverError } from 'lib/utils'
+import { percentage, UnexpectedNeverError } from 'lib/utils'
import { useCallback, useMemo } from 'react'
import { NewActionButton } from 'scenes/actions/NewActionButton'
import { countryCodeToFlag, countryCodeToName } from 'scenes/insights/views/WorldMap'
@@ -37,15 +40,72 @@ const toUtcOffsetFormat = (value: number): string => {
return `UTC${sign}${integerPart}${formattedMinutes}`
}
-const PercentageCell: QueryContextColumnComponent = ({ value }) => {
- if (typeof value === 'number') {
- return
{`${(value * 100).toFixed(1)}%`}
- }
- return null
-}
+type VariationCellProps = { isPercentage?: boolean; reverseColors?: boolean }
+const VariationCell = (
+ { isPercentage, reverseColors }: VariationCellProps = { isPercentage: false, reverseColors: false }
+): QueryContextColumnComponent => {
+ const formatNumber = (value: number): string =>
+ isPercentage ? `${(value * 100).toFixed(1)}%` : value.toLocaleString()
+
+ return function Cell({ value }) {
+ if (!value) {
+ return null
+ }
+
+ if (!Array.isArray(value)) {
+ return
{String(value)}
+ }
+
+ const [current, previous] = value as [number, number]
+ const pctChangeFromPrevious =
+ previous === 0 && current === 0 // Special case, render as flatline
+ ? 0
+ : current === null
+ ? null
+ : previous === null || previous === 0
+ ? Infinity
+ : current / previous - 1
+
+ const trend =
+ pctChangeFromPrevious === null
+ ? null
+ : pctChangeFromPrevious === 0
+ ? { Icon: IconTrendingFlat, color: getColorVar('muted') }
+ : pctChangeFromPrevious > 0
+ ? {
+ Icon: IconTrending,
+ color: reverseColors ? getColorVar('danger') : getColorVar('success'),
+ }
+ : {
+ Icon: IconTrendingDown,
+ color: reverseColors ? getColorVar('success') : getColorVar('danger'),
+ }
+
+ // If current === previous, say "increased by 0%"
+ const tooltip =
+ pctChangeFromPrevious !== null
+ ? `${current >= previous ? 'Increased' : 'Decreased'} by ${percentage(
+ Math.abs(pctChangeFromPrevious),
+ 0
+ )} since last period (from ${formatNumber(previous)} to ${formatNumber(current)})`
+ : null
-const NumericCell: QueryContextColumnComponent = ({ value }) => {
- return
{typeof value === 'number' ? value.toLocaleString() : String(value)}
+ return (
+
+
+
+ {formatNumber(current)}
+ {trend && (
+ // eslint-disable-next-line react/forbid-dom-props
+
+
+
+ )}
+
+
+
+ )
+ }
}
const BreakdownValueTitle: QueryContextColumnTitleComponent = (props) => {
@@ -227,48 +287,48 @@ export const webAnalyticsDataTableQueryContext: QueryContext = {
render: BreakdownValueCell,
},
bounce_rate: {
- title: 'Bounce Rate',
- render: PercentageCell,
+ title:
Bounce Rate,
+ render: VariationCell({ isPercentage: true, reverseColors: true }),
align: 'right',
},
views: {
- title: 'Views',
- render: NumericCell,
+ title:
Views,
+ render: VariationCell(),
align: 'right',
},
clicks: {
- title: 'Clicks',
- render: NumericCell,
+ title:
Clicks,
+ render: VariationCell(),
align: 'right',
},
visitors: {
- title: 'Visitors',
- render: NumericCell,
+ title:
Visitors,
+ render: VariationCell(),
align: 'right',
},
average_scroll_percentage: {
- title: 'Average Scroll',
- render: PercentageCell,
+ title:
Average Scroll,
+ render: VariationCell({ isPercentage: true }),
align: 'right',
},
scroll_gt80_percentage: {
- title: 'Deep Scroll Rate',
- render: PercentageCell,
+ title:
Deep Scroll Rate,
+ render: VariationCell({ isPercentage: true }),
align: 'right',
},
total_conversions: {
- title: 'Total Conversions',
- render: NumericCell,
+ title:
Total Conversions,
+ render: VariationCell(),
align: 'right',
},
conversion_rate: {
- title: 'Conversion Rate',
- render: PercentageCell,
+ title:
Conversion Rate,
+ render: VariationCell({ isPercentage: true }),
align: 'right',
},
converting_users: {
- title: 'Converting Users',
- render: NumericCell,
+ title:
Converting Users,
+ render: VariationCell(),
align: 'right',
},
action_name: {
diff --git a/posthog/hogql/functions/mapping.py b/posthog/hogql/functions/mapping.py
index 9eb0980b9d9330..d0f4755c67bad2 100644
--- a/posthog/hogql/functions/mapping.py
+++ b/posthog/hogql/functions/mapping.py
@@ -1007,6 +1007,7 @@ def compare_types(arg_types: list[ConstantType], sig_arg_types: tuple[ConstantTy
"argMaxMerge": HogQLFunctionMeta("argMaxMerge", 1, 1, aggregate=True),
"avgState": HogQLFunctionMeta("avgState", 1, 1, aggregate=True),
"avgMerge": HogQLFunctionMeta("avgMerge", 1, 1, aggregate=True),
+ "avgMergeIf": HogQLFunctionMeta("avgMergeIf", 2, 2, aggregate=True),
"avgWeighted": HogQLFunctionMeta("avgWeighted", 2, 2, aggregate=True),
"avgWeightedIf": HogQLFunctionMeta("avgWeightedIf", 3, 3, aggregate=True),
"avgArray": HogQLFunctionMeta("avgArrayOrNull", 1, 1, aggregate=True),
diff --git a/posthog/hogql_queries/web_analytics/stats_table.py b/posthog/hogql_queries/web_analytics/stats_table.py
index 8336a173c04bf0..1633c4389879d7 100644
--- a/posthog/hogql_queries/web_analytics/stats_table.py
+++ b/posthog/hogql_queries/web_analytics/stats_table.py
@@ -45,30 +45,42 @@ def to_query(self) -> ast.SelectQuery:
return self.to_path_scroll_bounce_query()
elif self.query.includeBounceRate:
return self.to_path_bounce_query()
+
if self.query.breakdownBy == WebStatsBreakdown.INITIAL_PAGE:
if self.query.includeBounceRate:
return self.to_entry_bounce_query()
if self._has_session_properties():
- self._to_main_query_with_session_properties()
+ return self._to_main_query_with_session_properties()
+
return self.to_main_query()
def to_main_query(self) -> ast.SelectQuery:
with self.timings.measure("stats_table_query"):
query = parse_select(
"""
+WITH
+ start_timestamp >= {date_from} AND start_timestamp < {date_to} AS current_period_segment,
+ start_timestamp >= {date_from_previous_period} AND start_timestamp < {date_from} AS previous_period_segment
SELECT
{processed_breakdown_value} AS "context.columns.breakdown_value",
- uniq(filtered_person_id) AS "context.columns.visitors",
- sum(filtered_pageview_count) AS "context.columns.views"
+ tuple(
+ uniqIf(filtered_person_id, current_period_segment),
+ uniqIf(filtered_person_id, previous_period_segment)
+ ) AS "context.columns.visitors",
+ tuple(
+ sumIf(filtered_pageview_count, current_period_segment),
+ sumIf(filtered_pageview_count, previous_period_segment)
+ ) AS "context.columns.views"
FROM (
SELECT
any(person_id) AS filtered_person_id,
count() AS filtered_pageview_count,
- {breakdown_value} AS breakdown_value
+ {breakdown_value} AS breakdown_value,
+ min(session.$start_timestamp) as start_timestamp
FROM events
WHERE and(
- timestamp >= {date_from},
+ timestamp >= {date_from_previous_period},
timestamp < {date_to},
events.event == '$pageview',
{all_properties},
@@ -87,10 +99,12 @@ def to_main_query(self) -> ast.SelectQuery:
"processed_breakdown_value": self._processed_breakdown_value(),
"where_breakdown": self.where_breakdown(),
"all_properties": self._all_properties(),
+ "date_from_previous_period": self._date_from_previous_period(),
"date_from": self._date_from(),
"date_to": self._date_to(),
},
)
+
assert isinstance(query, ast.SelectQuery)
if self._include_extra_aggregation_value():
@@ -102,19 +116,29 @@ def _to_main_query_with_session_properties(self) -> ast.SelectQuery:
with self.timings.measure("stats_table_query"):
query = parse_select(
"""
+WITH
+ start_timestamp >= {date_from} AND start_timestamp < {date_to} AS current_period_segment,
+ start_timestamp >= {date_from_previous_period} AND start_timestamp < {date_from} AS previous_period_segment
SELECT
{processed_breakdown_value} AS "context.columns.breakdown_value",
- uniq(filtered_person_id) AS "context.columns.visitors",
- sum(filtered_pageview_count) AS "context.columns.views"
+ tuple(
+ uniqIf(filtered_person_id, current_period_segment),
+ uniqIf(filtered_person_id, previous_period_segment)
+ ) AS "context.columns.visitors",
+ tuple(
+ sumIf(filtered_pageview_count, current_period_segment),
+ sumIf(filtered_pageview_count, previous_period_segment)
+ ) AS "context.columns.views"
FROM (
SELECT
any(person_id) AS filtered_person_id,
count() AS filtered_pageview_count,
{breakdown_value} AS breakdown_value,
- session.session_id AS session_id
+ session.session_id AS session_id,
+ min(session.$start_timestamp) as start_timestamp
FROM events
WHERE and(
- timestamp >= {date_from},
+ timestamp >= {date_from_previous_period},
timestamp < {date_to},
events.event == '$pageview',
{event_properties},
@@ -135,6 +159,7 @@ def _to_main_query_with_session_properties(self) -> ast.SelectQuery:
"where_breakdown": self.where_breakdown(),
"event_properties": self._event_properties(),
"session_properties": self._session_properties(),
+ "date_from_previous_period": self._date_from_previous_period(),
"date_from": self._date_from(),
"date_to": self._date_to(),
},
@@ -150,21 +175,34 @@ def to_entry_bounce_query(self) -> ast.SelectQuery:
with self.timings.measure("stats_table_query"):
query = parse_select(
"""
+WITH
+ start_timestamp >= {date_from} AND start_timestamp < {date_to} AS current_period_segment,
+ start_timestamp >= {date_from_previous_period} AND start_timestamp < {date_from} AS previous_period_segment
SELECT
breakdown_value AS "context.columns.breakdown_value",
- uniq(filtered_person_id) AS "context.columns.visitors",
- sum(filtered_pageview_count) AS "context.columns.views",
- avg(is_bounce) AS "context.columns.bounce_rate"
+ tuple(
+ uniqIf(filtered_person_id, current_period_segment),
+ uniqIf(filtered_person_id, previous_period_segment)
+ ) AS "context.columns.visitors",
+ tuple(
+ sumIf(filtered_pageview_count, current_period_segment),
+ sumIf(filtered_pageview_count, previous_period_segment)
+ ) AS "context.columns.views",
+ tuple(
+ avgIf(is_bounce, current_period_segment),
+ avgIf(is_bounce, previous_period_segment)
+ ) AS "context.columns.bounce_rate",
FROM (
SELECT
+ {bounce_breakdown} AS breakdown_value,
any(person_id) AS filtered_person_id,
count() AS filtered_pageview_count,
- {bounce_breakdown} AS breakdown_value,
any(session.$is_bounce) AS is_bounce,
- session.session_id AS session_id
+ session.session_id AS session_id,
+ min(session.$start_timestamp) as start_timestamp
FROM events
WHERE and(
- timestamp >= {date_from},
+ timestamp >= {date_from_previous_period},
timestamp < {date_to},
events.event == '$pageview',
{event_properties},
@@ -184,6 +222,7 @@ def to_entry_bounce_query(self) -> ast.SelectQuery:
"where_breakdown": self.where_breakdown(),
"session_properties": self._session_properties(),
"event_properties": self._event_properties(),
+ "date_from_previous_period": self._date_from_previous_period(),
"date_from": self._date_from(),
"date_to": self._date_to(),
},
@@ -198,27 +237,33 @@ def to_path_scroll_bounce_query(self) -> ast.SelectQuery:
with self.timings.measure("stats_table_bounce_query"):
query = parse_select(
"""
+WITH
+ start_timestamp >= {date_from} AND start_timestamp < {date_to} AS current_period_segment,
+ start_timestamp >= {date_from_previous_period} AND start_timestamp < {date_from} AS previous_period_segment
SELECT
counts.breakdown_value AS "context.columns.breakdown_value",
- counts.visitors AS "context.columns.visitors",
- counts.views AS "context.columns.views",
- bounce.bounce_rate AS "context.columns.bounce_rate",
- scroll.average_scroll_percentage AS "context.columns.average_scroll_percentage",
- scroll.scroll_gt80_percentage AS "context.columns.scroll_gt80_percentage"
+ tuple(counts.visitors, counts.previous_visitors) AS "context.columns.visitors",
+ tuple(counts.views, counts.previous_views) AS "context.columns.views",
+ tuple(bounce.bounce_rate, bounce.previous_bounce_rate) AS "context.columns.bounce_rate",
+ tuple(scroll.average_scroll_percentage, scroll.previous_average_scroll_percentage) AS "context.columns.average_scroll_percentage",
+ tuple(scroll.scroll_gt80_percentage, scroll.previous_scroll_gt80_percentage) AS "context.columns.scroll_gt80_percentage",
FROM (
SELECT
breakdown_value,
- uniq(filtered_person_id) AS visitors,
- sum(filtered_pageview_count) AS views
+ uniqIf(filtered_person_id, current_period_segment) AS visitors,
+ uniqIf(filtered_person_id, previous_period_segment) AS previous_visitors,
+ sumIf(filtered_pageview_count, current_period_segment) AS views,
+ sumIf(filtered_pageview_count, previous_period_segment) AS previous_views
FROM (
SELECT
any(person_id) AS filtered_person_id,
count() AS filtered_pageview_count,
{breakdown_value} AS breakdown_value,
- session.session_id AS session_id
+ session.session_id AS session_id,
+ min(session.$start_timestamp ) AS start_timestamp
FROM events
WHERE and(
- timestamp >= {date_from},
+ timestamp >= {date_from_previous_period},
timestamp < {date_to},
events.event == '$pageview',
{event_properties},
@@ -232,15 +277,17 @@ def to_path_scroll_bounce_query(self) -> ast.SelectQuery:
LEFT JOIN (
SELECT
breakdown_value,
- avg(is_bounce) AS bounce_rate
+ avgIf(is_bounce, current_period_segment) AS bounce_rate,
+ avgIf(is_bounce, previous_period_segment) AS previous_bounce_rate
FROM (
SELECT
{bounce_breakdown_value} AS breakdown_value, -- use $entry_pathname to find the bounce rate for sessions that started on this pathname
any(session.`$is_bounce`) AS is_bounce,
- session.session_id AS session_id
+ session.session_id AS session_id,
+ min(session.$start_timestamp) as start_timestamp
FROM events
WHERE and(
- timestamp >= {date_from},
+ timestamp >= {date_from_previous_period},
timestamp < {date_to},
events.event == '$pageview',
{event_properties},
@@ -255,8 +302,10 @@ def to_path_scroll_bounce_query(self) -> ast.SelectQuery:
LEFT JOIN (
SELECT
breakdown_value,
- avgMerge(average_scroll_percentage_state) AS average_scroll_percentage,
- avgMerge(scroll_gt80_percentage_state) AS scroll_gt80_percentage
+ avgMergeIf(average_scroll_percentage_state, current_period_segment) AS average_scroll_percentage,
+ avgMergeIf(average_scroll_percentage_state, previous_period_segment) AS previous_average_scroll_percentage,
+ avgMergeIf(scroll_gt80_percentage_state, current_period_segment) AS scroll_gt80_percentage,
+ avgMergeIf(scroll_gt80_percentage_state, previous_period_segment) AS previous_scroll_gt80_percentage
FROM (
SELECT
{scroll_breakdown_value} AS breakdown_value, -- use $prev_pageview_pathname to find the scroll depth when leaving this pathname
@@ -267,10 +316,11 @@ def to_path_scroll_bounce_query(self) -> ast.SelectQuery:
END
) AS scroll_gt80_percentage_state,
avgState(toFloat(events.properties.`$prev_pageview_max_scroll_percentage`)) as average_scroll_percentage_state,
- session.session_id AS session_id
+ session.session_id AS session_id,
+ min(session.$start_timestamp) AS start_timestamp
FROM events
WHERE and(
- timestamp >= {date_from},
+ timestamp >= {date_from_previous_period},
timestamp < {date_to},
or(events.event == '$pageview', events.event == '$pageleave'),
{event_properties_for_scroll},
@@ -291,6 +341,7 @@ def to_path_scroll_bounce_query(self) -> ast.SelectQuery:
"session_properties": self._session_properties(),
"event_properties": self._event_properties(),
"event_properties_for_scroll": self._event_properties_for_scroll(),
+ "date_from_previous_period": self._date_from_previous_period(),
"date_from": self._date_from(),
"date_to": self._date_to(),
"breakdown_value": self._counts_breakdown_value(),
@@ -308,25 +359,31 @@ def to_path_bounce_query(self) -> ast.SelectQuery:
with self.timings.measure("stats_table_scroll_query"):
query = parse_select(
"""
+WITH
+ start_timestamp >= {date_from} AND start_timestamp < {date_to} AS current_period_segment,
+ start_timestamp >= {date_from_previous_period} AND start_timestamp < {date_from} AS previous_period_segment
SELECT
counts.breakdown_value AS "context.columns.breakdown_value",
- counts.visitors AS "context.columns.visitors",
- counts.views AS "context.columns.views",
- bounce.bounce_rate AS "context.columns.bounce_rate"
+ tuple(counts.visitors, counts.previous_visitors) AS "context.columns.visitors",
+ tuple(counts.views, counts.previous_views) AS "context.columns.views",
+ tuple(bounce.bounce_rate, bounce.previous_bounce_rate) AS "context.columns.bounce_rate"
FROM (
SELECT
breakdown_value,
- uniq(filtered_person_id) AS visitors,
- sum(filtered_pageview_count) AS views
+ uniqIf(filtered_person_id, current_period_segment) AS visitors,
+ uniqIf(filtered_person_id, previous_period_segment) AS previous_visitors,
+ sumIf(filtered_pageview_count, current_period_segment) AS views,
+ sumIf(filtered_pageview_count, previous_period_segment) AS previous_views
FROM (
SELECT
any(person_id) AS filtered_person_id,
count() AS filtered_pageview_count,
{breakdown_value} AS breakdown_value,
- session.session_id AS session_id
+ session.session_id AS session_id,
+ min(session.$start_timestamp) AS start_timestamp
FROM events
WHERE and(
- timestamp >= {date_from},
+ timestamp >= {date_from_previous_period},
timestamp < {date_to},
events.event == '$pageview',
{event_properties},
@@ -340,15 +397,17 @@ def to_path_bounce_query(self) -> ast.SelectQuery:
LEFT JOIN (
SELECT
breakdown_value,
- avg(is_bounce) AS bounce_rate
+ avgIf(is_bounce, current_period_segment) AS bounce_rate,
+ avgIf(is_bounce, previous_period_segment) AS previous_bounce_rate
FROM (
SELECT
{bounce_breakdown_value} AS breakdown_value, -- use $entry_pathname to find the bounce rate for sessions that started on this pathname
any(session.`$is_bounce`) AS is_bounce,
- session.session_id AS session_id
+ session.session_id AS session_id,
+ min(session.$start_timestamp) AS start_timestamp
FROM events
WHERE and(
- timestamp >= {date_from},
+ timestamp >= {date_from_previous_period},
timestamp < {date_to},
events.event == '$pageview',
{event_properties},
@@ -370,6 +429,7 @@ def to_path_bounce_query(self) -> ast.SelectQuery:
"where_breakdown": self.where_breakdown(),
"session_properties": self._session_properties(),
"event_properties": self._event_properties(),
+ "date_from_previous_period": self._date_from_previous_period(),
"date_from": self._date_from(),
"date_to": self._date_to(),
"bounce_breakdown_value": self._bounce_entry_pathname_breakdown(),
@@ -433,6 +493,9 @@ def _date_to(self) -> ast.Expr:
def _date_from(self) -> ast.Expr:
return self.query_date_range.date_from_as_hogql()
+ def _date_from_previous_period(self) -> ast.Expr:
+ return self.query_date_range.previous_period_date_from_as_hogql()
+
def calculate(self):
query = self.to_query()
response = self.paginator.execute_hogql_query(
@@ -450,8 +513,11 @@ def calculate(self):
results,
{
0: self._join_with_aggregation_value, # breakdown_value
- 1: self._unsample, # views
- 2: self._unsample, # visitors
+ 1: lambda tuple, row: (self._unsample(tuple[0], row), self._unsample(tuple[1], row)), # Views (tuple)
+ 2: lambda tuple, row: (
+ self._unsample(tuple[0], row),
+ self._unsample(tuple[1], row),
+ ), # Visitors (tuple)
},
)
diff --git a/posthog/hogql_queries/web_analytics/test/test_web_stats_table.py b/posthog/hogql_queries/web_analytics/test/test_web_stats_table.py
index c021b6d1268bcf..ae4b48b0632c16 100644
--- a/posthog/hogql_queries/web_analytics/test/test_web_stats_table.py
+++ b/posthog/hogql_queries/web_analytics/test/test_web_stats_table.py
@@ -147,8 +147,8 @@ def test_increase_in_users(self):
self.assertEqual(
[
- ["/", 2, 2],
- ["/login", 1, 1],
+ ["/", (2, 0), (2, 0)],
+ ["/login", (1, 0), (1, 0)],
],
results,
)
@@ -168,9 +168,9 @@ def test_all_time(self):
self.assertEqual(
[
- ["/", 2, 2],
- ["/docs", 1, 1],
- ["/login", 1, 1],
+ ["/", (2, 0), (2, 0)],
+ ["/docs", (1, 0), (1, 0)],
+ ["/login", (1, 0), (1, 0)],
],
results,
)
@@ -195,7 +195,7 @@ def test_dont_filter_test_accounts(self):
results = self._run_web_stats_table_query("2023-12-01", "2023-12-03", filter_test_accounts=False).results
self.assertEqual(
- [["/", 1, 1], ["/login", 1, 1]],
+ [["/", (1, 0), (1, 0)], ["/login", (1, 0), (1, 0)]],
results,
)
@@ -235,7 +235,7 @@ def test_limit(self):
response_1 = self._run_web_stats_table_query("all", "2023-12-15", limit=1)
self.assertEqual(
[
- ["/", 2, 2],
+ ["/", (2, 0), (2, 0)],
],
response_1.results,
)
@@ -244,8 +244,8 @@ def test_limit(self):
response_2 = self._run_web_stats_table_query("all", "2023-12-15", limit=2)
self.assertEqual(
[
- ["/", 2, 2],
- ["/login", 1, 1],
+ ["/", (2, 0), (2, 0)],
+ ["/login", (1, 0), (1, 0)],
],
response_2.results,
)
@@ -280,10 +280,10 @@ def test_path_filters(self):
self.assertEqual(
[
- ["/cleaned/:id", 2, 2],
- ["/cleaned/:id/path/:id", 1, 1],
- ["/not-cleaned", 1, 1],
- ["/thing_c", 1, 1],
+ ["/cleaned/:id", (2, 0), (2, 0)],
+ ["/cleaned/:id/path/:id", (1, 0), (1, 0)],
+ ["/not-cleaned", (1, 0), (1, 0)],
+ ["/thing_c", (1, 0), (1, 0)],
],
results,
)
@@ -308,9 +308,9 @@ def test_scroll_depth_bounce_rate_one_user(self):
self.assertEqual(
[
- ["/a", 1, 1, 0, 0.1, 0],
- ["/b", 1, 1, None, 0.2, 0],
- ["/c", 1, 1, None, 0.9, 1],
+ ["/a", (1, 0), (1, 0), (0, None), (0.1, None), (0, None)],
+ ["/b", (1, 0), (1, 0), (None, None), (0.2, None), (0, None)],
+ ["/c", (1, 0), (1, 0), (None, None), (0.9, None), (1, None)],
],
results,
)
@@ -350,9 +350,9 @@ def test_scroll_depth_bounce_rate(self):
self.assertEqual(
[
- ["/a", 3, 4, 1 / 3, 0.5, 0.5],
- ["/b", 2, 2, None, 0.2, 0],
- ["/c", 2, 2, None, 0.9, 1],
+ ["/a", (3, 0), (4, 0), (1 / 3, None), (0.5, None), (0.5, None)],
+ ["/b", (2, 0), (2, 0), (None, None), (0.2, None), (0, None)],
+ ["/c", (2, 0), (2, 0), (None, None), (0.9, None), (1, None)],
],
results,
)
@@ -393,7 +393,7 @@ def test_scroll_depth_bounce_rate_with_filter(self):
self.assertEqual(
[
- ["/a", 3, 4, 1 / 3, 0.5, 0.5],
+ ["/a", (3, 0), (4, 0), (1 / 3, None), (0.5, None), (0.5, None)],
],
results,
)
@@ -423,9 +423,9 @@ def test_scroll_depth_bounce_rate_path_cleaning(self):
self.assertEqual(
[
- ["/a/:id", 1, 1, 0, 0.1, 0],
- ["/b/:id", 1, 1, None, 0.2, 0],
- ["/c/:id", 1, 1, None, 0.9, 1],
+ ["/a/:id", (1, 0), (1, 0), (0, None), (0.1, None), (0, None)],
+ ["/b/:id", (1, 0), (1, 0), (None, None), (0.2, None), (0, None)],
+ ["/c/:id", (1, 0), (1, 0), (None, None), (0.9, None), (1, None)],
],
results,
)
@@ -449,9 +449,9 @@ def test_bounce_rate_one_user(self):
self.assertEqual(
[
- ["/a", 1, 1, 0],
- ["/b", 1, 1, None],
- ["/c", 1, 1, None],
+ ["/a", (1, 0), (1, 0), (0, None)],
+ ["/b", (1, 0), (1, 0), (None, None)],
+ ["/c", (1, 0), (1, 0), (None, None)],
],
results,
)
@@ -490,9 +490,9 @@ def test_bounce_rate(self):
self.assertEqual(
[
- ["/a", 3, 4, 1 / 3],
- ["/b", 2, 2, None],
- ["/c", 2, 2, None],
+ ["/a", (3, 0), (4, 0), (1 / 3, None)],
+ ["/b", (2, 0), (2, 0), (None, None)],
+ ["/c", (2, 0), (2, 0), (None, None)],
],
results,
)
@@ -532,7 +532,7 @@ def test_bounce_rate_with_property(self):
self.assertEqual(
[
- ["/a", 3, 4, 1 / 3],
+ ["/a", (3, 0), (4, 0), (1 / 3, None)],
],
results,
)
@@ -561,9 +561,9 @@ def test_bounce_rate_path_cleaning(self):
self.assertEqual(
[
- ["/a/:id", 1, 1, 0],
- ["/b/:id", 1, 1, None],
- ["/c/:id", 1, 1, None],
+ ["/a/:id", (1, 0), (1, 0), (0, None)],
+ ["/b/:id", (1, 0), (1, 0), (None, None)],
+ ["/c/:id", (1, 0), (1, 0), (None, None)],
],
results,
)
@@ -587,7 +587,7 @@ def test_entry_bounce_rate_one_user(self):
self.assertEqual(
[
- ["/a", 1, 3, 0],
+ ["/a", (1, 0), (3, 0), (0, None)],
],
results,
)
@@ -626,7 +626,7 @@ def test_entry_bounce_rate(self):
self.assertEqual(
[
- ["/a", 3, 8, 1 / 3],
+ ["/a", (3, 0), (8, 0), (1 / 3, None)],
],
results,
)
@@ -666,7 +666,7 @@ def test_entry_bounce_rate_with_property(self):
self.assertEqual(
[
- ["/a", 3, 4, 1 / 3],
+ ["/a", (3, 0), (4, 0), (1 / 3, None)],
],
results,
)
@@ -695,7 +695,7 @@ def test_entry_bounce_rate_path_cleaning(self):
self.assertEqual(
[
- ["/a/:id", 1, 3, 0],
+ ["/a/:id", (1, 0), (3, 0), (0, None)],
],
results,
)
@@ -743,7 +743,10 @@ def test_source_medium_campaign(self):
).results
self.assertEqual(
- [["google / (none) / (none)", 1, 1], ["news.ycombinator.com / referral / (none)", 1, 1]],
+ [
+ ["google / (none) / (none)", (1, 0), (1, 0)],
+ ["news.ycombinator.com / referral / (none)", (1, 0), (1, 0)],
+ ],
results,
)
@@ -792,7 +795,7 @@ def test_null_in_utm_tags(self):
).results
self.assertEqual(
- [["google", 1.0, 1.0], [None, 1.0, 1.0]],
+ [["google", (1, 0), (1, 0)], [None, (1, 0), (1, 0)]],
results,
)
@@ -842,7 +845,7 @@ def test_is_not_set_filter(self):
).results
self.assertEqual(
- [[None, 1.0, 1.0]],
+ [[None, (1, 0), (1, 0)]],
results,
)
@@ -878,7 +881,7 @@ def test_same_user_multiple_sessions(self):
"2024-07-31",
breakdown_by=WebStatsBreakdown.INITIAL_UTM_SOURCE,
).results
- assert [["google", 1, 2]] == results_session
+ assert [["google", (1, 0), (2, 0)]] == results_session
# Try this with a query that uses event properties
results_event = self._run_web_stats_table_query(
@@ -886,13 +889,13 @@ def test_same_user_multiple_sessions(self):
"2024-07-31",
breakdown_by=WebStatsBreakdown.PAGE,
).results
- assert [["/path", 1, 2]] == results_event
+ assert [["/path", (1, 0), (2, 0)]] == results_event
# Try this with a query using the bounce rate
results_event = self._run_web_stats_table_query(
"all", "2024-07-31", breakdown_by=WebStatsBreakdown.PAGE, include_bounce_rate=True
).results
- assert [["/path", 1, 2, None]] == results_event
+ assert [["/path", (1, 0), (2, 0), (None, None)]] == results_event
# Try this with a query using the scroll depth
results_event = self._run_web_stats_table_query(
@@ -902,7 +905,7 @@ def test_same_user_multiple_sessions(self):
include_bounce_rate=True,
include_scroll_depth=True,
).results
- assert [["/path", 1, 2, None, None, None]] == results_event
+ assert [["/path", (1, 0), (2, 0), (None, None), (None, None), (None, None)]] == results_event
def test_no_session_id(self):
d1 = "d1"
@@ -935,13 +938,16 @@ def test_no_session_id(self):
).results
assert [] == results
- # Do show event property breakdowns of events of events with no session id
+ # Do show event property breakdowns of events with no session id
+ # but it will return 0 views because we depend on session.$start_timestamp
+ # to figure out the previous/current values
results = self._run_web_stats_table_query(
"all",
"2024-07-31",
breakdown_by=WebStatsBreakdown.PAGE,
).results
- assert [["/path", 1, 1]] == results
+
+ assert [["/path", (0, 0), (0, 0)]] == results
def test_cohort_test_filters(self):
d1 = "d1"
@@ -1003,7 +1009,7 @@ def test_cohort_test_filters(self):
breakdown_by=WebStatsBreakdown.PAGE,
).results
- assert results == [["/path1", 1, 1]]
+ assert results == [["/path1", (1, 0), (1, 0)]]
def test_language_filter(self):
d1, s1 = "d1", str(uuid7("2024-07-30"))
@@ -1079,45 +1085,61 @@ def test_language_filter(self):
# which is causing this to be flaky (en-GB happens sometimes),
# we'll instead assert on a reduced form where we're
# not counting the country, but only the locale
- # assert results == [["en-US", 1.0, 3.0], ["pt-BR", 1.0, 2.0], ["nl-", 1.0, 1.0]]
+ # assert results == [["en-US", (1, 0), (3, 0)], ["pt-BR", (1, 0), (2, 0)], ["nl-", (1, 0), (1, 0)]]
country_results = [result[0].split("-")[0] for result in results]
assert country_results == ["en", "pt", "nl"]
- def test_timezone_filter(self):
- date = "2024-07-30"
+ def test_timezone_filter_general(self):
+ before_date = "2024-07-14"
+ after_date = "2024-07-16"
- for idx, (distinct_id, session_id) in enumerate(
+ for idx, (distinct_id, before_session_id, after_session_id) in enumerate(
[
- ("UTC", str(uuid7(date))),
- ("Asia/Calcutta", str(uuid7(date))),
- ("America/New_York", str(uuid7(date))),
- ("America/Sao_Paulo", str(uuid7(date))),
+ ("UTC", str(uuid7(before_date)), str(uuid7(after_date))),
+ ("Asia/Calcutta", str(uuid7(before_date)), str(uuid7(after_date))),
+ ("America/New_York", str(uuid7(before_date)), str(uuid7(after_date))),
+ ("America/Sao_Paulo", str(uuid7(before_date)), str(uuid7(after_date))),
]
):
_create_person(
team_id=self.team.pk,
distinct_ids=[distinct_id],
- properties={"name": session_id, "email": f"{distinct_id}@example.com"},
+ properties={"name": before_session_id, "email": f"{distinct_id}@example.com"},
)
+ # Always one event in the before_date
+ _create_event(
+ team=self.team,
+ event="$pageview",
+ distinct_id=distinct_id,
+ timestamp=before_date,
+ properties={"$session_id": before_session_id, "$pathname": f"/path/landing", "$timezone": distinct_id},
+ )
+
+ # Several events in the actual range
for i in range(idx + 1):
_create_event(
team=self.team,
event="$pageview",
distinct_id=distinct_id,
- timestamp=date,
- properties={"$session_id": session_id, "$pathname": f"/path{i}", "$timezone": distinct_id},
+ timestamp=after_date,
+ properties={"$session_id": after_session_id, "$pathname": f"/path{i}", "$timezone": distinct_id},
)
results = self._run_web_stats_table_query(
- "all",
+ "2024-07-15", # Period is since July first, we create some events before that date, and some after
None,
breakdown_by=WebStatsBreakdown.TIMEZONE,
).results
# Brasilia UTC-3, New York UTC-4, Calcutta UTC+5:30, UTC
- assert results == [[-3.0, 1.0, 4.0], [-4.0, 1.0, 3.0], [5.5, 1.0, 2.0], [0.0, 1.0, 1.0]]
+ assert results == [
+ [-3, (1, 1), (4, 1)],
+ [-4, (1, 1), (3, 1)],
+ [5.5, (1, 1), (2, 1)],
+ [0, (1, 1), (1, 1)],
+ ]
def test_timezone_filter_dst_change(self):
did = "id"
@@ -1146,7 +1168,7 @@ def test_timezone_filter_dst_change(self):
).results
# Change from UTC-2 to UTC-3 in the middle of the night
- assert results == [[-3.0, 1.0, 4.0], [-2.0, 1.0, 2.0]]
+ assert results == [[-3, (1, 0), (4, 0)], [-2, (1, 0), (2, 0)]]
def test_timezone_filter_with_invalid_timezone(self):
date = "2024-07-30"