зеркало из https://github.com/github/docs.git
98 строки
4.1 KiB
JavaScript
98 строки
4.1 KiB
JavaScript
import path from 'path'
|
|
|
|
import got from 'got'
|
|
|
|
import patterns from '../lib/patterns.js'
|
|
import isArchivedVersion from '../lib/is-archived-version.js'
|
|
import { setFastlySurrogateKey, SURROGATE_ENUMS } from './set-fastly-surrogate-key.js'
|
|
import { archivedCacheControl } from './cache-control.js'
|
|
|
|
// This module handles requests for the CSS and JS assets for
|
|
// deprecated GitHub Enterprise versions by routing them to static content in
|
|
// help-docs-archived-enterprise-versions
|
|
//
|
|
// Note that as of GHES 3.2, we no longer store assets for deprecated versions
|
|
// in help-docs-archived-enterprise-versions. Instead, we store them in the
|
|
// Azure blob storage `githubdocs` in the `enterprise` container. All HTML files
|
|
// have been updated to use references to this blob storage for all assets.
|
|
//
|
|
// See also ./archived-enterprise-versions.js for non-CSS/JS paths
|
|
|
|
export default async function archivedEnterpriseVersionsAssets(req, res, next) {
|
|
// Only match asset paths
|
|
// This can be true on /enterprise/2.22/_next/static/foo.css
|
|
// or /_next/static/foo.css
|
|
if (!patterns.assetPaths.test(req.path)) return next()
|
|
|
|
// We now know the URL is either /enterprise/2.22/_next/static/foo.css
|
|
// or the regular /_next/static/foo.css. But we're only going to
|
|
// bother looking it up on https://github.github.com/help-docs-archived-enterprise-versions
|
|
// if the URL has the enterprise bit in it, or if the path was
|
|
// /_next/static/foo.css *and* its Referrer had the enterprise
|
|
// bit in it.
|
|
if (
|
|
!(
|
|
patterns.getEnterpriseVersionNumber.test(req.path) ||
|
|
patterns.getEnterpriseServerNumber.test(req.path) ||
|
|
patterns.getEnterpriseVersionNumber.test(req.get('referrer')) ||
|
|
patterns.getEnterpriseServerNumber.test(req.get('referrer'))
|
|
)
|
|
) {
|
|
return next()
|
|
}
|
|
|
|
// Now we know the URL is definitely not /_next/static/foo.css
|
|
// So it's probably /enterprise/2.22/_next/static/foo.css and we
|
|
// should see if we might find this in the proxied backend.
|
|
// But `isArchivedVersion()` will only return truthy if the
|
|
// Referrer header also indicates that the request for this static
|
|
// asset came from a page
|
|
const { isArchived, requestedVersion } = isArchivedVersion(req)
|
|
if (!isArchived) return next()
|
|
|
|
const assetPath = req.path.replace(`/enterprise/${requestedVersion}`, '')
|
|
const proxyPath = path.join('/', requestedVersion, assetPath)
|
|
|
|
try {
|
|
const r = await got(
|
|
`https://github.github.com/help-docs-archived-enterprise-versions${proxyPath}`
|
|
)
|
|
res.set('accept-ranges', 'bytes')
|
|
res.set('content-type', r.headers['content-type'])
|
|
res.set('content-length', r.headers['content-length'])
|
|
res.set('x-is-archived', 'true')
|
|
res.set('x-robots-tag', 'noindex')
|
|
|
|
// This cache configuration should match what we do for archived
|
|
// enterprise version URLs that are not assets.
|
|
archivedCacheControl(res)
|
|
setFastlySurrogateKey(res, SURROGATE_ENUMS.MANUAL)
|
|
|
|
return res.send(r.body)
|
|
} catch (err) {
|
|
// Primarly for the developers working on tests that mock
|
|
// requests. If you don't set up `nock` correctly, you might
|
|
// not realize that and think it failed for other reasons.
|
|
if (err.toString().includes('Nock: No match for request')) {
|
|
throw err
|
|
}
|
|
|
|
// It's important that we don't give up on this by returning a 404
|
|
// here. It's better to let this through in case the asset exists
|
|
// beyond the realm of archived enterprise versions.
|
|
// For example, image you load
|
|
// /enterprise-server@2.21/en/DOES/NOT/EXIST in your browser.
|
|
// Quickly, we discover that the proxying is failing because
|
|
// it didn't find a page called `/en/DOES/NOT/EXIST` over there.
|
|
// So, we proceed to render *our* 404 HTML page.
|
|
// Now, on that 404 page, it will reference static assets too.
|
|
// E.g. <link href="/_next/static/styles.css">
|
|
// These will thus be requested, with a Referrer header that
|
|
// forces us to give it a chance, but it'll find it can't find it
|
|
// but we mustn't return a 404 yet, because that
|
|
// /_next/static/styles.css will probably still succeed because the 404
|
|
// page is not that of the archived enterprise version.
|
|
return next()
|
|
}
|
|
}
|