react-static
Version:
A progressive static site generator for React
349 lines (276 loc) • 37.6 kB
JavaScript
;
var _interopRequireDefault = require("@babel/runtime/helpers/interopRequireDefault");
Object.defineProperty(exports, "__esModule", {
value: true
});
exports.getEmbeddedRouteInfoScript = getEmbeddedRouteInfoScript;
exports["default"] = void 0;
var _regenerator = _interopRequireDefault(require("@babel/runtime/regenerator"));
var _defineProperty2 = _interopRequireDefault(require("@babel/runtime/helpers/defineProperty"));
var _asyncToGenerator2 = _interopRequireDefault(require("@babel/runtime/helpers/asyncToGenerator"));
var _crypto = _interopRequireDefault(require("crypto"));
var _react = _interopRequireDefault(require("react"));
var _server = require("react-dom/server");
var _reactHelmet = _interopRequireDefault(require("react-helmet"));
var _reactUniversalComponent = require("react-universal-component");
var _webpackFlushChunks = _interopRequireDefault(require("webpack-flush-chunks"));
var _path = _interopRequireDefault(require("path"));
var _fsExtra = _interopRequireDefault(require("fs-extra"));
var _jsesc = _interopRequireDefault(require("jsesc"));
var _Redirect = _interopRequireDefault(require("./components/Redirect"));
var _plugins = _interopRequireDefault(require("./plugins"));
var _utils = require("../utils");
var _chunkBuilder = require("../utils/chunkBuilder");
var _HtmlWithMeta = _interopRequireDefault(require("./components/HtmlWithMeta"));
var _HeadWithMeta = _interopRequireDefault(require("./components/HeadWithMeta"));
var _BodyWithMeta = _interopRequireDefault(require("./components/BodyWithMeta"));
function ownKeys(object, enumerableOnly) { var keys = Object.keys(object); if (Object.getOwnPropertySymbols) { var symbols = Object.getOwnPropertySymbols(object); if (enumerableOnly) symbols = symbols.filter(function (sym) { return Object.getOwnPropertyDescriptor(object, sym).enumerable; }); keys.push.apply(keys, symbols); } return keys; }
function _objectSpread(target) { for (var i = 1; i < arguments.length; i++) { var source = arguments[i] != null ? arguments[i] : {}; if (i % 2) { ownKeys(Object(source), true).forEach(function (key) { (0, _defineProperty2["default"])(target, key, source[key]); }); } else if (Object.getOwnPropertyDescriptors) { Object.defineProperties(target, Object.getOwnPropertyDescriptors(source)); } else { ownKeys(Object(source)).forEach(function (key) { Object.defineProperty(target, key, Object.getOwnPropertyDescriptor(source, key)); }); } } return target; }
var cachedBasePath;
var cachedHrefReplace;
var cachedSrcReplace;
function getSHA256(str) {
var hash = _crypto["default"].createHash('sha256');
hash.update(str);
return hash.digest('base64');
}
function getSubresourceHash(str) {
var sha256 = getSHA256(str);
return "sha256-".concat(sha256);
}
function getEmbeddedRouteInfoScript(embeddedRouteInfo) {
var routeInfoJSON = (0, _jsesc["default"])(JSON.stringify(embeddedRouteInfo), {
isScriptContext: true,
wrap: true,
json: true
});
var script = "window.__routeInfo = JSON.parse(".concat(routeInfoJSON, ");");
var hash = getSubresourceHash(script);
return {
hash: hash,
script: script
};
}
var _default = /*#__PURE__*/function () {
var _exportRoute = (0, _asyncToGenerator2["default"])( /*#__PURE__*/_regenerator["default"].mark(function _callee(state) {
var _state, config, DocumentTemplate, route, siteData, clientStats, incremental, _state2, Comp, sharedHashesByProp, template, data, sharedData, routePath, remove, removeLocation, basePath, hrefReplace, srcReplace, routeInfo, embeddedRouteInfo, inlineScripts, meta, chunkNames, head, clientScripts, clientStyleSheets, clientCss, FinalComp, renderToStringAndExtract, appHtml, RenderedComp, DocumentHtml, html, publicPath, htmlFilename, routeInfoFilename, res;
return _regenerator["default"].wrap(function _callee$(_context) {
while (1) {
switch (_context.prev = _context.next) {
case 0:
_state = state, config = _state.config, DocumentTemplate = _state.DocumentTemplate, route = _state.route, siteData = _state.siteData, clientStats = _state.clientStats, incremental = _state.incremental;
_state2 = state, Comp = _state2.Comp;
sharedHashesByProp = route.sharedHashesByProp, template = route.template, data = route.data, sharedData = route.sharedData, routePath = route.path, remove = route.remove;
if (!(incremental && remove)) {
_context.next = 8;
break;
}
if (!((0, _utils.is404Path)(route.path) || route.path === '/')) {
_context.next = 6;
break;
}
throw new Error("You are attempting to incrementally remove the ".concat((0, _utils.is404Path)(route.path) ? '404' : 'index', " route from your export. This is currently not supported (or recommended) by React Static."));
case 6:
removeLocation = _path["default"].join(config.paths.DIST, route.path);
return _context.abrupt("return", _fsExtra["default"].remove(removeLocation));
case 8:
basePath = cachedBasePath || (cachedBasePath = config.basePath);
hrefReplace = cachedHrefReplace || (cachedHrefReplace = new RegExp("(href=[\"'])\\/(".concat(basePath ? "".concat(basePath, "\\/") : '', ")?([^\\/])"), 'gm'));
srcReplace = cachedSrcReplace || (cachedSrcReplace = new RegExp("(src=[\"'])\\/(".concat(basePath ? "".concat(basePath, "\\/") : '', ")?([^\\/])"), 'gm')); // This routeInfo will be saved to disk. It should only include the
// data and hashes to construct all of the props later.
_context.next = 13;
return _plugins["default"].routeInfo({
template: template,
sharedHashesByProp: sharedHashesByProp,
data: data,
path: routePath
}, state);
case 13:
routeInfo = _context.sent;
// This embeddedRouteInfo will be inlined into the HTML for this route.
// It should include all of the data, including shared data
embeddedRouteInfo = _objectSpread(_objectSpread({}, routeInfo), {}, {
sharedData: sharedData,
siteData: siteData
});
inlineScripts = {
routeInfo: getEmbeddedRouteInfoScript(embeddedRouteInfo)
};
state = _objectSpread(_objectSpread({}, state), {}, {
routeInfo: routeInfo,
embeddedRouteInfo: embeddedRouteInfo,
inlineScripts: inlineScripts
}); // Make a place to collect chunks, meta info and head tags
meta = {};
chunkNames = [];
head = {};
clientScripts = [];
clientStyleSheets = [];
clientCss = {};
// Get the react component from the Comp and pass it the export context. This
// uses reactContext under the hood to pass down the exportContext, since
// react's new context api doesn't survive across bundling.
Comp = config.disableRuntime ? Comp : Comp(embeddedRouteInfo);
if (route.redirect) {
FinalComp = function FinalComp() {
return /*#__PURE__*/_react["default"].createElement(_Redirect["default"], {
fromPath: route.path,
to: route.redirect
});
};
} else {
FinalComp = function FinalComp(props) {
return /*#__PURE__*/_react["default"].createElement(_reactUniversalComponent.ReportChunks, {
report: function report(chunkName) {
// if we are building to a absolute path we must make the detected
// chunkName relative and matching to the one we set in
// generateTemplates
if (!config.paths.DIST.startsWith(config.paths.ROOT)) {
chunkName = (0, _chunkBuilder.absoluteToRelativeChunkName)(config.paths.ROOT, chunkName);
}
chunkNames.push(chunkName);
}
}, /*#__PURE__*/_react["default"].createElement(Comp, props));
};
}
renderToStringAndExtract = function renderToStringAndExtract(comp) {
// Rend the app to string!
var appHtml = (0, _server.renderToString)(comp);
var _flushChunks = (0, _webpackFlushChunks["default"])(clientStats, {
chunkNames: chunkNames,
outputPath: config.paths.DIST
}),
scripts = _flushChunks.scripts,
stylesheets = _flushChunks.stylesheets,
css = _flushChunks.css;
clientScripts = scripts;
clientStyleSheets = stylesheets;
clientCss = css; // Extract head calls using Helmet synchronously right after renderToString
// to not introduce any race conditions in the meta data rendering
var helmet = _reactHelmet["default"].renderStatic();
head = {
htmlProps: helmet.htmlAttributes.toComponent(),
bodyProps: helmet.bodyAttributes.toComponent(),
base: helmet.base.toComponent(),
link: helmet.link.toComponent(),
meta: helmet.meta.toComponent(),
noscript: helmet.noscript.toComponent(),
script: helmet.script.toComponent(),
style: helmet.style.toComponent(),
title: helmet.title.toComponent()
};
return appHtml;
};
state = _objectSpread(_objectSpread({}, state), {}, {
meta: meta
});
_context.prev = 27;
_context.next = 30;
return _plugins["default"].beforeRenderToElement(FinalComp, state);
case 30:
FinalComp = _context.sent;
if (!config.renderToElement) {
_context.next = 33;
break;
}
throw new Error("config.renderToElement has been deprecated in favor of the " + "'beforeRenderToElement' or 'beforeRenderToHtml' hooks instead.");
case 33:
RenderedComp = /*#__PURE__*/_react["default"].createElement(FinalComp, null); // Run the beforeRenderToHtml hook
// Rum the Html hook
_context.next = 36;
return _plugins["default"].beforeRenderToHtml(RenderedComp, state);
case 36:
RenderedComp = _context.sent;
if (!config.renderToHtml) {
_context.next = 39;
break;
}
throw new Error("config.renderToHtml has been deprecated in favor of the " + "'beforeRenderToHtml' or 'beforeHtmlToDocument' hooks instead.");
case 39:
appHtml = renderToStringAndExtract(RenderedComp);
_context.next = 42;
return _plugins["default"].beforeHtmlToDocument(appHtml, state);
case 42:
appHtml = _context.sent;
_context.next = 50;
break;
case 45:
_context.prev = 45;
_context.t0 = _context["catch"](27);
if (_context.t0.then) {
_context.t0.message = 'Components are not allowed to suspend during static export. Please ' + 'make its data available synchronously and try again!';
}
_context.t0.message = "Failed exporting HTML for URL ".concat(route.path, " (").concat(route.template, "): ").concat(_context.t0.message);
throw _context.t0;
case 50:
state = _objectSpread(_objectSpread({}, state), {}, {
head: head,
clientScripts: clientScripts,
clientStyleSheets: clientStyleSheets,
clientCss: clientCss
});
_context.t1 = _server.renderToStaticMarkup;
_context.t2 = _react["default"];
_context.t3 = DocumentTemplate;
_context.next = 56;
return (0, _HtmlWithMeta["default"])(state);
case 56:
_context.t4 = _context.sent;
_context.next = 59;
return (0, _HeadWithMeta["default"])(state);
case 59:
_context.t5 = _context.sent;
_context.next = 62;
return (0, _BodyWithMeta["default"])(state);
case 62:
_context.t6 = _context.sent;
_context.t7 = state;
_context.t8 = {
Html: _context.t4,
Head: _context.t5,
Body: _context.t6,
state: _context.t7
};
_context.t9 = /*#__PURE__*/_react["default"].createElement("div", {
id: "root",
dangerouslySetInnerHTML: {
__html: appHtml
}
});
_context.t10 = _context.t2.createElement.call(_context.t2, _context.t3, _context.t8, _context.t9);
DocumentHtml = (0, _context.t1)(_context.t10);
// Render the html for the page inside of the base document.
html = "<!DOCTYPE html>".concat(DocumentHtml);
_context.next = 71;
return _plugins["default"].beforeDocumentToFile(html, state);
case 71:
html = _context.sent;
// If the siteRoot is set and we're not in staging, prefix all absolute URLs
// with the siteRoot
publicPath = (0, _utils.makePathAbsolute)(process.env.REACT_STATIC_PUBLIC_PATH);
if (process.env.REACT_STATIC_DISABLE_ROUTE_PREFIXING !== 'true') {
html = html.replace(hrefReplace, "$1".concat(publicPath, "$3"));
}
html = html.replace(srcReplace, "$1".concat(publicPath, "$3")); // If the route is a 404 page, write it directly to 404.html, instead of
// inside a directory.
htmlFilename = (0, _utils.is404Path)(route.path) ? _path["default"].join(config.paths.DIST, '404.html') : _path["default"].join(config.paths.DIST, route.path, 'index.html'); // Make the routeInfo sit right next to its companion html file
routeInfoFilename = _path["default"].join(config.paths.DIST, route.path, 'routeInfo.json');
_context.next = 79;
return Promise.all([_fsExtra["default"].outputFile(htmlFilename, html), !route.redirect ? _fsExtra["default"].outputJson(routeInfoFilename, routeInfo) : Promise.resolve()]);
case 79:
res = _context.sent;
return _context.abrupt("return", res);
case 81:
case "end":
return _context.stop();
}
}
}, _callee, null, [[27, 45]]);
}));
function exportRoute(_x) {
return _exportRoute.apply(this, arguments);
}
return exportRoute;
}();
exports["default"] = _default;
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/static/exportRoute.js"],"names":["cachedBasePath","cachedHrefReplace","cachedSrcReplace","getSHA256","str","hash","crypto","createHash","update","digest","getSubresourceHash","sha256","getEmbeddedRouteInfoScript","embeddedRouteInfo","routeInfoJSON","JSON","stringify","isScriptContext","wrap","json","script","state","config","DocumentTemplate","route","siteData","clientStats","incremental","Comp","sharedHashesByProp","template","data","sharedData","routePath","path","remove","Error","removeLocation","nodePath","join","paths","DIST","fs","basePath","hrefReplace","RegExp","srcReplace","plugins","routeInfo","inlineScripts","meta","chunkNames","head","clientScripts","clientStyleSheets","clientCss","disableRuntime","redirect","FinalComp","props","chunkName","startsWith","ROOT","push","renderToStringAndExtract","comp","appHtml","outputPath","scripts","stylesheets","css","helmet","Helmet","renderStatic","htmlProps","htmlAttributes","toComponent","bodyProps","bodyAttributes","base","link","noscript","style","title","beforeRenderToElement","renderToElement","RenderedComp","beforeRenderToHtml","renderToHtml","beforeHtmlToDocument","then","message","renderToStaticMarkup","__html","DocumentHtml","html","beforeDocumentToFile","publicPath","process","env","REACT_STATIC_PUBLIC_PATH","REACT_STATIC_DISABLE_ROUTE_PREFIXING","replace","htmlFilename","routeInfoFilename","Promise","all","outputFile","outputJson","resolve","res","exportRoute"],"mappings":"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","sourcesContent":["import crypto from 'crypto'\nimport React from 'react'\nimport { renderToString, renderToStaticMarkup } from 'react-dom/server'\nimport Helmet from 'react-helmet'\nimport { ReportChunks } from 'react-universal-component'\nimport flushChunks from 'webpack-flush-chunks'\nimport nodePath from 'path'\nimport fs from 'fs-extra'\nimport jsesc from 'jsesc'\n\nimport Redirect from './components/Redirect'\nimport plugins from './plugins'\nimport { makePathAbsolute, is404Path } from '../utils'\nimport { absoluteToRelativeChunkName } from '../utils/chunkBuilder'\n\nimport makeHtmlWithMeta from './components/HtmlWithMeta'\nimport makeHeadWithMeta from './components/HeadWithMeta'\nimport makeBodyWithMeta from './components/BodyWithMeta'\n\nlet cachedBasePath\nlet cachedHrefReplace\nlet cachedSrcReplace\n\nfunction getSHA256(str) {\n  const hash = crypto.createHash('sha256')\n  hash.update(str)\n  return hash.digest('base64')\n}\n\nfunction getSubresourceHash(str) {\n  const sha256 = getSHA256(str)\n  return `sha256-${sha256}`\n}\n\nexport function getEmbeddedRouteInfoScript(embeddedRouteInfo) {\n  const routeInfoJSON = jsesc(JSON.stringify(embeddedRouteInfo), {\n    isScriptContext: true,\n    wrap: true,\n    json: true,\n  })\n  const script = `window.__routeInfo = JSON.parse(${routeInfoJSON});`\n  const hash = getSubresourceHash(script)\n\n  return {\n    hash,\n    script,\n  }\n}\n\nexport default (async function exportRoute(state) {\n  const {\n    config,\n    DocumentTemplate,\n    route,\n    siteData,\n    clientStats,\n    incremental,\n  } = state\n\n  let { Comp } = state\n\n  const {\n    sharedHashesByProp,\n    template,\n    data,\n    sharedData,\n    path: routePath,\n    remove,\n  } = route\n\n  if (incremental && remove) {\n    if (is404Path(route.path) || route.path === '/') {\n      throw new Error(\n        `You are attempting to incrementally remove the ${\n          is404Path(route.path) ? '404' : 'index'\n        } route from your export. This is currently not supported (or recommended) by React Static.`\n      )\n    }\n    const removeLocation = nodePath.join(config.paths.DIST, route.path)\n    return fs.remove(removeLocation)\n  }\n\n  const basePath = cachedBasePath || (cachedBasePath = config.basePath)\n\n  const hrefReplace =\n    cachedHrefReplace ||\n    (cachedHrefReplace = new RegExp(\n      `(href=[\"'])\\\\/(${basePath ? `${basePath}\\\\/` : ''})?([^\\\\/])`,\n      'gm'\n    ))\n\n  const srcReplace =\n    cachedSrcReplace ||\n    (cachedSrcReplace = new RegExp(\n      `(src=[\"'])\\\\/(${basePath ? `${basePath}\\\\/` : ''})?([^\\\\/])`,\n      'gm'\n    ))\n\n  // This routeInfo will be saved to disk. It should only include the\n  // data and hashes to construct all of the props later.\n  const routeInfo = await plugins.routeInfo(\n    {\n      template,\n      sharedHashesByProp,\n      data,\n      path: routePath,\n    },\n    state\n  )\n\n  // This embeddedRouteInfo will be inlined into the HTML for this route.\n  // It should include all of the data, including shared data\n  const embeddedRouteInfo = {\n    ...routeInfo,\n    sharedData,\n    siteData,\n  }\n\n  const inlineScripts = {\n    routeInfo: getEmbeddedRouteInfoScript(embeddedRouteInfo),\n  }\n\n  state = {\n    ...state,\n    routeInfo,\n    embeddedRouteInfo,\n    inlineScripts,\n  }\n\n  // Make a place to collect chunks, meta info and head tags\n  const meta = {}\n  const chunkNames = []\n  let head = {}\n  let clientScripts = []\n  let clientStyleSheets = []\n  let clientCss = {}\n\n  let FinalComp\n\n  // Get the react component from the Comp and pass it the export context. This\n  // uses reactContext under the hood to pass down the exportContext, since\n  // react's new context api doesn't survive across bundling.\n  Comp = config.disableRuntime ? Comp : Comp(embeddedRouteInfo)\n\n  if (route.redirect) {\n    FinalComp = () => <Redirect fromPath={route.path} to={route.redirect} />\n  } else {\n    FinalComp = props => (\n      <ReportChunks\n        report={chunkName => {\n          // if we are building to a absolute path we must make the detected\n          // chunkName relative and matching to the one we set in\n          // generateTemplates\n          if (!config.paths.DIST.startsWith(config.paths.ROOT)) {\n            chunkName = absoluteToRelativeChunkName(\n              config.paths.ROOT,\n              chunkName\n            )\n          }\n\n          chunkNames.push(chunkName)\n        }}\n      >\n        <Comp {...props} />\n      </ReportChunks>\n    )\n  }\n\n  const renderToStringAndExtract = comp => {\n    // Rend the app to string!\n    const appHtml = renderToString(comp)\n    const { scripts, stylesheets, css } = flushChunks(clientStats, {\n      chunkNames,\n      outputPath: config.paths.DIST,\n    })\n\n    clientScripts = scripts\n    clientStyleSheets = stylesheets\n    clientCss = css\n    // Extract head calls using Helmet synchronously right after renderToString\n    // to not introduce any race conditions in the meta data rendering\n    const helmet = Helmet.renderStatic()\n    head = {\n      htmlProps: helmet.htmlAttributes.toComponent(),\n      bodyProps: helmet.bodyAttributes.toComponent(),\n      base: helmet.base.toComponent(),\n      link: helmet.link.toComponent(),\n      meta: helmet.meta.toComponent(),\n      noscript: helmet.noscript.toComponent(),\n      script: helmet.script.toComponent(),\n      style: helmet.style.toComponent(),\n      title: helmet.title.toComponent(),\n    }\n\n    return appHtml\n  }\n\n  let appHtml\n\n  state = {\n    ...state,\n    meta,\n  }\n\n  try {\n    FinalComp = await plugins.beforeRenderToElement(FinalComp, state)\n\n    if (config.renderToElement) {\n      throw new Error(\n        `config.renderToElement has been deprecated in favor of the ` +\n          `'beforeRenderToElement' or 'beforeRenderToHtml' hooks instead.`\n      )\n    }\n\n    let RenderedComp = <FinalComp />\n\n    // Run the beforeRenderToHtml hook\n    // Rum the Html hook\n    RenderedComp = await plugins.beforeRenderToHtml(RenderedComp, state)\n\n    if (config.renderToHtml) {\n      throw new Error(\n        `config.renderToHtml has been deprecated in favor of the ` +\n          `'beforeRenderToHtml' or 'beforeHtmlToDocument' hooks instead.`\n      )\n    }\n\n    appHtml = renderToStringAndExtract(RenderedComp)\n\n    appHtml = await plugins.beforeHtmlToDocument(appHtml, state)\n  } catch (error) {\n    if (error.then) {\n      error.message =\n        'Components are not allowed to suspend during static export. Please ' +\n        'make its data available synchronously and try again!'\n    }\n    error.message = `Failed exporting HTML for URL ${route.path} (${route.template}): ${error.message}`\n    throw error\n  }\n\n  state = {\n    ...state,\n    head,\n    clientScripts,\n    clientStyleSheets,\n    clientCss,\n  }\n\n  const DocumentHtml = renderToStaticMarkup(\n    <DocumentTemplate\n      Html={await makeHtmlWithMeta(state)}\n      Head={await makeHeadWithMeta(state)}\n      Body={await makeBodyWithMeta(state)}\n      state={state}\n    >\n      <div id=\"root\" dangerouslySetInnerHTML={{ __html: appHtml }} />\n    </DocumentTemplate>\n  )\n\n  // Render the html for the page inside of the base document.\n  let html = `<!DOCTYPE html>${DocumentHtml}`\n\n  html = await plugins.beforeDocumentToFile(html, state)\n\n  // If the siteRoot is set and we're not in staging, prefix all absolute URLs\n  // with the siteRoot\n  const publicPath = makePathAbsolute(process.env.REACT_STATIC_PUBLIC_PATH)\n  if (process.env.REACT_STATIC_DISABLE_ROUTE_PREFIXING !== 'true') {\n    html = html.replace(hrefReplace, `$1${publicPath}$3`)\n  }\n\n  html = html.replace(srcReplace, `$1${publicPath}$3`)\n\n  // If the route is a 404 page, write it directly to 404.html, instead of\n  // inside a directory.\n  const htmlFilename = is404Path(route.path)\n    ? nodePath.join(config.paths.DIST, '404.html')\n    : nodePath.join(config.paths.DIST, route.path, 'index.html')\n\n  // Make the routeInfo sit right next to its companion html file\n  const routeInfoFilename = nodePath.join(\n    config.paths.DIST,\n    route.path,\n    'routeInfo.json'\n  )\n\n  const res = await Promise.all([\n    fs.outputFile(htmlFilename, html),\n    !route.redirect\n      ? fs.outputJson(routeInfoFilename, routeInfo)\n      : Promise.resolve(),\n  ])\n  return res\n})\n"]}