/** * Copyright 2022 Google Inc. 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 * * https://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. */ import fs from 'fs'; import path from 'path'; import { MochaTestResult, TestExpectation, MochaResults, TestResult, } from './types.js'; export function extendProcessEnv(envs: object[]): NodeJS.ProcessEnv { return envs.reduce( (acc: object, item: object) => { Object.assign(acc, item); return acc; }, { ...process.env, } ) as NodeJS.ProcessEnv; } export function getFilename(file: string): string { return path.basename(file).replace(path.extname(file), ''); } export function readJSON(path: string): unknown { return JSON.parse(fs.readFileSync(path, 'utf-8')); } export function filterByPlatform( items: T[], platform: NodeJS.Platform ): T[] { return items.filter(item => { return item.platforms.includes(platform); }); } export function prettyPrintJSON(json: unknown): void { console.log(JSON.stringify(json, null, 2)); } export function filterByParameters( expectations: TestExpectation[], parameters: string[] ): TestExpectation[] { const querySet = new Set(parameters); return expectations.filter(ex => { return ex.parameters.every(param => { return querySet.has(param); }); }); } /** * The last expectation that matches an empty string as all tests pattern * or the name of the file or the whole name of the test the filter wins. */ export function findEffectiveExpectationForTest( expectations: TestExpectation[], result: MochaTestResult ): TestExpectation | undefined { return expectations .filter(expectation => { return ( '' === expectation.testIdPattern || getTestId(result.file) === expectation.testIdPattern || getTestId(result.file, result.fullTitle) === expectation.testIdPattern ); }) .pop(); } type RecommendedExpecation = { expectation: TestExpectation; test: MochaTestResult; action: 'remove' | 'add' | 'update'; }; export function isWildCardPattern(testIdPattern: string): boolean { testIdPattern = testIdPattern.trim(); return ( testIdPattern === '' || Boolean(testIdPattern.match(/^\[[a-zA-Z]+\.spec\]$/)) ); } export function getExpectationUpdates( results: MochaResults, expecations: TestExpectation[], context: { platforms: NodeJS.Platform[]; parameters: string[]; } ): RecommendedExpecation[] { const output: RecommendedExpecation[] = []; for (const pass of results.passes) { const expectationEntry = findEffectiveExpectationForTest(expecations, pass); if (expectationEntry && !expectationEntry.expectations.includes('PASS')) { output.push({ expectation: expectationEntry, test: pass, action: 'remove', }); } } for (const failure of results.failures) { const expectationEntry = findEffectiveExpectationForTest( expecations, failure ); // If the effective explanation is a wildcard, we recommend adding a new // expectation instead of updating the wildcard that might affect multiple // tests. if ( expectationEntry && !isWildCardPattern(expectationEntry.testIdPattern) ) { if ( !expectationEntry.expectations.includes( getTestResultForFailure(failure) ) ) { output.push({ expectation: { ...expectationEntry, expectations: [ ...expectationEntry.expectations, getTestResultForFailure(failure), ], }, test: failure, action: 'update', }); } } else { output.push({ expectation: { testIdPattern: getTestId(failure.file, failure.fullTitle), platforms: context.platforms, parameters: context.parameters, expectations: [getTestResultForFailure(failure)], }, test: failure, action: 'add', }); } } return output; } export function getTestResultForFailure( test: Pick ): TestResult { return test.err?.code === 'ERR_MOCHA_TIMEOUT' ? 'TIMEOUT' : 'FAIL'; } export function getTestId(file: string, fullTitle?: string): string { return fullTitle ? `[${getFilename(file)}] ${fullTitle}` : `[${getFilename(file)}]`; }