From f6f2f44d5126678c1ee323ce5278edbc0100cdef Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E4=B8=89=E5=92=B2=E6=99=BA=E5=AD=90?= Date: Fri, 24 Jun 2022 21:41:09 +0800 Subject: [PATCH] feat: backport supports cts and mts config (#8729) PR for v3: #8729 --- .../resolve-config/__tests__/serve.js | 12 ++- .../root/{vite.config.js => vite.config.ts} | 3 +- packages/vite/src/node/config.ts | 80 ++++++++----------- packages/vite/src/node/constants.ts | 9 +++ packages/vite/src/node/utils.ts | 2 + 5 files changed, 54 insertions(+), 52 deletions(-) rename packages/playground/resolve-config/root/{vite.config.js => vite.config.ts} (69%) diff --git a/packages/playground/resolve-config/__tests__/serve.js b/packages/playground/resolve-config/__tests__/serve.js index bd451d4cf6f6bc..4ad99dde41ff9f 100644 --- a/packages/playground/resolve-config/__tests__/serve.js +++ b/packages/playground/resolve-config/__tests__/serve.js @@ -8,7 +8,7 @@ const { testDir } = require('../../testUtils') const fromTestDir = (/** @type{string[]} */ ...p) => path.resolve(testDir, ...p) -const configNames = ['js', 'cjs', 'mjs', 'ts'] +const configNames = ['js', 'cjs', 'mjs', 'ts', 'mts', 'cts'] /** @param {string} root @param {boolean} isProd */ exports.serve = async function serve(root, isProd) { @@ -20,9 +20,9 @@ exports.serve = async function serve(root, isProd) { const pathToConf = fromTestDir(configName, `vite.config.${configName}`) await fs.copy(fromTestDir('root'), fromTestDir(configName)) - await fs.rename(fromTestDir(configName, 'vite.config.js'), pathToConf) + await fs.rename(fromTestDir(configName, 'vite.config.ts'), pathToConf) - if (configName === 'cjs') { + if (['cjs', 'cts'].includes(configName)) { const conf = await fs.readFile(pathToConf, 'utf8') await fs.writeFile( pathToConf, @@ -30,6 +30,12 @@ exports.serve = async function serve(root, isProd) { ) } + // Remove TS annotation for plain JavaScript file. + if (configName.endsWith('js')) { + const conf = await fs.readFile(pathToConf, 'utf8') + await fs.writeFile(pathToConf, conf.replace(': boolean', '')) + } + // copy directory and add package.json with "type": "module" await fs.copy(fromTestDir(configName), fromTestDir(`${configName}-module`)) await fs.writeJSON(fromTestDir(`${configName}-module`, 'package.json'), { diff --git a/packages/playground/resolve-config/root/vite.config.js b/packages/playground/resolve-config/root/vite.config.ts similarity index 69% rename from packages/playground/resolve-config/root/vite.config.js rename to packages/playground/resolve-config/root/vite.config.ts index ed72046f940d59..a6ad03b2122361 100644 --- a/packages/playground/resolve-config/root/vite.config.js +++ b/packages/playground/resolve-config/root/vite.config.ts @@ -1,5 +1,6 @@ +const __CONFIG_LOADED__: boolean = true export default { - define: { __CONFIG_LOADED__: true }, + define: { __CONFIG_LOADED__ }, logLevel: 'silent', build: { minify: false, diff --git a/packages/vite/src/node/config.ts b/packages/vite/src/node/config.ts index 1df307d6c017f4..d120bf21ce64eb 100644 --- a/packages/vite/src/node/config.ts +++ b/packages/vite/src/node/config.ts @@ -24,12 +24,18 @@ import { dynamicImport, isExternalUrl, isObject, + isTS, lookupFile, normalizePath } from './utils' import { resolvePlugins } from './plugins' import type { ESBuildOptions } from './plugins/esbuild' -import { CLIENT_ENTRY, DEFAULT_ASSETS_RE, ENV_ENTRY } from './constants' +import { + CLIENT_ENTRY, + DEFAULT_ASSETS_RE, + DEFAULT_CONFIG_FILES, + ENV_ENTRY +} from './constants' import type { InternalResolveOptions, ResolveOptions } from './plugins/resolve' import { resolvePlugin } from './plugins/resolve' import type { LogLevel, Logger } from './logger' @@ -863,56 +869,20 @@ export async function loadConfigFromFile( const getTime = () => `${(performance.now() - start).toFixed(2)}ms` let resolvedPath: string | undefined - let isTS = false - let isESM = false let dependencies: string[] = [] - // check package.json for type: "module" and set `isMjs` to true - try { - const pkg = lookupFile(configRoot, ['package.json']) - if (pkg && JSON.parse(pkg).type === 'module') { - isESM = true - } - } catch (e) {} - if (configFile) { // explicit config path is always resolved from cwd resolvedPath = path.resolve(configFile) - isTS = configFile.endsWith('.ts') - - if (configFile.endsWith('.mjs')) { - isESM = true - } } else { // implicit config file loaded from inline root (if present) // otherwise from cwd - const jsconfigFile = path.resolve(configRoot, 'vite.config.js') - if (fs.existsSync(jsconfigFile)) { - resolvedPath = jsconfigFile - } - - if (!resolvedPath) { - const mjsconfigFile = path.resolve(configRoot, 'vite.config.mjs') - if (fs.existsSync(mjsconfigFile)) { - resolvedPath = mjsconfigFile - isESM = true - } - } + for (const filename of DEFAULT_CONFIG_FILES) { + const filePath = path.resolve(configRoot, filename) + if (!fs.existsSync(filePath)) continue - if (!resolvedPath) { - const tsconfigFile = path.resolve(configRoot, 'vite.config.ts') - if (fs.existsSync(tsconfigFile)) { - resolvedPath = tsconfigFile - isTS = true - } - } - - if (!resolvedPath) { - const cjsConfigFile = path.resolve(configRoot, 'vite.config.cjs') - if (fs.existsSync(cjsConfigFile)) { - resolvedPath = cjsConfigFile - isESM = false - } + resolvedPath = filePath + break } } @@ -921,6 +891,19 @@ export async function loadConfigFromFile( return null } + let isESM = false + if (/\.m[jt]s$/.test(resolvedPath)) { + isESM = true + } else if (/\.c[jt]s$/.test(resolvedPath)) { + isESM = false + } else { + // check package.json for type: "module" and set `isESM` to true + try { + const pkg = lookupFile(configRoot, ['package.json']) + isESM = !!pkg && JSON.parse(pkg).type === 'module' + } catch (e) {} + } + try { let userConfig: UserConfigExport | undefined @@ -928,15 +911,16 @@ export async function loadConfigFromFile( const fileUrl = require('url').pathToFileURL(resolvedPath) const bundled = await bundleConfigFile(resolvedPath, true) dependencies = bundled.dependencies - if (isTS) { + + if (isTS(resolvedPath)) { // before we can register loaders without requiring users to run node // with --experimental-loader themselves, we have to do a hack here: // bundle the config file w/ ts transforms first, write it to disk, // load it with native Node ESM, then delete the file. - fs.writeFileSync(resolvedPath + '.js', bundled.code) - userConfig = (await dynamicImport(`${fileUrl}.js?t=${Date.now()}`)) + fs.writeFileSync(resolvedPath + '.mjs', bundled.code) + userConfig = (await dynamicImport(`${fileUrl}.mjs?t=${Date.now()}`)) .default - fs.unlinkSync(resolvedPath + '.js') + fs.unlinkSync(resolvedPath + '.mjs') debug(`TS + native esm config loaded in ${getTime()}`, fileUrl) } else { // using Function to avoid this from being compiled away by TS/Rollup @@ -1006,10 +990,10 @@ async function bundleConfigFile( { name: 'replace-import-meta', setup(build) { - build.onLoad({ filter: /\.[jt]s$/ }, async (args) => { + build.onLoad({ filter: /\.[cm]?[jt]s$/ }, async (args) => { const contents = await fs.promises.readFile(args.path, 'utf8') return { - loader: args.path.endsWith('.ts') ? 'ts' : 'js', + loader: isTS(args.path) ? 'ts' : 'js', contents: contents .replace( /\bimport\.meta\.url\b/g, diff --git a/packages/vite/src/node/constants.ts b/packages/vite/src/node/constants.ts index 9612cd8c96460d..ae32bf998b92c6 100644 --- a/packages/vite/src/node/constants.ts +++ b/packages/vite/src/node/constants.ts @@ -15,6 +15,15 @@ export const DEFAULT_EXTENSIONS = [ '.json' ] +export const DEFAULT_CONFIG_FILES = [ + 'vite.config.js', + 'vite.config.mjs', + 'vite.config.ts', + 'vite.config.cjs', + 'vite.config.mts', + 'vite.config.cts' +] + export const JS_TYPES_RE = /\.(?:j|t)sx?$|\.mjs$/ export const OPTIMIZABLE_ENTRY_RE = /\.(?:m?js|ts)$/ diff --git a/packages/vite/src/node/utils.ts b/packages/vite/src/node/utils.ts index 92d89d7adbe78f..efea40398e6c7f 100644 --- a/packages/vite/src/node/utils.ts +++ b/packages/vite/src/node/utils.ts @@ -842,3 +842,5 @@ function gracefulRemoveDir( export function emptyCssComments(raw: string) { return raw.replace(multilineCommentsRE, (s) => ' '.repeat(s.length)) } + +export const isTS = (filename: string): boolean => /\.[cm]?ts$/.test(filename)