Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat(experiment): advice banners #21433

Merged
merged 15 commits into from
Apr 16, 2024
3 changes: 2 additions & 1 deletion frontend/src/scenes/experiments/ExperimentView/Info.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -14,7 +14,7 @@ import { ProgressStatus } from '~/types'
import { StatusTag } from '../Experiment'
import { experimentLogic } from '../experimentLogic'
import { getExperimentStatus } from '../experimentsLogic'
import { ResultsTag } from './components'
import { ActionBanner, ResultsTag } from './components'

export function Info(): JSX.Element {
const { experiment } = useValues(experimentLogic)
Expand Down Expand Up @@ -102,6 +102,7 @@ export function Info(): JSX.Element {
compactButtons
/>
</div>
<ActionBanner />
</div>
)
}
44 changes: 21 additions & 23 deletions frontend/src/scenes/experiments/ExperimentView/ProgressBar.tsx
Original file line number Diff line number Diff line change
@@ -1,32 +1,28 @@
import '../Experiment.scss'

import { useValues } from 'kea'
import { dayjs } from 'lib/dayjs'
import { LemonProgress } from 'lib/lemon-ui/LemonProgress'
import { humanFriendlyNumber } from 'lib/utils'

import { FunnelStep, InsightType } from '~/types'
import { InsightType } from '~/types'

import { experimentLogic } from '../experimentLogic'
import { formatUnitByQuantity } from '../utils'

export function ProgressBar(): JSX.Element {
const { experiment, experimentResults, experimentInsightType } = useValues(experimentLogic)

// Parameters for experiment results
// don't use creation variables in results
const funnelResultsPersonsTotal =
experimentInsightType === InsightType.FUNNELS && experimentResults?.insight
? (experimentResults.insight as FunnelStep[][]).reduce(
(sum: number, variantResult: FunnelStep[]) => variantResult[0]?.count + sum,
0
)
: 0
const {
experiment,
experimentInsightType,
funnelResultsPersonsTotal,
recommendedSampleSize,
actualRunningTime,
recommendedRunningTime,
} = useValues(experimentLogic)

const experimentProgressPercent =
experimentInsightType === InsightType.FUNNELS
? ((funnelResultsPersonsTotal || 0) / (experiment?.parameters?.recommended_sample_size || 1)) * 100
: (dayjs().diff(experiment?.start_date, 'day') / (experiment?.parameters?.recommended_running_time || 1)) *
100
? (funnelResultsPersonsTotal / recommendedSampleSize) * 100
: (actualRunningTime / recommendedRunningTime) * 100

return (
<div>
Expand All @@ -48,32 +44,34 @@ export function ProgressBar(): JSX.Element {
<div className="flex justify-between mt-2">
{experiment.end_date ? (
<div>
Ran for <b>{dayjs(experiment.end_date).diff(experiment.start_date, 'day')}</b> days
Ran for <b>{actualRunningTime}</b> {formatUnitByQuantity(actualRunningTime, 'day')}
</div>
) : (
<div>
<b>{dayjs().diff(experiment.start_date, 'day')}</b> days running
<b>{actualRunningTime}</b> {formatUnitByQuantity(actualRunningTime, 'day')} running
</div>
)}
<div>
Goal: <b>{experiment?.parameters?.recommended_running_time ?? 'Unknown'}</b> days
Goal: <b>{recommendedRunningTime}</b> {formatUnitByQuantity(recommendedRunningTime, 'day')}
</div>
</div>
)}
{experimentInsightType === InsightType.FUNNELS && (
<div className="flex justify-between mt-2">
{experiment.end_date ? (
<div>
Saw <b>{humanFriendlyNumber(funnelResultsPersonsTotal)}</b> participants
Saw <b>{humanFriendlyNumber(funnelResultsPersonsTotal)}</b>{' '}
{formatUnitByQuantity(funnelResultsPersonsTotal, 'participant')}
</div>
) : (
<div>
<b>{humanFriendlyNumber(funnelResultsPersonsTotal)}</b> participants seen
<b>{humanFriendlyNumber(funnelResultsPersonsTotal)}</b>{' '}
{formatUnitByQuantity(funnelResultsPersonsTotal, 'participant')} seen
</div>
)}
<div>
Goal: <b>{humanFriendlyNumber(experiment?.parameters?.recommended_sample_size || 0)}</b>{' '}
participants
Goal: <b>{humanFriendlyNumber(recommendedSampleSize)}</b>{' '}
{formatUnitByQuantity(recommendedSampleSize, 'participant')}
</div>
</div>
)}
Expand Down
158 changes: 157 additions & 1 deletion frontend/src/scenes/experiments/ExperimentView/components.tsx
Original file line number Diff line number Diff line change
@@ -1,7 +1,7 @@
import '../Experiment.scss'

import { IconCheckbox } from '@posthog/icons'
import { LemonButton, LemonDivider, LemonTag, LemonTagType } from '@posthog/lemon-ui'
import { LemonBanner, LemonButton, LemonDivider, LemonTag, LemonTagType, Link } from '@posthog/lemon-ui'
import { Empty } from 'antd'
import { useActions, useValues } from 'kea'
import { AnimationType } from 'lib/animations/animations'
Expand Down Expand Up @@ -335,3 +335,159 @@ export function PageHeaderCustom(): JSX.Element {
/>
)
}

export function ActionBanner(): JSX.Element {
const {
experiment,
experimentInsightType,
experimentResults,
experimentLoading,
experimentResultsLoading,
isExperimentRunning,
areResultsSignificant,
isExperimentStopped,
funnelResultsPersonsTotal,
recommendedSampleSize,
actualRunningTime,
recommendedRunningTime,
getHighestProbabilityVariant,
} = useValues(experimentLogic)

const { archiveExperiment } = useActions(experimentLogic)

if (!experiment || experimentLoading || experimentResultsLoading) {
return <></>
}

// Draft
if (!isExperimentRunning) {
return (
<LemonBanner type="info" className="mt-4">
Your experiment is in draft mode. You can edit your variants, adjust release conditions, and{' '}
<Link className="font-semibold" to="https://posthog.com/docs/experiments/testing-and-launching">
test your feature flag
</Link>
. Once everything works as expected, you can launch your experiment. From that point, any new experiment
events will be counted towards the results.
</LemonBanner>
)
}

// Running, results present, not significant
if (isExperimentRunning && experimentResults && !isExperimentStopped && !areResultsSignificant) {
jurajmajerik marked this conversation as resolved.
Show resolved Hide resolved
// Results insignificant, but a large enough sample/running time has been achieved
// Further collection unlikely to change the result -> recommmend cutting the losses
if (
experimentInsightType === InsightType.FUNNELS &&
funnelResultsPersonsTotal > Math.max(recommendedSampleSize, 500) &&
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

thought of another idea, once we've fixed the recommendedSampleSize, we can replace this with 1.5*recommendedSampleSize to handle high volume insights well.

dayjs().diff(experiment.start_date, 'day') > 2 // at least 2 days running
) {
return (
<LemonBanner type="warning" className="mt-4">
You've reached a robust sample size for your experiment, but the results are still inconclusive.
jurajmajerik marked this conversation as resolved.
Show resolved Hide resolved
Continuing the experiment is unlikely to yield significant findings. It may be time to stop this
experiment.
</LemonBanner>
)
}
if (experimentInsightType === InsightType.TRENDS && actualRunningTime > Math.max(recommendedRunningTime, 7)) {
return (
<LemonBanner type="warning" className="mt-4">
Your experiment has been running long enough, but the results are still inconclusive. Continuing the
experiment is unlikely to yield significant findings. It may be time to stop this experiment.
</LemonBanner>
)
}

return (
<LemonBanner type="info" className="mt-4">
Your experiment is live and is collecting data, but hasn't yet reached the statistical significance
needed to make reliable decisions. It's important to wait for more data to avoid premature conclusions.
</LemonBanner>
)
}

// Running, results significant
if (isExperimentRunning && !isExperimentStopped && areResultsSignificant && experimentResults) {
const { probability } = experimentResults
const winningVariant = getHighestProbabilityVariant(experimentResults)
if (!winningVariant) {
return <></>
}

const winProbability = probability[winningVariant]

// Win probability only slightly over 0.9 and the recommended sample/time just met -> proceed with caution
if (
experimentInsightType === InsightType.FUNNELS &&
funnelResultsPersonsTotal > recommendedSampleSize + 50 &&
winProbability < 0.93
) {
return (
<LemonBanner type="info" className="mt-4">
You've achieved significant results, however, the sample size just meets the minimum requirements,
and the win probability is only marginally above 90%. To ensure more reliable outcomes, consider
running the experiment a bit longer.
</LemonBanner>
)
}

if (
experimentInsightType === InsightType.TRENDS &&
actualRunningTime > recommendedRunningTime + 2 &&
winProbability < 0.93
) {
return (
<LemonBanner type="info" className="mt-4">
You've achieved significant results, however, the running time just meets the minimum requirements,
and the win probability is only marginally above 90%. To ensure more reliable outcomes, consider
running the experiment a bit longer.
</LemonBanner>
)
}

return (
<LemonBanner type="success" className="mt-4">
Good news! Your experiment has gathered enough data to reach statistical significance, providing
jurajmajerik marked this conversation as resolved.
Show resolved Hide resolved
reliable results for decision making. Before taking any action, review relevant secondary metrics for
any unintended side effects. Once you're done, you can stop the experiment.
</LemonBanner>
)
}

// Stopped, results significant
if (isExperimentStopped && areResultsSignificant) {
return (
<LemonBanner type="success" className="mt-4">
You have stopped this experiment, and it is no longer collecting data. With significant results in hand,
you can now roll out the winning variant to all your users by adjusting the{' '}
<Link
target="_blank"
className="font-semibold"
to={experiment.feature_flag ? urls.featureFlag(experiment.feature_flag.id) : undefined}
>
{experiment.feature_flag?.key}
</Link>{' '}
feature flag.
</LemonBanner>
)
}

// Stopped, results not significant
if (isExperimentStopped && experimentResults && !areResultsSignificant) {
return (
<LemonBanner type="info" className="mt-4">
You have stopped this experiment, and it is no longer collecting data. Because your results are not
significant, we don't recommend drawing any conclusions from them. You can reset the experiment
(deleting the data collected so far) and restart the experiment at any point again. If this experiment
is no longer relevant, you can{' '}
<Link className="font-semibold" onClick={() => archiveExperiment()}>
archive it
</Link>
.
</LemonBanner>
)
}

return <></>
}
38 changes: 38 additions & 0 deletions frontend/src/scenes/experiments/experimentLogic.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -1107,6 +1107,44 @@ export const experimentLogic = kea<experimentLogicType>([
})
},
],
recommendedSampleSize: [
(s) => [s.experiment],
(experiment: Experiment): number => experiment?.parameters?.recommended_sample_size || 100,
jurajmajerik marked this conversation as resolved.
Show resolved Hide resolved
],
funnelResultsPersonsTotal: [
(s) => [s.experimentResults, s.experimentInsightType],
(experimentResults: ExperimentResults['result'], experimentInsightType: InsightType): number => {
if (experimentInsightType !== InsightType.FUNNELS || !experimentResults?.insight) {
return 0
}

let sum = 0
experimentResults.insight.forEach((variantResult) => {
if (variantResult[0]?.count) {
sum += variantResult[0].count
}
})
return sum
},
],
actualRunningTime: [
(s) => [s.experiment],
(experiment: Experiment): number => {
if (!experiment.start_date) {
return 0
}

if (experiment.end_date) {
return dayjs(experiment.end_date).diff(experiment.start_date, 'day')
}

return dayjs().diff(experiment.start_date, 'day')
},
],
recommendedRunningTime: [
(s) => [s.experiment],
(experiment: Experiment): number => experiment?.parameters?.recommended_running_time || 1,
],
}),
forms(({ actions, values }) => ({
experiment: {
Expand Down
4 changes: 4 additions & 0 deletions frontend/src/scenes/experiments/utils.ts
Original file line number Diff line number Diff line change
Expand Up @@ -17,3 +17,7 @@ export const transformResultFilters = (filters: Partial<FilterType>): Partial<Fi
display: ChartDisplayType.ActionsLineGraphCumulative,
}),
})

export function formatUnitByQuantity(value: number, unit: string): string {
return value === 1 ? unit : unit + 's'
}
Loading