UNPKG

react-static

Version:

A progressive static site generator for React

349 lines (276 loc) 37.6 kB
"use strict"; 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":";;;;;;;;;;;;;;;;AAAA;;AACA;;AACA;;AACA;;AACA;;AACA;;AACA;;AACA;;AACA;;AAEA;;AACA;;AACA;;AACA;;AAEA;;AACA;;AACA;;;;;;AAEA,IAAIA,cAAJ;AACA,IAAIC,iBAAJ;AACA,IAAIC,gBAAJ;;AAEA,SAASC,SAAT,CAAmBC,GAAnB,EAAwB;AACtB,MAAMC,IAAI,GAAGC,mBAAOC,UAAP,CAAkB,QAAlB,CAAb;;AACAF,EAAAA,IAAI,CAACG,MAAL,CAAYJ,GAAZ;AACA,SAAOC,IAAI,CAACI,MAAL,CAAY,QAAZ,CAAP;AACD;;AAED,SAASC,kBAAT,CAA4BN,GAA5B,EAAiC;AAC/B,MAAMO,MAAM,GAAGR,SAAS,CAACC,GAAD,CAAxB;AACA,0BAAiBO,MAAjB;AACD;;AAEM,SAASC,0BAAT,CAAoCC,iBAApC,EAAuD;AAC5D,MAAMC,aAAa,GAAG,uBAAMC,IAAI,CAACC,SAAL,CAAeH,iBAAf,CAAN,EAAyC;AAC7DI,IAAAA,eAAe,EAAE,IAD4C;AAE7DC,IAAAA,IAAI,EAAE,IAFuD;AAG7DC,IAAAA,IAAI,EAAE;AAHuD,GAAzC,CAAtB;AAKA,MAAMC,MAAM,6CAAsCN,aAAtC,OAAZ;AACA,MAAMT,IAAI,GAAGK,kBAAkB,CAACU,MAAD,CAA/B;AAEA,SAAO;AACLf,IAAAA,IAAI,EAAJA,IADK;AAELe,IAAAA,MAAM,EAANA;AAFK,GAAP;AAID;;;mGAEe,iBAA2BC,KAA3B;AAAA;;AAAA;AAAA;AAAA;AAAA;AAAA,qBAQVA,KARU,EAEZC,MAFY,UAEZA,MAFY,EAGZC,gBAHY,UAGZA,gBAHY,EAIZC,KAJY,UAIZA,KAJY,EAKZC,QALY,UAKZA,QALY,EAMZC,WANY,UAMZA,WANY,EAOZC,WAPY,UAOZA,WAPY;AAAA,sBAUCN,KAVD,EAURO,IAVQ,WAURA,IAVQ;AAaZC,YAAAA,kBAbY,GAmBVL,KAnBU,CAaZK,kBAbY,EAcZC,QAdY,GAmBVN,KAnBU,CAcZM,QAdY,EAeZC,IAfY,GAmBVP,KAnBU,CAeZO,IAfY,EAgBZC,UAhBY,GAmBVR,KAnBU,CAgBZQ,UAhBY,EAiBNC,SAjBM,GAmBVT,KAnBU,CAiBZU,IAjBY,EAkBZC,MAlBY,GAmBVX,KAnBU,CAkBZW,MAlBY;;AAAA,kBAqBVR,WAAW,IAAIQ,MArBL;AAAA;AAAA;AAAA;;AAAA,kBAsBR,sBAAUX,KAAK,CAACU,IAAhB,KAAyBV,KAAK,CAACU,IAAN,KAAe,GAtBhC;AAAA;AAAA;AAAA;;AAAA,kBAuBJ,IAAIE,KAAJ,0DAEF,sBAAUZ,KAAK,CAACU,IAAhB,IAAwB,KAAxB,GAAgC,OAF9B,gGAvBI;;AAAA;AA6BNG,YAAAA,cA7BM,GA6BWC,iBAASC,IAAT,CAAcjB,MAAM,CAACkB,KAAP,CAAaC,IAA3B,EAAiCjB,KAAK,CAACU,IAAvC,CA7BX;AAAA,6CA8BLQ,oBAAGP,MAAH,CAAUE,cAAV,CA9BK;;AAAA;AAiCRM,YAAAA,QAjCQ,GAiCG3C,cAAc,KAAKA,cAAc,GAAGsB,MAAM,CAACqB,QAA7B,CAjCjB;AAmCRC,YAAAA,WAnCQ,GAoCZ3C,iBAAiB,KAChBA,iBAAiB,GAAG,IAAI4C,MAAJ,2BACDF,QAAQ,aAAMA,QAAN,WAAsB,EAD7B,iBAEnB,IAFmB,CADJ,CApCL;AA0CRG,YAAAA,UA1CQ,GA2CZ5C,gBAAgB,KACfA,gBAAgB,GAAG,IAAI2C,MAAJ,0BACDF,QAAQ,aAAMA,QAAN,WAAsB,EAD7B,iBAElB,IAFkB,CADJ,CA3CJ,EAiDd;AACA;;AAlDc;AAAA,mBAmDUI,oBAAQC,SAAR,CACtB;AACElB,cAAAA,QAAQ,EAARA,QADF;AAEED,cAAAA,kBAAkB,EAAlBA,kBAFF;AAGEE,cAAAA,IAAI,EAAJA,IAHF;AAIEG,cAAAA,IAAI,EAAED;AAJR,aADsB,EAOtBZ,KAPsB,CAnDV;;AAAA;AAmDR2B,YAAAA,SAnDQ;AA6Dd;AACA;AACMnC,YAAAA,iBA/DQ,mCAgETmC,SAhES;AAiEZhB,cAAAA,UAAU,EAAVA,UAjEY;AAkEZP,cAAAA,QAAQ,EAARA;AAlEY;AAqERwB,YAAAA,aArEQ,GAqEQ;AACpBD,cAAAA,SAAS,EAAEpC,0BAA0B,CAACC,iBAAD;AADjB,aArER;AAyEdQ,YAAAA,KAAK,mCACAA,KADA;AAEH2B,cAAAA,SAAS,EAATA,SAFG;AAGHnC,cAAAA,iBAAiB,EAAjBA,iBAHG;AAIHoC,cAAAA,aAAa,EAAbA;AAJG,cAAL,CAzEc,CAgFd;;AACMC,YAAAA,IAjFQ,GAiFD,EAjFC;AAkFRC,YAAAA,UAlFQ,GAkFK,EAlFL;AAmFVC,YAAAA,IAnFU,GAmFH,EAnFG;AAoFVC,YAAAA,aApFU,GAoFM,EApFN;AAqFVC,YAAAA,iBArFU,GAqFU,EArFV;AAsFVC,YAAAA,SAtFU,GAsFE,EAtFF;AA0Fd;AACA;AACA;AACA3B,YAAAA,IAAI,GAAGN,MAAM,CAACkC,cAAP,GAAwB5B,IAAxB,GAA+BA,IAAI,CAACf,iBAAD,CAA1C;;AAEA,gBAAIW,KAAK,CAACiC,QAAV,EAAoB;AAClBC,cAAAA,SAAS,GAAG;AAAA,oCAAM,gCAAC,oBAAD;AAAU,kBAAA,QAAQ,EAAElC,KAAK,CAACU,IAA1B;AAAgC,kBAAA,EAAE,EAAEV,KAAK,CAACiC;AAA1C,kBAAN;AAAA,eAAZ;AACD,aAFD,MAEO;AACLC,cAAAA,SAAS,GAAG,mBAAAC,KAAK;AAAA,oCACf,gCAAC,qCAAD;AACE,kBAAA,MAAM,EAAE,gBAAAC,SAAS,EAAI;AACnB;AACA;AACA;AACA,wBAAI,CAACtC,MAAM,CAACkB,KAAP,CAAaC,IAAb,CAAkBoB,UAAlB,CAA6BvC,MAAM,CAACkB,KAAP,CAAasB,IAA1C,CAAL,EAAsD;AACpDF,sBAAAA,SAAS,GAAG,+CACVtC,MAAM,CAACkB,KAAP,CAAasB,IADH,EAEVF,SAFU,CAAZ;AAID;;AAEDT,oBAAAA,UAAU,CAACY,IAAX,CAAgBH,SAAhB;AACD;AAbH,gCAeE,gCAAC,IAAD,EAAUD,KAAV,CAfF,CADe;AAAA,eAAjB;AAmBD;;AAEKK,YAAAA,wBAvHQ,GAuHmB,SAA3BA,wBAA2B,CAAAC,IAAI,EAAI;AACvC;AACA,kBAAMC,OAAO,GAAG,4BAAeD,IAAf,CAAhB;;AAFuC,iCAGD,oCAAYvC,WAAZ,EAAyB;AAC7DyB,gBAAAA,UAAU,EAAVA,UAD6D;AAE7DgB,gBAAAA,UAAU,EAAE7C,MAAM,CAACkB,KAAP,CAAaC;AAFoC,eAAzB,CAHC;AAAA,kBAG/B2B,OAH+B,gBAG/BA,OAH+B;AAAA,kBAGtBC,WAHsB,gBAGtBA,WAHsB;AAAA,kBAGTC,GAHS,gBAGTA,GAHS;;AAQvCjB,cAAAA,aAAa,GAAGe,OAAhB;AACAd,cAAAA,iBAAiB,GAAGe,WAApB;AACAd,cAAAA,SAAS,GAAGe,GAAZ,CAVuC,CAWvC;AACA;;AACA,kBAAMC,MAAM,GAAGC,wBAAOC,YAAP,EAAf;;AACArB,cAAAA,IAAI,GAAG;AACLsB,gBAAAA,SAAS,EAAEH,MAAM,CAACI,cAAP,CAAsBC,WAAtB,EADN;AAELC,gBAAAA,SAAS,EAAEN,MAAM,CAACO,cAAP,CAAsBF,WAAtB,EAFN;AAGLG,gBAAAA,IAAI,EAAER,MAAM,CAACQ,IAAP,CAAYH,WAAZ,EAHD;AAILI,gBAAAA,IAAI,EAAET,MAAM,CAACS,IAAP,CAAYJ,WAAZ,EAJD;AAKL1B,gBAAAA,IAAI,EAAEqB,MAAM,CAACrB,IAAP,CAAY0B,WAAZ,EALD;AAMLK,gBAAAA,QAAQ,EAAEV,MAAM,CAACU,QAAP,CAAgBL,WAAhB,EANL;AAOLxD,gBAAAA,MAAM,EAAEmD,MAAM,CAACnD,MAAP,CAAcwD,WAAd,EAPH;AAQLM,gBAAAA,KAAK,EAAEX,MAAM,CAACW,KAAP,CAAaN,WAAb,EARF;AASLO,gBAAAA,KAAK,EAAEZ,MAAM,CAACY,KAAP,CAAaP,WAAb;AATF,eAAP;AAYA,qBAAOV,OAAP;AACD,aAlJa;;AAsJd7C,YAAAA,KAAK,mCACAA,KADA;AAEH6B,cAAAA,IAAI,EAAJA;AAFG,cAAL;AAtJc;AAAA;AAAA,mBA4JMH,oBAAQqC,qBAAR,CAA8B1B,SAA9B,EAAyCrC,KAAzC,CA5JN;;AAAA;AA4JZqC,YAAAA,SA5JY;;AAAA,iBA8JRpC,MAAM,CAAC+D,eA9JC;AAAA;AAAA;AAAA;;AAAA,kBA+JJ,IAAIjD,KAAJ,CACJ,gIADI,CA/JI;;AAAA;AAqKRkD,YAAAA,YArKQ,gBAqKO,gCAAC,SAAD,OArKP,EAuKZ;AACA;;AAxKY;AAAA,mBAyKSvC,oBAAQwC,kBAAR,CAA2BD,YAA3B,EAAyCjE,KAAzC,CAzKT;;AAAA;AAyKZiE,YAAAA,YAzKY;;AAAA,iBA2KRhE,MAAM,CAACkE,YA3KC;AAAA;AAAA;AAAA;;AAAA,kBA4KJ,IAAIpD,KAAJ,CACJ,4HADI,CA5KI;;AAAA;AAkLZ8B,YAAAA,OAAO,GAAGF,wBAAwB,CAACsB,YAAD,CAAlC;AAlLY;AAAA,mBAoLIvC,oBAAQ0C,oBAAR,CAA6BvB,OAA7B,EAAsC7C,KAAtC,CApLJ;;AAAA;AAoLZ6C,YAAAA,OApLY;AAAA;AAAA;;AAAA;AAAA;AAAA;;AAsLZ,gBAAI,YAAMwB,IAAV,EAAgB;AACd,0BAAMC,OAAN,GACE,wEACA,sDAFF;AAGD;;AACD,wBAAMA,OAAN,2CAAiDnE,KAAK,CAACU,IAAvD,eAAgEV,KAAK,CAACM,QAAtE,gBAAoF,YAAM6D,OAA1F;AA3LY;;AAAA;AA+LdtE,YAAAA,KAAK,mCACAA,KADA;AAEH+B,cAAAA,IAAI,EAAJA,IAFG;AAGHC,cAAAA,aAAa,EAAbA,aAHG;AAIHC,cAAAA,iBAAiB,EAAjBA,iBAJG;AAKHC,cAAAA,SAAS,EAATA;AALG,cAAL;AA/Lc,0BAuMOqC,4BAvMP;AAAA;AAAA,0BAwMX,gBAxMW;AAAA;AAAA,mBAyME,8BAAiBvE,KAAjB,CAzMF;;AAAA;AAAA;AAAA;AAAA,mBA0ME,8BAAiBA,KAAjB,CA1MF;;AAAA;AAAA;AAAA;AAAA,mBA2ME,8BAAiBA,KAAjB,CA3MF;;AAAA;AAAA;AAAA,0BA4MHA,KA5MG;AAAA;AAyMV,cAAA,IAzMU;AA0MV,cAAA,IA1MU;AA2MV,cAAA,IA3MU;AA4MV,cAAA,KA5MU;AAAA;AAAA,uCA8MV;AAAK,cAAA,EAAE,EAAC,MAAR;AAAe,cAAA,uBAAuB,EAAE;AAAEwE,gBAAAA,MAAM,EAAE3B;AAAV;AAAxC,cA9MU;AAAA;AAuMR4B,YAAAA,YAvMQ;AAkNd;AACIC,YAAAA,IAnNU,4BAmNeD,YAnNf;AAAA;AAAA,mBAqND/C,oBAAQiD,oBAAR,CAA6BD,IAA7B,EAAmC1E,KAAnC,CArNC;;AAAA;AAqNd0E,YAAAA,IArNc;AAuNd;AACA;AACME,YAAAA,UAzNQ,GAyNK,6BAAiBC,OAAO,CAACC,GAAR,CAAYC,wBAA7B,CAzNL;;AA0Nd,gBAAIF,OAAO,CAACC,GAAR,CAAYE,oCAAZ,KAAqD,MAAzD,EAAiE;AAC/DN,cAAAA,IAAI,GAAGA,IAAI,CAACO,OAAL,CAAa1D,WAAb,cAA+BqD,UAA/B,QAAP;AACD;;AAEDF,YAAAA,IAAI,GAAGA,IAAI,CAACO,OAAL,CAAaxD,UAAb,cAA8BmD,UAA9B,QAAP,CA9Nc,CAgOd;AACA;;AACMM,YAAAA,YAlOQ,GAkOO,sBAAU/E,KAAK,CAACU,IAAhB,IACjBI,iBAASC,IAAT,CAAcjB,MAAM,CAACkB,KAAP,CAAaC,IAA3B,EAAiC,UAAjC,CADiB,GAEjBH,iBAASC,IAAT,CAAcjB,MAAM,CAACkB,KAAP,CAAaC,IAA3B,EAAiCjB,KAAK,CAACU,IAAvC,EAA6C,YAA7C,CApOU,EAsOd;;AACMsE,YAAAA,iBAvOQ,GAuOYlE,iBAASC,IAAT,CACxBjB,MAAM,CAACkB,KAAP,CAAaC,IADW,EAExBjB,KAAK,CAACU,IAFkB,EAGxB,gBAHwB,CAvOZ;AAAA;AAAA,mBA6OIuE,OAAO,CAACC,GAAR,CAAY,CAC5BhE,oBAAGiE,UAAH,CAAcJ,YAAd,EAA4BR,IAA5B,CAD4B,EAE5B,CAACvE,KAAK,CAACiC,QAAP,GACIf,oBAAGkE,UAAH,CAAcJ,iBAAd,EAAiCxD,SAAjC,CADJ,GAEIyD,OAAO,CAACI,OAAR,EAJwB,CAAZ,CA7OJ;;AAAA;AA6ORC,YAAAA,GA7OQ;AAAA,6CAmPPA,GAnPO;;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,G;;WAAeC,W;;;;SAAAA,W","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"]}