forked from github/docs
-
Notifications
You must be signed in to change notification settings - Fork 0
/
context.js
96 lines (88 loc) · 4.19 KB
/
context.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
import languages from '#src/languages/lib/languages.js'
import enterpriseServerReleases from '#src/versions/lib/enterprise-server-releases.js'
import { allVersions } from '#src/versions/lib/all-versions.js'
import { productMap } from '../lib/all-products.js'
import {
getVersionStringFromPath,
getProductStringFromPath,
getCategoryStringFromPath,
getPathWithoutLanguage,
getPathWithoutVersion,
} from '../lib/path-utils.js'
import productNames from '../lib/product-names.js'
import warmServer from '../lib/warm-server.js'
import searchVersions from '../src/search/lib/versions.js'
import nonEnterpriseDefaultVersion from '#src/versions/lib/non-enterprise-default-version.js'
import { getDataByLanguage, getUIDataMerged } from '../lib/get-data.js'
// This doesn't change just because the request changes, so compute it once.
const enterpriseServerVersions = Object.keys(allVersions).filter((version) =>
version.startsWith('enterprise-server@'),
)
// Supply all route handlers with a baseline `req.context` object
// Note that additional middleware in middleware/index.js adds to this context object
export default async function contextualize(req, res, next) {
// Ensure that we load some data only once on first request
const { redirects, siteTree, pages: pageMap } = await warmServer()
req.context = {}
req.context.process = { env: {} }
// define each context property explicitly for code-search friendliness
// e.g. searches for "req.context.page" will include results from this file
req.context.currentLanguage = req.language
req.context.userLanguage = req.userLanguage
req.context.currentVersion = getVersionStringFromPath(req.pagePath)
req.context.currentVersionObj = allVersions[req.context.currentVersion]
req.context.currentProduct = getProductStringFromPath(req.pagePath)
req.context.currentCategory = getCategoryStringFromPath(req.pagePath)
req.context.productMap = productMap
req.context.allVersions = allVersions
req.context.currentPathWithoutLanguage = getPathWithoutLanguage(req.pagePath)
// define property for writers to link to the current page in a different version
// includes any type of rendered page not just "articles"
req.context.currentArticle = getPathWithoutVersion(req.context.currentPathWithoutLanguage)
req.context.currentPath = req.pagePath
req.context.query = req.query
req.context.languages = languages
req.context.productNames = productNames
req.context.enterpriseServerReleases = enterpriseServerReleases
req.context.enterpriseServerVersions = enterpriseServerVersions
req.context.redirects = redirects
req.context.site = {
data: {
ui: getUIDataMerged(req.language),
},
}
req.context.getDottedData = (dottedPath) => getDataByLanguage(dottedPath, req.language)
req.context.siteTree = siteTree
req.context.pages = pageMap
req.context.searchVersions = searchVersions
req.context.nonEnterpriseDefaultVersion = nonEnterpriseDefaultVersion
req.context.initialRestVersioningReleaseDate =
allVersions[nonEnterpriseDefaultVersion].apiVersions[0]
const restDate = new Date(req.context.initialRestVersioningReleaseDate)
req.context.initialRestVersioningReleaseDateLong = restDate.toUTCString().split(' 00:')[0]
// Conditionally add this for non-English pages so what inside the
// `Page.render` method, when it calls out to `renderContentWithFallback`
// it can be able to fall back get original content from English if there's
// some runtime rendering error from the translation.
if (req.language !== 'en') {
// The reason this is a function is because most of the time, we don't
// need to know the English equivalent. It only comes into play if a
// translated
req.context.getEnglishPage = (context) => {
if (!context.enPage) {
const { page } = context
if (!page) {
throw new Error("The 'page' has not been put into the context yet.")
}
const enPath = context.currentPath.replace(`/${page.languageCode}`, '/en')
const enPage = context.pages[enPath]
if (!enPage) {
throw new Error(`Unable to find equivalent English page by the path '${enPath}'`)
}
context.enPage = enPage
}
return context.enPage
}
}
return next()
}