import { build, InlineConfig, type PluginOption } from 'vite'; import { resolve } from 'path'; import { fileURLToPath } from 'url'; import jisonPlugin from './jisonPlugin.js'; import jsonSchemaPlugin from './jsonSchemaPlugin.js'; import { readFileSync } from 'fs'; import typescript from '@rollup/plugin-typescript'; import { visualizer } from 'rollup-plugin-visualizer'; import type { TemplateType } from 'rollup-plugin-visualizer/dist/plugin/template-types.js'; import istanbul from 'vite-plugin-istanbul'; const visualize = process.argv.includes('--visualize'); const watch = process.argv.includes('--watch'); const mermaidOnly = process.argv.includes('--mermaid'); const coverage = process.env.VITE_COVERAGE === 'true'; const __dirname = fileURLToPath(new URL('.', import.meta.url)); const sourcemap = false; type OutputOptions = Exclude< Exclude['rollupOptions'], undefined >['output']; const visualizerOptions = (packageName: string, core = false): PluginOption[] => { if (packageName !== 'mermaid' || !visualize) { return []; } return ['network', 'treemap', 'sunburst'].map( (chartType) => visualizer({ filename: `./stats/${chartType}${core ? '.core' : ''}.html`, template: chartType as TemplateType, gzipSize: true, brotliSize: true, }) as PluginOption ); }; const packageOptions = { mermaid: { name: 'mermaid', packageName: 'mermaid', file: 'mermaid.ts', }, 'mermaid-example-diagram': { name: 'mermaid-example-diagram', packageName: 'mermaid-example-diagram', file: 'detector.ts', }, 'mermaid-zenuml': { name: 'mermaid-zenuml', packageName: 'mermaid-zenuml', file: 'detector.ts', }, }; interface BuildOptions { minify: boolean | 'esbuild'; core?: boolean; watch?: boolean; entryName: keyof typeof packageOptions; } export const getBuildConfig = ({ minify, core, watch, entryName }: BuildOptions): InlineConfig => { const external: (string | RegExp)[] = ['require', 'fs', 'path']; console.log(entryName, packageOptions[entryName]); const { name, file, packageName } = packageOptions[entryName]; let output: OutputOptions = [ { name, format: 'esm', sourcemap, entryFileNames: `${name}.esm${minify ? '.min' : ''}.mjs`, }, { name, format: 'umd', sourcemap, entryFileNames: `${name}${minify ? '.min' : ''}.js`, }, ]; if (core) { const { dependencies } = JSON.parse( readFileSync(resolve(__dirname, `../packages/${packageName}/package.json`), 'utf-8') ); // Core build is used to generate file without bundled dependencies. // This is used by downstream projects to bundle dependencies themselves. // Ignore dependencies and any dependencies of dependencies // Adapted from the RegEx used by `rollup-plugin-node` external.push(new RegExp('^(?:' + Object.keys(dependencies).join('|') + ')(?:/.+)?$')); // This needs to be an array. Otherwise vite will build esm & umd with same name and overwrite esm with umd. output = [ { name, format: 'esm', sourcemap, entryFileNames: `${name}.core.mjs`, }, ]; } const config: InlineConfig = { configFile: false, build: { emptyOutDir: false, outDir: resolve(__dirname, `../packages/${packageName}/dist`), lib: { entry: resolve(__dirname, `../packages/${packageName}/src/${file}`), name, // the proper extensions will be added fileName: name, }, minify, rollupOptions: { external, output, }, }, define: { 'import.meta.vitest': 'undefined', }, resolve: { extensions: [], }, plugins: [ jisonPlugin(), jsonSchemaPlugin(), // handles `.schema.yaml` files // @ts-expect-error According to the type definitions, rollup plugins are incompatible with vite typescript({ compilerOptions: { declaration: false } }), istanbul({ exclude: ['node_modules', 'test/', '__mocks__'], extension: ['.js', '.ts'], requireEnv: true, forceBuildInstrument: coverage, }), ...visualizerOptions(packageName, core), ], }; if (watch && config.build) { config.build.watch = { include: ['packages/mermaid-example-diagram/src/**', 'packages/mermaid/src/**'], }; } return config; }; const buildPackage = async (entryName: keyof typeof packageOptions) => { await build(getBuildConfig({ minify: false, entryName })); await build(getBuildConfig({ minify: 'esbuild', entryName })); await build(getBuildConfig({ minify: false, core: true, entryName })); }; const main = async () => { const packageNames = Object.keys(packageOptions) as (keyof typeof packageOptions)[]; for (const pkg of packageNames.filter((pkg) => !mermaidOnly || pkg === 'mermaid')) { await buildPackage(pkg); } }; if (watch) { build(getBuildConfig({ minify: false, watch, core: false, entryName: 'mermaid' })); if (!mermaidOnly) { build(getBuildConfig({ minify: false, watch, entryName: 'mermaid-example-diagram' })); build(getBuildConfig({ minify: false, watch, entryName: 'mermaid-zenuml' })); } } else if (visualize) { await build(getBuildConfig({ minify: false, core: true, entryName: 'mermaid' })); await build(getBuildConfig({ minify: false, core: false, entryName: 'mermaid' })); } else { void main(); }