react-static
Version:
A progressive static site generator for React
515 lines (413 loc) • 19.9 kB
JavaScript
;
Object.defineProperty(exports, "__esModule", {
value: true
});
exports.buildProductionBundles = exports.startDevServer = exports.buildCompiler = exports.reloadRoutes = undefined;
var _regenerator = require('babel-runtime/regenerator');
var _regenerator2 = _interopRequireDefault(_regenerator);
var _slicedToArray = function () { function sliceIterator(arr, i) { var _arr = []; var _n = true; var _d = false; var _e = undefined; try { for (var _i = arr[Symbol.iterator](), _s; !(_n = (_s = _i.next()).done); _n = true) { _arr.push(_s.value); if (i && _arr.length === i) break; } } catch (err) { _d = true; _e = err; } finally { try { if (!_n && _i["return"]) _i["return"](); } finally { if (_d) throw _e; } } return _arr; } return function (arr, i) { if (Array.isArray(arr)) { return arr; } else if (Symbol.iterator in Object(arr)) { return sliceIterator(arr, i); } else { throw new TypeError("Invalid attempt to destructure non-iterable instance"); } }; }();
var _extends = Object.assign || function (target) { for (var i = 1; i < arguments.length; i++) { var source = arguments[i]; for (var key in source) { if (Object.prototype.hasOwnProperty.call(source, key)) { target[key] = source[key]; } } } return target; };
var buildCompiler = exports.buildCompiler = function () {
var _ref2 = _asyncToGenerator( /*#__PURE__*/_regenerator2.default.mark(function _callee(_ref3) {
var config = _ref3.config,
stage = _ref3.stage;
return _regenerator2.default.wrap(function _callee$(_context) {
while (1) {
switch (_context.prev = _context.next) {
case 0:
return _context.abrupt('return', (0, _webpack2.default)(webpackConfig({ config: config, stage: stage })));
case 1:
case 'end':
return _context.stop();
}
}
}, _callee, this);
}));
return function buildCompiler(_x) {
return _ref2.apply(this, arguments);
};
}();
// Starts the development server
var startDevServer = exports.startDevServer = function () {
var _ref4 = _asyncToGenerator( /*#__PURE__*/_regenerator2.default.mark(function _callee6(_ref5) {
var _this = this;
var config = _ref5.config;
var devCompiler, first, intendedPort, port, messagePort, host, devServerConfig, timefix, devServer, socket;
return _regenerator2.default.wrap(function _callee6$(_context6) {
while (1) {
switch (_context6.prev = _context6.next) {
case 0:
_context6.next = 2;
return buildCompiler({ config: config, stage: 'dev' });
case 2:
devCompiler = _context6.sent;
first = true;
// Default to localhost:3000, or use a custom combo if defined in static.config.js
// or environment variables
intendedPort = config.devServer && config.devServer.port || process.env.PORT || 3000;
_context6.next = 7;
return (0, _utils.findAvailablePort)(Number(intendedPort));
case 7:
port = _context6.sent;
_context6.next = 10;
return (0, _utils.findAvailablePort)(4000, [port]);
case 10:
messagePort = _context6.sent;
if (intendedPort !== port) {
console.time(_chalk2.default.red('=> Warning! Port ' + intendedPort + ' is not available. Using port ' + _chalk2.default.green(intendedPort) + ' instead!'));
}
host = config.devServer && config.devServer.host || process.env.HOST || 'http://localhost';
devServerConfig = _extends({
hot: true,
disableHostCheck: true,
contentBase: [config.paths.PUBLIC, config.paths.DIST],
publicPath: '/',
historyApiFallback: true,
compress: false,
quiet: true
}, config.devServer, {
watchOptions: _extends({
ignored: /node_modules/
}, config.devServer ? config.devServer.watchOptions || {} : {}),
before: function before(app) {
// Serve the site data
app.get('/__react-static__/getMessagePort', function () {
var _ref6 = _asyncToGenerator( /*#__PURE__*/_regenerator2.default.mark(function _callee2(req, res) {
return _regenerator2.default.wrap(function _callee2$(_context2) {
while (1) {
switch (_context2.prev = _context2.next) {
case 0:
res.json({
port: messagePort
});
case 1:
case 'end':
return _context2.stop();
}
}
}, _callee2, _this);
}));
return function (_x3, _x4) {
return _ref6.apply(this, arguments);
};
}());
app.get('/__react-static__/siteData', function () {
var _ref7 = _asyncToGenerator( /*#__PURE__*/_regenerator2.default.mark(function _callee3(req, res, next) {
var siteData;
return _regenerator2.default.wrap(function _callee3$(_context3) {
while (1) {
switch (_context3.prev = _context3.next) {
case 0:
_context3.prev = 0;
_context3.next = 3;
return config.getSiteData({ dev: true });
case 3:
siteData = _context3.sent;
res.json(siteData);
_context3.next = 12;
break;
case 7:
_context3.prev = 7;
_context3.t0 = _context3['catch'](0);
res.status(500);
res.json(_context3.t0);
next(_context3.t0);
case 12:
case 'end':
return _context3.stop();
}
}
}, _callee3, _this, [[0, 7]]);
}));
return function (_x5, _x6, _x7) {
return _ref7.apply(this, arguments);
};
}());
// Since routes may change during dev, this function can rebuild all of the config
// routes. It also references the original config when possible, to make sure it
// uses any up to date getData callback generated from new or replacement routes.
reloadWebpackRoutes = function reloadWebpackRoutes() {
// Serve each routes data
config.routes.forEach(function (_ref8) {
var routePath = _ref8.path;
app.get('/__react-static__/routeInfo/' + encodeURI(routePath === '/' ? '' : routePath), function () {
var _ref9 = _asyncToGenerator( /*#__PURE__*/_regenerator2.default.mark(function _callee4(req, res, next) {
var route, allProps;
return _regenerator2.default.wrap(function _callee4$(_context4) {
while (1) {
switch (_context4.prev = _context4.next) {
case 0:
// Make sure we have the most up to date route from the config, not
// an out of dat object.
route = config.routes.find(function (d) {
return d.path === routePath;
});
_context4.prev = 1;
if (!route.getData) {
_context4.next = 8;
break;
}
_context4.next = 5;
return route.getData({ dev: true });
case 5:
_context4.t0 = _context4.sent;
_context4.next = 9;
break;
case 8:
_context4.t0 = {};
case 9:
allProps = _context4.t0;
res.json(_extends({}, route, {
allProps: allProps
}));
_context4.next = 17;
break;
case 13:
_context4.prev = 13;
_context4.t1 = _context4['catch'](1);
res.status(500);
next(_context4.t1);
case 17:
case 'end':
return _context4.stop();
}
}
}, _callee4, _this, [[1, 13]]);
}));
return function (_x8, _x9, _x10) {
return _ref9.apply(this, arguments);
};
}());
});
};
reloadWebpackRoutes();
if (config.devServer && config.devServer.before) {
config.devServer.before(app);
}
},
port: port,
host: host
});
timefix = 11000;
devCompiler.plugin('watch-run', function (watching, callback) {
watching.startTime += timefix;
callback();
});
devCompiler.plugin('invalid', function () {
console.time(_chalk2.default.green('=> [\u2713] Build Complete'));
console.log('=> Rebuilding...');
});
devCompiler.plugin('done', function (stats) {
var messages = (0, _formatWebpackMessages2.default)(stats.toJson({}, true));
var isSuccessful = !messages.errors.length && !messages.warnings.length;
if (isSuccessful) {
console.timeEnd(_chalk2.default.green('=> [\u2713] Build Complete'));
if (first) {
first = false;
console.log(_chalk2.default.green('=> [\u2713] App serving at'), host + ':' + port);
stats.startTime -= timefix;
if (config.onStart) {
config.onStart({ devServerConfig: devServerConfig });
}
}
}
if (messages.errors.length) {
console.log(_chalk2.default.red('Failed to build! Fix any errors and try again!'));
messages.errors.forEach(function (message) {
console.log(message);
console.log();
});
}
if (messages.warnings.length) {
console.log(_chalk2.default.yellow('Built complete with warnings.'));
console.log();
messages.warnings.forEach(function (message) {
console.log(message);
console.log();
});
}
});
console.log('=> Building App Bundle...');
console.time(_chalk2.default.green('=> [\u2713] Build Complete'));
// Start the webpack dev server
devServer = new _webpackDevServer2.default(devCompiler, devServerConfig);
// Start the messages socket
socket = (0, _socket2.default)();
socket.listen(messagePort);
resolvedReloadRoutes = function () {
var _ref10 = _asyncToGenerator( /*#__PURE__*/_regenerator2.default.mark(function _callee5(paths) {
return _regenerator2.default.wrap(function _callee5$(_context5) {
while (1) {
switch (_context5.prev = _context5.next) {
case 0:
_context5.next = 2;
return (0, _.prepareRoutes)(config, { dev: true });
case 2:
if (!paths) {
paths = config.routes.map(function (route) {
return route.path;
});
}
paths = paths.map(_shared.cleanPath);
reloadWebpackRoutes();
socket.emit('message', { type: 'reloadRoutes', paths: paths });
case 6:
case 'end':
return _context5.stop();
}
}
}, _callee5, _this);
}));
return function resolvedReloadRoutes(_x11) {
return _ref10.apply(this, arguments);
};
}();
return _context6.abrupt('return', new Promise(function (resolve, reject) {
devServer.listen(port, function (err) {
if (err) {
return reject(err);
}
resolve();
});
}));
case 25:
case 'end':
return _context6.stop();
}
}
}, _callee6, this);
}));
return function startDevServer(_x2) {
return _ref4.apply(this, arguments);
};
}();
var buildProductionBundles = exports.buildProductionBundles = function () {
var _ref11 = _asyncToGenerator( /*#__PURE__*/_regenerator2.default.mark(function _callee7(_ref12) {
var config = _ref12.config;
return _regenerator2.default.wrap(function _callee7$(_context7) {
while (1) {
switch (_context7.prev = _context7.next) {
case 0:
return _context7.abrupt('return', new Promise(function (resolve, reject) {
(0, _webpack2.default)([webpackConfig({ config: config, stage: 'prod' }), webpackConfig({ config: config, stage: 'node' })]).run(function (err, stats) {
if (err) {
console.log(_chalk2.default.red(err.stack || err));
if (err.details) {
console.log(_chalk2.default.red(err.details));
}
return reject(err);
}
stats.toJson('verbose');
var _stats$stats = _slicedToArray(stats.stats, 2),
prodStats = _stats$stats[0],
nodeStats = _stats$stats[1];
checkBuildStats('prod', prodStats);
checkBuildStats('node', nodeStats);
function checkBuildStats(stage, stageStats) {
var buildErrors = stageStats.hasErrors();
var buildWarnings = stageStats.hasWarnings();
if (buildErrors || buildWarnings) {
console.log(stageStats.toString({
context: config.context,
performance: false,
hash: false,
timings: true,
entrypoints: false,
chunkOrigins: false,
chunkModules: false,
colors: true
}));
if (buildErrors) {
console.log(_chalk2.default.red.bold('\n => There were ERRORS during the ' + stage + ' build stage! :(\n => Fix them and try again!\n '));
} else if (buildWarnings) {
console.log(_chalk2.default.yellow.bold('\n => There were WARNINGS during the ' + stage + ' build stage!\n '));
}
}
}
var prodStatsJson = prodStats.toJson();
_fsExtra2.default.outputFileSync(_path2.default.join(config.paths.DIST, 'client-stats.json'), JSON.stringify(prodStatsJson, null, 2));
resolve(prodStatsJson);
});
}));
case 1:
case 'end':
return _context7.stop();
}
}
}, _callee7, this);
}));
return function buildProductionBundles(_x12) {
return _ref11.apply(this, arguments);
};
}();
exports.webpackConfig = webpackConfig;
var _webpack = require('webpack');
var _webpack2 = _interopRequireDefault(_webpack);
var _path = require('path');
var _path2 = _interopRequireDefault(_path);
var _formatWebpackMessages = require('react-dev-utils/formatWebpackMessages');
var _formatWebpackMessages2 = _interopRequireDefault(_formatWebpackMessages);
var _chalk = require('chalk');
var _chalk2 = _interopRequireDefault(_chalk);
var _webpackDevServer = require('webpack-dev-server');
var _webpackDevServer2 = _interopRequireDefault(_webpackDevServer);
var _socket = require('socket.io');
var _socket2 = _interopRequireDefault(_socket);
var _fsExtra = require('fs-extra');
var _fsExtra2 = _interopRequireDefault(_fsExtra);
var _rules = require('./rules');
var _utils = require('../../utils');
var _shared = require('../../utils/shared');
var _ = require('../');
function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
function _asyncToGenerator(fn) { return function () { var gen = fn.apply(this, arguments); return new Promise(function (resolve, reject) { function step(key, arg) { try { var info = gen[key](arg); var value = info.value; } catch (error) { reject(error); return; } if (info.done) { resolve(value); } else { return Promise.resolve(value).then(function (value) { step("next", value); }, function (err) { step("throw", err); }); } } return step("next"); }); }; } /* eslint-disable import/no-dynamic-require, react/no-danger, import/no-mutable-exports */
// import errorOverlayMiddleware from 'react-dev-utils/errorOverlayMiddleware'
//
var resolvedReloadRoutes = void 0;
var reloadWebpackRoutes = void 0;
var reloadRoutes = function reloadRoutes() {
if (!resolvedReloadRoutes) {
// Not ready yet, so just wait
return;
}
resolvedReloadRoutes.apply(undefined, arguments);
};
exports.reloadRoutes = reloadRoutes;
// Builds a compiler using a stage preset, then allows extension via
// webpackConfigurator
function webpackConfig(_ref) {
var config = _ref.config,
stage = _ref.stage;
var webpackConfig = void 0;
if (stage === 'dev') {
webpackConfig = require('./webpack.config.dev').default({ config: config });
} else if (stage === 'prod') {
webpackConfig = require('./webpack.config.prod').default({
config: config
});
} else if (stage === 'node') {
webpackConfig = require('./webpack.config.prod').default({
config: config,
isNode: true
});
} else {
throw new Error('A stage is required when building a compiler.');
}
var defaultLoaders = (0, _rules.getStagedRules)({ config: config, stage: stage });
if (config.webpack) {
var transformers = config.webpack;
if (!Array.isArray(config.webpack)) {
transformers = [config.webpack];
}
transformers.forEach(function (transformer) {
var modifiedConfig = transformer(webpackConfig, {
stage: stage,
defaultLoaders: defaultLoaders
});
if (modifiedConfig) {
webpackConfig = modifiedConfig;
}
});
}
return webpackConfig;
}