Skip to content

Commit 1fa5506

Browse files
authored
Merge pull request #36242 from github/repo-sync
Repo sync
2 parents 0a03e4f + 624bd0d commit 1fa5506

File tree

10 files changed

+247
-156
lines changed

10 files changed

+247
-156
lines changed

src/events/components/events.ts

+4-1
Original file line numberDiff line numberDiff line change
@@ -116,7 +116,10 @@ export function sendEvent<T extends EventType>({
116116
os_preference: Cookies.get('osPreferred'),
117117
code_display_preference: Cookies.get('annotate-mode'),
118118

119-
experiment_variation: getExperimentVariationForContext(getMetaContent('path-language')),
119+
experiment_variation: getExperimentVariationForContext(
120+
getMetaContent('path-language'),
121+
getMetaContent('path-version'),
122+
),
120123

121124
// Event grouping
122125
event_group_key: eventGroupKey,

src/events/components/experiments/README.md

+1-6
Original file line numberDiff line numberDiff line change
@@ -68,19 +68,14 @@ In the code that displays the search bar, you can use the `shouldShowExperiment`
6868
Example:
6969

7070
```typescript
71-
import { useRouter } from 'next/router'
7271
import { useShouldShowExperiment } from '@/events/components/experiments/useShouldShowExperiment'
7372
import { EXPERIMENTS } from '@/events/components/experiments/experiments'
7473
import { ClassicSearchBar } from "@/search/components/ClassicSearchBar.tsx"
7574
import { NewSearchBar } from "@/search/components/NewSearchBar.tsx"
7675

7776
export function SearchBar() {
78-
const router = useRouter()
7977
// Users who were randomly placed in the `treatment` group will be shown the experiment
80-
const { shouldShow: shouldShowNewSearch } = useShouldShowExperiment(
81-
EXPERIMENTS.ai_search_experiment,
82-
router.locale
83-
)
78+
const { shouldShow: shouldShowNewSearch } = useShouldShowExperiment(EXPERIMENTS.ai_search_experiment)
8479

8580
if (shouldShowNewSearch) {
8681
return (

src/events/components/experiments/experiment.ts

+37-6
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,7 @@
11
import murmur from 'imurmurhash'
22
import {
33
CONTROL_VARIATION,
4+
EXPERIMENTS,
45
ExperimentNames,
56
TREATMENT_VARIATION,
67
getActiveExperiments,
@@ -12,6 +13,7 @@ let experimentsInitialized = false
1213
export function shouldShowExperiment(
1314
experimentKey: ExperimentNames | { key: ExperimentNames },
1415
locale: string,
16+
version: string,
1517
isStaff: boolean,
1618
) {
1719
// Accept either EXPERIMENTS.<experiment_key> or EXPERIMENTS.<experiment_key>.key
@@ -40,8 +42,10 @@ export function shouldShowExperiment(
4042
return controlGroup === TREATMENT_VARIATION
4143
// Otherwise use the regular logic to determine if the user is in the treatment group
4244
} else if (
43-
experiment.limitToLanguages?.length &&
44-
experiment.limitToLanguages.includes(locale)
45+
(experiment.limitToLanguages?.length
46+
? experiment.limitToLanguages.includes(locale)
47+
: true) &&
48+
(experiment.limitToVersions?.length ? experiment.limitToVersions.includes(version) : true)
4549
) {
4650
return (
4751
getExperimentControlGroupFromSession(
@@ -98,8 +102,8 @@ export function getExperimentControlGroupFromSession(
98102
return modHash < percentToGetExperiment ? TREATMENT_VARIATION : CONTROL_VARIATION
99103
}
100104

101-
export function getExperimentVariationForContext(locale: string): string {
102-
const experiments = getActiveExperiments(locale)
105+
export function getExperimentVariationForContext(locale: string, version: string): string {
106+
const experiments = getActiveExperiments(locale, version)
103107
for (const experiment of experiments) {
104108
if (experiment.includeVariationInContext) {
105109
return getExperimentControlGroupFromSession(
@@ -113,11 +117,38 @@ export function getExperimentVariationForContext(locale: string): string {
113117
return ''
114118
}
115119

116-
export function initializeExperiments(locale: string) {
120+
export function initializeExperiments(
121+
locale: string,
122+
currentVersion: string,
123+
allVersions: { [key: string]: { version: string } },
124+
) {
117125
if (experimentsInitialized) return
118126
experimentsInitialized = true
119127

120-
const experiments = getActiveExperiments(locale)
128+
// Replace any occurrence of 'enterprise-server@latest' with the actual latest version
129+
for (const [experimentKey, experiment] of Object.entries(EXPERIMENTS)) {
130+
if (experiment.limitToVersions?.includes('enterprise-server@latest')) {
131+
// Sort the versions in descending order so that the latest enterprise-server version is first
132+
const latestEnterpriseServerVersion = Object.keys(allVersions)
133+
.filter((version) => version.startsWith('enterprise-server@'))
134+
.sort((a, b) => {
135+
const aVersion = a.split('@')[1]
136+
const bVersion = b.split('@')[1]
137+
return Number(bVersion) - Number(aVersion)
138+
})[0]
139+
if (latestEnterpriseServerVersion) {
140+
EXPERIMENTS[experimentKey as ExperimentNames].limitToVersions =
141+
experiment.limitToVersions.map((version) =>
142+
version.replace(
143+
'enterprise-server@latest',
144+
allVersions[latestEnterpriseServerVersion].version,
145+
),
146+
)
147+
}
148+
}
149+
}
150+
151+
const experiments = getActiveExperiments(locale, currentVersion)
121152

122153
if (experiments.length && process.env.NODE_ENV === 'development') {
123154
console.log(

src/events/components/experiments/experiments.ts

+32-6
Original file line numberDiff line numberDiff line change
@@ -9,6 +9,7 @@ type Experiment = {
99
// Only one experiment's control group (variation) can be included in the context at a time
1010
includeVariationInContext?: boolean
1111
limitToLanguages?: string[]
12+
limitToVersions?: string[]
1213
alwaysShowForStaff: boolean
1314
}
1415

@@ -22,6 +23,11 @@ export const EXPERIMENTS = {
2223
percentOfUsersToGetExperiment: 0, // 10% of users will get the experiment
2324
includeVariationInContext: true, // All events will include the `experiment_variation` of the `ai_search_experiment`
2425
limitToLanguages: ['en'], // Only users with the `en` language will be included in the experiment
26+
limitToVersions: [
27+
'free-pro-team@latest',
28+
'enterprise-cloud@latest',
29+
'enterprise-server@latest',
30+
], // Only enable for versions
2531
alwaysShowForStaff: false, // When set to true, staff will always see the experiment (determined by the `staffonly` cookie)
2632
},
2733
/* Add new experiments here, example:
@@ -31,17 +37,37 @@ export const EXPERIMENTS = {
3137
percentOfUsersToGetExperiment: 10, // 10% of users will randomly get the experiment
3238
includeVariationInContext: true, // All events will include the `experiment_variation` of the `example_experiment`
3339
limitToLanguages: ['en'], // Only users with the `en` language will be included in the experiment
40+
limitToVersions: [
41+
'free-pro-team@latest',
42+
'enterprise-cloud@latest',
43+
'enterprise-server@latest',
44+
], // Only enable for the latest versions
3445
alwaysShowForStaff: true, // When set to true, staff will always see the experiment (determined by the `staffonly` cookie)
3546
}
3647
*/
3748
} as Record<ExperimentNames, Experiment>
3849

39-
export function getActiveExperiments(locale: string): Experiment[] {
50+
export function getActiveExperiments(locale: string, version?: string): Experiment[] {
4051
return Object.values(EXPERIMENTS).filter((experiment) => {
41-
return (
42-
experiment.isActive &&
43-
(locale === 'all' ||
44-
(experiment.limitToLanguages?.length ? experiment.limitToLanguages.includes(locale) : true))
45-
)
52+
if (locale === 'all') {
53+
return true
54+
}
55+
56+
let include = true
57+
if (!experiment.isActive) {
58+
include = false
59+
}
60+
61+
// Only include experiment if it's supported for the current language
62+
if (experiment.limitToLanguages?.length && !experiment.limitToLanguages.includes(locale)) {
63+
include = false
64+
}
65+
66+
// Only include experiment if it's supported for the current version
67+
if (experiment.limitToVersions?.length && !experiment.limitToVersions.includes(version || '')) {
68+
include = false
69+
}
70+
71+
return include
4672
})
4773
}
Original file line numberDiff line numberDiff line change
@@ -1,22 +1,30 @@
11
import { useEffect, useState } from 'react'
2+
import { useRouter } from 'next/router'
23
import { shouldShowExperiment } from './experiment'
34
import { ExperimentNames } from './experiments'
45
import { getIsStaff } from '../dotcom-cookies'
6+
import { useMainContext } from '@/frame/components/context/MainContext'
57

6-
export function useShouldShowExperiment(
7-
experimentKey: ExperimentNames | { key: ExperimentNames },
8-
locale: string,
9-
) {
8+
export function useShouldShowExperiment(experimentKey: ExperimentNames | { key: ExperimentNames }) {
109
if (typeof experimentKey === 'object') {
1110
experimentKey = experimentKey.key
1211
}
1312

1413
const [showExperiment, setShowExperiment] = useState(false)
14+
const router = useRouter()
15+
const mainContext = useMainContext()
1516

1617
useEffect(() => {
1718
const updateShouldShow = async () => {
1819
const isStaff = await getIsStaff()
19-
setShowExperiment(shouldShowExperiment(experimentKey, locale, isStaff))
20+
setShowExperiment(
21+
shouldShowExperiment(
22+
experimentKey,
23+
router.locale || '',
24+
mainContext.currentVersion || '',
25+
isStaff,
26+
),
27+
)
2028
}
2129

2230
updateShouldShow()
@@ -27,7 +35,7 @@ export function useShouldShowExperiment(
2735
return () => {
2836
window.removeEventListener('controlGroupOverrideChanged', updateShouldShow)
2937
}
30-
}, [experimentKey])
38+
}, [experimentKey, router.locale, mainContext.currentVersion])
3139

3240
return showExperiment
3341
}

src/frame/components/page-header/Header.tsx

+1-4
Original file line numberDiff line numberDiff line change
@@ -51,10 +51,7 @@ export const Header = () => {
5151
const { width } = useInnerWindowWidth()
5252
const returnFocusRef = useRef(null)
5353

54-
const showNewSearch = useShouldShowExperiment(
55-
EXPERIMENTS.ai_search_experiment,
56-
router.locale as string,
57-
)
54+
const showNewSearch = useShouldShowExperiment(EXPERIMENTS.ai_search_experiment)
5855

5956
useEffect(() => {
6057
function onScroll() {

src/frame/pages/app.tsx

+5-1
Original file line numberDiff line numberDiff line change
@@ -25,7 +25,11 @@ const MyApp = ({ Component, pageProps, languagesContext }: MyAppProps) => {
2525

2626
useEffect(() => {
2727
initializeEvents()
28-
initializeExperiments(router.locale as string)
28+
initializeExperiments(
29+
router.locale as string,
30+
pageProps.mainContext.currentVersion,
31+
pageProps.mainContext.allVersions,
32+
)
2933
}, [])
3034

3135
useEffect(() => {

src/search/components/helpers/ai-search-links-json.ts

+10-1
Original file line numberDiff line numberDiff line change
@@ -53,7 +53,16 @@ function extractMarkdownLinks(markdownResponse: string) {
5353
urls.push(match[2])
5454
}
5555

56-
return urls
56+
// Filter out any invalid URLs
57+
return urls.filter((url) => {
58+
try {
59+
new URL(url)
60+
return true
61+
// eslint-disable-next-line @typescript-eslint/no-unused-vars
62+
} catch (e) {
63+
return false
64+
}
65+
})
5766
}
5867

5968
// Given a Docs URL, extract the product name

0 commit comments

Comments
 (0)