Skip to content

Commit 9423555

Browse files
authored
Merge pull request #36186 from github/repo-sync
Repo sync
2 parents 0c0544a + b20385d commit 9423555

File tree

11 files changed

+212
-10
lines changed

11 files changed

+212
-10
lines changed
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,57 @@
1+
apiVersion: apps/v1
2+
kind: Deployment
3+
metadata:
4+
name: webapp
5+
spec:
6+
replicas: 2
7+
selector:
8+
matchLabels:
9+
app: webapp
10+
template:
11+
metadata:
12+
labels:
13+
app: webapp
14+
annotations:
15+
# Our internal logs aren't structured so we use logfmt_sloppy to just log stdout and error
16+
# See https://thehub.github.com/epd/engineering/dev-practicals/observability/logging/ for more details
17+
fluentbit.io/parser: logfmt_sloppy
18+
observability.github.com/splunk_index: docs-internal
19+
spec:
20+
dnsPolicy: Default
21+
containers:
22+
- name: webapp
23+
image: docs-internal
24+
resources:
25+
requests:
26+
cpu: 1000m
27+
memory: 4500Mi
28+
limits:
29+
cpu: 8000m
30+
memory: 16Gi
31+
ports:
32+
- name: http
33+
containerPort: 4000
34+
protocol: TCP
35+
envFrom:
36+
- secretRef:
37+
name: vault-secrets
38+
- configMapRef:
39+
name: kube-cluster-metadata
40+
# application-config is created at deploy time from
41+
# configuration set in config/moda/configuration/*/env.yaml
42+
- configMapRef:
43+
name: application-config
44+
# Zero-downtime deploys
45+
# https://thehub.github.com/engineering/products-and-services/internal/moda/feature-documentation/pod-lifecycle/#required-prestop-hook
46+
# https://kubernetes.io/docs/concepts/containers/container-lifecycle-hooks/#container-hooks
47+
lifecycle:
48+
preStop:
49+
exec:
50+
command: ['sleep', '5']
51+
readinessProbe:
52+
initialDelaySeconds: 5
53+
httpGet:
54+
# WARNING: This should be updated to a meaningful endpoint for your application which will return a 200 once the app is fully started.
55+
# See: https://kubernetes.io/docs/tasks/configure-pod-container/configure-liveness-readiness-startup-probes/#define-readiness-probes
56+
path: /healthcheck
57+
port: http
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,19 @@
1+
apiVersion: v1
2+
kind: Service
3+
metadata:
4+
name: webapp
5+
labels:
6+
service: webapp
7+
annotations:
8+
moda.github.net/domain-name: 'docs-internal-%environment%.service.%region%.github.net'
9+
# HTTP app reachable inside GitHub's network (employee website)
10+
moda.github.net/load-balancer-type: internal-http
11+
spec:
12+
ports:
13+
- name: http
14+
port: 4000
15+
protocol: TCP
16+
targetPort: http
17+
selector:
18+
app: webapp
19+
type: LoadBalancer

config/kubernetes/production/deployments/webapp.yaml

+1-1
Original file line numberDiff line numberDiff line change
@@ -37,7 +37,7 @@ spec:
3737
name: vault-secrets
3838
- configMapRef:
3939
name: kube-cluster-metadata
40-
# application-config is crated at deploy time from
40+
# application-config is created at deploy time from
4141
# configuration set in config/moda/configuration/*/env.yaml
4242
- configMapRef:
4343
name: application-config
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,9 @@
1+
data:
2+
MODA_APP_NAME: docs-internal
3+
NODE_ENV: production
4+
NODE_OPTIONS: '--max-old-space-size=4096'
5+
PORT: '4000'
6+
ENABLED_LANGUAGES: 'en,zh,es,pt,ru,ja,fr,de,ko'
7+
RATE_LIMIT_MAX: '21'
8+
# Moda uses a non-default port for sending datadog metrics
9+
DD_DOGSTATSD_PORT: '28125'
Original file line numberDiff line numberDiff line change
@@ -1,12 +1,12 @@
11
data:
22
MODA_APP_NAME: docs-internal
3-
# Identifies the service deployment environment as production
4-
# Equivalent to HEAVEN_DEPLOYED_ENV === 'production'
5-
MODA_PROD_SERVICE_ENV: 'true'
63
NODE_ENV: production
74
NODE_OPTIONS: '--max-old-space-size=4096'
85
PORT: '4000'
96
ENABLED_LANGUAGES: 'en,zh,es,pt,ru,ja,fr,de,ko'
107
RATE_LIMIT_MAX: '21'
118
# Moda uses a non-default port for sending datadog metrics
129
DD_DOGSTATSD_PORT: '28125'
10+
# Identifies the service deployment environment as production
11+
# Equivalent to HEAVEN_DEPLOYED_ENV === 'production'
12+
MODA_PROD_SERVICE_ENV: 'true'

config/moda/deployment.yaml

+21
Original file line numberDiff line numberDiff line change
@@ -7,6 +7,27 @@ environments:
77
profile: general
88
region: iad
99

10+
- name: staging-cedar
11+
require_pipeline: false
12+
notify_still_locked: true # Notify last person to lock this after an hour
13+
cluster_selector:
14+
profile: general
15+
region: iad
16+
17+
- name: staging-pine
18+
require_pipeline: false
19+
notify_still_locked: true # Notify last person to lock this after an hour
20+
cluster_selector:
21+
profile: general
22+
region: iad
23+
24+
- name: staging-spruce
25+
require_pipeline: false
26+
notify_still_locked: true # Notify last person to lock this after an hour
27+
cluster_selector:
28+
profile: general
29+
region: iad
30+
1031
required_builds:
1132
- docs-internal-moda-config-bundle / docs-internal-moda-config-bundle
1233
- docs-internal-docker-image / docs-internal-docker-image
+69
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,69 @@
1+
// We cannot use Cookies.get() on the frontend for httpOnly cookies
2+
// so we need to make a request to the server to get the cookies
3+
4+
type DotcomCookies = {
5+
dotcomUsername?: string
6+
isStaff?: boolean
7+
}
8+
9+
let cachedCookies: DotcomCookies | null = null
10+
let inFlightPromise: Promise<DotcomCookies> | null = null
11+
let tries = 0
12+
13+
const GET_COOKIES_ENDPOINT = '/api/cookies'
14+
const MAX_TRIES = 3
15+
16+
// Fetches httpOnly cookies from the server and cache the result
17+
// We use an in-flight promise to avoid duplicate requests
18+
async function fetchCookies(): Promise<DotcomCookies> {
19+
if (cachedCookies) {
20+
return cachedCookies
21+
}
22+
23+
// If request is already in progress, return the same promise
24+
if (inFlightPromise) {
25+
return inFlightPromise
26+
}
27+
28+
if (tries > MAX_TRIES) {
29+
// In prod, fail without a serious error
30+
console.error('Failed to fetch cookies after 3 tries')
31+
// In dev, be loud about the issue
32+
if (process.env.NODE_ENV === 'development') {
33+
throw new Error('Failed to fetch cookies after 3 tries')
34+
}
35+
36+
return Promise.resolve({})
37+
}
38+
39+
inFlightPromise = fetch(GET_COOKIES_ENDPOINT)
40+
.then((response) => {
41+
tries++
42+
if (!response.ok) {
43+
throw new Error(`Failed to fetch cookies: ${response.statusText}`)
44+
}
45+
return response.json() as Promise<DotcomCookies>
46+
})
47+
.then((data) => {
48+
cachedCookies = data
49+
return data
50+
})
51+
.finally(() => {
52+
// Clear the in-flight promise regardless of success or failure
53+
// On success, subsequent calls will return the cached value
54+
// On failure, subsequent calls will retry the request up to MAX_TRIES times
55+
inFlightPromise = null
56+
})
57+
58+
return inFlightPromise
59+
}
60+
61+
export async function getIsStaff(): Promise<boolean> {
62+
const cookies = await fetchCookies()
63+
return cookies.isStaff || false
64+
}
65+
66+
export async function getDotcomUsername(): Promise<string> {
67+
const cookies = await fetchCookies()
68+
return cookies.dotcomUsername || ''
69+
}

src/events/components/experiments/experiment.ts

+2-3
Original file line numberDiff line numberDiff line change
@@ -6,13 +6,13 @@ import {
66
getActiveExperiments,
77
} from './experiments'
88
import { getUserEventsId } from '../events'
9-
import Cookies from 'src/frame/components/lib/cookies'
109

1110
let experimentsInitialized = false
1211

1312
export function shouldShowExperiment(
1413
experimentKey: ExperimentNames | { key: ExperimentNames },
1514
locale: string,
15+
isStaff: boolean,
1616
) {
1717
// Accept either EXPERIMENTS.<experiment_key> or EXPERIMENTS.<experiment_key>.key
1818
if (typeof experimentKey === 'object') {
@@ -25,8 +25,7 @@ export function shouldShowExperiment(
2525
if (experiment.key === experimentKey) {
2626
// If the user has staffonly cookie, and staff override is true, show the experiment
2727
if (experiment.alwaysShowForStaff) {
28-
const staffCookie = Cookies.get('staffonly')
29-
if (staffCookie && staffCookie.startsWith('yes')) {
28+
if (isStaff) {
3029
console.log(`Staff cookie is set, showing '${experiment.key}' experiment`)
3130
return true
3231
}

src/events/components/experiments/useShouldShowExperiment.ts

+4-2
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,7 @@
11
import { useEffect, useState } from 'react'
22
import { shouldShowExperiment } from './experiment'
33
import { ExperimentNames } from './experiments'
4+
import { getIsStaff } from '../dotcom-cookies'
45

56
export function useShouldShowExperiment(
67
experimentKey: ExperimentNames | { key: ExperimentNames },
@@ -13,8 +14,9 @@ export function useShouldShowExperiment(
1314
const [showExperiment, setShowExperiment] = useState(false)
1415

1516
useEffect(() => {
16-
const updateShouldShow = () => {
17-
setShowExperiment(shouldShowExperiment(experimentKey, locale))
17+
const updateShouldShow = async () => {
18+
const isStaff = await getIsStaff()
19+
setShowExperiment(shouldShowExperiment(experimentKey, locale, isStaff))
1820
}
1921

2022
updateShouldShow()

src/frame/middleware/api.ts

+12
Original file line numberDiff line numberDiff line change
@@ -9,6 +9,7 @@ import pageInfo from '@/pageinfo/middleware'
99
import pageList from '@/pagelist/middleware'
1010
import webhooks from '@/webhooks/middleware/webhooks.js'
1111
import { ExtendedRequest } from '@/types'
12+
import { noCacheControl } from './cache-control'
1213

1314
const router = express.Router()
1415

@@ -56,6 +57,17 @@ if (process.env.ELASTICSEARCH_URL) {
5657
)
5758
}
5859

60+
// We need access to specific httpOnly cookies set on github.com from the client
61+
// The only way to access these on the client is to fetch them from the server
62+
router.get('/cookies', (req, res) => {
63+
noCacheControl(res)
64+
const cookies = {
65+
isStaff: Boolean(req.cookies?.staffonly?.startsWith('yes')) || false,
66+
dotcomUsername: req.cookies?.dotcom_user || '',
67+
}
68+
return res.json(cookies)
69+
})
70+
5971
router.get('*', (req, res) => {
6072
res.status(404).json({ error: `${req.path} not found` })
6173
})

src/search/components/input/SearchOverlay.tsx

+15-1
Original file line numberDiff line numberDiff line change
@@ -38,6 +38,7 @@ import { AutocompleteSearchHit } from '@/search/types'
3838
import { useAISearchAutocomplete } from '@/search/components/hooks/useAISearchAutocomplete'
3939
import { AskAIResults } from './AskAIResults'
4040
import { uuidv4 } from '@/events/components/events'
41+
import { getIsStaff } from '@/events/components/dotcom-cookies'
4142

4243
type Props = {
4344
searchOverlayOpen: boolean
@@ -465,7 +466,20 @@ export function SearchOverlay({
465466
backgroundColor: 'var(--overlay-bg-color)',
466467
}}
467468
/>
468-
<Link as="button">Give Feedback</Link>
469+
<Link
470+
onClick={async () => {
471+
if (await getIsStaff()) {
472+
// Hubbers users use an internal discussion for feedback
473+
window.open('https://github.com/github/docs-team/discussions/4952', '_blank')
474+
} else {
475+
// TODO: On ship date set this value
476+
// window.open('TODO', '_blank')
477+
}
478+
}}
479+
as="button"
480+
>
481+
{t('search.overlay.give_feedback')}
482+
</Link>
469483
</footer>
470484
</Overlay>
471485
</>

0 commit comments

Comments
 (0)