From 38425ba2e9e0ce08755c8463464d9a74968d7659 Mon Sep 17 00:00:00 2001 From: Andrew Clark Date: Fri, 14 Oct 2022 18:07:29 -0400 Subject: [PATCH 1/3] Move Fizz inline instructions to unified module Instead of a separate module per instruction, this exports all of them from a unified module. In the next step, I'll add a script to generate this new module. --- .../src/server/ReactDOMServerFormatConfig.js | 10 +- ...tDOMFizzInstructionSetInlineCodeStrings.js | 212 ++++++++++++++++++ .../clientRenderFunctionString.js | 38 ---- .../completeBoundaryFunctionString.js | 82 ------- .../completeSegmentFunctionString.js | 31 --- .../styleInsertionFunctionString.js | 100 --------- 6 files changed, 218 insertions(+), 255 deletions(-) create mode 100644 packages/react-dom-bindings/src/server/fizz-instruction-set/ReactDOMFizzInstructionSetInlineCodeStrings.js delete mode 100644 packages/react-dom-bindings/src/server/fizz-instruction-set/clientRenderFunctionString.js delete mode 100644 packages/react-dom-bindings/src/server/fizz-instruction-set/completeBoundaryFunctionString.js delete mode 100644 packages/react-dom-bindings/src/server/fizz-instruction-set/completeSegmentFunctionString.js delete mode 100644 packages/react-dom-bindings/src/server/fizz-instruction-set/styleInsertionFunctionString.js diff --git a/packages/react-dom-bindings/src/server/ReactDOMServerFormatConfig.js b/packages/react-dom-bindings/src/server/ReactDOMServerFormatConfig.js index ae5ccf75e7109..f5c95eaa37c69 100644 --- a/packages/react-dom-bindings/src/server/ReactDOMServerFormatConfig.js +++ b/packages/react-dom-bindings/src/server/ReactDOMServerFormatConfig.js @@ -74,10 +74,12 @@ export { hoistResourcesToRoot, } from './ReactDOMFloatServer'; -import completeSegmentFunction from './fizz-instruction-set/completeSegmentFunctionString'; -import completeBoundaryFunction from './fizz-instruction-set/completeBoundaryFunctionString'; -import styleInsertionFunction from './fizz-instruction-set/styleInsertionFunctionString'; -import clientRenderFunction from './fizz-instruction-set/clientRenderFunctionString'; +import { + clientRenderBoundary as clientRenderFunction, + completeBoundary as completeBoundaryFunction, + completeBoundaryWithStyles as styleInsertionFunction, + completeSegment as completeSegmentFunction, +} from './fizz-instruction-set/ReactDOMFizzInstructionSetInlineCodeStrings'; import ReactDOMSharedInternals from 'shared/ReactDOMSharedInternals'; const ReactDOMCurrentDispatcher = ReactDOMSharedInternals.Dispatcher; diff --git a/packages/react-dom-bindings/src/server/fizz-instruction-set/ReactDOMFizzInstructionSetInlineCodeStrings.js b/packages/react-dom-bindings/src/server/fizz-instruction-set/ReactDOMFizzInstructionSetInlineCodeStrings.js new file mode 100644 index 0000000000000..c6668e3f8cf30 --- /dev/null +++ b/packages/react-dom-bindings/src/server/fizz-instruction-set/ReactDOMFizzInstructionSetInlineCodeStrings.js @@ -0,0 +1,212 @@ +// Instruction Set + +// The following code is the source scripts that we then minify and inline below, +// with renamed function names that we hope don't collide: + +// const COMMENT_NODE = 8; +// const SUSPENSE_START_DATA = '$'; +// const SUSPENSE_END_DATA = '/$'; +// const SUSPENSE_PENDING_START_DATA = '$?'; +// const SUSPENSE_FALLBACK_START_DATA = '$!'; +// const LOADED = 'l'; +// const ERRORED = 'e'; + +// function clientRenderBoundary(suspenseBoundaryID, errorDigest, errorMsg, errorComponentStack) { +// // Find the fallback's first element. +// const suspenseIdNode = document.getElementById(suspenseBoundaryID); +// if (!suspenseIdNode) { +// // The user must have already navigated away from this tree. +// // E.g. because the parent was hydrated. +// return; +// } +// // Find the boundary around the fallback. This is always the previous node. +// const suspenseNode = suspenseIdNode.previousSibling; +// // Tag it to be client rendered. +// suspenseNode.data = SUSPENSE_FALLBACK_START_DATA; +// // assign error metadata to first sibling +// let dataset = suspenseIdNode.dataset; +// if (errorDigest) dataset.dgst = errorDigest; +// if (errorMsg) dataset.msg = errorMsg; +// if (errorComponentStack) dataset.stck = errorComponentStack; +// // Tell React to retry it if the parent already hydrated. +// if (suspenseNode._reactRetry) { +// suspenseNode._reactRetry(); +// } +// } + +// resourceMap = new Map(); +// function completeBoundaryWithStyles(suspenseBoundaryID, contentID, styles) { +// const precedences = new Map(); +// const thisDocument = document; +// let lastResource, node; + +// // Seed the precedence list with existing resources +// let nodes = thisDocument.querySelectorAll('link[data-rprec]'); +// for (let i = 0;node = nodes[i++];) { +// precedences.set(node.dataset.rprec, lastResource = node); +// } + +// let i = 0; +// let dependencies = []; +// let style, href, precedence, attr, loadingState, resourceEl; + +// function setStatus(s) { +// this.s = s; +// } + +// while (style = styles[i++]) { +// let j = 0; +// href = style[j++]; +// // We check if this resource is already in our resourceMap and reuse it if so. +// // If it is already loaded we don't return it as a depenendency since there is nothing +// // to wait for +// loadingState = resourceMap.get(href); +// if (loadingState) { +// if (loadingState.s !== 'l') { +// dependencies.push(loadingState); +// } +// continue; +// } + +// // We construct our new resource element, looping over remaining attributes if any +// // setting them to the Element. +// resourceEl = thisDocument.createElement("link"); +// resourceEl.href = href; +// resourceEl.rel = 'stylesheet'; +// resourceEl.dataset.rprec = precedence = style[j++]; +// while(attr = style[j++]) { +// resourceEl.setAttribute(attr, style[j++]); +// } + +// // We stash a pending promise in our map by href which will resolve or reject +// // when the underlying resource loads or errors. We add it to the dependencies +// // array to be returned. +// loadingState = resourceEl._p = new Promise((re, rj) => { +// resourceEl.onload = re; +// resourceEl.onerror = rj; +// }) +// loadingState.then( +// setStatus.bind(loadingState, LOADED), +// setStatus.bind(loadingState, ERRORED) +// ); +// resourceMap.set(href, loadingState); +// dependencies.push(loadingState); + +// // The prior style resource is the last one placed at a given +// // precedence or the last resource itself which may be null. +// // We grab this value and then update the last resource for this +// // precedence to be the inserted element, updating the lastResource +// // pointer if needed. +// let prior = precedences.get(precedence) || lastResource; +// if (prior === lastResource) { +// lastResource = resourceEl +// } +// precedences.set(precedence, resourceEl) + +// // Finally, we insert the newly constructed instance at an appropriate location +// // in the Document. +// if (prior) { +// prior.parentNode.insertBefore(resourceEl, prior.nextSibling); +// } else { +// let head = thisDocument.head; +// head.insertBefore(resourceEl, head.firstChild); +// } +// } + +// Promise.all(dependencies).then( +// completeBoundary.bind(null, suspenseBoundaryID, contentID, ''), +// completeBoundary.bind(null, suspenseBoundaryID, contentID, "Resource failed to load") +// ); +// } + +// function completeBoundary(suspenseBoundaryID, contentID, errorDigest) { +// const contentNode = document.getElementById(contentID); +// // We'll detach the content node so that regardless of what happens next we don't leave in the tree. +// // This might also help by not causing recalcing each time we move a child from here to the target. +// contentNode.parentNode.removeChild(contentNode); + +// // Find the fallback's first element. +// const suspenseIdNode = document.getElementById(suspenseBoundaryID); +// if (!suspenseIdNode) { +// // The user must have already navigated away from this tree. +// // E.g. because the parent was hydrated. That's fine there's nothing to do +// // but we have to make sure that we already deleted the container node. +// return; +// } +// // Find the boundary around the fallback. This is always the previous node. +// const suspenseNode = suspenseIdNode.previousSibling; + +// if (!errorDigest) { +// // Clear all the existing children. This is complicated because +// // there can be embedded Suspense boundaries in the fallback. +// // This is similar to clearSuspenseBoundary in ReactDOMHostConfig. +// // TODO: We could avoid this if we never emitted suspense boundaries in fallback trees. +// // They never hydrate anyway. However, currently we support incrementally loading the fallback. +// const parentInstance = suspenseNode.parentNode; +// let node = suspenseNode.nextSibling; +// let depth = 0; +// do { +// if (node && node.nodeType === COMMENT_NODE) { +// const data = node.data; +// if (data === SUSPENSE_END_DATA) { +// if (depth === 0) { +// break; +// } else { +// depth--; +// } +// } else if ( +// data === SUSPENSE_START_DATA || +// data === SUSPENSE_PENDING_START_DATA || +// data === SUSPENSE_FALLBACK_START_DATA +// ) { +// depth++; +// } +// } + +// const nextNode = node.nextSibling; +// parentInstance.removeChild(node); +// node = nextNode; +// } while (node); + +// const endOfBoundary = node; + +// // Insert all the children from the contentNode between the start and end of suspense boundary. +// while (contentNode.firstChild) { +// parentInstance.insertBefore(contentNode.firstChild, endOfBoundary); +// } + +// suspenseNode.data = SUSPENSE_START_DATA; +// } else { +// suspenseNode.data = SUSPENSE_FALLBACK_START_DATA; +// suspenseIdNode.setAttribute('data-dgst', errorDigest) +// } + +// if (suspenseNode._reactRetry) { +// suspenseNode._reactRetry(); +// } +// } + +// function completeSegment(containerID, placeholderID) { +// const segmentContainer = document.getElementById(containerID); +// const placeholderNode = document.getElementById(placeholderID); +// // We always expect both nodes to exist here because, while we might +// // have navigated away from the main tree, we still expect the detached +// // tree to exist. +// segmentContainer.parentNode.removeChild(segmentContainer); +// while (segmentContainer.firstChild) { +// placeholderNode.parentNode.insertBefore( +// segmentContainer.firstChild, +// placeholderNode, +// ); +// } +// placeholderNode.parentNode.removeChild(placeholderNode); +// } + +export const clientRenderBoundary = + '$RX=function(b,c,d,e){var a=document.getElementById(b);a&&(b=a.previousSibling,b.data="$!",a=a.dataset,c&&(a.dgst=c),d&&(a.msg=d),e&&(a.stck=e),b._reactRetry&&b._reactRetry())};'; +export const completeBoundary = + '$RC=function(b,c,e){c=document.getElementById(c);c.parentNode.removeChild(c);var a=document.getElementById(b);if(a){b=a.previousSibling;if(e)b.data="$!",a.setAttribute("data-dgst",e);else{e=b.parentNode;a=b.nextSibling;var f=0;do{if(a&&8===a.nodeType){var d=a.data;if("/$"===d)if(0===f)break;else f--;else"$"!==d&&"$?"!==d&&"$!"!==d||f++}d=a.nextSibling;e.removeChild(a);a=d}while(a);for(;c.firstChild;)e.insertBefore(c.firstChild,a);b.data="$"}b._reactRetry&&b._reactRetry()}};'; +export const completeBoundaryWithStyles = + '$RM=new Map;\n$RR=function(p,q,v){function r(l){this.s=l}for(var t=$RC,u=$RM,m=new Map,n=document,g,e,f=n.querySelectorAll("link[data-rprec]"),d=0;e=f[d++];)m.set(e.dataset.rprec,g=e);e=0;f=[];for(var c,h,b,a;c=v[e++];){var k=0;h=c[k++];if(b=u.get(h))"l"!==b.s&&f.push(b);else{a=n.createElement("link");a.href=h;a.rel="stylesheet";for(a.dataset.rprec=d=c[k++];b=c[k++];)a.setAttribute(b,c[k++]);b=a._p=new Promise(function(l,w){a.onload=l;a.onerror=w});b.then(r.bind(b,"l"),r.bind(b,"e"));u.set(h,\nb);f.push(b);c=m.get(d)||g;c===g&&(g=a);m.set(d,a);c?c.parentNode.insertBefore(a,c.nextSibling):(d=n.head,d.insertBefore(a,d.firstChild))}}Promise.all(f).then(t.bind(null,p,q,""),t.bind(null,p,q,"Resource failed to load"))};'; +export const completeSegment = + '$RS=function(a,b){a=document.getElementById(a);b=document.getElementById(b);for(a.parentNode.removeChild(a);a.firstChild;)b.parentNode.insertBefore(a.firstChild,b);b.parentNode.removeChild(b)};'; diff --git a/packages/react-dom-bindings/src/server/fizz-instruction-set/clientRenderFunctionString.js b/packages/react-dom-bindings/src/server/fizz-instruction-set/clientRenderFunctionString.js deleted file mode 100644 index d2c2a4b8efbd4..0000000000000 --- a/packages/react-dom-bindings/src/server/fizz-instruction-set/clientRenderFunctionString.js +++ /dev/null @@ -1,38 +0,0 @@ -// Instruction Set - -// The following code is the source scripts that we then minify and inline below, -// with renamed function names that we hope don't collide: - -// const COMMENT_NODE = 8; -// const SUSPENSE_START_DATA = '$'; -// const SUSPENSE_END_DATA = '/$'; -// const SUSPENSE_PENDING_START_DATA = '$?'; -// const SUSPENSE_FALLBACK_START_DATA = '$!'; -// const LOADED = 'l'; -// const ERRORED = 'e'; - -// function clientRenderBoundary(suspenseBoundaryID, errorDigest, errorMsg, errorComponentStack) { -// // Find the fallback's first element. -// const suspenseIdNode = document.getElementById(suspenseBoundaryID); -// if (!suspenseIdNode) { -// // The user must have already navigated away from this tree. -// // E.g. because the parent was hydrated. -// return; -// } -// // Find the boundary around the fallback. This is always the previous node. -// const suspenseNode = suspenseIdNode.previousSibling; -// // Tag it to be client rendered. -// suspenseNode.data = SUSPENSE_FALLBACK_START_DATA; -// // assign error metadata to first sibling -// let dataset = suspenseIdNode.dataset; -// if (errorDigest) dataset.dgst = errorDigest; -// if (errorMsg) dataset.msg = errorMsg; -// if (errorComponentStack) dataset.stck = errorComponentStack; -// // Tell React to retry it if the parent already hydrated. -// if (suspenseNode._reactRetry) { -// suspenseNode._reactRetry(); -// } -// } - -// TODO: Generate this file with a build step. -export default 'function $RX(b,c,d,e){var a=document.getElementById(b);a&&(b=a.previousSibling,b.data="$!",a=a.dataset,c&&(a.dgst=c),d&&(a.msg=d),e&&(a.stck=e),b._reactRetry&&b._reactRetry())}'; diff --git a/packages/react-dom-bindings/src/server/fizz-instruction-set/completeBoundaryFunctionString.js b/packages/react-dom-bindings/src/server/fizz-instruction-set/completeBoundaryFunctionString.js deleted file mode 100644 index cd1def26fcecb..0000000000000 --- a/packages/react-dom-bindings/src/server/fizz-instruction-set/completeBoundaryFunctionString.js +++ /dev/null @@ -1,82 +0,0 @@ -// Instruction Set - -// The following code is the source scripts that we then minify and inline below, -// with renamed function names that we hope don't collide: - -// const COMMENT_NODE = 8; -// const SUSPENSE_START_DATA = '$'; -// const SUSPENSE_END_DATA = '/$'; -// const SUSPENSE_PENDING_START_DATA = '$?'; -// const SUSPENSE_FALLBACK_START_DATA = '$!'; -// const LOADED = 'l'; -// const ERRORED = 'e'; - -// function completeBoundary(suspenseBoundaryID, contentID, errorDigest) { -// const contentNode = document.getElementById(contentID); -// // We'll detach the content node so that regardless of what happens next we don't leave in the tree. -// // This might also help by not causing recalcing each time we move a child from here to the target. -// contentNode.parentNode.removeChild(contentNode); - -// // Find the fallback's first element. -// const suspenseIdNode = document.getElementById(suspenseBoundaryID); -// if (!suspenseIdNode) { -// // The user must have already navigated away from this tree. -// // E.g. because the parent was hydrated. That's fine there's nothing to do -// // but we have to make sure that we already deleted the container node. -// return; -// } -// // Find the boundary around the fallback. This is always the previous node. -// const suspenseNode = suspenseIdNode.previousSibling; - -// if (!errorDigest) { -// // Clear all the existing children. This is complicated because -// // there can be embedded Suspense boundaries in the fallback. -// // This is similar to clearSuspenseBoundary in ReactDOMHostConfig. -// // TODO: We could avoid this if we never emitted suspense boundaries in fallback trees. -// // They never hydrate anyway. However, currently we support incrementally loading the fallback. -// const parentInstance = suspenseNode.parentNode; -// let node = suspenseNode.nextSibling; -// let depth = 0; -// do { -// if (node && node.nodeType === COMMENT_NODE) { -// const data = node.data; -// if (data === SUSPENSE_END_DATA) { -// if (depth === 0) { -// break; -// } else { -// depth--; -// } -// } else if ( -// data === SUSPENSE_START_DATA || -// data === SUSPENSE_PENDING_START_DATA || -// data === SUSPENSE_FALLBACK_START_DATA -// ) { -// depth++; -// } -// } - -// const nextNode = node.nextSibling; -// parentInstance.removeChild(node); -// node = nextNode; -// } while (node); - -// const endOfBoundary = node; - -// // Insert all the children from the contentNode between the start and end of suspense boundary. -// while (contentNode.firstChild) { -// parentInstance.insertBefore(contentNode.firstChild, endOfBoundary); -// } - -// suspenseNode.data = SUSPENSE_START_DATA; -// } else { -// suspenseNode.data = SUSPENSE_FALLBACK_START_DATA; -// suspenseIdNode.setAttribute('data-dgst', errorDigest) -// } - -// if (suspenseNode._reactRetry) { -// suspenseNode._reactRetry(); -// } -// } - -// TODO: Generate this file with a build step. -export default 'function $RC(b,c,d){c=document.getElementById(c);c.parentNode.removeChild(c);var a=document.getElementById(b);if(a){b=a.previousSibling;if(d)b.data="$!",a.setAttribute("data-dgst",d);else{d=b.parentNode;a=b.nextSibling;var e=0;do{if(a&&a.nodeType===8){var h=a.data;if(h==="/$")if(0===e)break;else e--;else h!=="$"&&h!=="$?"&&h!=="$!"||e++}h=a.nextSibling;d.removeChild(a);a=h}while(a);for(;c.firstChild;)d.insertBefore(c.firstChild,a);b.data="$"}b._reactRetry&&b._reactRetry()}}'; diff --git a/packages/react-dom-bindings/src/server/fizz-instruction-set/completeSegmentFunctionString.js b/packages/react-dom-bindings/src/server/fizz-instruction-set/completeSegmentFunctionString.js deleted file mode 100644 index d88785341b739..0000000000000 --- a/packages/react-dom-bindings/src/server/fizz-instruction-set/completeSegmentFunctionString.js +++ /dev/null @@ -1,31 +0,0 @@ -// Instruction Set - -// The following code is the source scripts that we then minify and inline below, -// with renamed function names that we hope don't collide: - -// const COMMENT_NODE = 8; -// const SUSPENSE_START_DATA = '$'; -// const SUSPENSE_END_DATA = '/$'; -// const SUSPENSE_PENDING_START_DATA = '$?'; -// const SUSPENSE_FALLBACK_START_DATA = '$!'; -// const LOADED = 'l'; -// const ERRORED = 'e'; - -// function completeSegment(containerID, placeholderID) { -// const segmentContainer = document.getElementById(containerID); -// const placeholderNode = document.getElementById(placeholderID); -// // We always expect both nodes to exist here because, while we might -// // have navigated away from the main tree, we still expect the detached -// // tree to exist. -// segmentContainer.parentNode.removeChild(segmentContainer); -// while (segmentContainer.firstChild) { -// placeholderNode.parentNode.insertBefore( -// segmentContainer.firstChild, -// placeholderNode, -// ); -// } -// placeholderNode.parentNode.removeChild(placeholderNode); -// } - -// TODO: Generate this file with a build step. -export default 'function $RS(a,b){a=document.getElementById(a);b=document.getElementById(b);for(a.parentNode.removeChild(a);a.firstChild;)b.parentNode.insertBefore(a.firstChild,b);b.parentNode.removeChild(b)}'; diff --git a/packages/react-dom-bindings/src/server/fizz-instruction-set/styleInsertionFunctionString.js b/packages/react-dom-bindings/src/server/fizz-instruction-set/styleInsertionFunctionString.js deleted file mode 100644 index d143400aa0ebe..0000000000000 --- a/packages/react-dom-bindings/src/server/fizz-instruction-set/styleInsertionFunctionString.js +++ /dev/null @@ -1,100 +0,0 @@ -// Instruction Set - -// The following code is the source scripts that we then minify and inline below, -// with renamed function names that we hope don't collide: - -// const COMMENT_NODE = 8; -// const SUSPENSE_START_DATA = '$'; -// const SUSPENSE_END_DATA = '/$'; -// const SUSPENSE_PENDING_START_DATA = '$?'; -// const SUSPENSE_FALLBACK_START_DATA = '$!'; -// const LOADED = 'l'; -// const ERRORED = 'e'; - -// resourceMap = new Map(); -// function completeBoundaryWithStyles(suspenseBoundaryID, contentID, styles) { -// const precedences = new Map(); -// const thisDocument = document; -// let lastResource, node; - -// // Seed the precedence list with existing resources -// let nodes = thisDocument.querySelectorAll('link[data-rprec]'); -// for (let i = 0;node = nodes[i++];) { -// precedences.set(node.dataset.rprec, lastResource = node); -// } - -// let i = 0; -// let dependencies = []; -// let style, href, precedence, attr, loadingState, resourceEl; - -// function setStatus(s) { -// this.s = s; -// } - -// while (style = styles[i++]) { -// let j = 0; -// href = style[j++]; -// // We check if this resource is already in our resourceMap and reuse it if so. -// // If it is already loaded we don't return it as a depenendency since there is nothing -// // to wait for -// loadingState = resourceMap.get(href); -// if (loadingState) { -// if (loadingState.s !== 'l') { -// dependencies.push(loadingState); -// } -// continue; -// } - -// // We construct our new resource element, looping over remaining attributes if any -// // setting them to the Element. -// resourceEl = thisDocument.createElement("link"); -// resourceEl.href = href; -// resourceEl.rel = 'stylesheet'; -// resourceEl.dataset.rprec = precedence = style[j++]; -// while(attr = style[j++]) { -// resourceEl.setAttribute(attr, style[j++]); -// } - -// // We stash a pending promise in our map by href which will resolve or reject -// // when the underlying resource loads or errors. We add it to the dependencies -// // array to be returned. -// loadingState = resourceEl._p = new Promise((re, rj) => { -// resourceEl.onload = re; -// resourceEl.onerror = rj; -// }) -// loadingState.then( -// setStatus.bind(loadingState, LOADED), -// setStatus.bind(loadingState, ERRORED) -// ); -// resourceMap.set(href, loadingState); -// dependencies.push(loadingState); - -// // The prior style resource is the last one placed at a given -// // precedence or the last resource itself which may be null. -// // We grab this value and then update the last resource for this -// // precedence to be the inserted element, updating the lastResource -// // pointer if needed. -// let prior = precedences.get(precedence) || lastResource; -// if (prior === lastResource) { -// lastResource = resourceEl -// } -// precedences.set(precedence, resourceEl) - -// // Finally, we insert the newly constructed instance at an appropriate location -// // in the Document. -// if (prior) { -// prior.parentNode.insertBefore(resourceEl, prior.nextSibling); -// } else { -// let head = thisDocument.head; -// head.insertBefore(resourceEl, head.firstChild); -// } -// } - -// Promise.all(dependencies).then( -// completeBoundary.bind(null, suspenseBoundaryID, contentID, ''), -// completeBoundary.bind(null, suspenseBoundaryID, contentID, "Resource failed to load") -// ); -// } - -// TODO: Generate this file with a build step. -export default '$RM=new Map;function $RR(p,q,t){function r(l){this.s=l}for(var m=new Map,n=document,g,e,f=n.querySelectorAll("link[data-rprec]"),d=0;e=f[d++];)m.set(e.dataset.rprec,g=e);e=0;f=[];for(var c,h,b,a;c=t[e++];){var k=0;h=c[k++];if(b=$RM.get(h))"l"!==b.s&&f.push(b);else{a=n.createElement("link");a.href=h;a.rel="stylesheet";for(a.dataset.rprec=d=c[k++];b=c[k++];)a.setAttribute(b,c[k++]);b=a._p=new Promise(function(l,u){a.onload=l;a.onerror=u});b.then(r.bind(b,"l"),r.bind(b,"e"));$RM.set(h,b);f.push(b);c=m.get(d)||g;c===g&&(g=a);m.set(d,a);c?c.parentNode.insertBefore(a,c.nextSibling):(d=n.head,d.insertBefore(a,d.firstChild))}}Promise.all(f).then($RC.bind(null,p,q,""),$RC.bind(null,p,q,"Resource failed to load"))}'; From e1287984e1be46aa73526303eeab5b64378c37bc Mon Sep 17 00:00:00 2001 From: Andrew Clark Date: Fri, 14 Oct 2022 18:15:30 -0400 Subject: [PATCH 2/3] Add script to generate inline Fizz runtime This adds a script to generate the inline Fizz runtime. Previously, the runtime source was in an inline comment, and a compiled version of the instructions were hardcoded as strings into the Fizz implementation, where they are injected into the HTML stream. I've moved the source for the instructions to a regular JavaScript module. A script compiles the instructions with Closure, then generates another module that exports the compiled instructions as strings. Then the Fizz runtime imports the instructions from the generated module. To build the instructions, run: yarn generate-inline-fizz-runtime In the next step, I'll add a CI check to verify that the generated files are up to date. --- package.json | 3 +- .../ReactDOMFizzInlineClientRenderBoundary.js | 5 + .../ReactDOMFizzInlineCompleteBoundary.js | 5 + ...DOMFizzInlineCompleteBoundaryWithStyles.js | 7 + .../ReactDOMFizzInlineCompleteSegment.js | 5 + .../ReactDOMFizzInstructionSet.js | 224 ++++++++++++++++++ ...tDOMFizzInstructionSetInlineCodeStrings.js | 207 +--------------- .../rollup/generate-inline-fizz-runtime.js | 88 +++++++ 8 files changed, 339 insertions(+), 205 deletions(-) create mode 100644 packages/react-dom-bindings/src/server/fizz-instruction-set/ReactDOMFizzInlineClientRenderBoundary.js create mode 100644 packages/react-dom-bindings/src/server/fizz-instruction-set/ReactDOMFizzInlineCompleteBoundary.js create mode 100644 packages/react-dom-bindings/src/server/fizz-instruction-set/ReactDOMFizzInlineCompleteBoundaryWithStyles.js create mode 100644 packages/react-dom-bindings/src/server/fizz-instruction-set/ReactDOMFizzInlineCompleteSegment.js create mode 100644 packages/react-dom-bindings/src/server/fizz-instruction-set/ReactDOMFizzInstructionSet.js create mode 100644 scripts/rollup/generate-inline-fizz-runtime.js diff --git a/package.json b/package.json index b3288286d7842..12c3022c2a360 100644 --- a/package.json +++ b/package.json @@ -145,7 +145,8 @@ "download-build": "node ./scripts/release/download-experimental-build.js", "download-build-for-head": "node ./scripts/release/download-experimental-build.js --commit=$(git rev-parse HEAD)", "download-build-in-codesandbox-ci": "cd scripts/release && yarn install && cd ../../ && yarn download-build-for-head || yarn build-combined --type=node react/index react-dom/index react-dom/src/server react-dom/test-utils scheduler/index react/jsx-runtime react/jsx-dev-runtime", - "check-release-dependencies": "node ./scripts/release/check-release-dependencies" + "check-release-dependencies": "node ./scripts/release/check-release-dependencies", + "generate-inline-fizz-runtime": "node ./scripts/rollup/generate-inline-fizz-runtime.js" }, "resolutions": { "react-is": "npm:react-is" diff --git a/packages/react-dom-bindings/src/server/fizz-instruction-set/ReactDOMFizzInlineClientRenderBoundary.js b/packages/react-dom-bindings/src/server/fizz-instruction-set/ReactDOMFizzInlineClientRenderBoundary.js new file mode 100644 index 0000000000000..96f750a8a46f3 --- /dev/null +++ b/packages/react-dom-bindings/src/server/fizz-instruction-set/ReactDOMFizzInlineClientRenderBoundary.js @@ -0,0 +1,5 @@ +import {clientRenderBoundary} from './ReactDOMFizzInstructionSet'; + +// This is a string so Closure's advanced compilation mode doesn't mangle it. +// eslint-disable-next-line dot-notation +window['$RX'] = clientRenderBoundary; diff --git a/packages/react-dom-bindings/src/server/fizz-instruction-set/ReactDOMFizzInlineCompleteBoundary.js b/packages/react-dom-bindings/src/server/fizz-instruction-set/ReactDOMFizzInlineCompleteBoundary.js new file mode 100644 index 0000000000000..ed85f4e70a795 --- /dev/null +++ b/packages/react-dom-bindings/src/server/fizz-instruction-set/ReactDOMFizzInlineCompleteBoundary.js @@ -0,0 +1,5 @@ +import {completeBoundary} from './ReactDOMFizzInstructionSet'; + +// This is a string so Closure's advanced compilation mode doesn't mangle it. +// eslint-disable-next-line dot-notation +window['$RC'] = completeBoundary; diff --git a/packages/react-dom-bindings/src/server/fizz-instruction-set/ReactDOMFizzInlineCompleteBoundaryWithStyles.js b/packages/react-dom-bindings/src/server/fizz-instruction-set/ReactDOMFizzInlineCompleteBoundaryWithStyles.js new file mode 100644 index 0000000000000..62760ee543b5d --- /dev/null +++ b/packages/react-dom-bindings/src/server/fizz-instruction-set/ReactDOMFizzInlineCompleteBoundaryWithStyles.js @@ -0,0 +1,7 @@ +import {completeBoundaryWithStyles} from './ReactDOMFizzInstructionSet'; + +// This is a string so Closure's advanced compilation mode doesn't mangle it. +// eslint-disable-next-line dot-notation +window['$RM'] = new Map(); +// eslint-disable-next-line dot-notation +window['$RR'] = completeBoundaryWithStyles; diff --git a/packages/react-dom-bindings/src/server/fizz-instruction-set/ReactDOMFizzInlineCompleteSegment.js b/packages/react-dom-bindings/src/server/fizz-instruction-set/ReactDOMFizzInlineCompleteSegment.js new file mode 100644 index 0000000000000..dbccb338b50e1 --- /dev/null +++ b/packages/react-dom-bindings/src/server/fizz-instruction-set/ReactDOMFizzInlineCompleteSegment.js @@ -0,0 +1,5 @@ +import {completeSegment} from './ReactDOMFizzInstructionSet'; + +// This is a string so Closure's advanced compilation mode doesn't mangle it. +// eslint-disable-next-line dot-notation +window['$RS'] = completeSegment; diff --git a/packages/react-dom-bindings/src/server/fizz-instruction-set/ReactDOMFizzInstructionSet.js b/packages/react-dom-bindings/src/server/fizz-instruction-set/ReactDOMFizzInstructionSet.js new file mode 100644 index 0000000000000..9f7ad8bd034cf --- /dev/null +++ b/packages/react-dom-bindings/src/server/fizz-instruction-set/ReactDOMFizzInstructionSet.js @@ -0,0 +1,224 @@ +/* eslint-disable dot-notation */ + +const COMMENT_NODE = 8; +const SUSPENSE_START_DATA = '$'; +const SUSPENSE_END_DATA = '/$'; +const SUSPENSE_PENDING_START_DATA = '$?'; +const SUSPENSE_FALLBACK_START_DATA = '$!'; +const LOADED = 'l'; +const ERRORED = 'e'; + +// TODO: Symbols that are referenced outside this module use dynamic accessor +// notation instead of dot notation to prevent Closure's advanced compilation +// mode from renaming. We could use extern files instead, but I couldn't get it +// working. Closure converts it to a dot access anyway, though, so it's not an +// urgent issue. + +export function clientRenderBoundary( + suspenseBoundaryID, + errorDigest, + errorMsg, + errorComponentStack, +) { + // Find the fallback's first element. + const suspenseIdNode = document.getElementById(suspenseBoundaryID); + if (!suspenseIdNode) { + // The user must have already navigated away from this tree. + // E.g. because the parent was hydrated. + return; + } + // Find the boundary around the fallback. This is always the previous node. + const suspenseNode = suspenseIdNode.previousSibling; + // Tag it to be client rendered. + suspenseNode.data = SUSPENSE_FALLBACK_START_DATA; + // assign error metadata to first sibling + const dataset = suspenseIdNode.dataset; + if (errorDigest) dataset['dgst'] = errorDigest; + if (errorMsg) dataset['msg'] = errorMsg; + if (errorComponentStack) dataset['stck'] = errorComponentStack; + // Tell React to retry it if the parent already hydrated. + if (suspenseNode['_reactRetry']) { + suspenseNode['_reactRetry'](); + } +} + +export function completeBoundaryWithStyles( + suspenseBoundaryID, + contentID, + styles, +) { + // TODO: In the non-inline version of the runtime, these don't need to be read + // from the global scope. + const completeBoundaryImpl = window['$RC']; + const resourceMap = window['$RM']; + + const precedences = new Map(); + const thisDocument = document; + let lastResource, node; + + // Seed the precedence list with existing resources + const nodes = thisDocument.querySelectorAll('link[data-rprec]'); + for (let i = 0; (node = nodes[i++]); ) { + precedences.set(node.dataset['rprec'], (lastResource = node)); + } + + let i = 0; + const dependencies = []; + let style, href, precedence, attr, loadingState, resourceEl; + + function setStatus(s) { + this['s'] = s; + } + + while ((style = styles[i++])) { + let j = 0; + href = style[j++]; + // We check if this resource is already in our resourceMap and reuse it if so. + // If it is already loaded we don't return it as a depenendency since there is nothing + // to wait for + loadingState = resourceMap.get(href); + if (loadingState) { + if (loadingState['s'] !== 'l') { + dependencies.push(loadingState); + } + continue; + } + + // We construct our new resource element, looping over remaining attributes if any + // setting them to the Element. + resourceEl = thisDocument.createElement('link'); + resourceEl.href = href; + resourceEl.rel = 'stylesheet'; + resourceEl.dataset['rprec'] = precedence = style[j++]; + while ((attr = style[j++])) { + resourceEl.setAttribute(attr, style[j++]); + } + + // We stash a pending promise in our map by href which will resolve or reject + // when the underlying resource loads or errors. We add it to the dependencies + // array to be returned. + loadingState = resourceEl['_p'] = new Promise((re, rj) => { + resourceEl.onload = re; + resourceEl.onerror = rj; + }); + loadingState.then( + setStatus.bind(loadingState, LOADED), + setStatus.bind(loadingState, ERRORED), + ); + resourceMap.set(href, loadingState); + dependencies.push(loadingState); + + // The prior style resource is the last one placed at a given + // precedence or the last resource itself which may be null. + // We grab this value and then update the last resource for this + // precedence to be the inserted element, updating the lastResource + // pointer if needed. + const prior = precedences.get(precedence) || lastResource; + if (prior === lastResource) { + lastResource = resourceEl; + } + precedences.set(precedence, resourceEl); + + // Finally, we insert the newly constructed instance at an appropriate location + // in the Document. + if (prior) { + prior.parentNode.insertBefore(resourceEl, prior.nextSibling); + } else { + const head = thisDocument.head; + head.insertBefore(resourceEl, head.firstChild); + } + } + + Promise.all(dependencies).then( + completeBoundaryImpl.bind(null, suspenseBoundaryID, contentID, ''), + completeBoundaryImpl.bind( + null, + suspenseBoundaryID, + contentID, + 'Resource failed to load', + ), + ); +} + +export function completeBoundary(suspenseBoundaryID, contentID, errorDigest) { + const contentNode = document.getElementById(contentID); + // We'll detach the content node so that regardless of what happens next we don't leave in the tree. + // This might also help by not causing recalcing each time we move a child from here to the target. + contentNode.parentNode.removeChild(contentNode); + + // Find the fallback's first element. + const suspenseIdNode = document.getElementById(suspenseBoundaryID); + if (!suspenseIdNode) { + // The user must have already navigated away from this tree. + // E.g. because the parent was hydrated. That's fine there's nothing to do + // but we have to make sure that we already deleted the container node. + return; + } + // Find the boundary around the fallback. This is always the previous node. + const suspenseNode = suspenseIdNode.previousSibling; + + if (!errorDigest) { + // Clear all the existing children. This is complicated because + // there can be embedded Suspense boundaries in the fallback. + // This is similar to clearSuspenseBoundary in ReactDOMHostConfig. + // TODO: We could avoid this if we never emitted suspense boundaries in fallback trees. + // They never hydrate anyway. However, currently we support incrementally loading the fallback. + const parentInstance = suspenseNode.parentNode; + let node = suspenseNode.nextSibling; + let depth = 0; + do { + if (node && node.nodeType === COMMENT_NODE) { + const data = node.data; + if (data === SUSPENSE_END_DATA) { + if (depth === 0) { + break; + } else { + depth--; + } + } else if ( + data === SUSPENSE_START_DATA || + data === SUSPENSE_PENDING_START_DATA || + data === SUSPENSE_FALLBACK_START_DATA + ) { + depth++; + } + } + + const nextNode = node.nextSibling; + parentInstance.removeChild(node); + node = nextNode; + } while (node); + + const endOfBoundary = node; + + // Insert all the children from the contentNode between the start and end of suspense boundary. + while (contentNode.firstChild) { + parentInstance.insertBefore(contentNode.firstChild, endOfBoundary); + } + + suspenseNode.data = SUSPENSE_START_DATA; + } else { + suspenseNode.data = SUSPENSE_FALLBACK_START_DATA; + suspenseIdNode.setAttribute('data-dgst', errorDigest); + } + + if (suspenseNode['_reactRetry']) { + suspenseNode['_reactRetry'](); + } +} + +export function completeSegment(containerID, placeholderID) { + const segmentContainer = document.getElementById(containerID); + const placeholderNode = document.getElementById(placeholderID); + // We always expect both nodes to exist here because, while we might + // have navigated away from the main tree, we still expect the detached + // tree to exist. + segmentContainer.parentNode.removeChild(segmentContainer); + while (segmentContainer.firstChild) { + placeholderNode.parentNode.insertBefore( + segmentContainer.firstChild, + placeholderNode, + ); + } + placeholderNode.parentNode.removeChild(placeholderNode); +} diff --git a/packages/react-dom-bindings/src/server/fizz-instruction-set/ReactDOMFizzInstructionSetInlineCodeStrings.js b/packages/react-dom-bindings/src/server/fizz-instruction-set/ReactDOMFizzInstructionSetInlineCodeStrings.js index c6668e3f8cf30..6862f77651a04 100644 --- a/packages/react-dom-bindings/src/server/fizz-instruction-set/ReactDOMFizzInstructionSetInlineCodeStrings.js +++ b/packages/react-dom-bindings/src/server/fizz-instruction-set/ReactDOMFizzInstructionSetInlineCodeStrings.js @@ -1,207 +1,6 @@ -// Instruction Set - -// The following code is the source scripts that we then minify and inline below, -// with renamed function names that we hope don't collide: - -// const COMMENT_NODE = 8; -// const SUSPENSE_START_DATA = '$'; -// const SUSPENSE_END_DATA = '/$'; -// const SUSPENSE_PENDING_START_DATA = '$?'; -// const SUSPENSE_FALLBACK_START_DATA = '$!'; -// const LOADED = 'l'; -// const ERRORED = 'e'; - -// function clientRenderBoundary(suspenseBoundaryID, errorDigest, errorMsg, errorComponentStack) { -// // Find the fallback's first element. -// const suspenseIdNode = document.getElementById(suspenseBoundaryID); -// if (!suspenseIdNode) { -// // The user must have already navigated away from this tree. -// // E.g. because the parent was hydrated. -// return; -// } -// // Find the boundary around the fallback. This is always the previous node. -// const suspenseNode = suspenseIdNode.previousSibling; -// // Tag it to be client rendered. -// suspenseNode.data = SUSPENSE_FALLBACK_START_DATA; -// // assign error metadata to first sibling -// let dataset = suspenseIdNode.dataset; -// if (errorDigest) dataset.dgst = errorDigest; -// if (errorMsg) dataset.msg = errorMsg; -// if (errorComponentStack) dataset.stck = errorComponentStack; -// // Tell React to retry it if the parent already hydrated. -// if (suspenseNode._reactRetry) { -// suspenseNode._reactRetry(); -// } -// } - -// resourceMap = new Map(); -// function completeBoundaryWithStyles(suspenseBoundaryID, contentID, styles) { -// const precedences = new Map(); -// const thisDocument = document; -// let lastResource, node; - -// // Seed the precedence list with existing resources -// let nodes = thisDocument.querySelectorAll('link[data-rprec]'); -// for (let i = 0;node = nodes[i++];) { -// precedences.set(node.dataset.rprec, lastResource = node); -// } - -// let i = 0; -// let dependencies = []; -// let style, href, precedence, attr, loadingState, resourceEl; - -// function setStatus(s) { -// this.s = s; -// } - -// while (style = styles[i++]) { -// let j = 0; -// href = style[j++]; -// // We check if this resource is already in our resourceMap and reuse it if so. -// // If it is already loaded we don't return it as a depenendency since there is nothing -// // to wait for -// loadingState = resourceMap.get(href); -// if (loadingState) { -// if (loadingState.s !== 'l') { -// dependencies.push(loadingState); -// } -// continue; -// } - -// // We construct our new resource element, looping over remaining attributes if any -// // setting them to the Element. -// resourceEl = thisDocument.createElement("link"); -// resourceEl.href = href; -// resourceEl.rel = 'stylesheet'; -// resourceEl.dataset.rprec = precedence = style[j++]; -// while(attr = style[j++]) { -// resourceEl.setAttribute(attr, style[j++]); -// } - -// // We stash a pending promise in our map by href which will resolve or reject -// // when the underlying resource loads or errors. We add it to the dependencies -// // array to be returned. -// loadingState = resourceEl._p = new Promise((re, rj) => { -// resourceEl.onload = re; -// resourceEl.onerror = rj; -// }) -// loadingState.then( -// setStatus.bind(loadingState, LOADED), -// setStatus.bind(loadingState, ERRORED) -// ); -// resourceMap.set(href, loadingState); -// dependencies.push(loadingState); - -// // The prior style resource is the last one placed at a given -// // precedence or the last resource itself which may be null. -// // We grab this value and then update the last resource for this -// // precedence to be the inserted element, updating the lastResource -// // pointer if needed. -// let prior = precedences.get(precedence) || lastResource; -// if (prior === lastResource) { -// lastResource = resourceEl -// } -// precedences.set(precedence, resourceEl) - -// // Finally, we insert the newly constructed instance at an appropriate location -// // in the Document. -// if (prior) { -// prior.parentNode.insertBefore(resourceEl, prior.nextSibling); -// } else { -// let head = thisDocument.head; -// head.insertBefore(resourceEl, head.firstChild); -// } -// } - -// Promise.all(dependencies).then( -// completeBoundary.bind(null, suspenseBoundaryID, contentID, ''), -// completeBoundary.bind(null, suspenseBoundaryID, contentID, "Resource failed to load") -// ); -// } - -// function completeBoundary(suspenseBoundaryID, contentID, errorDigest) { -// const contentNode = document.getElementById(contentID); -// // We'll detach the content node so that regardless of what happens next we don't leave in the tree. -// // This might also help by not causing recalcing each time we move a child from here to the target. -// contentNode.parentNode.removeChild(contentNode); - -// // Find the fallback's first element. -// const suspenseIdNode = document.getElementById(suspenseBoundaryID); -// if (!suspenseIdNode) { -// // The user must have already navigated away from this tree. -// // E.g. because the parent was hydrated. That's fine there's nothing to do -// // but we have to make sure that we already deleted the container node. -// return; -// } -// // Find the boundary around the fallback. This is always the previous node. -// const suspenseNode = suspenseIdNode.previousSibling; - -// if (!errorDigest) { -// // Clear all the existing children. This is complicated because -// // there can be embedded Suspense boundaries in the fallback. -// // This is similar to clearSuspenseBoundary in ReactDOMHostConfig. -// // TODO: We could avoid this if we never emitted suspense boundaries in fallback trees. -// // They never hydrate anyway. However, currently we support incrementally loading the fallback. -// const parentInstance = suspenseNode.parentNode; -// let node = suspenseNode.nextSibling; -// let depth = 0; -// do { -// if (node && node.nodeType === COMMENT_NODE) { -// const data = node.data; -// if (data === SUSPENSE_END_DATA) { -// if (depth === 0) { -// break; -// } else { -// depth--; -// } -// } else if ( -// data === SUSPENSE_START_DATA || -// data === SUSPENSE_PENDING_START_DATA || -// data === SUSPENSE_FALLBACK_START_DATA -// ) { -// depth++; -// } -// } - -// const nextNode = node.nextSibling; -// parentInstance.removeChild(node); -// node = nextNode; -// } while (node); - -// const endOfBoundary = node; - -// // Insert all the children from the contentNode between the start and end of suspense boundary. -// while (contentNode.firstChild) { -// parentInstance.insertBefore(contentNode.firstChild, endOfBoundary); -// } - -// suspenseNode.data = SUSPENSE_START_DATA; -// } else { -// suspenseNode.data = SUSPENSE_FALLBACK_START_DATA; -// suspenseIdNode.setAttribute('data-dgst', errorDigest) -// } - -// if (suspenseNode._reactRetry) { -// suspenseNode._reactRetry(); -// } -// } - -// function completeSegment(containerID, placeholderID) { -// const segmentContainer = document.getElementById(containerID); -// const placeholderNode = document.getElementById(placeholderID); -// // We always expect both nodes to exist here because, while we might -// // have navigated away from the main tree, we still expect the detached -// // tree to exist. -// segmentContainer.parentNode.removeChild(segmentContainer); -// while (segmentContainer.firstChild) { -// placeholderNode.parentNode.insertBefore( -// segmentContainer.firstChild, -// placeholderNode, -// ); -// } -// placeholderNode.parentNode.removeChild(placeholderNode); -// } - +// This is a generated file. The source files are in react-dom-bindings/src/server/fizz-instruction-set. +// The build script is at scripts/rollup/generate-inline-fizz-runtime.js. +// Run `yarn generate-inline-fizz-runtime` to generate. export const clientRenderBoundary = '$RX=function(b,c,d,e){var a=document.getElementById(b);a&&(b=a.previousSibling,b.data="$!",a=a.dataset,c&&(a.dgst=c),d&&(a.msg=d),e&&(a.stck=e),b._reactRetry&&b._reactRetry())};'; export const completeBoundary = diff --git a/scripts/rollup/generate-inline-fizz-runtime.js b/scripts/rollup/generate-inline-fizz-runtime.js new file mode 100644 index 0000000000000..f798a6f5b093f --- /dev/null +++ b/scripts/rollup/generate-inline-fizz-runtime.js @@ -0,0 +1,88 @@ +'use strict'; + +const fs = require('fs'); +const ClosureCompiler = require('google-closure-compiler').compiler; +const prettier = require('prettier'); + +const instructionDir = + './packages/react-dom-bindings/src/server/fizz-instruction-set'; + +// This is the name of the generated file that exports the inline instruction +// set as strings. +const inlineCodeStringsFilename = + instructionDir + '/ReactDOMFizzInstructionSetInlineCodeStrings.js'; + +const config = [ + { + entry: 'ReactDOMFizzInlineClientRenderBoundary.js', + exportName: 'clientRenderBoundary', + }, + { + entry: 'ReactDOMFizzInlineCompleteBoundary.js', + exportName: 'completeBoundary', + }, + { + entry: 'ReactDOMFizzInlineCompleteBoundaryWithStyles.js', + exportName: 'completeBoundaryWithStyles', + }, + { + entry: 'ReactDOMFizzInlineCompleteSegment.js', + exportName: 'completeSegment', + }, +]; + +const prettierConfig = require('../../.prettierrc.js'); + +async function main() { + const exportStatements = await Promise.all( + config.map(async ({entry, exportName}) => { + const fullEntryPath = instructionDir + '/' + entry; + const compiler = new ClosureCompiler({ + entry_point: fullEntryPath, + js: [fullEntryPath, instructionDir + '/ReactDOMFizzInstructionSet.js'], + compilation_level: 'ADVANCED', + module_resolution: 'NODE', + // This is necessary to prevent Closure from inlining a Promise polyfill + rewrite_polyfills: false, + }); + + const code = await new Promise((resolve, reject) => { + compiler.run((exitCode, stdOut, stdErr) => { + if (exitCode !== 0) { + reject(new Error(stdErr)); + } else { + resolve(stdOut); + } + }); + }); + + return `export const ${exportName} = ${JSON.stringify(code.trim())};`; + }) + ); + + let outputCode = [ + '// This is a generated file. The source files are in react-dom-bindings/src/server/fizz-instruction-set.', + '// The build script is at scripts/rollup/generate-inline-fizz-runtime.js.', + '// Run `yarn generate-inline-fizz-runtime` to generate.', + ...exportStatements, + ].join('\n'); + + // This replaces "window.$globalVar" with "$globalVar". There's probably a + // better way to do this with Closure, with externs or something, but I + // couldn't figure it out. Good enough for now. This only affects the inline + // Fizz runtime, and should break immediately if there were a mistake, so I'm + // not too worried about it. + outputCode = outputCode.replaceAll( + /window\.(\$[A-z0-9_]*)/g, + (_, variableName) => variableName + ); + + const prettyOutputCode = prettier.format(outputCode, prettierConfig); + + fs.writeFileSync(inlineCodeStringsFilename, prettyOutputCode, 'utf8'); +} + +main().catch(err => { + console.error(err); + process.exit(1); +}); From c9640b942b73cb25310173bbd9f3e104e6cbc5a8 Mon Sep 17 00:00:00 2001 From: Andrew Clark Date: Fri, 14 Oct 2022 18:20:14 -0400 Subject: [PATCH 3/3] Check in CI if generated Fizz runtime is in sync The generated Fizz runtime is checked into source. In CI, we'll ensure it stays in sync by running the script and confirming nothing changed. --- .circleci/config.yml | 16 ++++++++++++++++ 1 file changed, 16 insertions(+) diff --git a/.circleci/config.yml b/.circleci/config.yml index f9ae56ca58905..bc7c90086f596 100644 --- a/.circleci/config.yml +++ b/.circleci/config.yml @@ -367,6 +367,19 @@ jobs: command: | yarn extract-errors git diff --quiet || (echo "Found unminified errors. Either update the error codes map or disable error minification for the affected build, if appropriate." && false) + + check_generated_fizz_runtime: + docker: *docker + environment: *environment + steps: + - checkout + - attach_workspace: *attach_workspace + - *restore_node_modules + - run: + name: Confirm generated inline Fizz runtime is up to date + command: | + yarn generate-inline-fizz-runtime + git diff --quiet || (echo "There was a change to the Fizz runtime. Run `yarn generate-inline-fizz-runtime` and check in the result." && false) yarn_test: docker: *docker @@ -494,6 +507,9 @@ workflows: - sync_reconciler_forks: requires: - setup + - check_generated_fizz_runtime: + requires: + - setup - yarn_lint: requires: - setup