2023-08-01 16:07:52 +00:00
|
|
|
import fs from 'fs-extra';
|
2023-09-03 02:59:33 +00:00
|
|
|
import chalk from 'chalk';
|
|
|
|
import ncc from '@vercel/ncc';
|
2023-08-01 16:07:52 +00:00
|
|
|
import path from 'path';
|
2023-09-03 02:59:33 +00:00
|
|
|
import react from '@vitejs/plugin-react';
|
2023-08-01 16:07:52 +00:00
|
|
|
import { build as tsupBuild } from 'tsup';
|
|
|
|
import { build as viteBuild } from 'vite';
|
2023-09-03 02:59:33 +00:00
|
|
|
import fg from 'fast-glob';
|
2023-08-01 16:07:52 +00:00
|
|
|
import cssInjectedByJsPlugin from 'vite-plugin-css-injected-by-js';
|
2023-09-03 02:59:33 +00:00
|
|
|
import { transformAsync } from '@babel/core';
|
|
|
|
|
2023-08-01 16:07:52 +00:00
|
|
|
import {
|
|
|
|
buildCheck,
|
2023-09-03 02:59:33 +00:00
|
|
|
checkFileSize,
|
|
|
|
checkRequire,
|
2023-08-01 16:07:52 +00:00
|
|
|
getExcludePackages,
|
|
|
|
getIncludePackages,
|
2023-09-03 02:59:33 +00:00
|
|
|
getPackagesFromFiles,
|
2023-08-01 16:07:52 +00:00
|
|
|
getSourcePackages,
|
|
|
|
} from './utils/buildPluginUtils';
|
|
|
|
import { getDepsConfig } from './utils/getDepsConfig';
|
2023-09-03 02:59:33 +00:00
|
|
|
import { EsbuildSupportExts, globExcludeFiles } from './constant';
|
|
|
|
import { PkgLog, getPackageJson } from './utils';
|
2023-08-01 16:07:52 +00:00
|
|
|
|
2023-09-03 02:59:33 +00:00
|
|
|
const serverGlobalFiles: string[] = ['src/**', '!src/client/**', ...globExcludeFiles];
|
|
|
|
const clientGlobalFiles: string[] = ['src/**', '!src/server/**', ...globExcludeFiles];
|
|
|
|
const dynamicImportRegexp = /import\((["'])(.*?)\1\)/g;
|
2023-08-01 16:07:52 +00:00
|
|
|
|
|
|
|
const external = [
|
|
|
|
// nocobase
|
|
|
|
'@nocobase/acl',
|
|
|
|
'@nocobase/actions',
|
|
|
|
'@nocobase/auth',
|
|
|
|
'@nocobase/cache',
|
|
|
|
'@nocobase/client',
|
|
|
|
'@nocobase/database',
|
|
|
|
'@nocobase/evaluators',
|
|
|
|
'@nocobase/logger',
|
|
|
|
'@nocobase/resourcer',
|
|
|
|
'@nocobase/sdk',
|
|
|
|
'@nocobase/server',
|
|
|
|
'@nocobase/test',
|
|
|
|
'@nocobase/utils',
|
|
|
|
|
|
|
|
// @nocobase/auth
|
|
|
|
'jsonwebtoken',
|
|
|
|
|
|
|
|
// @nocobase/cache
|
|
|
|
'cache-manager',
|
|
|
|
|
|
|
|
// @nocobase/database
|
|
|
|
'sequelize',
|
|
|
|
'umzug',
|
|
|
|
'async-mutex',
|
|
|
|
|
|
|
|
// @nocobase/evaluators
|
|
|
|
'@formulajs/formulajs',
|
|
|
|
'mathjs',
|
|
|
|
|
|
|
|
// @nocobase/logger
|
|
|
|
'winston',
|
|
|
|
'winston-daily-rotate-file',
|
|
|
|
|
|
|
|
// koa
|
|
|
|
'koa',
|
|
|
|
'@koa/cors',
|
|
|
|
'@koa/router',
|
|
|
|
'multer',
|
|
|
|
'@koa/multer',
|
|
|
|
'koa-bodyparser',
|
|
|
|
'koa-static',
|
|
|
|
'koa-send',
|
|
|
|
|
|
|
|
// react
|
|
|
|
'react',
|
|
|
|
'react-dom',
|
|
|
|
'react/jsx-runtime',
|
|
|
|
|
|
|
|
// react-router
|
|
|
|
'react-router',
|
|
|
|
'react-router-dom',
|
|
|
|
|
|
|
|
// antd
|
|
|
|
'antd',
|
|
|
|
'antd-style',
|
|
|
|
'@ant-design/icons',
|
|
|
|
'@ant-design/cssinjs',
|
|
|
|
|
|
|
|
// i18next
|
|
|
|
'i18next',
|
|
|
|
'react-i18next',
|
|
|
|
|
|
|
|
// dnd-kit 相关
|
|
|
|
'@dnd-kit/accessibility',
|
|
|
|
'@dnd-kit/core',
|
|
|
|
'@dnd-kit/modifiers',
|
|
|
|
'@dnd-kit/sortable',
|
|
|
|
'@dnd-kit/utilities',
|
|
|
|
|
|
|
|
// formily 相关
|
|
|
|
'@formily/antd-v5',
|
|
|
|
'@formily/core',
|
|
|
|
'@formily/react',
|
|
|
|
'@formily/json-schema',
|
|
|
|
'@formily/path',
|
|
|
|
'@formily/validator',
|
|
|
|
'@formily/shared',
|
|
|
|
'@formily/reactive',
|
|
|
|
'@formily/reactive-react',
|
|
|
|
|
|
|
|
// utils
|
|
|
|
'dayjs',
|
|
|
|
'mysql2',
|
|
|
|
'pg',
|
|
|
|
'pg-hstore',
|
|
|
|
'sqlite3',
|
|
|
|
'supertest',
|
|
|
|
'axios',
|
|
|
|
'@emotion/css',
|
|
|
|
'ahooks',
|
|
|
|
'lodash',
|
|
|
|
'china-division',
|
|
|
|
];
|
|
|
|
const pluginPrefix = (
|
|
|
|
process.env.PLUGIN_PACKAGE_PREFIX || '@nocobase/plugin-,@nocobase/preset-,@nocobase/plugin-pro-'
|
|
|
|
).split(',');
|
|
|
|
|
|
|
|
const target_dir = 'dist';
|
|
|
|
|
2023-09-03 02:59:33 +00:00
|
|
|
export function deleteJsFiles(cwd: string, log: PkgLog) {
|
2023-08-01 16:07:52 +00:00
|
|
|
log('delete babel js files');
|
|
|
|
const jsFiles = fg.globSync(['**/*', '!**/*.d.ts', '!node_modules'], {
|
|
|
|
cwd: path.join(cwd, target_dir),
|
|
|
|
absolute: true,
|
|
|
|
});
|
|
|
|
jsFiles.forEach((item) => {
|
|
|
|
fs.unlinkSync(item);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2023-09-03 02:59:33 +00:00
|
|
|
export async function buildServerDeps(cwd: string, serverFiles: string[], log: PkgLog) {
|
|
|
|
log('build plugin server dependencies');
|
2023-08-01 16:07:52 +00:00
|
|
|
const outDir = path.join(cwd, target_dir, 'node_modules');
|
2023-09-03 02:59:33 +00:00
|
|
|
const serverFileSource = serverFiles.map((item) => fs.readFileSync(item, 'utf-8'));
|
|
|
|
const sourcePackages = getSourcePackages(serverFileSource);
|
2023-08-01 16:07:52 +00:00
|
|
|
const includePackages = getIncludePackages(sourcePackages, external, pluginPrefix);
|
|
|
|
const excludePackages = getExcludePackages(sourcePackages, external, pluginPrefix);
|
|
|
|
|
|
|
|
let tips = [];
|
|
|
|
if (includePackages.length) {
|
|
|
|
tips.push(
|
2023-09-03 02:59:33 +00:00
|
|
|
`These packages ${chalk.yellow(includePackages.join(', '))} will be ${chalk.italic(
|
2023-08-01 16:07:52 +00:00
|
|
|
'bundled',
|
|
|
|
)} to dist/node_modules.`,
|
|
|
|
);
|
|
|
|
}
|
|
|
|
if (excludePackages.length) {
|
2023-09-03 02:59:33 +00:00
|
|
|
tips.push(`These packages ${chalk.yellow(excludePackages.join(', '))} will be ${chalk.italic('exclude')}.`);
|
2023-08-01 16:07:52 +00:00
|
|
|
}
|
|
|
|
tips.push(`For more information, please refer to: ${chalk.blue('https://docs.nocobase.com/development/deps')}.`);
|
|
|
|
log(tips.join(' '));
|
|
|
|
|
|
|
|
if (!includePackages.length) return;
|
|
|
|
|
|
|
|
const deps = getDepsConfig(cwd, outDir, includePackages, external);
|
|
|
|
|
|
|
|
// bundle deps
|
|
|
|
for (const dep of Object.keys(deps)) {
|
|
|
|
const { outputDir, mainFile, pkg, nccConfig, depDir } = deps[dep];
|
|
|
|
const outputPackageJson = path.join(outputDir, 'package.json');
|
|
|
|
|
|
|
|
// cache check
|
|
|
|
if (fs.existsSync(outputPackageJson)) {
|
|
|
|
const outputPackage = require(outputPackageJson);
|
|
|
|
if (outputPackage.version === pkg.version) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// copy package
|
|
|
|
await fs.copy(depDir, outputDir, { errorOnExist: false });
|
|
|
|
|
|
|
|
// delete files
|
|
|
|
const deleteFiles = fg.sync(
|
|
|
|
[
|
|
|
|
'./**/*.map',
|
|
|
|
'./**/*.js.map',
|
|
|
|
'./**/*.md',
|
|
|
|
'./**/*.mjs',
|
|
|
|
'./**/*.png',
|
|
|
|
'./**/*.jpg',
|
|
|
|
'./**/*.jpeg',
|
|
|
|
'./**/*.gif',
|
|
|
|
'./**/*/.bin',
|
|
|
|
'./**/*/bin',
|
|
|
|
'./**/*/LICENSE',
|
|
|
|
'./**/*/tsconfig.json',
|
|
|
|
],
|
|
|
|
{ cwd: outputDir, absolute: true },
|
|
|
|
);
|
|
|
|
|
|
|
|
deleteFiles.forEach((file) => {
|
|
|
|
fs.unlinkSync(file);
|
|
|
|
});
|
|
|
|
|
|
|
|
await ncc(dep, nccConfig).then(
|
|
|
|
({ code, assets }: { code: string; assets: Record<string, { source: string; permissions: number }> }) => {
|
|
|
|
// emit dist file
|
|
|
|
fs.writeFileSync(mainFile, code, 'utf-8');
|
|
|
|
|
|
|
|
// emit assets
|
|
|
|
Object.entries(assets).forEach(([name, item]) => {
|
|
|
|
fs.writeFileSync(path.join(outputDir, name), item.source, {
|
|
|
|
encoding: 'utf-8',
|
|
|
|
mode: item.permissions,
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
// emit package.json
|
|
|
|
fs.writeFileSync(
|
|
|
|
outputPackageJson,
|
|
|
|
JSON.stringify({
|
|
|
|
...pkg,
|
|
|
|
_lastModified: new Date().toISOString(),
|
|
|
|
}),
|
|
|
|
'utf-8',
|
|
|
|
);
|
|
|
|
},
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-09-03 02:59:33 +00:00
|
|
|
export async function buildPluginServer(cwd: string, sourcemap: boolean, log: PkgLog) {
|
|
|
|
log('build plugin server source');
|
2023-08-01 16:07:52 +00:00
|
|
|
const packageJson = getPackageJson(cwd);
|
|
|
|
const serverFiles = fg.globSync(serverGlobalFiles, { cwd, absolute: true });
|
|
|
|
buildCheck({ cwd, packageJson, entry: 'server', files: serverFiles, log });
|
2023-09-03 02:59:33 +00:00
|
|
|
const otherExts = Array.from(new Set(serverFiles.map((item) => path.extname(item)).filter((item) => !EsbuildSupportExts.includes(item))));
|
|
|
|
if (otherExts.length) {
|
|
|
|
log('%s will not be processed, only be copied to the dist directory.', chalk.yellow(otherExts.join(',')));
|
|
|
|
}
|
2023-08-01 16:07:52 +00:00
|
|
|
|
|
|
|
await tsupBuild({
|
|
|
|
entry: serverFiles,
|
|
|
|
splitting: false,
|
|
|
|
clean: false,
|
|
|
|
bundle: false,
|
|
|
|
silent: true,
|
2023-09-03 02:59:33 +00:00
|
|
|
treeshake: false,
|
2023-08-01 16:07:52 +00:00
|
|
|
target: 'node16',
|
2023-09-03 02:59:33 +00:00
|
|
|
sourcemap,
|
2023-08-01 16:07:52 +00:00
|
|
|
outDir: path.join(cwd, target_dir),
|
|
|
|
format: 'cjs',
|
|
|
|
skipNodeModulesBundle: true,
|
2023-09-03 02:59:33 +00:00
|
|
|
loader: {
|
|
|
|
...otherExts.reduce((prev, cur) => ({ ...prev, [cur]: 'copy' }), {}),
|
|
|
|
'.json': 'copy',
|
|
|
|
},
|
2023-08-01 16:07:52 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
await buildServerDeps(cwd, serverFiles, log);
|
|
|
|
}
|
|
|
|
|
2023-09-03 02:59:33 +00:00
|
|
|
export function transformClientFilesToAmd(outDir: string, outputFileName: string, packageName: string, log: PkgLog) {
|
|
|
|
const files = fs.readdirSync(outDir);
|
|
|
|
|
|
|
|
return Promise.all(files.map(async file => {
|
|
|
|
const filePath = path.join(outDir, file);
|
|
|
|
|
|
|
|
// 只编译 JavaScript 文件
|
|
|
|
if (path.extname(filePath) === '.mjs' || path.extname(filePath) === '.js') {
|
|
|
|
let fileContent = fs.readFileSync(filePath, 'utf-8');
|
|
|
|
|
|
|
|
// import('./dayjs.mjs') => import(window.staticBaseUrl + '/@nocobase/plugin-acl/dayjs.mjs?noExt')
|
|
|
|
if (file === outputFileName) {
|
|
|
|
fileContent = fileContent.replace(dynamicImportRegexp, (match, _1, dynamicImportPath) => {
|
|
|
|
let absolutePath = path.join(outDir, dynamicImportPath).replace(outDir, '');
|
|
|
|
if (absolutePath.startsWith(path.sep)) {
|
|
|
|
absolutePath = absolutePath.slice(1);
|
|
|
|
}
|
|
|
|
return `import(window.staticBaseUrl + '/${packageName}/${absolutePath}?noExt')`;
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
const { code } = await transformAsync(fileContent, {
|
|
|
|
presets: [['@babel/preset-env', {
|
|
|
|
modules: 'amd',
|
|
|
|
targets: {
|
|
|
|
'edge': 88,
|
|
|
|
'firefox': 78,
|
|
|
|
'chrome': 87,
|
|
|
|
'safari': 14,
|
|
|
|
}
|
|
|
|
}]],
|
|
|
|
plugins: ['@babel/plugin-transform-modules-amd'],
|
|
|
|
});
|
|
|
|
fs.writeFileSync(filePath, code, 'utf-8');
|
|
|
|
}
|
|
|
|
}));
|
|
|
|
}
|
2023-08-01 16:07:52 +00:00
|
|
|
|
2023-09-03 02:59:33 +00:00
|
|
|
export async function buildPluginClient(cwd: string, sourcemap: boolean, log: PkgLog) {
|
|
|
|
log('build plugin client');
|
2023-08-01 16:07:52 +00:00
|
|
|
const packageJson = getPackageJson(cwd);
|
|
|
|
const clientFiles = fg.globSync(clientGlobalFiles, { cwd, absolute: true });
|
2023-09-03 02:59:33 +00:00
|
|
|
const clientFileSource = clientFiles.map((item) => fs.readFileSync(item, 'utf-8'));
|
|
|
|
const sourcePackages = getPackagesFromFiles(clientFileSource);
|
2023-08-01 16:07:52 +00:00
|
|
|
const excludePackages = getExcludePackages(sourcePackages, external, pluginPrefix);
|
|
|
|
|
2023-09-03 02:59:33 +00:00
|
|
|
checkRequire(clientFiles, log);
|
2023-08-01 16:07:52 +00:00
|
|
|
buildCheck({ cwd, packageJson, entry: 'client', files: clientFiles, log });
|
2023-09-03 02:59:33 +00:00
|
|
|
const hasDynamicImport = false;
|
|
|
|
// const hasDynamicImport = clientFileSource.some((source) => {
|
|
|
|
// return source.match(dynamicImportRegexp);
|
|
|
|
// });
|
2023-08-01 16:07:52 +00:00
|
|
|
const outDir = path.join(cwd, target_dir, 'client');
|
|
|
|
|
|
|
|
const globals = excludePackages.reduce<Record<string, string>>((prev, curr) => {
|
|
|
|
if (curr.startsWith('@nocobase')) {
|
|
|
|
prev[`${curr}/client`] = curr;
|
|
|
|
}
|
|
|
|
prev[curr] = curr;
|
|
|
|
return prev;
|
|
|
|
}, {});
|
|
|
|
|
2023-09-03 02:59:33 +00:00
|
|
|
const entry = fg.globSync('src/client/index.{ts,tsx,js,jsx}', { absolute: true, cwd });
|
2023-08-01 16:07:52 +00:00
|
|
|
const outputFileName = 'index.js';
|
2023-09-03 02:59:33 +00:00
|
|
|
await viteBuild({
|
2023-08-01 16:07:52 +00:00
|
|
|
mode: 'production',
|
|
|
|
define: {
|
|
|
|
'process.env.NODE_ENV': JSON.stringify('production'),
|
|
|
|
},
|
|
|
|
logLevel: 'warn',
|
|
|
|
build: {
|
2023-09-03 02:59:33 +00:00
|
|
|
minify: true,
|
2023-08-01 16:07:52 +00:00
|
|
|
outDir,
|
|
|
|
cssCodeSplit: false,
|
2023-09-03 02:59:33 +00:00
|
|
|
emptyOutDir: true,
|
|
|
|
sourcemap,
|
2023-08-01 16:07:52 +00:00
|
|
|
lib: {
|
|
|
|
entry,
|
2023-09-03 02:59:33 +00:00
|
|
|
formats: [hasDynamicImport ? 'es' : 'umd'],
|
2023-08-01 16:07:52 +00:00
|
|
|
name: packageJson.name,
|
|
|
|
fileName: () => outputFileName,
|
|
|
|
},
|
2023-09-03 02:59:33 +00:00
|
|
|
target: ['es2015', 'edge88', 'firefox78', 'chrome87', 'safari14'],
|
2023-08-01 16:07:52 +00:00
|
|
|
rollupOptions: {
|
|
|
|
cache: true,
|
2023-09-03 02:59:33 +00:00
|
|
|
external: [...Object.keys(globals), 'react', 'react/jsx-runtime'],
|
2023-08-01 16:07:52 +00:00
|
|
|
output: {
|
|
|
|
exports: 'named',
|
2023-09-03 02:59:33 +00:00
|
|
|
globals: {
|
|
|
|
react: 'React',
|
|
|
|
'react/jsx-runtime': 'jsxRuntime',
|
|
|
|
...globals,
|
|
|
|
},
|
2023-08-01 16:07:52 +00:00
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
plugins: [
|
|
|
|
react(),
|
|
|
|
cssInjectedByJsPlugin({ styleId: packageJson.name }),
|
|
|
|
],
|
|
|
|
});
|
2023-09-03 02:59:33 +00:00
|
|
|
|
|
|
|
checkFileSize(outDir, log);
|
|
|
|
|
|
|
|
// if (hasDynamicImport) {
|
|
|
|
// await transformClientFilesToAmd(outDir, outputFileName, packageJson.name, log);
|
|
|
|
// }
|
|
|
|
}
|
|
|
|
|
|
|
|
export async function buildPlugin(cwd: string, sourcemap: boolean, log: PkgLog) {
|
|
|
|
await buildPluginClient(cwd, sourcemap, log);
|
|
|
|
await buildPluginServer(cwd, sourcemap, log);
|
|
|
|
const buildFile = path.join(cwd, 'build.js');
|
|
|
|
if (fs.existsSync(buildFile)) {
|
|
|
|
log('build others');
|
|
|
|
try {
|
|
|
|
await require(buildFile).run(log);
|
|
|
|
} catch (error) {
|
|
|
|
console.error(error);
|
|
|
|
}
|
|
|
|
}
|
2023-08-01 16:07:52 +00:00
|
|
|
}
|