viewer.js 5.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222
  1. "use strict";
  2. function asyncGeneratorStep(gen, resolve, reject, _next, _throw, key, arg) { try { var info = gen[key](arg); var value = info.value; } catch (error) { reject(error); return; } if (info.done) { resolve(value); } else { Promise.resolve(value).then(_next, _throw); } }
  3. function _asyncToGenerator(fn) { return function () { var self = this, args = arguments; return new Promise(function (resolve, reject) { var gen = fn.apply(self, args); function _next(value) { asyncGeneratorStep(gen, resolve, reject, _next, _throw, "next", value); } function _throw(err) { asyncGeneratorStep(gen, resolve, reject, _next, _throw, "throw", err); } _next(undefined); }); }; }
  4. const path = require('path');
  5. const fs = require('fs');
  6. const http = require('http');
  7. const WebSocket = require('ws');
  8. const _ = require('lodash');
  9. const express = require('express');
  10. const ejs = require('ejs');
  11. const opener = require('opener');
  12. const mkdir = require('mkdirp');
  13. const {
  14. bold
  15. } = require('chalk');
  16. const Logger = require('./Logger');
  17. const analyzer = require('./analyzer');
  18. const projectRoot = path.resolve(__dirname, '..');
  19. const assetsRoot = path.join(projectRoot, 'public');
  20. module.exports = {
  21. startServer,
  22. generateReport,
  23. // deprecated
  24. start: startServer
  25. };
  26. function startServer(_x, _x2) {
  27. return _startServer.apply(this, arguments);
  28. }
  29. function _startServer() {
  30. _startServer = _asyncToGenerator(function* (bundleStats, opts) {
  31. const {
  32. port = 8888,
  33. host = '127.0.0.1',
  34. openBrowser = true,
  35. bundleDir = null,
  36. logger = new Logger(),
  37. defaultSizes = 'parsed',
  38. excludeAssets = null
  39. } = opts || {};
  40. const analyzerOpts = {
  41. logger,
  42. excludeAssets
  43. };
  44. let chartData = getChartData(analyzerOpts, bundleStats, bundleDir);
  45. if (!chartData) return;
  46. const app = express(); // Explicitly using our `ejs` dependency to render templates
  47. // Fixes #17
  48. app.engine('ejs', require('ejs').renderFile);
  49. app.set('view engine', 'ejs');
  50. app.set('views', `${projectRoot}/views`);
  51. app.use(express.static(`${projectRoot}/public`));
  52. app.use('/', (req, res) => {
  53. res.render('viewer', {
  54. mode: 'server',
  55. get chartData() {
  56. return chartData;
  57. },
  58. defaultSizes,
  59. enableWebSocket: true,
  60. // Helpers
  61. escapeJson
  62. });
  63. });
  64. const server = http.createServer(app);
  65. yield new Promise(resolve => {
  66. server.listen(port, host, () => {
  67. resolve();
  68. const url = `http://${host}:${server.address().port}`;
  69. logger.info(`${bold('Webpack Bundle Analyzer')} is started at ${bold(url)}\n` + `Use ${bold('Ctrl+C')} to close it`);
  70. if (openBrowser) {
  71. opener(url);
  72. }
  73. });
  74. });
  75. const wss = new WebSocket.Server({
  76. server
  77. });
  78. wss.on('connection', ws => {
  79. ws.on('error', err => {
  80. // Ignore network errors like `ECONNRESET`, `EPIPE`, etc.
  81. if (err.errno) return;
  82. logger.info(err.message);
  83. });
  84. });
  85. return {
  86. ws: wss,
  87. http: server,
  88. updateChartData
  89. };
  90. function updateChartData(bundleStats) {
  91. const newChartData = getChartData(analyzerOpts, bundleStats, bundleDir);
  92. if (!newChartData) return;
  93. chartData = newChartData;
  94. wss.clients.forEach(client => {
  95. if (client.readyState === WebSocket.OPEN) {
  96. client.send(JSON.stringify({
  97. event: 'chartDataUpdated',
  98. data: newChartData
  99. }));
  100. }
  101. });
  102. }
  103. });
  104. return _startServer.apply(this, arguments);
  105. }
  106. function generateReport(_x3, _x4) {
  107. return _generateReport.apply(this, arguments);
  108. }
  109. function _generateReport() {
  110. _generateReport = _asyncToGenerator(function* (bundleStats, opts) {
  111. const {
  112. openBrowser = true,
  113. reportFilename = 'report.html',
  114. bundleDir = null,
  115. logger = new Logger(),
  116. defaultSizes = 'parsed',
  117. excludeAssets = null
  118. } = opts || {};
  119. const chartData = getChartData({
  120. logger,
  121. excludeAssets
  122. }, bundleStats, bundleDir);
  123. if (!chartData) return;
  124. yield new Promise((resolve, reject) => {
  125. ejs.renderFile(`${projectRoot}/views/viewer.ejs`, {
  126. mode: 'static',
  127. chartData,
  128. defaultSizes,
  129. enableWebSocket: false,
  130. // Helpers
  131. assetContent: getAssetContent,
  132. escapeJson
  133. }, (err, reportHtml) => {
  134. try {
  135. if (err) {
  136. logger.error(err);
  137. reject(err);
  138. return;
  139. }
  140. const reportFilepath = path.resolve(bundleDir || process.cwd(), reportFilename);
  141. mkdir.sync(path.dirname(reportFilepath));
  142. fs.writeFileSync(reportFilepath, reportHtml);
  143. logger.info(`${bold('Webpack Bundle Analyzer')} saved report to ${bold(reportFilepath)}`);
  144. if (openBrowser) {
  145. opener(`file://${reportFilepath}`);
  146. }
  147. resolve();
  148. } catch (e) {
  149. reject(e);
  150. }
  151. });
  152. });
  153. });
  154. return _generateReport.apply(this, arguments);
  155. }
  156. function getAssetContent(filename) {
  157. const assetPath = path.join(assetsRoot, filename);
  158. if (!assetPath.startsWith(assetsRoot)) {
  159. throw new Error(`"${filename}" is outside of the assets root`);
  160. }
  161. return fs.readFileSync(assetPath, 'utf8');
  162. }
  163. /**
  164. * Escapes `<` characters in JSON to safely use it in `<script>` tag.
  165. */
  166. function escapeJson(json) {
  167. return JSON.stringify(json).replace(/</gu, '\\u003c');
  168. }
  169. function getChartData(analyzerOpts, ...args) {
  170. let chartData;
  171. const {
  172. logger
  173. } = analyzerOpts;
  174. try {
  175. chartData = analyzer.getViewerData(...args, analyzerOpts);
  176. } catch (err) {
  177. logger.error(`Could't analyze webpack bundle:\n${err}`);
  178. logger.debug(err.stack);
  179. chartData = null;
  180. }
  181. if (_.isPlainObject(chartData) && _.isEmpty(chartData)) {
  182. logger.error("Could't find any javascript bundles in provided stats file");
  183. chartData = null;
  184. }
  185. return chartData;
  186. }