2023-01-26 22:20:05 +01:00
|
|
|
|
/**
|
|
|
|
|
|
* Copyright Microsoft Corporation. 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.
|
|
|
|
|
|
*/
|
|
|
|
|
|
|
2023-03-03 16:49:19 +01:00
|
|
|
|
import fs from 'fs';
|
2023-01-26 22:20:05 +01:00
|
|
|
|
import path from 'path';
|
2023-03-03 16:49:19 +01:00
|
|
|
|
import readline from 'readline';
|
2023-01-27 21:44:15 +01:00
|
|
|
|
import type { Reporter, TestError } from '../../types/testReporter';
|
2023-02-01 21:33:42 +01:00
|
|
|
|
import { InProcessLoaderHost, OutOfProcessLoaderHost } from './loaderHost';
|
2023-02-01 00:59:13 +01:00
|
|
|
|
import { Suite } from '../common/test';
|
|
|
|
|
|
import type { TestCase } from '../common/test';
|
2023-01-30 23:34:48 +01:00
|
|
|
|
import type { FullConfigInternal, FullProjectInternal } from '../common/types';
|
2023-03-03 16:49:19 +01:00
|
|
|
|
import { createFileMatcherFromArguments, createFileFiltersFromArguments, createTitleMatcher, errorWithFile, forceRegExp } from '../util';
|
2023-01-26 22:20:05 +01:00
|
|
|
|
import type { Matcher, TestFileFilter } from '../util';
|
2023-02-02 00:25:26 +01:00
|
|
|
|
import { buildProjectsClosure, collectFilesForProject, filterProjects } from './projectUtils';
|
2023-01-27 21:44:15 +01:00
|
|
|
|
import { requireOrImport } from '../common/transform';
|
2023-02-07 00:52:14 +01:00
|
|
|
|
import { buildFileSuiteForProject, filterByFocusedLine, filterByTestIds, filterOnly, filterTestsRemoveEmptySuites } from '../common/suiteUtils';
|
2023-03-14 18:41:37 +01:00
|
|
|
|
import { createTestGroups, filterForShard, type TestGroup } from './testGroups';
|
2023-03-03 00:09:50 +01:00
|
|
|
|
import { dependenciesForTestFile } from '../common/compilationCache';
|
2023-01-26 22:20:05 +01:00
|
|
|
|
|
2023-03-14 18:41:37 +01:00
|
|
|
|
export type ProjectWithTestGroups = {
|
|
|
|
|
|
project: FullProjectInternal;
|
|
|
|
|
|
projectSuite: Suite;
|
|
|
|
|
|
testGroups: TestGroup[];
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
export async function loadAllTests(mode: 'out-of-process' | 'in-process', config: FullConfigInternal, projectsToIgnore: Set<FullProjectInternal>, additionalFileMatcher: Matcher | undefined, errors: TestError[], shouldFilterOnly: boolean): Promise<{ rootSuite: Suite, projectsWithTestGroups: ProjectWithTestGroups[] }> {
|
2023-02-03 18:11:02 +01:00
|
|
|
|
const projects = filterProjects(config.projects, config._internal.cliProjectFilter);
|
2023-02-01 00:59:13 +01:00
|
|
|
|
|
2023-03-03 16:49:19 +01:00
|
|
|
|
// Interpret cli parameters.
|
|
|
|
|
|
const cliFileFilters = createFileFiltersFromArguments(config._internal.cliArgs);
|
|
|
|
|
|
const grepMatcher = config._internal.cliGrep ? createTitleMatcher(forceRegExp(config._internal.cliGrep)) : () => true;
|
|
|
|
|
|
const grepInvertMatcher = config._internal.cliGrepInvert ? createTitleMatcher(forceRegExp(config._internal.cliGrepInvert)) : () => false;
|
|
|
|
|
|
const cliTitleMatcher = (title: string) => !grepInvertMatcher(title) && grepMatcher(title);
|
|
|
|
|
|
const cliFileMatcher = config._internal.cliArgs.length ? createFileMatcherFromArguments(config._internal.cliArgs) : null;
|
|
|
|
|
|
|
2023-03-14 18:41:37 +01:00
|
|
|
|
const filesToRunByProject = new Map<FullProjectInternal, string[]>();
|
2023-02-01 00:59:13 +01:00
|
|
|
|
let topLevelProjects: FullProjectInternal[];
|
|
|
|
|
|
let dependencyProjects: FullProjectInternal[];
|
2023-02-01 21:33:42 +01:00
|
|
|
|
// Collect files, categorize top level and dependency projects.
|
2023-02-01 00:59:13 +01:00
|
|
|
|
{
|
2023-02-01 21:33:42 +01:00
|
|
|
|
const fsCache = new Map();
|
2023-03-03 16:49:19 +01:00
|
|
|
|
const sourceMapCache = new Map();
|
2023-02-01 21:33:42 +01:00
|
|
|
|
|
2023-02-01 00:59:13 +01:00
|
|
|
|
// First collect all files for the projects in the command line, don't apply any file filters.
|
|
|
|
|
|
const allFilesForProject = new Map<FullProjectInternal, string[]>();
|
|
|
|
|
|
for (const project of projects) {
|
2023-02-03 18:11:02 +01:00
|
|
|
|
if (projectsToIgnore.has(project))
|
|
|
|
|
|
continue;
|
2023-02-01 21:33:42 +01:00
|
|
|
|
const files = await collectFilesForProject(project, fsCache);
|
2023-02-01 00:59:13 +01:00
|
|
|
|
allFilesForProject.set(project, files);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// Filter files based on the file filters, eliminate the empty projects.
|
|
|
|
|
|
for (const [project, files] of allFilesForProject) {
|
2023-03-03 16:49:19 +01:00
|
|
|
|
const matchedFiles = await Promise.all(files.map(async file => {
|
|
|
|
|
|
if (additionalFileMatcher && !additionalFileMatcher(file))
|
|
|
|
|
|
return;
|
|
|
|
|
|
if (cliFileMatcher) {
|
|
|
|
|
|
if (!cliFileMatcher(file) && !await isPotentiallyJavaScriptFileWithSourceMap(file, sourceMapCache))
|
|
|
|
|
|
return;
|
|
|
|
|
|
}
|
|
|
|
|
|
return file;
|
|
|
|
|
|
}));
|
|
|
|
|
|
const filteredFiles = matchedFiles.filter(Boolean) as string[];
|
2023-02-01 00:59:13 +01:00
|
|
|
|
if (filteredFiles.length)
|
|
|
|
|
|
filesToRunByProject.set(project, filteredFiles);
|
|
|
|
|
|
}
|
2023-02-02 00:25:26 +01:00
|
|
|
|
|
|
|
|
|
|
const projectClosure = buildProjectsClosure([...filesToRunByProject.keys()]);
|
2023-03-14 18:41:37 +01:00
|
|
|
|
topLevelProjects = projectClosure.filter(p => p._internal.type === 'top-level' && !projectsToIgnore.has(p));
|
|
|
|
|
|
dependencyProjects = projectClosure.filter(p => p._internal.type === 'dependency' && !projectsToIgnore.has(p));
|
2023-02-03 18:11:02 +01:00
|
|
|
|
|
2023-02-01 00:59:13 +01:00
|
|
|
|
// (Re-)add all files for dependent projects, disregard filters.
|
|
|
|
|
|
for (const project of dependencyProjects) {
|
2023-03-14 18:41:37 +01:00
|
|
|
|
filesToRunByProject.delete(project);
|
2023-02-01 21:33:42 +01:00
|
|
|
|
const files = allFilesForProject.get(project) || await collectFilesForProject(project, fsCache);
|
2023-02-01 00:59:13 +01:00
|
|
|
|
filesToRunByProject.set(project, files);
|
|
|
|
|
|
}
|
2023-01-30 23:34:48 +01:00
|
|
|
|
}
|
2023-01-26 22:20:05 +01:00
|
|
|
|
|
2023-02-01 00:59:13 +01:00
|
|
|
|
// Load all test files and create a preprocessed root. Child suites are files there.
|
2023-03-03 00:09:50 +01:00
|
|
|
|
const fileSuites: Suite[] = [];
|
2023-02-01 21:33:42 +01:00
|
|
|
|
{
|
2023-02-07 23:08:17 +01:00
|
|
|
|
const loaderHost = mode === 'out-of-process' ? new OutOfProcessLoaderHost(config) : new InProcessLoaderHost(config);
|
2023-02-01 21:33:42 +01:00
|
|
|
|
const allTestFiles = new Set<string>();
|
|
|
|
|
|
for (const files of filesToRunByProject.values())
|
|
|
|
|
|
files.forEach(file => allTestFiles.add(file));
|
|
|
|
|
|
for (const file of allTestFiles) {
|
|
|
|
|
|
const fileSuite = await loaderHost.loadTestFile(file, errors);
|
2023-03-03 00:09:50 +01:00
|
|
|
|
fileSuites.push(fileSuite);
|
2023-02-01 21:33:42 +01:00
|
|
|
|
}
|
|
|
|
|
|
await loaderHost.stop();
|
2023-03-03 00:09:50 +01:00
|
|
|
|
|
|
|
|
|
|
// Check that no test file imports another test file.
|
|
|
|
|
|
// Loader must be stopped first, since it popuplates the dependency tree.
|
|
|
|
|
|
for (const file of allTestFiles) {
|
|
|
|
|
|
for (const dependency of dependenciesForTestFile(file)) {
|
|
|
|
|
|
if (allTestFiles.has(dependency)) {
|
|
|
|
|
|
const importer = path.relative(config.rootDir, file);
|
|
|
|
|
|
const importee = path.relative(config.rootDir, dependency);
|
|
|
|
|
|
errors.push({
|
|
|
|
|
|
message: `Error: test file "${importer}" should not import test file "${importee}"`,
|
|
|
|
|
|
location: { file, line: 1, column: 1 },
|
|
|
|
|
|
});
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
2023-02-01 21:33:42 +01:00
|
|
|
|
}
|
2023-01-26 22:20:05 +01:00
|
|
|
|
|
|
|
|
|
|
// Complain about duplicate titles.
|
2023-03-03 00:09:50 +01:00
|
|
|
|
errors.push(...createDuplicateTitlesErrors(config, fileSuites));
|
2023-01-26 22:20:05 +01:00
|
|
|
|
|
2023-02-01 00:59:13 +01:00
|
|
|
|
// Create root suites with clones for the projects.
|
|
|
|
|
|
const rootSuite = new Suite('', 'root');
|
|
|
|
|
|
|
2023-03-14 18:41:37 +01:00
|
|
|
|
// First iterate top-level projects to focus only, then add all other projects.
|
|
|
|
|
|
for (const project of topLevelProjects)
|
|
|
|
|
|
rootSuite._addSuite(await createProjectSuite(fileSuites, project, { cliFileFilters, cliTitleMatcher, testIdMatcher: config._internal.testIdMatcher }, filesToRunByProject.get(project)!));
|
2023-01-26 22:20:05 +01:00
|
|
|
|
|
|
|
|
|
|
// Complain about only.
|
|
|
|
|
|
if (config.forbidOnly) {
|
2023-02-01 00:59:13 +01:00
|
|
|
|
const onlyTestsAndSuites = rootSuite._getOnlyItems();
|
2023-01-26 22:20:05 +01:00
|
|
|
|
if (onlyTestsAndSuites.length > 0)
|
|
|
|
|
|
errors.push(...createForbidOnlyErrors(onlyTestsAndSuites));
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2023-03-14 18:41:37 +01:00
|
|
|
|
// Filter only for top-level projects.
|
2023-02-13 20:13:30 +01:00
|
|
|
|
if (shouldFilterOnly)
|
|
|
|
|
|
filterOnly(rootSuite);
|
2023-01-26 22:20:05 +01:00
|
|
|
|
|
2023-03-14 18:41:37 +01:00
|
|
|
|
// Create test groups for top-level projects.
|
|
|
|
|
|
let projectsWithTestGroups: ProjectWithTestGroups[] = [];
|
|
|
|
|
|
for (const projectSuite of rootSuite.suites) {
|
|
|
|
|
|
const project = projectSuite.project() as FullProjectInternal;
|
|
|
|
|
|
const testGroups = createTestGroups(projectSuite, config.workers);
|
|
|
|
|
|
projectsWithTestGroups.push({ project, projectSuite, testGroups });
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// Shard only the top-level projects.
|
|
|
|
|
|
if (config.shard) {
|
|
|
|
|
|
const allTestGroups: TestGroup[] = [];
|
|
|
|
|
|
for (const { testGroups } of projectsWithTestGroups)
|
|
|
|
|
|
allTestGroups.push(...testGroups);
|
|
|
|
|
|
const shardedTestGroups = filterForShard(config.shard, allTestGroups);
|
|
|
|
|
|
|
|
|
|
|
|
const shardedTests = new Set<TestCase>();
|
|
|
|
|
|
for (const group of shardedTestGroups) {
|
|
|
|
|
|
for (const test of group.tests)
|
|
|
|
|
|
shardedTests.add(test);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// Update project suites and test groups.
|
|
|
|
|
|
for (const p of projectsWithTestGroups) {
|
|
|
|
|
|
p.testGroups = p.testGroups.filter(group => shardedTestGroups.has(group));
|
|
|
|
|
|
filterTestsRemoveEmptySuites(p.projectSuite, test => shardedTests.has(test));
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// Remove now-empty top-level projects.
|
|
|
|
|
|
projectsWithTestGroups = projectsWithTestGroups.filter(p => p.testGroups.length > 0);
|
|
|
|
|
|
|
|
|
|
|
|
// Re-build the closure, project set might have changed.
|
|
|
|
|
|
const shardedProjectClosure = buildProjectsClosure(projectsWithTestGroups.map(p => p.project));
|
|
|
|
|
|
topLevelProjects = shardedProjectClosure.filter(p => p._internal.type === 'top-level' && !projectsToIgnore.has(p));
|
|
|
|
|
|
dependencyProjects = shardedProjectClosure.filter(p => p._internal.type === 'dependency' && !projectsToIgnore.has(p));
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2023-02-01 00:59:13 +01:00
|
|
|
|
// Prepend the projects that are dependencies.
|
|
|
|
|
|
for (const project of dependencyProjects) {
|
2023-03-03 00:09:50 +01:00
|
|
|
|
const projectSuite = await createProjectSuite(fileSuites, project, { cliFileFilters: [], cliTitleMatcher: undefined }, filesToRunByProject.get(project)!);
|
2023-03-14 18:41:37 +01:00
|
|
|
|
rootSuite._prependSuite(projectSuite);
|
|
|
|
|
|
const testGroups = createTestGroups(projectSuite, config.workers);
|
|
|
|
|
|
projectsWithTestGroups.push({ project, projectSuite, testGroups });
|
2023-02-01 00:59:13 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
2023-03-14 18:41:37 +01:00
|
|
|
|
return { rootSuite, projectsWithTestGroups };
|
2023-02-01 00:59:13 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
2023-03-14 18:41:37 +01:00
|
|
|
|
async function createProjectSuite(fileSuites: Suite[], project: FullProjectInternal, options: { cliFileFilters: TestFileFilter[], cliTitleMatcher?: Matcher, testIdMatcher?: Matcher }, files: string[]): Promise<Suite> {
|
2023-02-01 21:33:42 +01:00
|
|
|
|
const fileSuitesMap = new Map<string, Suite>();
|
2023-03-14 18:41:37 +01:00
|
|
|
|
for (const fileSuite of fileSuites)
|
2023-02-01 21:33:42 +01:00
|
|
|
|
fileSuitesMap.set(fileSuite._requireFile, fileSuite);
|
2023-02-01 00:59:13 +01:00
|
|
|
|
|
|
|
|
|
|
const projectSuite = new Suite(project.name, 'project');
|
|
|
|
|
|
projectSuite._projectConfig = project;
|
2023-02-02 00:25:26 +01:00
|
|
|
|
if (project._internal.fullyParallel)
|
2023-02-01 00:59:13 +01:00
|
|
|
|
projectSuite._parallelMode = 'parallel';
|
|
|
|
|
|
for (const file of files) {
|
2023-02-01 21:33:42 +01:00
|
|
|
|
const fileSuite = fileSuitesMap.get(file);
|
2023-02-01 00:59:13 +01:00
|
|
|
|
if (!fileSuite)
|
|
|
|
|
|
continue;
|
|
|
|
|
|
for (let repeatEachIndex = 0; repeatEachIndex < project.repeatEach; repeatEachIndex++) {
|
|
|
|
|
|
const builtSuite = buildFileSuiteForProject(project, fileSuite, repeatEachIndex);
|
|
|
|
|
|
projectSuite._addSuite(builtSuite);
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
2023-02-07 00:52:14 +01:00
|
|
|
|
|
2023-02-03 18:11:02 +01:00
|
|
|
|
filterByFocusedLine(projectSuite, options.cliFileFilters);
|
2023-02-07 00:52:14 +01:00
|
|
|
|
filterByTestIds(projectSuite, options.testIdMatcher);
|
2023-02-01 00:59:13 +01:00
|
|
|
|
|
|
|
|
|
|
const grepMatcher = createTitleMatcher(project.grep);
|
|
|
|
|
|
const grepInvertMatcher = project.grepInvert ? createTitleMatcher(project.grepInvert) : null;
|
|
|
|
|
|
|
|
|
|
|
|
const titleMatcher = (test: TestCase) => {
|
|
|
|
|
|
const grepTitle = test.titlePath().join(' ');
|
|
|
|
|
|
if (grepInvertMatcher?.(grepTitle))
|
|
|
|
|
|
return false;
|
2023-02-03 18:11:02 +01:00
|
|
|
|
return grepMatcher(grepTitle) && (!options.cliTitleMatcher || options.cliTitleMatcher(grepTitle));
|
2023-02-01 00:59:13 +01:00
|
|
|
|
};
|
|
|
|
|
|
|
2023-03-02 22:32:23 +01:00
|
|
|
|
filterTestsRemoveEmptySuites(projectSuite, titleMatcher);
|
|
|
|
|
|
return projectSuite;
|
2023-01-26 22:20:05 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
function createForbidOnlyErrors(onlyTestsAndSuites: (TestCase | Suite)[]): TestError[] {
|
|
|
|
|
|
const errors: TestError[] = [];
|
|
|
|
|
|
for (const testOrSuite of onlyTestsAndSuites) {
|
|
|
|
|
|
// Skip root and file.
|
|
|
|
|
|
const title = testOrSuite.titlePath().slice(2).join(' ');
|
|
|
|
|
|
const error: TestError = {
|
|
|
|
|
|
message: `Error: focused item found in the --forbid-only mode: "${title}"`,
|
|
|
|
|
|
location: testOrSuite.location!,
|
|
|
|
|
|
};
|
|
|
|
|
|
errors.push(error);
|
|
|
|
|
|
}
|
|
|
|
|
|
return errors;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2023-02-01 21:33:42 +01:00
|
|
|
|
function createDuplicateTitlesErrors(config: FullConfigInternal, fileSuites: Suite[]): TestError[] {
|
2023-01-26 22:20:05 +01:00
|
|
|
|
const errors: TestError[] = [];
|
2023-02-01 21:33:42 +01:00
|
|
|
|
for (const fileSuite of fileSuites) {
|
2023-01-26 22:20:05 +01:00
|
|
|
|
const testsByFullTitle = new Map<string, TestCase>();
|
|
|
|
|
|
for (const test of fileSuite.allTests()) {
|
2023-02-01 21:33:42 +01:00
|
|
|
|
const fullTitle = test.titlePath().slice(1).join(' › ');
|
2023-01-26 22:20:05 +01:00
|
|
|
|
const existingTest = testsByFullTitle.get(fullTitle);
|
|
|
|
|
|
if (existingTest) {
|
|
|
|
|
|
const error: TestError = {
|
|
|
|
|
|
message: `Error: duplicate test title "${fullTitle}", first declared in ${buildItemLocation(config.rootDir, existingTest)}`,
|
|
|
|
|
|
location: test.location,
|
|
|
|
|
|
};
|
|
|
|
|
|
errors.push(error);
|
|
|
|
|
|
}
|
|
|
|
|
|
testsByFullTitle.set(fullTitle, test);
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
return errors;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
function buildItemLocation(rootDir: string, testOrSuite: Suite | TestCase) {
|
|
|
|
|
|
if (!testOrSuite.location)
|
|
|
|
|
|
return '';
|
|
|
|
|
|
return `${path.relative(rootDir, testOrSuite.location.file)}:${testOrSuite.location.line}`;
|
|
|
|
|
|
}
|
2023-01-27 21:44:15 +01:00
|
|
|
|
|
|
|
|
|
|
async function requireOrImportDefaultFunction(file: string, expectConstructor: boolean) {
|
|
|
|
|
|
let func = await requireOrImport(file);
|
|
|
|
|
|
if (func && typeof func === 'object' && ('default' in func))
|
|
|
|
|
|
func = func['default'];
|
|
|
|
|
|
if (typeof func !== 'function')
|
|
|
|
|
|
throw errorWithFile(file, `file must export a single ${expectConstructor ? 'class' : 'function'}.`);
|
|
|
|
|
|
return func;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
export function loadGlobalHook(config: FullConfigInternal, file: string): Promise<(config: FullConfigInternal) => any> {
|
|
|
|
|
|
return requireOrImportDefaultFunction(path.resolve(config.rootDir, file), false);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
export function loadReporter(config: FullConfigInternal, file: string): Promise<new (arg?: any) => Reporter> {
|
|
|
|
|
|
return requireOrImportDefaultFunction(path.resolve(config.rootDir, file), true);
|
|
|
|
|
|
}
|
2023-03-03 16:49:19 +01:00
|
|
|
|
|
|
|
|
|
|
async function isPotentiallyJavaScriptFileWithSourceMap(file: string, cache: Map<string, boolean>): Promise<boolean> {
|
|
|
|
|
|
if (!file.endsWith('.js'))
|
|
|
|
|
|
return false;
|
|
|
|
|
|
if (cache.has(file))
|
|
|
|
|
|
return cache.get(file)!;
|
|
|
|
|
|
|
|
|
|
|
|
try {
|
|
|
|
|
|
const stream = fs.createReadStream(file);
|
|
|
|
|
|
const rl = readline.createInterface({ input: stream, crlfDelay: Infinity });
|
|
|
|
|
|
let lastLine: string | undefined;
|
|
|
|
|
|
rl.on('line', line => {
|
|
|
|
|
|
lastLine = line;
|
|
|
|
|
|
});
|
|
|
|
|
|
await new Promise((fulfill, reject) => {
|
|
|
|
|
|
rl.on('close', fulfill);
|
|
|
|
|
|
rl.on('error', reject);
|
|
|
|
|
|
stream.on('error', reject);
|
|
|
|
|
|
});
|
|
|
|
|
|
const hasSourceMap = !!lastLine && lastLine.startsWith('//# sourceMappingURL=');
|
|
|
|
|
|
cache.set(file, hasSourceMap);
|
|
|
|
|
|
return hasSourceMap;
|
|
|
|
|
|
} catch (e) {
|
|
|
|
|
|
cache.set(file, true);
|
|
|
|
|
|
return true;
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|