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

core(fr): add base fraggle rock snapshot runner #11748

Merged
merged 5 commits into from
Dec 8, 2020
Merged
Show file tree
Hide file tree
Changes from 2 commits
Commits
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
1 change: 1 addition & 0 deletions jest.config.js
Original file line number Diff line number Diff line change
Expand Up @@ -22,6 +22,7 @@ module.exports = {
testMatch: [
'**/lighthouse-core/**/*-test.js',
'**/lighthouse-cli/**/*-test.js',
'**/lighthouse-core/test/fraggle-rock/**/*-test-pptr.js',
'**/lighthouse-treemap/**/*-test.js',
'**/lighthouse-treemap/**/*-test-pptr.js',
'**/lighthouse-viewer/**/*-test.js',
Expand Down
6 changes: 4 additions & 2 deletions lighthouse-cli/test/fixtures/static-server.js
Original file line number Diff line number Diff line change
Expand Up @@ -22,6 +22,8 @@ const HEADER_SAFELIST = new Set(['x-robots-tag', 'link']);
const lhRootDirPath = path.join(__dirname, '../../../');

class Server {
baseDir = __dirname;

constructor() {
this._server = http.createServer(this._requestHandler.bind(this));
/** @type {(data: string) => string=} */
Expand Down Expand Up @@ -64,8 +66,7 @@ class Server {
const requestUrl = parseURL(request.url);
const filePath = requestUrl.pathname;
const queryString = requestUrl.search && parseQueryString(requestUrl.search.slice(1));
let absoluteFilePath = path.join(__dirname, filePath);

let absoluteFilePath = path.join(this.baseDir, filePath);
const sendResponse = (statusCode, data) => {
// Used by Smokerider.
if (this._dataTransformer) data = this._dataTransformer(data);
Expand Down Expand Up @@ -227,6 +228,7 @@ if (require.main === module) {
console.log(`offline: listening on http://localhost:${offlinePort}`);
Copy link
Collaborator

Choose a reason for hiding this comment

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

one day we should split this off into a smoke-static-server-entry.js or something ..

} else {
module.exports = {
Server,
server: serverForOnline,
serverForOffline,
};
Expand Down
83 changes: 83 additions & 0 deletions lighthouse-core/fraggle-rock/api.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,83 @@
/**
* @license Copyright 2020 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';

const Driver = require('./gather/driver.js');
const Runner = require('../runner.js');
const Config = require('../config/config.js');

/**
* @param {LH.Gatherer.GathererInstance} gatherer
* @return {gatherer is LH.Gatherer.FRGathererInstance}
*/
function isFRGatherer(gatherer) {
Copy link
Member

Choose a reason for hiding this comment

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

Are all FR gatherers going to support snapshot mode?

Copy link
Collaborator Author

Choose a reason for hiding this comment

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

No, not all, see full compat table for which ones, but the next step after #11759 is the TODO that's here about using gatherer.meta to annotate whether a gatherer supports it or not.

Copy link
Member

Choose a reason for hiding this comment

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

The name isFRGatherer makes it sound like this will detect FR gatherers, but it's used to detect snapshot support on L60. WDYT of renaming this to something like isFRSnapshotGatherer?

Copy link
Collaborator Author

Choose a reason for hiding this comment

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

The name isFRGatherer makes it sound like this will detect FR gatherers

Excellent, that's exactly what it's supposed to be doing :) purely a type guard on whether a gatherer instance supports the FR gatherer properties.

but it's used to detect snapshot support on L60

I consider this function not detecting snapshot support at all which is why the TODO is there. Any suggestion for alternative wording there to make it clearer? Snapshot detection support is 2-3 PRs away from being fixed and won't affect any results in the meantime.

Copy link
Member

Choose a reason for hiding this comment

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

I think this issue has more to do with me misunderstanding the comment on L59 than the comment's wording. It's fine to leave as is :)

// TODO(FR-COMPAT): use configuration on gatherer.meta to detect interface compatibility
return gatherer.name === 'Accessibility';
}

/** @param {{page: import('puppeteer').Page, config?: LH.Config.Json}} options */
async function snapshot(options) {
const config = new Config(options.config);
const driver = new Driver(options.page);
await driver.connect();

const url = await options.page.url();

return Runner.run(
async () => {
/** @type {LH.BaseArtifacts} */
const baseArtifacts = {
fetchTime: new Date().toJSON(),
LighthouseRunWarnings: [],
URL: {requestedUrl: url, finalUrl: url},
Timing: [],
Stacks: [],
settings: config.settings,
// TODO(FR-COMPAT): convert these to regular artifacts
HostFormFactor: 'mobile',
TestedAsMobileDevice: true,
HostUserAgent: 'unknown',
patrickhulce marked this conversation as resolved.
Show resolved Hide resolved
NetworkUserAgent: 'unknown',
BenchmarkIndex: 0,
InstallabilityErrors: {errors: []},
traces: {},
devtoolsLogs: {},
WebAppManifest: null,
PageLoadError: null,
};

const gatherers = (config.passes || [])
.map(pass => pass.gatherers)
.reduce((a, b) => a.concat(b), []);
patrickhulce marked this conversation as resolved.
Show resolved Hide resolved

/** @type {Partial<LH.GathererArtifacts>} */
const artifacts = {};

for (const {instance} of gatherers) {
// TODO(FR-COMPAT): use configuration on gatherer.meta to detect snapshot support
if (!isFRGatherer(instance)) continue;

/** @type {keyof LH.GathererArtifacts} */
const artifactName = instance.name;
const artifact = await Promise.resolve()
patrickhulce marked this conversation as resolved.
Show resolved Hide resolved
.then(() => instance.afterPass({driver}))
.catch(err => err);

artifacts[artifactName] = artifact;
}

return /** @type {LH.Artifacts} */ ({...baseArtifacts, ...artifacts}); // Cast to drop Partial<>
Copy link
Member

Choose a reason for hiding this comment

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

I assume you're already thinking about how these types are mostly useless (sometimes worse than useless) in a world where we can't usually assume they're all actually there...

Copy link
Collaborator Author

Choose a reason for hiding this comment

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

Important to note this section was just copied from our existing gathering so it's not a degradation :)

I have not invested significant time on ideating how to improve the types here, no. I don't really consider the pre-existing temporary lie of LH.Artifacts at the runner level to be high on the list of type migrations that need to be done to get FR off the ground. Do you think this type flaw we have today is much more important in the FR world to warrant addressing soon?

},
{
url,
config,
}
);
}

module.exports = {
snapshot,
};
3 changes: 1 addition & 2 deletions lighthouse-core/gather/gather-runner.js
Original file line number Diff line number Diff line change
Expand Up @@ -35,12 +35,11 @@ const str_ = i18n.createMessageInstanceIdFn(__filename, UIStrings);

/** @typedef {import('../gather/driver.js')} Driver */

/** @typedef {import('./gatherers/gatherer.js').PhaseResult} PhaseResult */
/**
* Each entry in each gatherer result array is the output of a gatherer phase:
* `beforePass`, `pass`, and `afterPass`. Flattened into an `LH.Artifacts` in
* `collectArtifacts`.
* @typedef {Record<keyof LH.GathererArtifacts, Array<PhaseResult|Promise<PhaseResult>>>} GathererResults
* @typedef {Record<keyof LH.GathererArtifacts, Array<LH.Gatherer.PhaseResult>>} GathererResults
*/
/** @typedef {Array<[keyof GathererResults, GathererResults[keyof GathererResults]]>} GathererResultsEntries */

Expand Down
3 changes: 3 additions & 0 deletions lighthouse-core/gather/gatherers/accessibility.js
Original file line number Diff line number Diff line change
Expand Up @@ -92,6 +92,9 @@ function runA11yChecks() {
});
}

/**
* @implements {LH.Gatherer.FRGathererInstance}
*/
class Accessibility extends Gatherer {
/**
* @param {LH.Gatherer.FRTransitionalContext} passContext
Expand Down
10 changes: 5 additions & 5 deletions lighthouse-core/gather/gatherers/gatherer.js
Original file line number Diff line number Diff line change
Expand Up @@ -5,8 +5,6 @@
*/
'use strict';

/** @typedef {void|LH.GathererArtifacts[keyof LH.GathererArtifacts]} PhaseResult */

/**
* Base class for all gatherers; defines pass lifecycle methods. The artifact
* from the gatherer is the last not-undefined value returned by a lifecycle
Expand All @@ -16,6 +14,8 @@
* If an Error is thrown (or a Promise that rejects on an Error),
* the runner will treat it as an error internal to the gatherer and
* continue execution of any remaining gatherers.
*
* @implements {LH.Gatherer.GathererInstance}
*/
class Gatherer {
/**
Expand All @@ -31,15 +31,15 @@ class Gatherer {
/**
* Called before navigation to target url.
* @param {LH.Gatherer.PassContext} passContext
* @return {PhaseResult|Promise<PhaseResult>}
* @return {LH.Gatherer.PhaseResult}
*/
beforePass(passContext) { }

/**
* Called after target page is loaded. If a trace is enabled for this pass,
* the trace is still being recorded.
* @param {LH.Gatherer.PassContext} passContext
* @return {PhaseResult|Promise<PhaseResult>}
* @return {LH.Gatherer.PhaseResult}
*/
pass(passContext) { }

Expand All @@ -49,7 +49,7 @@ class Gatherer {
* and record of network activity are provided in `loadData`.
* @param {LH.Gatherer.PassContext} passContext
* @param {LH.Gatherer.LoadData} loadData
* @return {PhaseResult|Promise<PhaseResult>}
* @return {LH.Gatherer.PhaseResult}
*/
afterPass(passContext, loadData) { }

Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,40 @@
<!doctype html>
<!--
* Copyright 2020 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.
-->
<html lang="en">
<head>
<title>Interactive Onclick Tester</title>
<meta charset="utf-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0, minimum-scale=1.0">
</head>
<body>
Hello, Fraggle Rock!

This page has no accessibility violations until the mouse is clicked.

<template id="click-target">
<button>Click to add violations</button>
</template>

<template id="violations">
<input type="text" />
</template>

<script>
function addTemplate(selector) {
/** @type {HTMLTemplateElement} */
const template = document.querySelector(selector);
document.body.appendChild(template.content.cloneNode(true));
}

document.addEventListener('click', () => {
addTemplate('#violations');
});

addTemplate('#click-target');
</script>
</body>
</html>
84 changes: 84 additions & 0 deletions lighthouse-core/test/fraggle-rock/api-test-pptr.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,84 @@
/**
Copy link
Member

Choose a reason for hiding this comment

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

if this ends up expanding to a bunch of new smoke tests, are you thinking about moving them somewhere else/called somehow else instead of in the midst of all the (mostly) unit tests?

Copy link
Collaborator Author

Choose a reason for hiding this comment

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

called somehow else, yes definitely 👍

moved somewhere else, no wasn't planning on it.

* @license Copyright 2020 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';

/* eslint-env jest */

const path = require('path');
const lighthouse = require('../../fraggle-rock/api.js');
const puppeteer = require('puppeteer');
const StaticServer = require('../../../lighthouse-cli/test/fixtures/static-server.js').Server;

jest.setTimeout(90 * 1000);
Copy link
Member

Choose a reason for hiding this comment

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

90_000 y'all, step into the future :)


describe('Fraggle Rock API', () => {
/** @type {InstanceType<StaticServer>} */
let server;
/** @type {import('puppeteer').Browser} */
let browser;
/** @type {import('puppeteer').Page} */
let page;
/** @type {string} */
let serverBaseUrl;

beforeAll(async () => {
server = new StaticServer();
await server.listen(0, '127.0.0.1');
serverBaseUrl = `http://localhost:${server.getPort()}`;
browser = await puppeteer.launch({
headless: true,
});
});

beforeEach(async () => {
page = await browser.newPage();
});

afterEach(async () => {
await page.close();
});

afterAll(async () => {
await browser.close();
await server.close();
});

describe('snapshot', () => {
beforeEach(() => {
server.baseDir = path.join(__dirname, '../fixtures/fraggle-rock/snapshot-basic');
});

it('should compute accessibility results on the page as-is', async () => {
await page.goto(`${serverBaseUrl}/onclick.html`);
// Wait for the javascript to run
await page.waitForSelector('button');
await page.click('button');
// Wait for the violations to appear
await page.waitForSelector('input');

const result = await lighthouse.snapshot({page});
if (!result) throw new Error('Lighthouse failed to produce a result');

const {lhr} = result;
const accessibility = lhr.categories.accessibility;
expect(accessibility.score).toBeLessThan(1);

const auditResults = accessibility.auditRefs.map(ref => lhr.audits[ref.id]);
const irrelevantDisplayModes = new Set(['notApplicable', 'manual']);
const applicableAudits = auditResults
.filter(audit => !irrelevantDisplayModes.has(audit.scoreDisplayMode));

const erroredAudits = applicableAudits
.filter(audit => audit.score === null);
expect(erroredAudits).toHaveLength(0);

const failedAuditIds = applicableAudits
.filter(audit => audit.score !== null && audit.score < 1)
.map(audit => audit.id);
expect(failedAuditIds).toContain('label');
});
});
});
1 change: 1 addition & 0 deletions tsconfig.json
Original file line number Diff line number Diff line change
Expand Up @@ -34,6 +34,7 @@
"lighthouse-core/test/gather/driver/execution-context-test.js",
"lighthouse-core/test/fraggle-rock/gather/session-test.js",
"lighthouse-core/test/fraggle-rock/gather/driver-test.js",
"lighthouse-core/test/fraggle-rock/api-test-pptr.js",
"lighthouse-core/test/gather/driver-test.js",
"lighthouse-core/test/gather/gather-runner-test.js",
"lighthouse-core/test/audits/script-treemap-data-test.js"
Expand Down
18 changes: 11 additions & 7 deletions types/config.d.ts
Original file line number Diff line number Diff line change
Expand Up @@ -4,9 +4,12 @@
* 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.
*/

import Gatherer = require('../lighthouse-core/gather/gatherers/gatherer.js');
import Audit = require('../lighthouse-core/audits/audit.js');

interface ClassOf<T> {
new (): T;
}

declare global {
module LH {
module Config {
Expand Down Expand Up @@ -41,12 +44,13 @@ declare global {
path: string;
options?: {};
} | {
implementation: typeof Gatherer;
implementation: ClassOf<Gatherer.GathererInstance>;
Copy link
Member

Choose a reason for hiding this comment

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

What does changing this do? typeof Gatherer already is a ClassOf<Gatherer.GathererInstance>

Copy link
Collaborator Author

Choose a reason for hiding this comment

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

options?: {};
} | {
instance: InstanceType<typeof Gatherer>;
Copy link
Member

Choose a reason for hiding this comment

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

this is working around an old bug and should be able to just be instance: Gatherer, fwiw

instance: Gatherer.GathererInstance;
options?: {};
} | Gatherer | typeof Gatherer | string;
} | Gatherer.GathererInstance | ClassOf<Gatherer.GathererInstance> | string;


export interface CategoryJson {
title: string | IcuMessage;
Expand Down Expand Up @@ -87,8 +91,8 @@ declare global {
}

export interface GathererDefn {
implementation?: typeof Gatherer;
instance: InstanceType<typeof Gatherer>;
implementation?: ClassOf<Gatherer.GathererInstance>;
instance: Gatherer.GathererInstance;
path?: string;
}

Expand Down Expand Up @@ -125,4 +129,4 @@ declare global {
}

// empty export to keep file a module
export {}
export {};
15 changes: 15 additions & 0 deletions types/gatherer.d.ts
Original file line number Diff line number Diff line change
Expand Up @@ -51,6 +51,21 @@ declare global {
trace?: Trace;
}

type PhaseResult_ = void|LH.GathererArtifacts[keyof LH.GathererArtifacts]
export type PhaseResult = PhaseResult_ | Promise<PhaseResult_>

export interface GathererInstance {
Copy link
Member

Choose a reason for hiding this comment

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

same here. The js class is the base class and the interface. Duplicating it here doesn't seem to do more with the type and adds to maintenance/distancing types from jsdocs, etc

Copy link
Collaborator Author

Choose a reason for hiding this comment

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

Decoupling the implementation and the interface is the goal here to distinguish the new from the old and differentiate which gatherers support which models of execution in a relatively trivial and lightweight way.

Is there a counter proposal in there I'm missing for how incremental transition for gatherers will be supported? I'm happy to move the docs to the interface if that's what you're asking.

name: keyof LH.GathererArtifacts;
beforePass(context: LH.Gatherer.PassContext): PhaseResult;
pass(context: LH.Gatherer.PassContext): PhaseResult;
afterPass(context: LH.Gatherer.PassContext, loadData: LH.Gatherer.LoadData): PhaseResult;
}

export interface FRGathererInstance {
name: keyof LH.GathererArtifacts;
afterPass(context: FRTransitionalContext): PhaseResult;
}

namespace Simulation {
export type GraphNode = import('../lighthouse-core/lib/dependency-graph/base-node').Node;
export type GraphNetworkNode = _NetworkNode;
Expand Down