mirror of
https://github.com/Tencent/tmagic-editor.git
synced 2025-04-06 03:57:56 +08:00
feat(cli): 新增tamgic-cli,用于runtime 依赖生成
This commit is contained in:
parent
f1a8097e06
commit
f18e7b275d
3
.gitignore
vendored
3
.gitignore
vendored
@ -2,6 +2,9 @@
|
|||||||
node_modules
|
node_modules
|
||||||
dist
|
dist
|
||||||
|
|
||||||
|
.temp
|
||||||
|
.cache
|
||||||
|
|
||||||
# local env files
|
# local env files
|
||||||
.env.local
|
.env.local
|
||||||
.env.*.local
|
.env.*.local
|
||||||
|
24
packages/cli/.gitignore
vendored
Normal file
24
packages/cli/.gitignore
vendored
Normal file
@ -0,0 +1,24 @@
|
|||||||
|
.DS_Store
|
||||||
|
node_modules
|
||||||
|
lib
|
||||||
|
|
||||||
|
# local env files
|
||||||
|
.env.local
|
||||||
|
.env.*.local
|
||||||
|
|
||||||
|
# Log files
|
||||||
|
npm-debug.log*
|
||||||
|
yarn-debug.log*
|
||||||
|
yarn-error.log*
|
||||||
|
pnpm-debug.log*
|
||||||
|
|
||||||
|
# Editor directories and files
|
||||||
|
.idea
|
||||||
|
.vscode
|
||||||
|
*.suo
|
||||||
|
*.ntvs*
|
||||||
|
*.njsproj
|
||||||
|
*.sln
|
||||||
|
*.sw?
|
||||||
|
|
||||||
|
coverage
|
10
packages/cli/bin/tmagic.js
Executable file
10
packages/cli/bin/tmagic.js
Executable file
@ -0,0 +1,10 @@
|
|||||||
|
#!/usr/bin/env node
|
||||||
|
|
||||||
|
const { cli } = require('../lib');
|
||||||
|
|
||||||
|
cli({
|
||||||
|
source: process.cwd(),
|
||||||
|
packages: {},
|
||||||
|
componentFileAffix: '',
|
||||||
|
cleanTemp: true,
|
||||||
|
});
|
37
packages/cli/package.json
Normal file
37
packages/cli/package.json
Normal file
@ -0,0 +1,37 @@
|
|||||||
|
{
|
||||||
|
"version": "1.1.0-beta.5",
|
||||||
|
"name": "@tmagic/cli",
|
||||||
|
"main": "lib/index.js",
|
||||||
|
"types": "lib/index.d.ts",
|
||||||
|
"license": "Apache-2.0",
|
||||||
|
"scripts": {
|
||||||
|
"build": "tsc -b tsconfig.build.json",
|
||||||
|
"clean": "rimraf lib *.tsbuildinfo"
|
||||||
|
},
|
||||||
|
"bin": {
|
||||||
|
"tmagic": "bin/tmagic.js"
|
||||||
|
},
|
||||||
|
"files": [
|
||||||
|
"bin",
|
||||||
|
"lib"
|
||||||
|
],
|
||||||
|
"engines": {
|
||||||
|
"node": ">=14"
|
||||||
|
},
|
||||||
|
"repository": {
|
||||||
|
"type": "git",
|
||||||
|
"url": "https://github.com/Tencent/tmagic-editor.git"
|
||||||
|
},
|
||||||
|
"dependencies": {
|
||||||
|
"@vuepress/cli": "^2.0.0-beta.49",
|
||||||
|
"cac": "^6.7.12",
|
||||||
|
"chalk": "^4.1.0",
|
||||||
|
"chokidar": "^3.5.3",
|
||||||
|
"fs-extra": "^10.1.0",
|
||||||
|
"recast": "^0.21.1",
|
||||||
|
"tslib": "^2.4.0"
|
||||||
|
},
|
||||||
|
"devDependencies": {
|
||||||
|
"@types/fs-extra": "^9.0.13"
|
||||||
|
}
|
||||||
|
}
|
40
packages/cli/src/Core.ts
Normal file
40
packages/cli/src/Core.ts
Normal file
@ -0,0 +1,40 @@
|
|||||||
|
import path from 'path';
|
||||||
|
|
||||||
|
import fs from 'fs-extra';
|
||||||
|
|
||||||
|
import { UserConfig } from './types';
|
||||||
|
import { prepareEntryFile, resolveAppPackages } from './utils';
|
||||||
|
|
||||||
|
export default class Core {
|
||||||
|
public version = require('../package.json').version;
|
||||||
|
|
||||||
|
public options: UserConfig;
|
||||||
|
|
||||||
|
public moduleMainFilePath = {
|
||||||
|
componentMap: {},
|
||||||
|
pluginMap: {},
|
||||||
|
configMap: {},
|
||||||
|
valueMap: {},
|
||||||
|
eventMap: {},
|
||||||
|
};
|
||||||
|
|
||||||
|
public dir = {
|
||||||
|
temp: () => path.resolve(this.options.source, 'src/.tmagic'),
|
||||||
|
};
|
||||||
|
|
||||||
|
constructor(options: UserConfig) {
|
||||||
|
this.options = options;
|
||||||
|
}
|
||||||
|
|
||||||
|
public async writeTemp(file: string, content: string) {
|
||||||
|
await fs.outputFile(path.resolve(this.dir.temp(), file), content);
|
||||||
|
}
|
||||||
|
|
||||||
|
public async init() {
|
||||||
|
this.moduleMainFilePath = resolveAppPackages(this);
|
||||||
|
}
|
||||||
|
|
||||||
|
public async prepare() {
|
||||||
|
await prepareEntryFile(this);
|
||||||
|
}
|
||||||
|
}
|
44
packages/cli/src/cli.ts
Normal file
44
packages/cli/src/cli.ts
Normal file
@ -0,0 +1,44 @@
|
|||||||
|
import { allowTs, info } from '@vuepress/cli';
|
||||||
|
import { cac } from 'cac';
|
||||||
|
import chalk from 'chalk';
|
||||||
|
|
||||||
|
import { scripts } from './commands';
|
||||||
|
import { UserConfig } from './types';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Wrap raw command to catch errors and exit process
|
||||||
|
*/
|
||||||
|
const wrapCommand = (cmd: (...args: any[]) => Promise<void>): typeof cmd => {
|
||||||
|
const wrappedCommand: typeof cmd = (...args) =>
|
||||||
|
cmd(...args).catch((err) => {
|
||||||
|
console.error(chalk.red(err.stack));
|
||||||
|
process.exit(1);
|
||||||
|
});
|
||||||
|
return wrappedCommand;
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Vuepress cli
|
||||||
|
*/
|
||||||
|
export const cli = (defaultAppConfig: UserConfig): void => {
|
||||||
|
// allow ts files globally
|
||||||
|
allowTs();
|
||||||
|
|
||||||
|
// create cac instance
|
||||||
|
const program = cac('tmagic');
|
||||||
|
|
||||||
|
// display core version and cli version
|
||||||
|
const versionCli = require('../package.json').version;
|
||||||
|
program.version(`tmagic/cli@${versionCli}`);
|
||||||
|
|
||||||
|
// display help message
|
||||||
|
program.help();
|
||||||
|
|
||||||
|
// register `dev` command
|
||||||
|
program.command('entry', 'Start development server').action(wrapCommand(scripts(defaultAppConfig)));
|
||||||
|
|
||||||
|
// register `info` command
|
||||||
|
program.command('info', 'Display environment information').action(wrapCommand(info));
|
||||||
|
|
||||||
|
program.parse(process.argv);
|
||||||
|
};
|
48
packages/cli/src/commands/index.ts
Normal file
48
packages/cli/src/commands/index.ts
Normal file
@ -0,0 +1,48 @@
|
|||||||
|
import path from 'path';
|
||||||
|
|
||||||
|
import { loadUserConfig } from '@vuepress/cli';
|
||||||
|
import fs from 'fs-extra';
|
||||||
|
|
||||||
|
import App from '../Core';
|
||||||
|
import { UserConfig } from '../types';
|
||||||
|
|
||||||
|
export const scripts = (defaultAppConfig: UserConfig) => {
|
||||||
|
const entry = async (): Promise<void> => {
|
||||||
|
if (process.env.NODE_ENV === undefined) {
|
||||||
|
process.env.NODE_ENV = 'development';
|
||||||
|
}
|
||||||
|
|
||||||
|
// resolve user config file
|
||||||
|
const userConfigPath = [
|
||||||
|
path.resolve(process.cwd(), 'tmagic.config.ts'),
|
||||||
|
path.resolve(process.cwd(), 'tmagic.config.js'),
|
||||||
|
path.resolve(process.cwd(), 'tmagic.config.cjs'),
|
||||||
|
].find((item) => fs.pathExistsSync(item));
|
||||||
|
|
||||||
|
const userConfig = await loadUserConfig(userConfigPath);
|
||||||
|
|
||||||
|
// resolve the final app config to use
|
||||||
|
const appConfig = {
|
||||||
|
...defaultAppConfig,
|
||||||
|
...userConfig,
|
||||||
|
};
|
||||||
|
|
||||||
|
if (appConfig === null) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// create vuepress app
|
||||||
|
const app = new App(appConfig);
|
||||||
|
|
||||||
|
// clean temp and cache
|
||||||
|
if (appConfig.cleanTemp === true) {
|
||||||
|
await fs.remove(app.dir.temp());
|
||||||
|
}
|
||||||
|
|
||||||
|
// initialize and prepare
|
||||||
|
await app.init();
|
||||||
|
await app.prepare();
|
||||||
|
};
|
||||||
|
|
||||||
|
return entry;
|
||||||
|
};
|
3
packages/cli/src/index.ts
Normal file
3
packages/cli/src/index.ts
Normal file
@ -0,0 +1,3 @@
|
|||||||
|
export * from './cli';
|
||||||
|
export * from './utils';
|
||||||
|
export * from './types';
|
40
packages/cli/src/types.ts
Normal file
40
packages/cli/src/types.ts
Normal file
@ -0,0 +1,40 @@
|
|||||||
|
export enum EntryType {
|
||||||
|
CONFIG = 'config',
|
||||||
|
VALUE = 'value',
|
||||||
|
COMPONENT = 'component',
|
||||||
|
EVENT = 'event',
|
||||||
|
PLUGIN = 'plugin',
|
||||||
|
}
|
||||||
|
|
||||||
|
export enum PackageType {
|
||||||
|
COMPONENT = '1',
|
||||||
|
PLUGIN = '2',
|
||||||
|
COMPONENT_PACKAGE = '3',
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface Entry {
|
||||||
|
[EntryType.CONFIG]?: string;
|
||||||
|
[EntryType.VALUE]?: string;
|
||||||
|
[EntryType.COMPONENT]?: string;
|
||||||
|
[EntryType.EVENT]?: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface OptionEntry {
|
||||||
|
type: string;
|
||||||
|
entry: Entry;
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface EntryFile {
|
||||||
|
entries: OptionEntry[];
|
||||||
|
entryFile: string;
|
||||||
|
type: EntryType;
|
||||||
|
componentFileAffix: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface UserConfig {
|
||||||
|
source: string;
|
||||||
|
scripts: Record<string, string>;
|
||||||
|
packages: Record<string, any>;
|
||||||
|
componentFileAffix: string;
|
||||||
|
cleanTemp: boolean;
|
||||||
|
}
|
3
packages/cli/src/utils/defineUserConfig.ts
Normal file
3
packages/cli/src/utils/defineUserConfig.ts
Normal file
@ -0,0 +1,3 @@
|
|||||||
|
import { UserConfig } from '../types';
|
||||||
|
|
||||||
|
export const defineConfig = (config: Partial<UserConfig>): Partial<UserConfig> => config;
|
3
packages/cli/src/utils/index.ts
Normal file
3
packages/cli/src/utils/index.ts
Normal file
@ -0,0 +1,3 @@
|
|||||||
|
export * from './defineUserConfig';
|
||||||
|
export * from './prepareEntryFile';
|
||||||
|
export * from './resolveAppPackages';
|
53
packages/cli/src/utils/prepareEntryFile.ts
Normal file
53
packages/cli/src/utils/prepareEntryFile.ts
Normal file
@ -0,0 +1,53 @@
|
|||||||
|
import * as recast from 'recast';
|
||||||
|
|
||||||
|
import type App from '../Core';
|
||||||
|
import { EntryType } from '../types';
|
||||||
|
|
||||||
|
export const prepareEntryFile = (app: App) => {
|
||||||
|
const { componentMap = {}, pluginMap = {}, configMap = {}, valueMap = {}, eventMap = {} } = app.moduleMainFilePath;
|
||||||
|
const { componentFileAffix } = app.options;
|
||||||
|
|
||||||
|
app.writeTemp('comp-entry.ts', generateContent(EntryType.COMPONENT, componentMap, componentFileAffix));
|
||||||
|
app.writeTemp('plugin-entry.ts', generateContent(EntryType.PLUGIN, pluginMap));
|
||||||
|
app.writeTemp('config-entry.ts', generateContent(EntryType.CONFIG, configMap));
|
||||||
|
app.writeTemp('value-entry.ts', generateContent(EntryType.VALUE, valueMap));
|
||||||
|
app.writeTemp('event-entry.ts', generateContent(EntryType.EVENT, eventMap));
|
||||||
|
};
|
||||||
|
|
||||||
|
const generateContent = (type: EntryType, map: Record<string, string>, componentFileAffix = '') => {
|
||||||
|
const list: string[] = [];
|
||||||
|
const importDeclarations: string[] = [];
|
||||||
|
|
||||||
|
Object.entries(map).forEach(([key, packagePath]) => {
|
||||||
|
const name = makeCamelCase(key);
|
||||||
|
importDeclarations.push(
|
||||||
|
`import ${name} from '${packagePath}${packagePath.endsWith(componentFileAffix) ? '' : componentFileAffix}'`,
|
||||||
|
);
|
||||||
|
list.push(`'${key}': ${name}`);
|
||||||
|
});
|
||||||
|
|
||||||
|
const exportToken = `${type}s`;
|
||||||
|
const capitalToken = exportToken.charAt(0).toUpperCase() + exportToken.slice(1);
|
||||||
|
|
||||||
|
return prettyCode(`${importDeclarations.join(';')}
|
||||||
|
const ${exportToken}: Record<string, any> = {
|
||||||
|
${list.join(',')}
|
||||||
|
}
|
||||||
|
window.magicPreset${capitalToken} = ${exportToken};
|
||||||
|
export default ${exportToken};
|
||||||
|
`);
|
||||||
|
};
|
||||||
|
|
||||||
|
const prettyCode = (code: string) =>
|
||||||
|
recast.prettyPrint(recast.parse(code.replace(/\\/g, '/'), { parser: require('recast/parsers/typescript') }), {
|
||||||
|
tabWidth: 2,
|
||||||
|
trailingComma: true,
|
||||||
|
quote: 'single',
|
||||||
|
}).code;
|
||||||
|
|
||||||
|
const makeCamelCase = function (name: string): string {
|
||||||
|
if (typeof name !== 'string') {
|
||||||
|
return '';
|
||||||
|
}
|
||||||
|
return name.replace(/-(\w)/g, ($0, $1) => $1.toUpperCase());
|
||||||
|
};
|
364
packages/cli/src/utils/resolveAppPackages.ts
Normal file
364
packages/cli/src/utils/resolveAppPackages.ts
Normal file
@ -0,0 +1,364 @@
|
|||||||
|
import { execSync } from 'child_process';
|
||||||
|
import path from 'path';
|
||||||
|
import { exit } from 'process';
|
||||||
|
|
||||||
|
import fs from 'fs-extra';
|
||||||
|
import * as recast from 'recast';
|
||||||
|
|
||||||
|
import type App from '../Core';
|
||||||
|
import { Entry, EntryType, PackageType } from '../types';
|
||||||
|
|
||||||
|
interface TypeAssertion {
|
||||||
|
type: string;
|
||||||
|
imports: any[];
|
||||||
|
}
|
||||||
|
|
||||||
|
interface ParseEntryOption {
|
||||||
|
ast: any;
|
||||||
|
package: string;
|
||||||
|
indexPath: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
export const resolveAppPackages = (app: App) => {
|
||||||
|
const componentMap: Record<string, string> = {};
|
||||||
|
const configMap: Record<string, string> = {};
|
||||||
|
const eventMap: Record<string, string> = {};
|
||||||
|
const valueMap: Record<string, string> = {};
|
||||||
|
const pluginMap: Record<string, string> = {};
|
||||||
|
|
||||||
|
Object.entries(app.options.packages).forEach(([key, packagePath]) => {
|
||||||
|
installPackage(packagePath, app.options.source);
|
||||||
|
|
||||||
|
const indexPath = require.resolve(packagePath);
|
||||||
|
const indexCode = fs.readFileSync(indexPath, { encoding: 'utf-8', flag: 'r' });
|
||||||
|
const ast = recast.parse(indexCode, { parser: require('recast/parsers/typescript') });
|
||||||
|
const result = typeAssertion({ ast, indexPath });
|
||||||
|
|
||||||
|
const setItem = (key: string, entry: Entry) => {
|
||||||
|
if (entry.component) componentMap[key] = entry.component;
|
||||||
|
if (entry.config) configMap[key] = entry.config;
|
||||||
|
if (entry.event) eventMap[key] = entry.event;
|
||||||
|
if (entry.value) valueMap[key] = entry.value;
|
||||||
|
};
|
||||||
|
|
||||||
|
if (result.type === PackageType.COMPONENT) {
|
||||||
|
// 组件
|
||||||
|
setItem(key, parseEntry({ ast, package: packagePath, indexPath }));
|
||||||
|
} else if (result.type === PackageType.PLUGIN) {
|
||||||
|
// 插件
|
||||||
|
pluginMap[key] = packagePath;
|
||||||
|
} else if (result.type === PackageType.COMPONENT_PACKAGE) {
|
||||||
|
// 组件&插件包
|
||||||
|
result.imports.forEach((i) => {
|
||||||
|
const affixReg = new RegExp(`${app.options.componentFileAffix}$`);
|
||||||
|
if (affixReg.test(i.indexPath)) {
|
||||||
|
componentMap[i.type] = i.indexPath;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
const indexCode = fs.readFileSync(i.indexPath, { encoding: 'utf-8', flag: 'r' });
|
||||||
|
const ast = recast.parse(indexCode);
|
||||||
|
if (typeAssertion({ ast, indexPath }).type === PackageType.PLUGIN) {
|
||||||
|
// 插件
|
||||||
|
pluginMap[i.type] = i.indexPath;
|
||||||
|
} else {
|
||||||
|
// 组件
|
||||||
|
setItem(i.type, parseEntry({ ast, package: `${module} | ${i.name}`, indexPath: i.indexPath }));
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
return {
|
||||||
|
componentMap,
|
||||||
|
configMap,
|
||||||
|
eventMap,
|
||||||
|
valueMap,
|
||||||
|
pluginMap,
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
const installPackage = function (module: string, cwd: string) {
|
||||||
|
try {
|
||||||
|
// window下需要将路径中\转换成/
|
||||||
|
execSync(`node -e "require.resolve('${module.replace(/\\/g, '/')}')"`, { stdio: 'ignore' });
|
||||||
|
} catch (e) {
|
||||||
|
execSync(`npm install ${module}`, {
|
||||||
|
stdio: 'inherit',
|
||||||
|
cwd,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 1 判断是否组件&插件包
|
||||||
|
* 2 判断是组件还是插件
|
||||||
|
* 3 组件插件分开写入 comp-entry.ts
|
||||||
|
* @param {*} ast
|
||||||
|
* @param {String} indexPath
|
||||||
|
* @return {Object} { type: '', imports: [] } 返回传入组件的类型。如果是组件包,imports 中包含所有子组件的入口文件路径
|
||||||
|
*/
|
||||||
|
const typeAssertion = function ({ ast, indexPath }: { ast: any; indexPath: string }): TypeAssertion {
|
||||||
|
const n = recast.types.namedTypes;
|
||||||
|
|
||||||
|
const result = {
|
||||||
|
type: '',
|
||||||
|
imports: [],
|
||||||
|
};
|
||||||
|
|
||||||
|
const { importDeclarations, variableDeclarations, exportDefaultName, exportDefaultNode } =
|
||||||
|
getAssertionTokenByTraverse(ast);
|
||||||
|
|
||||||
|
if (exportDefaultName) {
|
||||||
|
importDeclarations.every((node) => {
|
||||||
|
const [specifier] = node.specifiers;
|
||||||
|
|
||||||
|
// 从 import 语句中找到 export default 的变量,认为是组件
|
||||||
|
if (n.ImportDefaultSpecifier.check(specifier) && specifier.local?.name === exportDefaultName) {
|
||||||
|
result.type = PackageType.COMPONENT;
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
});
|
||||||
|
|
||||||
|
if (result.type) return result;
|
||||||
|
|
||||||
|
variableDeclarations.every((node) => {
|
||||||
|
const [variable] = node.declarations;
|
||||||
|
|
||||||
|
// 从声明变量语句中找到 export default 的变量,认为是组件包
|
||||||
|
if (
|
||||||
|
n.Identifier.check(variable.id) &&
|
||||||
|
variable.id.name === exportDefaultName &&
|
||||||
|
n.ObjectExpression.check(variable.init)
|
||||||
|
) {
|
||||||
|
if (isPlugin(variable.init.properties)) {
|
||||||
|
result.type = PackageType.PLUGIN;
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
// 从组件包声明中找到对应子组件入口文件
|
||||||
|
getComponentPackageImports({ result, properties: variable.init.properties, indexPath, importDeclarations });
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
if (exportDefaultNode) {
|
||||||
|
if (isPlugin((exportDefaultNode as any).properties)) {
|
||||||
|
result.type = PackageType.PLUGIN;
|
||||||
|
} else {
|
||||||
|
getComponentPackageImports({
|
||||||
|
result,
|
||||||
|
properties: (exportDefaultNode as any).properties,
|
||||||
|
indexPath,
|
||||||
|
importDeclarations,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return result;
|
||||||
|
};
|
||||||
|
|
||||||
|
const getAssertionTokenByTraverse = (ast: any) => {
|
||||||
|
const importDeclarations: any[] = [];
|
||||||
|
const variableDeclarations: any[] = [];
|
||||||
|
const n = recast.types.namedTypes;
|
||||||
|
|
||||||
|
let exportDefaultName = '';
|
||||||
|
let exportDefaultNode = undefined;
|
||||||
|
|
||||||
|
recast.types.visit(ast, {
|
||||||
|
visitImportDeclaration(p) {
|
||||||
|
importDeclarations.push(p.node);
|
||||||
|
this.traverse(p);
|
||||||
|
},
|
||||||
|
visitVariableDeclaration(p) {
|
||||||
|
variableDeclarations.push(p.node);
|
||||||
|
this.traverse(p);
|
||||||
|
},
|
||||||
|
visitExportDefaultDeclaration(p) {
|
||||||
|
const { node } = p;
|
||||||
|
const { declaration } = node;
|
||||||
|
|
||||||
|
// 导出的是变量名
|
||||||
|
if (n.Identifier.check(declaration)) {
|
||||||
|
exportDefaultName = declaration.name;
|
||||||
|
}
|
||||||
|
|
||||||
|
// 导出的是对象的字面量
|
||||||
|
if (n.ObjectExpression.check(declaration)) {
|
||||||
|
exportDefaultNode = declaration;
|
||||||
|
}
|
||||||
|
|
||||||
|
this.traverse(p);
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
return {
|
||||||
|
importDeclarations,
|
||||||
|
variableDeclarations,
|
||||||
|
exportDefaultName,
|
||||||
|
exportDefaultNode,
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
const isPlugin = function (properties: any[]) {
|
||||||
|
const [match] = properties.filter((property) => property.key.name === 'install');
|
||||||
|
|
||||||
|
return !!match;
|
||||||
|
};
|
||||||
|
|
||||||
|
const getComponentPackageImports = function ({
|
||||||
|
result,
|
||||||
|
properties,
|
||||||
|
indexPath,
|
||||||
|
importDeclarations,
|
||||||
|
}: {
|
||||||
|
result: TypeAssertion;
|
||||||
|
properties: any[];
|
||||||
|
indexPath: string;
|
||||||
|
importDeclarations: any[];
|
||||||
|
}) {
|
||||||
|
const n = recast.types.namedTypes;
|
||||||
|
|
||||||
|
result.type = PackageType.COMPONENT_PACKAGE;
|
||||||
|
result.imports = [];
|
||||||
|
|
||||||
|
properties.forEach((property) => {
|
||||||
|
const [propertyMatch] = importDeclarations.filter((i) => {
|
||||||
|
const [specifier] = i.specifiers;
|
||||||
|
|
||||||
|
if (n.ImportDefaultSpecifier.check(specifier) && specifier.local?.name === property.value.name) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
return false;
|
||||||
|
});
|
||||||
|
|
||||||
|
if (propertyMatch) {
|
||||||
|
result.imports.push({
|
||||||
|
type: property.key.name,
|
||||||
|
name: propertyMatch.specifiers[0].local.name,
|
||||||
|
indexPath: getIndexPath(path.resolve(path.dirname(indexPath), propertyMatch.source.value)),
|
||||||
|
});
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
return result;
|
||||||
|
};
|
||||||
|
|
||||||
|
const getIndexPath = function (entry: string) {
|
||||||
|
if (fs.lstatSync(entry).isFile()) {
|
||||||
|
return entry;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (fs.lstatSync(entry).isDirectory()) {
|
||||||
|
const files = fs.readdirSync(entry);
|
||||||
|
const [index] = files.filter((file) => file.split('.')[0] === 'index');
|
||||||
|
|
||||||
|
return path.resolve(entry, index);
|
||||||
|
}
|
||||||
|
|
||||||
|
return entry;
|
||||||
|
};
|
||||||
|
|
||||||
|
const parseEntry = function ({ ast, package: module, indexPath }: ParseEntryOption) {
|
||||||
|
if (!ast.program) {
|
||||||
|
console.log(`${module} 入口文件不合法`);
|
||||||
|
return exit(1);
|
||||||
|
}
|
||||||
|
|
||||||
|
const tokens = getASTTokenByTraverse({ ast, indexPath });
|
||||||
|
let { config, value, event } = tokens;
|
||||||
|
const { importComponentSource, importComponentToken, exportDefaultToken } = tokens;
|
||||||
|
|
||||||
|
if (!config) {
|
||||||
|
console.log(`${module} ${EntryType.CONFIG} 文件声明不合法`);
|
||||||
|
return exit(1);
|
||||||
|
}
|
||||||
|
if (!value) {
|
||||||
|
console.log(`${module} ${EntryType.VALUE} 文件声明不合法`);
|
||||||
|
return exit(1);
|
||||||
|
}
|
||||||
|
if (!event) {
|
||||||
|
// event 非必须,不需要 exit
|
||||||
|
console.log(`${module} ${EntryType.EVENT} 文件声明缺失`);
|
||||||
|
}
|
||||||
|
|
||||||
|
const findIndex = importComponentToken.indexOf(exportDefaultToken);
|
||||||
|
let component = '';
|
||||||
|
if (findIndex > -1) {
|
||||||
|
component = path.resolve(path.dirname(indexPath), importComponentSource[findIndex]);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!component) {
|
||||||
|
console.log(`${module} ${EntryType.COMPONENT} 文件声明不合法`);
|
||||||
|
return exit(1);
|
||||||
|
}
|
||||||
|
|
||||||
|
const reg = /^.*[/\\]node_modules[/\\](.*)/;
|
||||||
|
[, config] = config.match(reg) || [, config];
|
||||||
|
[, value] = value.match(reg) || [, value];
|
||||||
|
[, component] = component.match(reg) || [, component];
|
||||||
|
[, event] = event.match(reg) || [, event];
|
||||||
|
|
||||||
|
return {
|
||||||
|
config,
|
||||||
|
value,
|
||||||
|
component,
|
||||||
|
event,
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
const getASTTokenByTraverse = ({ ast, indexPath }: { ast: any; indexPath: string }) => {
|
||||||
|
let config = '';
|
||||||
|
let value = '';
|
||||||
|
let event = '';
|
||||||
|
const importComponentToken: string[] = [];
|
||||||
|
const importComponentSource: any[] = [];
|
||||||
|
let exportDefaultToken = '';
|
||||||
|
|
||||||
|
recast.types.visit(ast, {
|
||||||
|
visitImportDeclaration(p) {
|
||||||
|
const { node } = p;
|
||||||
|
const { specifiers, source } = node;
|
||||||
|
|
||||||
|
importComponentToken.push(specifiers?.[0].local?.name || '');
|
||||||
|
importComponentSource.push(source.value);
|
||||||
|
|
||||||
|
this.traverse(p);
|
||||||
|
},
|
||||||
|
visitExportNamedDeclaration(p) {
|
||||||
|
const { node } = p;
|
||||||
|
const { specifiers, source } = node;
|
||||||
|
const name = specifiers?.[0].exported.name.toLowerCase();
|
||||||
|
|
||||||
|
if (name === EntryType.VALUE) {
|
||||||
|
value = path.resolve(path.dirname(indexPath), `${source?.value}`);
|
||||||
|
} else if (name === EntryType.CONFIG) {
|
||||||
|
config = path.resolve(path.dirname(indexPath), `${source?.value}`);
|
||||||
|
} else if (name === EntryType.EVENT) {
|
||||||
|
event = path.resolve(path.dirname(indexPath), `${source?.value}`);
|
||||||
|
}
|
||||||
|
|
||||||
|
this.traverse(p);
|
||||||
|
},
|
||||||
|
visitExportDefaultDeclaration(p) {
|
||||||
|
const { node } = p;
|
||||||
|
const { declaration } = node as any;
|
||||||
|
exportDefaultToken = `${declaration.name}`;
|
||||||
|
this.traverse(p);
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
return {
|
||||||
|
config,
|
||||||
|
value,
|
||||||
|
event,
|
||||||
|
importComponentToken,
|
||||||
|
importComponentSource,
|
||||||
|
exportDefaultToken,
|
||||||
|
};
|
||||||
|
};
|
9
packages/cli/tsconfig.build.json
Normal file
9
packages/cli/tsconfig.build.json
Normal file
@ -0,0 +1,9 @@
|
|||||||
|
{
|
||||||
|
"extends": "../../tsconfig.json",
|
||||||
|
"compilerOptions": {
|
||||||
|
"module": "CommonJS",
|
||||||
|
"rootDir": "./src",
|
||||||
|
"outDir": "./lib",
|
||||||
|
},
|
||||||
|
"include": ["./src"],
|
||||||
|
}
|
1001
pnpm-lock.yaml
generated
1001
pnpm-lock.yaml
generated
File diff suppressed because it is too large
Load Diff
Loading…
x
Reference in New Issue
Block a user