Skip to content

Commit

Permalink
Merge pull request #397 from realityking/templates
Browse files Browse the repository at this point in the history
Replace ejs templates with a simple js file
  • Loading branch information
valscion committed Nov 30, 2020
2 parents 37f048a + 084cc02 commit 302df4f
Show file tree
Hide file tree
Showing 6 changed files with 85 additions and 167 deletions.
3 changes: 3 additions & 0 deletions CHANGELOG.md
Expand Up @@ -13,6 +13,9 @@ _Note: Gaps between patch versions are faulty, broken or test releases._
## UNRELEASED

<!-- Add changelog entries for new changes under this section -->
* **Improvement**
* A number of improvements to reduce the number of dependencies ([#391](https://github.com/webpack-contrib/webpack-bundle-analyzer/pull/391), [#396](https://github.com/webpack-contrib/webpack-bundle-analyzer/pull/396), [#397](https://github.com/webpack-contrib/webpack-bundle-analyzer/pull/397))

* **Bug Fix**
* Prevent crashes for bundles generated from webpack array configs. ([#394](https://github.com/webpack-contrib/webpack-bundle-analyzer/pull/394) by [@ctavan](https://github.com/ctavan))
* Fix `non-asset` assets causing analyze failure. ([#385](https://github.com/webpack-contrib/webpack-bundle-analyzer/issues/385) by [@ZKHelloworld](https://github.com/ZKHelloworld))
Expand Down
89 changes: 8 additions & 81 deletions package-lock.json

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

4 changes: 1 addition & 3 deletions package.json
Expand Up @@ -30,15 +30,13 @@
"files": [
"public",
"lib",
"src",
"views"
"src"
],
"dependencies": {
"acorn": "^8.0.4",
"acorn-walk": "^8.0.0",
"chalk": "^4.1.0",
"commander": "^6.2.0",
"ejs": "^3.1.5",
"express": "^4.17.1",
"filesize": "^6.1.0",
"gzip-size": "^6.0.0",
Expand Down
57 changes: 50 additions & 7 deletions views/viewer.ejs → src/template.js
@@ -1,22 +1,65 @@
<!DOCTYPE html>
/* eslint-disable max-len */
const path = require('path');
const fs = require('fs');

const _ = require('lodash');

const projectRoot = path.resolve(__dirname, '..');
const assetsRoot = path.join(projectRoot, 'public');

exports.renderViewer = renderViewer;

/**
* Escapes `<` characters in JSON to safely use it in `<script>` tag.
*/
function escapeJson(json) {
return JSON.stringify(json).replace(/</gu, '\\u003c');
}

function getAssetContent(filename) {
const assetPath = path.join(assetsRoot, filename);

if (!assetPath.startsWith(assetsRoot)) {
throw new Error(`"${filename}" is outside of the assets root`);
}

return fs.readFileSync(assetPath, 'utf8');
}

function html(strings, ...values) {
return strings.map((string, index) => `${string}${values[index] || ''}`).join('');
}

function getScript(filename, mode) {
if (mode === 'static') {
return `<!-- ${_.escape(filename)} -->
<script>${getAssetContent(filename)}</script>`;
} else {
return `<script src="${_.escape(filename)}"></script>`;
}
}

function renderViewer({title, enableWebSocket, chartData, defaultSizes, mode} = {}) {
return html`<!DOCTYPE html>
<html>
<head>
<meta charset="UTF-8"/>
<meta name="viewport" content="width=device-width, initial-scale=1"/>
<title><%= title %></title>
<title>${_.escape(title)}</title>
<link rel="shortcut icon" href="" type="image/x-icon" />
<script>
window.enableWebSocket = <%- escapeJson(enableWebSocket) %>;
window.enableWebSocket = ${escapeJson(enableWebSocket)};
</script>
<%- include('script', { filename: 'viewer.js' }) %>
${getScript('viewer.js', mode)}
</head>
<body>
<div id="app"></div>
<script>
window.chartData = <%- escapeJson(chartData) %>;
window.defaultSizes = <%- escapeJson(defaultSizes) %>;
window.chartData = ${escapeJson(chartData)};
window.defaultSizes = ${escapeJson(defaultSizes)};
</script>
</body>
</html>
</html>`;
}
91 changes: 23 additions & 68 deletions src/viewer.js
Expand Up @@ -5,15 +5,14 @@ const http = require('http');
const WebSocket = require('ws');
const _ = require('lodash');
const express = require('express');
const ejs = require('ejs');
const {bold} = require('chalk');

const Logger = require('./Logger');
const analyzer = require('./analyzer');
const {open} = require('./utils');
const {renderViewer} = require('./template');

const projectRoot = path.resolve(__dirname, '..');
const assetsRoot = path.join(projectRoot, 'public');

function resolveTitle(reportTitle) {
if (typeof reportTitle === 'function') {
Expand Down Expand Up @@ -50,24 +49,18 @@ async function startServer(bundleStats, opts) {
if (!chartData) return;

const app = express();

// Explicitly using our `ejs` dependency to render templates
// Fixes #17
app.engine('ejs', require('ejs').renderFile);
app.set('view engine', 'ejs');
app.set('views', `${projectRoot}/views`);
app.use(express.static(`${projectRoot}/public`));

app.use('/', (req, res) => {
res.render('viewer', {
app.get('/', (req, res) => {
res.writeHead(200, {'Content-Type': 'text/html'});
const html = renderViewer({
mode: 'server',
title: resolveTitle(reportTitle),
get chartData() { return chartData },
chartData,
defaultSizes,
enableWebSocket: true,
// Helpers
escapeJson
enableWebSocket: true
});
return res.end(html);
});

const server = http.createServer(app);
Expand Down Expand Up @@ -139,44 +132,23 @@ async function generateReport(bundleStats, opts) {

if (!chartData) return;

await new Promise((resolve, reject) => {
ejs.renderFile(
`${projectRoot}/views/viewer.ejs`,
{
mode: 'static',
title: resolveTitle(reportTitle),
chartData,
defaultSizes,
enableWebSocket: false,
// Helpers
assetContent: getAssetContent,
escapeJson
},
(err, reportHtml) => {
try {
if (err) {
logger.error(err);
reject(err);
return;
}

const reportFilepath = path.resolve(bundleDir || process.cwd(), reportFilename);

fs.mkdirSync(path.dirname(reportFilepath), {recursive: true});
fs.writeFileSync(reportFilepath, reportHtml);

logger.info(`${bold('Webpack Bundle Analyzer')} saved report to ${bold(reportFilepath)}`);

if (openBrowser) {
open(`file://${reportFilepath}`, logger);
}
resolve();
} catch (e) {
reject(e);
}
}
);
const reportHtml = renderViewer({
mode: 'static',
title: resolveTitle(reportTitle),
chartData,
defaultSizes,
enableWebSocket: false
});
const reportFilepath = path.resolve(bundleDir || process.cwd(), reportFilename);

fs.mkdirSync(path.dirname(reportFilepath), {recursive: true});
fs.writeFileSync(reportFilepath, reportHtml);

logger.info(`${bold('Webpack Bundle Analyzer')} saved report to ${bold(reportFilepath)}`);

if (openBrowser) {
open(`file://${reportFilepath}`, logger);
}
}

async function generateJSONReport(bundleStats, opts) {
Expand All @@ -192,23 +164,6 @@ async function generateJSONReport(bundleStats, opts) {
logger.info(`${bold('Webpack Bundle Analyzer')} saved JSON report to ${bold(reportFilename)}`);
}

function getAssetContent(filename) {
const assetPath = path.join(assetsRoot, filename);

if (!assetPath.startsWith(assetsRoot)) {
throw new Error(`"${filename}" is outside of the assets root`);
}

return fs.readFileSync(assetPath, 'utf8');
}

/**
* Escapes `<` characters in JSON to safely use it in `<script>` tag.
*/
function escapeJson(json) {
return JSON.stringify(json).replace(/</gu, '\\u003c');
}

function getChartData(analyzerOpts, ...args) {
let chartData;
const {logger} = analyzerOpts;
Expand Down
8 changes: 0 additions & 8 deletions views/script.ejs

This file was deleted.

0 comments on commit 302df4f

Please sign in to comment.