Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

report: autogenerate components.js from templates.html #12803

Merged
merged 55 commits into from
Aug 9, 2021
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
55 commits
Select commit Hold shift + click to select a range
bc277e3
wip
connorjclark Jul 15, 2021
3018199
start using dom stuff
connorjclark Jul 15, 2021
4ace55d
git rid of createTextNode
connorjclark Jul 15, 2021
8944fdd
tweaks
connorjclark Jul 15, 2021
ec87013
better args serialz
connorjclark Jul 15, 2021
4d1f12c
load templates file
connorjclark Jul 15, 2021
da5ac10
fragment
connorjclark Jul 16, 2021
f3cc791
lol
connorjclark Jul 16, 2021
3f473cf
attributeNS
connorjclark Jul 16, 2021
de88af4
lol
connorjclark Jul 16, 2021
2df95e3
add a innerHTML diff check
paulirish Jul 16, 2021
189c120
bring in DOM library
paulirish Jul 16, 2021
f9649c8
create good function name
paulirish Jul 16, 2021
f3e1069
move fn
paulirish Jul 16, 2021
ae4551e
some tlc
connorjclark Jul 16, 2021
0afbe2b
mostly working
connorjclark Jul 16, 2021
00ccf4f
async main
connorjclark Jul 16, 2021
f9fd13d
fix NS class
connorjclark Jul 16, 2021
f3b4495
lol
connorjclark Jul 16, 2021
7a56d4a
rm empty div
connorjclark Jul 16, 2021
024a55a
create components.js
connorjclark Jul 16, 2021
daccad9
reorder attributes
paulirish Jul 16, 2021
7ffdfaa
actually need to do createTextNode bc textContent blows away children
connorjclark Jul 16, 2021
3a2566d
Merge branch 'make-component-script' of github.com:GoogleChrome/light…
connorjclark Jul 16, 2021
bac811b
fix
connorjclark Jul 16, 2021
25aefd5
fix
connorjclark Jul 16, 2021
9c1a41b
minor
paulirish Jul 16, 2021
8afdcff
rm fallback
connorjclark Jul 16, 2021
f2653ca
better component names.
connorjclark Jul 16, 2021
b91c79e
snippets. refactor script a bit
connorjclark Jul 16, 2021
e4c1c1d
fix assert
connorjclark Jul 16, 2021
4e941cf
rename fn
connorjclark Jul 16, 2021
a6aff91
delete template context
connorjclark Jul 16, 2021
3399419
feedback
connorjclark Jul 20, 2021
2efb03f
wip
connorjclark Jul 21, 2021
0baf3dd
fix
connorjclark Jul 21, 2021
4f4e31b
Merge remote-tracking branch 'origin/master' into components
connorjclark Jul 22, 2021
38eb541
tsc
connorjclark Jul 22, 2021
7d00ae9
more append
connorjclark Jul 22, 2021
374d066
fix test
connorjclark Jul 22, 2021
5b14c17
Apply suggestions from code review
connorjclark Jul 22, 2021
a12de0f
snapshot
connorjclark Jul 22, 2021
2123847
Met pushrge branch 'components' of github.com:GoogleChrome/lighthouse…
connorjclark Jul 23, 2021
82c0f5a
pr
connorjclark Jul 23, 2021
227124e
update
connorjclark Jul 23, 2021
35ecad4
exec ctx test
connorjclark Jul 23, 2021
ca1211a
fix clients, remote templates.html from tests
connorjclark Jul 23, 2021
31fbea0
Merge remote-tracking branch 'origin/master' into components
connorjclark Jul 30, 2021
50faaa6
more ws
connorjclark Jul 30, 2021
7099549
update
connorjclark Jul 30, 2021
ae0f3d4
Merge remote-tracking branch 'origin/master' into components
connorjclark Aug 2, 2021
85a46a8
some comments
connorjclark Aug 3, 2021
7241e8d
comments
connorjclark Aug 3, 2021
fe66b3b
rm
connorjclark Aug 3, 2021
127b554
fix
connorjclark Aug 3, 2021
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
2 changes: 1 addition & 1 deletion build/build-dt-report-resources.js
Original file line number Diff line number Diff line change
Expand Up @@ -32,7 +32,7 @@ fs.mkdirSync(distDir);
writeFile('report.js', htmlReportAssets.REPORT_JAVASCRIPT);
writeFile('report.css', htmlReportAssets.REPORT_CSS);
writeFile('standalone-template.html', htmlReportAssets.REPORT_TEMPLATE);
writeFile('templates.html', htmlReportAssets.REPORT_TEMPLATES);
writeFile('templates.html', '<div>Empty file. Remove on next roll to CDT</div>');
writeFile('report.d.ts', 'export {}');
writeFile('report-generator.d.ts', 'export {}');

Expand Down
212 changes: 212 additions & 0 deletions build/build-report-components.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,212 @@
/**
* @license Copyright 2021 The Lighthouse Authors. All Rights Reserved.
* Licensed under the Apache License, Version 2.0 (the "License"); you may not use this file except in compliance with the License. You may obtain a copy of the License at http://www.apache.org/licenses/LICENSE-2.0
* Unless required by applicable law or agreed to in writing, software distributed under the License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the License for the specific language governing permissions and limitations under the License.
*/
'use strict';

/**
* @typedef CompiledComponent
* @property {HTMLTemplateElement} tmpEl
* @property {string} componentName
* @property {string} functionName
* @property {string} functionCode
*/

const fs = require('fs');
const jsdom = require('jsdom');
const {LH_ROOT} = require('../root.js');

const html = fs.readFileSync(LH_ROOT + '/report/assets/templates.html', 'utf-8');
const {window} = new jsdom.JSDOM(html);
const tmplEls = window.document.querySelectorAll('template');

/**
* @param {string} str
*/
function upperFirst(str) {
return str.charAt(0).toUpperCase() + str.substr(1);
}

/**
* @param {string} functionName
* @param {string[]} bodyLines
* @param {string[]} parameterNames
*/
function createFunctionCode(functionName, bodyLines, parameterNames = []) {
const body = bodyLines.map(l => ` ${l}`).join('\n');
const functionCode = `function ${functionName}(${parameterNames.join(', ')}) {\n${body}\n}`;
return functionCode;
}

/**
* @param {ChildNode} childNode
* @return {string|undefined}
*/
function normalizeTextNodeText(childNode) {
// Just for typescript.
if (!childNode.parentElement) return;
// Just for typescript. If a text node has no text, it's trivially not significant.
if (!childNode.textContent) return;

let textContent = childNode.textContent || '';
// Consecutive whitespace is redundant, unless in certain elements.
if (!['PRE', 'STYLE'].includes(childNode.parentElement.tagName)) {
textContent = textContent.replace(/\s+/g, ' ');
}

return textContent;
}

/**
* @param {HTMLTemplateElement} tmpEl
* @return {CompiledComponent}
*/
function compileTemplate(tmpEl) {
const elemToVarNames = new Map();
const lines = [];

/**
* @param {Element} el
* @return {string}
*/
function makeOrGetVarName(el) {
const varName = elemToVarNames.get(el) || ('el' + elemToVarNames.size);
elemToVarNames.set(el, varName);
return varName;
}

/**
* @param {Element} el
*/
function process(el) {
const isSvg = el.namespaceURI && el.namespaceURI.endsWith('/svg');
const namespaceURI = isSvg ? el.namespaceURI : '';
const tagName = el.localName;
const className = el.classList.toString();

let createElementFnName = 'createElement';
const args = [tagName];
if (className) {
args.push(className);
}
if (namespaceURI) {
createElementFnName = 'createElementNS';
args.unshift(namespaceURI);
}

const varName = makeOrGetVarName(el);
const argsSerialzed =
args.map(arg => arg === undefined ? 'undefined' : JSON.stringify(arg)).join(', ');
lines.push(`const ${varName} = dom.${createElementFnName}(${argsSerialzed});`);

if (el.getAttributeNames) {
for (const attr of el.getAttributeNames() || []) {
if (attr === 'class') continue;

lines.push(`${varName}.setAttribute('${attr}', '${el.getAttribute(attr)}');`);
}
}

/** @type {string[]} */
const childNodesToAppend = [];
for (const childNode of el.childNodes) {
if (childNode.nodeType === window.Node.COMMENT_NODE) continue;

if (childNode.nodeType === window.Node.TEXT_NODE) {
if (!childNode.parentElement) continue;

const textContent = normalizeTextNodeText(childNode);
if (!textContent) continue;

// Escaped string value for JS.
childNodesToAppend.push(JSON.stringify(textContent));
continue;
}

if (!(childNode instanceof /** @type {typeof Element} */ (window.Element))) {
throw new Error(`Expected ${childNode} to be an element`);
}
process(childNode);

const childVarName = elemToVarNames.get(childNode);
if (childVarName) childNodesToAppend.push(childVarName);
}

if (childNodesToAppend.length) {
lines.push(`${varName}.append(${childNodesToAppend.join(',')});`);
}
}

const fragmentVarName = makeOrGetVarName(tmpEl);
lines.push(`const ${fragmentVarName} = dom.document().createDocumentFragment();`);

for (const topLevelEl of tmpEl.content.children) {
process(topLevelEl);
lines.push(`${fragmentVarName}.append(${makeOrGetVarName(topLevelEl)});`);
}

lines.push(`return ${fragmentVarName};`);

const componentName = tmpEl.id;
const functionName = `create${upperFirst(componentName)}Component`;
const jsdoc = `
/**
* @param {DOM_} dom
*/`;
const functionCode = jsdoc + '\n' + createFunctionCode(functionName, lines, ['dom']);
return {tmpEl, componentName, functionName, functionCode};
}

/**
* @param {CompiledComponent[]} compiledTemplates
* @return {string}
*/
function makeGenericCreateComponentFunctionCode(compiledTemplates) {
const lines = [];

lines.push('switch (componentName) {');
for (const {componentName, functionName} of compiledTemplates) {
lines.push(` case '${componentName}': return ${functionName}(dom);`);
}
lines.push('}');
lines.push('throw new Error(\'unexpected component: \' + componentName)');

const paramType = compiledTemplates.map(t => `'${t.componentName}'`).join('|');
const jsdoc = `
/** @typedef {${paramType}} ComponentName */
/**
* @param {DOM_} dom
* @param {ComponentName} componentName
* @return {DocumentFragment}
*/`;
return jsdoc + '\nexport ' +
createFunctionCode('createComponent', lines, ['dom', 'componentName']);
}

async function main() {
const compiledTemplates = [...tmplEls].map(compileTemplate);
compiledTemplates.sort((a, b) => a.componentName.localeCompare(b.componentName));
const code = `
'use strict';

// auto-generated by build/build-report-components.js

// must import as DOM_ to avoid redeclaring 'DOM' export in bundle.d.ts, otherwise
// yarn test-devtools will fail.
/** @typedef {import('./dom.js').DOM} DOM_ */

/* eslint-disable max-len */

${compiledTemplates.map(t => t.functionCode).join('\n')}

${makeGenericCreateComponentFunctionCode(compiledTemplates)}
`.trim();
fs.writeFileSync(LH_ROOT + '/report/renderer/components.js', code);
}

if (require.main === module) {
main();
}

module.exports = {normalizeTextNodeText};
3 changes: 0 additions & 3 deletions build/build-viewer.js
Original file line number Diff line number Diff line change
Expand Up @@ -38,9 +38,6 @@ async function run() {
name: 'viewer',
appDir: `${LH_ROOT}/lighthouse-viewer/app`,
html: {path: 'index.html'},
htmlReplacements: {
'%%LIGHTHOUSE_TEMPLATES%%': htmlReportAssets.REPORT_TEMPLATES,
},
stylesheets: [
htmlReportAssets.REPORT_CSS,
{path: 'styles/*'},
Expand Down
3 changes: 0 additions & 3 deletions clients/devtools-report-assets.js
Original file line number Diff line number Diff line change
Expand Up @@ -30,7 +30,4 @@ module.exports = {
get REPORT_TEMPLATE() {
return cachedResources.get('third_party/lighthouse/report-assets/standalone-template.html');
},
get REPORT_TEMPLATES() {
return cachedResources.get('third_party/lighthouse/report-assets/templates.html');
},
};
1 change: 0 additions & 1 deletion lighthouse-viewer/app/src/lighthouse-report-viewer.js
Original file line number Diff line number Diff line change
Expand Up @@ -231,7 +231,6 @@ class LighthouseReportViewer {
features.initFeatures(json);
} catch (e) {
logger.error(`Error rendering report: ${e.message}`);
dom.resetTemplates(); // TODO(bckenny): hack
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

does the cache persist across loading different reports? That was the original need for this hack :/

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

We make a new DOM for every new report render.

container.textContent = '';
throw e;
} finally {
Expand Down
3 changes: 2 additions & 1 deletion package.json
Original file line number Diff line number Diff line change
Expand Up @@ -23,7 +23,7 @@
"build-smokehouse-bundle": "node ./build/build-smokehouse-bundle.js",
"build-lr": "yarn reset-link && node ./build/build-lightrider-bundles.js",
"build-pack": "bash build/build-pack.sh",
"build-report": "node build/build-report.js",
"build-report": "node build/build-report-components.js && yarn eslint --fix report/renderer/components.js && node build/build-report.js",
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

so now every yarn start we're running eslint? 😞

reminder me again why a watch is terrible? 😅

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

running eslint on one file, it just takes a second
image

why a watch is terrible?

Extra hoops to jump through for development is never good. running yarn start ... is simpler than remembering to open a new terminal and run a watch command

don't feel strongly. i say we vote on it.

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

The watch question is a good one (as someone who generally dislikes watch commands :) and we should definitely look at what we're doing in all our build steps and the efficiency of them, but this seems ok until we have #12689 mostly settled and know more of the lay of the land?

Personally I've been spoiled by esbuild and would love to get building everything down to like < 1s total, but we'll see what's possible :)

"build-treemap": "node ./build/build-treemap.js",
"build-viewer": "node ./build/build-viewer.js",
"reset-link": "(yarn unlink || true) && yarn link && yarn link lighthouse",
Expand Down Expand Up @@ -105,6 +105,7 @@
"@types/google.analytics": "0.0.39",
"@types/jest": "^24.0.9",
"@types/jpeg-js": "^0.3.0",
"@types/jsdom": "^16.2.13",
"@types/lodash.clonedeep": "^4.5.6",
"@types/lodash.get": "^4.4.6",
"@types/lodash.isequal": "^4.5.2",
Expand Down
12 changes: 5 additions & 7 deletions report/assets/styles.css

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

Loading