feat(cli): support replace ext in dynamic imports (#12046)

* feat(get-deps): import('../foo.vue') => import('../foo.mjs') 的替换

* feat(get-deps): import('../foo.vue') => import('../foo.mjs') 的替换

* feat(get-deps): 对应正则的优化

* feat(get-deps): 对应正则的优化

* feat(get-deps): 对应正则的优化之前的太宽泛了

* feat(get-deps): 对应正则的优化之前的太宽泛了

* feat(get-deps): 对应正则的优化之前的太宽泛了

---------

Co-authored-by: sunguohui <sunguohui@vcom.com>
This commit is contained in:
suncohey 2023-07-29 11:29:46 +08:00 committed by GitHub
parent 49e1e9f73e
commit 3acee8bf2b
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23

View File

@ -8,6 +8,9 @@ let existsCache: Record<string, boolean> = {};
// https://regexr.com/47jlq // https://regexr.com/47jlq
const IMPORT_RE = const IMPORT_RE =
/import\s+?(?:(?:(?:[\w*\s{},]*)\s+from(\s+)?)|)(?:(?:".*?")|(?:'.*?'))[\s]*?(?:;|$|)/g; /import\s+?(?:(?:(?:[\w*\s{},]*)\s+from(\s+)?)|)(?:(?:".*?")|(?:'.*?'))[\s]*?(?:;|$|)/g;
const IMPORT_NOF_RE = /import\s*\(\s*?['"].+?['"]\)/g;
const EXPORT_FROM_RE = const EXPORT_FROM_RE =
/@?export\s+?(?:(?:(?:[\w*\s{},]*)\s+from(\s+)?)|)(?:(?:".*?")|(?:'.*?'))[\s]*?(?:;|$|)/g; /@?export\s+?(?:(?:(?:[\w*\s{},]*)\s+from(\s+)?)|)(?:(?:".*?")|(?:'.*?'))[\s]*?(?:;|$|)/g;
@ -16,6 +19,11 @@ function matchImports(code: string): string[] {
return imports.filter((line) => !line.includes('import type')); return imports.filter((line) => !line.includes('import type'));
} }
function matchImportsNof(code: string): string[] {
const imports = code.match(IMPORT_NOF_RE) || [];
return imports.filter((line) => !line.includes('import type'));
}
function matchExportFroms(code: string): string[] { function matchExportFroms(code: string): string[] {
const exportFroms = code.match(EXPORT_FROM_RE) || []; const exportFroms = code.match(EXPORT_FROM_RE) || [];
return exportFroms.filter((line) => !line.includes('export type')); return exportFroms.filter((line) => !line.includes('export type'));
@ -97,17 +105,23 @@ export function getDeps(filePath: string) {
/** /**
* 1. Replace .vue extension * 1. Replace .vue extension
* @example "import App from 'App.vue';" => "import App from 'App.xxx';" * @example "import App from 'App.vue';" => "import App from 'App.xxx';"
* @example "defineAsyncComponent(() => import('../xxx.vue'))" => "defineAsyncComponent(() => import('../xxx.xxx'));"
* *
* 2. if using .mjs or .cjs, complete the import path * 2. if using .mjs or .cjs, complete the import path
* @example import './foo' -> import './foo.mjs' * @example import './foo' -> import './foo.mjs'
* @example import './foo' -> import './foo/index.mjs' * @example import './foo' -> import './foo/index.mjs'
* @example "defineAsyncComponent(() => import('../foo.vue'))" => "defineAsyncComponent(() => import('../foo.mjs'));"
*/ */
export function replaceScriptImportExt( export function replaceScriptImportExt(
code: string, code: string,
filePath: string, filePath: string,
ext: string, ext: string,
) { ) {
const imports = [...matchImports(code), ...matchExportFroms(code)]; const imports = [
...matchImports(code),
...matchImportsNof(code),
...matchExportFroms(code),
];
const updateImport = (index: number, newImport: string) => { const updateImport = (index: number, newImport: string) => {
code = code.replace(imports[index], newImport); code = code.replace(imports[index], newImport);