From e29b6a051e89e23945e2acfdfca7057978929715 Mon Sep 17 00:00:00 2001 From: Evan You Date: Tue, 7 Mar 2023 20:18:40 +0800 Subject: [PATCH] feat(data-loader): pass watched files into load() --- __tests__/e2e/static-data/static.data.ts | 10 ++--- src/node/plugins/staticDataPlugin.ts | 50 +++++++++++++----------- 2 files changed, 31 insertions(+), 29 deletions(-) diff --git a/__tests__/e2e/static-data/static.data.ts b/__tests__/e2e/static-data/static.data.ts index a9ae896bd15d..e88b49b44472 100644 --- a/__tests__/e2e/static-data/static.data.ts +++ b/__tests__/e2e/static-data/static.data.ts @@ -1,21 +1,17 @@ import fs from 'fs' -import path from 'path' -import { fileURLToPath } from 'url' - -const dirname = path.dirname(fileURLToPath(import.meta.url)) type Data = Record[] export declare const data: Data export default { watch: ['./data/*'], - async load(): Promise { + async load(files: string[]): Promise { const foo = fs.readFileSync( - path.resolve(dirname, './data/foo.json'), + files.find((f) => f.endsWith('foo.json'))!, 'utf-8' ) const bar = fs.readFileSync( - path.resolve(dirname, './data/bar.json'), + files.find((f) => f.endsWith('bar.json'))!, 'utf-8' ) return [JSON.parse(foo), JSON.parse(bar)] diff --git a/src/node/plugins/staticDataPlugin.ts b/src/node/plugins/staticDataPlugin.ts index e8fa50d7d44e..4907d5e9472b 100644 --- a/src/node/plugins/staticDataPlugin.ts +++ b/src/node/plugins/staticDataPlugin.ts @@ -6,22 +6,18 @@ import { } from 'vite' import path, { dirname, resolve } from 'path' import { isMatch } from 'micromatch' +import glob from 'fast-glob' const loaderMatch = /\.data\.(j|t)s$/ let server: ViteDevServer interface LoaderModule { - watch: string[] | string | undefined - load: () => any + watch?: string[] | string + load: (watchedFiles: string[]) => any } -interface CachedLoaderModule { - pattern: string[] | undefined - loader: () => any -} - -const idToLoaderModulesMap: Record = +const idToLoaderModulesMap: Record = Object.create(null) const depToLoaderModuleIdMap: Record = Object.create(null) @@ -59,12 +55,12 @@ export const staticDataPlugin: Plugin = { } const base = dirname(id) - let pattern: string[] | undefined - let loader: () => any + let watch: LoaderModule['watch'] + let load: LoaderModule['load'] const existing = idToLoaderModulesMap[id] if (existing) { - ;({ pattern, loader } = existing) + ;({ watch, load } = existing) } else { // use vite's load config util as a away to load Node.js file with // TS & native ESM support @@ -78,26 +74,34 @@ export const staticDataPlugin: Plugin = { } const loaderModule = res?.config as LoaderModule - pattern = + watch = typeof loaderModule.watch === 'string' ? [loaderModule.watch] : loaderModule.watch - if (pattern) { - pattern = pattern.map((p) => { + if (watch) { + watch = watch.map((p) => { return p.startsWith('.') ? normalizePath(resolve(base, p)) : normalizePath(p) }) } - loader = loaderModule.load + load = loaderModule.load } // load the data - const data = await loader() + let watchedFiles + if (watch) { + watchedFiles = ( + await glob(watch, { + ignore: ['**/node_modules/**', '**/dist/**'] + }) + ).sort() + } + const data = await load(watchedFiles || []) // record loader module for HMR if (server) { - idToLoaderModulesMap[id] = { pattern, loader } + idToLoaderModulesMap[id] = { watch, load } } const result = `export const data = JSON.parse(${JSON.stringify( @@ -112,9 +116,11 @@ export const staticDataPlugin: Plugin = { transform(_code, id) { if (server && loaderMatch.test(id)) { // register this module as a glob importer - const { pattern } = idToLoaderModulesMap[id]! - if (pattern) { - ;(server as any)._importGlobMap.set(id, [pattern]) + const { watch } = idToLoaderModulesMap[id]! + if (watch) { + ;(server as any)._importGlobMap.set(id, [ + Array.isArray(watch) ? watch : [watch] + ]) } } return null @@ -132,8 +138,8 @@ export const staticDataPlugin: Plugin = { } for (const id in idToLoaderModulesMap) { - const { pattern } = idToLoaderModulesMap[id]! - if (pattern && isMatch(file, pattern)) { + const { watch } = idToLoaderModulesMap[id]! + if (watch && isMatch(file, watch)) { ctx.modules.push(server.moduleGraph.getModuleById(id)!) } }