-
Notifications
You must be signed in to change notification settings - Fork 513
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(ssr): remove external files dependencies #10885
Changes from 11 commits
38e51ae
e31f711
4722a8f
3f53f72
8568ff7
da96be8
e4ab749
3a7feb4
74a488b
9fb44aa
00346b7
7c80156
8d3f145
7ffbf3b
616a17d
593d82e
cdee340
4e5e120
3e77fc2
0af8d62
c1708e3
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -19,7 +19,9 @@ const appPackage = JSON.parse(fs.readFileSync(resolveApp("package.json"))); | |
const publicUrlOrPath = getPublicUrlOrPath( | ||
process.env.NODE_ENV === "development", | ||
appPackage.homepage, | ||
process.env.PUBLIC_URL | ||
process.env.PUBLIC_URL || | ||
process.env.BASE_URL || | ||
"https://developer.mozilla.org/" | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Add fallback(s). We should replace PUBLIC_URL everywhere, but that's to much for this PR: |
||
); | ||
|
||
const buildPath = process.env.BUILD_PATH || "build"; | ||
|
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -9,13 +9,13 @@ | |
of the file that has a hash in it. | ||
--> | ||
|
||
<link rel="icon" href="%PUBLIC_URL%/favicon-48x48.png" /> | ||
<link rel="icon" href="/favicon-48x48.png" /> | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Just to be safe. Those are relative right now because |
||
|
||
<link rel="apple-touch-icon" href="%PUBLIC_URL%/apple-touch-icon.png" /> | ||
<link rel="apple-touch-icon" href="/apple-touch-icon.png" /> | ||
|
||
<meta name="theme-color" content="#ffffff" /> | ||
|
||
<link rel="manifest" href="%PUBLIC_URL%/manifest.json" /> | ||
<link rel="manifest" href="/manifest.json" /> | ||
|
||
<link | ||
rel="search" | ||
|
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -15,6 +15,7 @@ import printBuildError from "react-dev-utils/printBuildError.js"; | |
|
||
import configFactory from "../config/webpack.config.js"; | ||
import paths from "../config/paths.js"; | ||
import { hashSomeStaticFilesForClientBuild } from "./postprocess-client-build.js"; | ||
|
||
// Makes the script crash on unhandled rejections instead of silently | ||
// ignoring them. In the future, promise rejections that are not handled will | ||
|
@@ -120,6 +121,17 @@ checkBrowsers(paths.appPath, isInteractive) | |
} | ||
} | ||
) | ||
.then(async () => { | ||
const { results } = await hashSomeStaticFilesForClientBuild(paths.appBuild); | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. This is now where the old "optimizeClientBuild" runs. Just after we build the client. |
||
console.log( | ||
chalk.green( | ||
`Hashed ${results.length} files in ${path.join( | ||
paths.appBuild, | ||
"index.html" | ||
)}` | ||
) | ||
); | ||
}) | ||
.catch((err) => { | ||
if (err && err.message) { | ||
console.log(err.message); | ||
|
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -9,7 +9,7 @@ import path from "node:path"; | |
import cheerio from "cheerio"; | ||
import md5File from "md5-file"; | ||
|
||
export async function runOptimizeClientBuild(buildRoot) { | ||
export async function hashSomeStaticFilesForClientBuild(buildRoot) { | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Maybe this optimized in the past. Now it merely hashes files. |
||
const indexHtmlFilePath = path.join(buildRoot, "index.html"); | ||
const indexHtml = fs.readFileSync(indexHtmlFilePath, "utf-8"); | ||
|
||
|
@@ -28,15 +28,6 @@ export async function runOptimizeClientBuild(buildRoot) { | |
} | ||
href = element.attribs.content; | ||
attributeKey = "content"; | ||
// This is an unfortunate hack. The value for the | ||
// <meta property=og:image content=...> needs to be an absolute URL. | ||
// We tested with a relative URL and it seems it doesn't work in Twitter. | ||
// So we hardcode the URL to be our production domain so the URL is | ||
// always absolute. | ||
// Yes, this makes it a bit weird to use a build of this on a dev, | ||
// stage, preview, or a local build. Especially if the hashed URL doesn't | ||
// always work. But it's a fair price to pay. | ||
hrefPrefix = "https://developer.mozilla.org"; | ||
} else { | ||
href = element.attribs.href; | ||
if (!href) { | ||
|
@@ -75,7 +66,7 @@ export async function runOptimizeClientBuild(buildRoot) { | |
const splitName = filePath.split(extName); | ||
const hashedFilePath = `${splitName[0]}.${hash}${extName}`; | ||
fs.copyFileSync(filePath, hashedFilePath); | ||
const hashedHref = filePathToHref(buildRoot, hashedFilePath); | ||
const hashedHref = filePathToHref(buildRoot, hashedFilePath, href); | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Ad the href to restore the domain. |
||
results.push({ | ||
filePath, | ||
href, | ||
|
@@ -101,8 +92,18 @@ export async function runOptimizeClientBuild(buildRoot) { | |
} | ||
|
||
// Turn 'C:\Path\to\client\build\favicon.ico' to '/favicon.ico' | ||
function filePathToHref(root, filePath) { | ||
return `${filePath.replace(root, "").replace(path.sep, "/")}`; | ||
function filePathToHref(root, filePath, href) { | ||
let dummyOrExistingUrl = new URL(href, "http://localhost.example"); | ||
dummyOrExistingUrl.pathname = ""; | ||
let url = new URL( | ||
`${filePath.replace(root, "").replace(path.sep, "/")}`, | ||
dummyOrExistingUrl | ||
); | ||
if (url.hostname === "localhost.example") { | ||
return url.pathname; | ||
} else { | ||
return url.href; | ||
} | ||
} | ||
|
||
// Turn '/favicon.ico' to 'C:\Path\to\client\build\favicon.ico' | ||
|
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -22,14 +22,15 @@ | |
"build:curriculum": "cross-env NODE_ENV=production NODE_OPTIONS='--no-warnings=ExperimentalWarning --loader ts-node/esm' node build/build-curriculum.ts", | ||
"build:dist": "tsc -p tsconfig.dist.json", | ||
"build:glean": "cd client && cross-env VIRTUAL_ENV=venv glean translate src/telemetry/metrics.yaml src/telemetry/pings.yaml -f typescript -o src/telemetry/generated", | ||
"build:prepare": "yarn build:client && yarn build:ssr && yarn tool optimize-client-build && yarn tool google-analytics-code && yarn tool popularities && yarn tool spas && yarn tool gather-git-history && yarn tool build-robots-txt", | ||
"build:ssr": "cd ssr && webpack --mode=production", | ||
"build:prepare": "yarn build:client && yarn build:ssr && yarn tool popularities && yarn tool spas && yarn tool gather-git-history && yarn tool build-robots-txt", | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. optimize-client-build moved into the client build step. |
||
"build:ssr": "cross-env NODE_ENV=production NODE_OPTIONS='--no-warnings=ExperimentalWarning --loader ts-node/esm' node ssr/prepare.ts && cd ssr && webpack --mode=production", | ||
"build:sw": "cd client/pwa && yarn && yarn build:prod", | ||
"build:sw-dev": "cd client/pwa && yarn && yarn build", | ||
"check:tsc": "find . -name 'tsconfig.json' ! -wholename '**/node_modules/**' -print0 | xargs -n1 -P 2 -0 sh -c 'cd `dirname $0` && echo \"🔄 $(pwd)\" && npx tsc --noEmit && echo \"☑️ $(pwd)\" || exit 255'", | ||
"dev": "yarn build:prepare && nf -j Procfile.dev start", | ||
"eslint": "eslint .", | ||
"filecheck": "cross-env NODE_OPTIONS='--no-warnings=ExperimentalWarning --loader ts-node/esm' node filecheck/cli.ts", | ||
"ga": "cross-env NODE_OPTIONS='--no-warnings=ExperimentalWarning --loader ts-node/esm' node ./tool/ga.ts", | ||
"install:all": "find . -mindepth 2 -name 'yarn.lock' ! -wholename '**/node_modules/**' -print0 | xargs -n1 -0 sh -cx 'yarn --cwd $(dirname $0) install'", | ||
"install:all:npm": "find . -mindepth 2 -name 'package-lock.json' ! -wholename '**/node_modules/**' -print0 | xargs -n1 -0 sh -cx 'npm --prefix $(dirname $0) install'", | ||
"jest": "node --experimental-vm-modules --expose-gc ./node_modules/.bin/jest --logHeapUsage", | ||
|
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1 @@ | ||
include.ts |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,47 @@ | ||
import fs from "node:fs"; | ||
import { | ||
BUILD_OUT_ROOT, | ||
GOOGLE_ANALYTICS_MEASUREMENT_ID, | ||
} from "../libs/env/index.js"; | ||
import path from "node:path"; | ||
|
||
export async function generateGA() { | ||
const outFile = path.join(BUILD_OUT_ROOT, "static", "js", "gtag.js"); | ||
const measurementIds = | ||
GOOGLE_ANALYTICS_MEASUREMENT_ID.split(",").filter(Boolean); | ||
if (measurementIds.length) { | ||
const dntHelperCode = fs | ||
.readFileSync( | ||
new URL("mozilla.dnthelper.min.js", import.meta.url), | ||
"utf-8" | ||
) | ||
.trim(); | ||
|
||
const firstMeasurementId = measurementIds[0]; | ||
const gaScriptURL = `https://www.googletagmanager.com/gtag/js?id=${encodeURIComponent(firstMeasurementId)}`; | ||
|
||
const code = ` | ||
// Mozilla DNT Helper | ||
${dntHelperCode} | ||
// Load GA unless DNT is enabled. | ||
if (Mozilla && !Mozilla.dntEnabled()) { | ||
window.dataLayer = window.dataLayer || []; | ||
function gtag(){dataLayer.push(arguments);} | ||
gtag('js', new Date()); | ||
${measurementIds | ||
.map((id) => `gtag('config', '${id}', { 'anonymize_ip': true });`) | ||
.join("\n ")} | ||
|
||
var gaScript = document.createElement('script'); | ||
gaScript.async = true; | ||
gaScript.src = '${gaScriptURL}'; | ||
document.head.appendChild(gaScript); | ||
}`.trim(); | ||
fs.writeFileSync(outFile, `${code}\n`, "utf-8"); | ||
console.log( | ||
`Generated ${outFile} for SSR rendering using ${GOOGLE_ANALYTICS_MEASUREMENT_ID}.` | ||
); | ||
} else { | ||
console.log("No Google Analytics code file generated"); | ||
} | ||
} |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,4 @@ | ||
export const WEBFONT_TAGS: string; | ||
export const GTAG_PATH: null | string; | ||
export const BASE_URL: string; | ||
export const ALWAYS_ALLOW_ROBOTS: boolean; |
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Extracted code from |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,73 @@ | ||
import fs from "node:fs"; | ||
import path from "node:path"; | ||
import { fileURLToPath } from "node:url"; | ||
import { | ||
ALWAYS_ALLOW_ROBOTS, | ||
BUILD_OUT_ROOT, | ||
BASE_URL, | ||
} from "../libs/env/index.js"; | ||
import { generateGA } from "./ga.js"; | ||
|
||
const dirname = path.dirname(fileURLToPath(new URL(".", import.meta.url))); | ||
const clientBuildRoot = path.resolve(dirname, "client/build"); | ||
|
||
function extractWebFontURLs() { | ||
const urls: string[] = []; | ||
const manifest = JSON.parse( | ||
fs.readFileSync(path.join(clientBuildRoot, "asset-manifest.json"), "utf-8") | ||
); | ||
for (const entrypoint of manifest.entrypoints) { | ||
if (!entrypoint.endsWith(".css")) continue; | ||
const css = fs.readFileSync( | ||
path.join(clientBuildRoot, entrypoint), | ||
"utf-8" | ||
); | ||
const generator = extractCSSURLs(css, (url) => url.endsWith(".woff2")); | ||
urls.push(...generator); | ||
} | ||
return [...new Set(urls)]; | ||
} | ||
|
||
function* extractCSSURLs(css, filterFunction) { | ||
for (const match of css.matchAll(/url\((.*?)\)/g)) { | ||
const url = match[1]; | ||
if (filterFunction(url)) { | ||
yield url; | ||
} | ||
} | ||
} | ||
|
||
function webfontTags(webfontURLs): string { | ||
return webfontURLs | ||
.map( | ||
(url) => | ||
`<link rel="preload" as="font" type="font/woff2" href="${url}" crossorigin>` | ||
) | ||
.join(""); | ||
} | ||
|
||
function gtagScriptPath(relPath = "/static/js/gtag.js") { | ||
const filePath = relPath.split("/").slice(1).join(path.sep); | ||
if (fs.existsSync(path.join(BUILD_OUT_ROOT, filePath))) { | ||
return relPath; | ||
} | ||
return null; | ||
} | ||
|
||
function prepare() { | ||
const webfontURLs = extractWebFontURLs(); | ||
const tags = webfontTags(webfontURLs); | ||
const gtagPath = gtagScriptPath(); | ||
|
||
fs.writeFileSync( | ||
path.join(dirname, "ssr", "include.ts"), | ||
` | ||
export const WEBFONT_TAGS = ${JSON.stringify(tags)}; | ||
export const GTAG_PATH = ${JSON.stringify(gtagPath)}; | ||
export const BASE_URL = ${JSON.stringify(BASE_URL)}; | ||
export const ALWAYS_ALLOW_ROBOTS = ${JSON.stringify(ALWAYS_ALLOW_ROBOTS)}; | ||
` | ||
); | ||
} | ||
|
||
generateGA().then(() => prepare()); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Since the file moved.