blob: 41f399a3c71e89593ee29f27ca56f81cfb570c0f [file] [log] [blame]
// Copyright 2020 The Chromium Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
'use strict';
const espree = require('@typescript-eslint/parser');
const fs = require('fs');
const path = require('path');
const parseOptions = {
ecmaVersion: 'latest',
sourceType: 'module',
range: true,
};
/**
* Determines if a node is a class declaration.
* If className is provided, node must also match class name.
*/
function isClassNameDeclaration(node, className) {
const isClassDeclaration = node.type === 'ExportNamedDeclaration' && node.declaration.type === 'ClassDeclaration';
if (className) {
return isClassDeclaration && node.declaration.id.name === className;
}
return isClassDeclaration;
}
/**
* Determines if a node is an typescript enum declaration.
* If enumName is provided, node must also match enum name.
*/
function isEnumDeclaration(node, enumName) {
const isEnumDeclaration = node.type === 'ExportNamedDeclaration' && node.declaration.type === 'TSEnumDeclaration';
if (enumName) {
return isEnumDeclaration && node.declaration.id.name === enumName;
}
return isEnumDeclaration;
}
/**
* Finds a function declaration node inside a class declaration node
*/
function findFunctionInClass(classNode, functionName) {
for (const node of classNode.declaration.body.body) {
if (node.key.name === functionName) {
return node;
}
}
return null;
}
/**
* Determines if AST Node is a call to register a DevtoolsExperiment
*/
function isExperimentRegistrationCall(node) {
return (
node.expression && node.expression.type === 'CallExpression' &&
node.expression.callee.property.name === 'register');
}
/**
* Extract the enum Root.Runtime.ExperimentName to a map
*/
function getExperimentNameEnum(mainImplFile) {
const mainAST = espree.parse(mainImplFile, parseOptions);
let experimentNameEnum;
for (const node of mainAST.body) {
if (isEnumDeclaration(node, 'ExperimentName')) {
experimentNameEnum = node;
break;
}
}
const map = new Map();
if (!experimentNameEnum) {
return map;
}
for (const member of experimentNameEnum.declaration.members) {
map.set(member.id.name, member.initializer.value);
}
return map;
}
/**
* Determine if node is of the form Root.Runtime.ExperimentName.NAME, and if so
* return NAME as string.
*/
function isExperimentNameReference(node) {
if (node.type !== 'MemberExpression') {
return false;
}
if (node.object.type !== 'MemberExpression' || node.object.property?.name !== 'ExperimentName') {
return false;
}
if (node.object.object.type !== 'MemberExpression' || node.object.object.property?.name !== 'Runtime') {
return false;
}
if (node.object.object.object.type !== 'Identifier' || node.object.object.object.name !== 'Root') {
return false;
}
return node.property.name;
}
/**
* Gets list of experiments registered in MainImpl.js.
*/
function getMainImplExperimentList(mainImplFile, experimentNames) {
const mainAST = espree.parse(mainImplFile, parseOptions);
// Find MainImpl Class node
let mainImplClassNode;
for (const node of mainAST.body) {
if (isClassNameDeclaration(node, 'MainImpl')) {
mainImplClassNode = node;
break;
}
}
if (!mainImplClassNode) {
return null;
}
// Find function in MainImpl Class
const initializeExperimentNode = findFunctionInClass(
mainImplClassNode,
'initializeExperiments',
);
if (!initializeExperimentNode) {
return null;
}
// Get list of experiments
const experiments = [];
for (const statement of initializeExperimentNode.value.body.body) {
if (isExperimentRegistrationCall(statement)) {
// Experiment name is first argument of registration call
const experimentNameArg = statement.expression.arguments[0];
// The experiment name can either be a literal, e.g. 'fooExperiment'..
if (experimentNameArg.type === 'Literal') {
experiments.push(experimentNameArg.value);
} else {
// .. or a member of Root.Runtime.ExperimentName.
const experimentName = isExperimentNameReference(experimentNameArg);
if (experimentName) {
const translatedName = experimentNames.get(experimentName);
if (!translatedName) {
console.log(
'Failed to resolve Root.Runtime.ExperimentName.${experimentName} to a string',
);
process.exit(1);
}
experiments.push(translatedName);
} else {
console.log(
'Unexpected argument to Root.Runtime.experiments.register: ',
experimentNameArg,
);
process.exit(1);
}
}
}
}
return experiments.length ? experiments : null;
}
/**
* Determines if AST Node is the DevtoolsExperiments Enum declaration
*/
function isExperimentEnumDeclaration(node) {
return (node.type === 'ExportNamedDeclaration' && node?.declaration?.id?.name === 'DevtoolsExperiments');
}
/**
* Gets list of experiments registered in UserMetrics.ts
*/
function getUserMetricExperimentList(userMetricsFile) {
const userMetricsAST = espree.parse(userMetricsFile, parseOptions);
for (const node of userMetricsAST.body) {
if (isExperimentEnumDeclaration(node)) {
return node.declaration.members.filter(member => member.id.name !== 'MAX_VALUE')
.filter(
member => member.id.type === 'Literal' || member.id.type === 'Identifier',
)
.map(member => member.id.value ?? member.id.name);
}
}
return null;
}
/**
* Compares list of experiments, fires error if an experiment is registered without telemetry entry.
*/
function compareExperimentLists(mainImplList, userMetricsList) {
// Ensure both lists are valid
let errorFound = false;
if (!mainImplList) {
console.log(
'Changes to Devtools Experiment registration have prevented this check from finding registered experiments.',
);
console.log(
'Please update scripts/check_experiments.js to account for the new experiment registration.',
);
errorFound = true;
}
if (!userMetricsList) {
console.log(
'Changes to Devtools Experiment UserMetrics enum have prevented this check from finding experiments registered for telemetry.',
);
console.log(
'Please update scripts/check_experiments.js to account for the new experiment telemetry format.',
);
errorFound = true;
}
if (errorFound) {
process.exit(1);
}
// Ensure both lists match
const missingTelemetry = mainImplList.filter(
experiment => !userMetricsList.includes(experiment),
);
const staleTelemetry = userMetricsList.filter(
experiment => !mainImplList.includes(experiment),
);
if (missingTelemetry.length) {
console.log(
'Devtools Experiments have been added without corresponding histogram update!',
);
console.log(missingTelemetry.join('\n'));
console.log(
'Please ensure that the DevtoolsExperiments enum in UserMetrics.ts is updated with the new experiment.',
);
console.log(
'Please ensure that a corresponding CL is opened against chromium.src/tools/metrics/histograms/metadata/dev/enums.xml to update the DevtoolsExperiments enum',
);
errorFound = true;
}
if (staleTelemetry.length) {
console.log(
'Devtools Experiments that are no longer registered are still listed in the telemetry enum!',
);
console.log(staleTelemetry.join('\n'));
console.log(
'Please ensure that the DevtoolsExperiments enum in UserMetrics.ts is updated to remove these stale experiments.',
);
errorFound = true;
}
if (errorFound) {
process.exit(1);
}
console.log('DevTools Experiment Telemetry checker passed.');
}
function main() {
const mainImplPath = path.resolve(
__dirname,
'..',
'front_end',
'entrypoints',
'main',
'MainImpl.ts',
);
const mainImplFile = fs.readFileSync(mainImplPath, 'utf-8');
const userMetricsPath = path.resolve(
__dirname,
'..',
'front_end',
'core',
'host',
'UserMetrics.ts',
);
const userMetricsFile = fs.readFileSync(userMetricsPath, 'utf-8');
const runtimePath = path.resolve(
__dirname,
'..',
'front_end',
'core',
'root',
'Runtime.ts',
);
const runtimeFile = fs.readFileSync(runtimePath, 'utf-8');
const experimentNames = getExperimentNameEnum(runtimeFile);
compareExperimentLists(
getMainImplExperimentList(mainImplFile, experimentNames),
getUserMetricExperimentList(userMetricsFile),
);
}
main();