UNPKG

chrome-devtools-frontend

Version:
158 lines (137 loc) 5.35 kB
// Copyright 2022 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. // @ts-check const path = require('path'); /** * `path.dirname` does not include trailing slashes. If we would always * use `path.dirname` and then later perform comparisons on the paths that * it returns, we could run into false positives. For example, given the * the following two paths: * * front_end/timeline_model/TimelineModel.js * front_end/timeline/Timeline.js * * And that would have the following values for `path.dirname`: * * front_end/timeline_model * front_end/timeline * * If we would do a simple `.startswith` on the `path.dirname` of both of * these paths, then the first path would start with the dirname of the * second. However, they *are* part of different folders. To fix that problem, * we need to force a path separator after each folder. That makes sure we * and up with the following comparison of path dirnames: * * front_end/timeline_model/ * front_end/timeline/ * * Now, the first path does *not* start with the second one, as expected. * * @param {string} file * @return {string} */ function dirnameWithSeparator(file) { return path.dirname(file) + path.sep; } function devtoolsPlugin(source, importer) { if (!importer) { return null; } if (source === '../../lib/codemirror' || !source.startsWith('.')) { // These are imported via require(...), but we don't use // @rollup/plugin-commonjs. So this check is not necessary for rollup. But // need to have this for esbuild as it doesn't ignore require(...). return { id: source, external: true, }; } const currentDirectory = path.normalize(dirnameWithSeparator(importer)); const importedFilelocation = path.normalize(path.join(currentDirectory, source)); const importedFileDirectory = dirnameWithSeparator(importedFilelocation); // Generated files are part of other directories, as they are only imported once if (path.basename(importedFileDirectory) === 'generated') { return null; } // An import is considered external (and therefore a separate // bundle) if its filename matches its immediate parent's folder // name (without the extension). For example: // `import * as Components from './components/components.js'` = external // `import * as UI from '../ui/ui.js'` = external // `import * as Lit from '../third_party/lit/lit.js'` = external // `import {DataGrid} from './components/DataGrid.js'` = not external // `import * as Components from './components/foo.js'` = not external // Note that we can't do a simple check for only `third_party`, as in Chromium // our full path is `third_party/devtools-frontend/src/`, which thus *always* // includes third_party. It also not possible to use the current directory // as a check for the import, as the import will be different in Chromium and // would therefore not match the path of `__dirname`. // These should be removed because the new heuristic _should_ deal with these // e.g. it'll pick up third_party/lit/lit.js is its own entrypoint // The CodeMirror addons look like bundles (addon/comment/comment.js) but are not. if (importedFileDirectory.includes(path.join('front_end', 'third_party', 'codemirror', 'package'))) { return null; } // The LightHouse bundle shouldn't be processed by `terser` again, as it is uniquely built if (importedFilelocation.includes(path.join('front_end', 'third_party', 'lighthouse', 'lighthouse-dt-bundle.js'))) { return { id: importedFilelocation, external: true, }; } if (importedFileDirectory.includes(path.join('front_end', 'third_party', 'puppeteer', 'package'))) { // Ignore possible dynamic imports from the Node folder. if (importedFileDirectory.includes(path.join('front_end', 'third_party', 'puppeteer', 'package', 'lib', 'esm', 'puppeteer', 'node'))) { return { id: importedFilelocation, external: true, }; } return { id: importedFilelocation, external: false, }; } if (importedFileDirectory.includes(path.join('front_end', 'third_party', 'puppeteer-replay', 'package'))) { return { id: importedFilelocation, external: false, }; } const importedFileName = path.basename(importedFilelocation, '.js'); const importedFileParentDirectory = path.basename(path.dirname(importedFilelocation)); const isExternal = importedFileName === importedFileParentDirectory; return { id: importedFilelocation, external: isExternal, }; } function esbuildPlugin(outdir) { return args => { // args.importer is absolute path in esbuild. const res = devtoolsPlugin(args.path, args.importer); if (!res) { return null; } if (res.external) { // res.id can be both of absolutized local JavaScript path or node's // builtin module (e.g. 'fs', 'path'), and only relativize the path in // former case. if (path.isAbsolute(res.id)) { res.id = './' + path.relative(outdir, res.id); } return { external: res.external, path: res.id, }; } return { path: res.id, }; }; } module.exports = { devtoolsPlugin, esbuildPlugin };