amis-rpc-design/libs/amis/packages/amis-formula/scripts/genDoc.ts
2023-10-07 19:42:30 +08:00

155 lines
3.8 KiB
TypeScript
Raw Blame History

This file contains ambiguous Unicode characters

This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

import path from 'path';
import fs from 'fs';
import doctrine from 'doctrine';
const workDir = path.resolve(path.dirname(__dirname));
const jsFile = path.join(workDir, 'src/evalutor.ts');
const outputFile = path.join(workDir, 'src/doc.ts');
const outputMD = path.join(workDir, 'src/doc.md');
function getFormulaComments(contents: string) {
const comments: Array<{
fnName: string;
comments: string;
}> = [];
contents.replace(/\/\*[\s\S]+?\*\//g, (_, index, input) => {
const pos = index + _.length;
const following = input.substring(pos, pos + 200);
if (/^\s*fn(\w+)\(/.test(following)) {
comments.push({
fnName: RegExp.$1,
comments: _
});
}
return _;
});
return comments;
}
function formatType(tag: any): string {
// console.log(tag);
if (tag.type === 'RestType') {
return `...${formatType(tag.expression)}`;
} else if (tag.type === 'TypeApplication') {
return `Array<${tag.applications
.map((item: any) => formatType(item))
.join(',')}>`;
}
return tag.name;
}
async function main(...params: Array<any>) {
const contents = fs.readFileSync(jsFile, 'utf8');
const comments = getFormulaComments(contents);
const result = comments.map(item => {
const ast = doctrine.parse(item.comments, {
unwrap: true
});
const result: any = {
name: item.fnName,
description: ast.description
};
let example = '';
let params: Array<any> = [];
let returns: any = undefined;
let namespace = '';
ast.tags.forEach(tag => {
if (tag.title === 'example') {
example = tag.description!;
} else if (tag.title === 'namespace') {
namespace = tag.name!;
} else if (tag.title === 'param') {
params.push({
type: formatType(tag.type),
name: tag.name,
description: tag.description
});
} else if (tag.title === 'returns') {
returns = {
type: formatType(tag.type),
description: tag.description
};
}
});
result.example = example;
result.params = params;
result.returns = returns;
result.namespace = namespace;
return result;
});
fs.writeFileSync(
outputFile,
`/**\n * 公式文档 请运行 \`npm run genDoc\` 自动生成\n */\nexport const doc: ${[
`{`,
` name: string;`,
` description: string;`,
` example: string;`,
` params: {`,
` type: string;`,
` name: string;`,
` description: string | null;`,
` }[];`,
` returns: {`,
` type: string;`,
` description: string | null;`,
` };`,
` namespace: string;`,
`}[]`
].join('\n')} = ${JSON.stringify(result, null, 2).replace(
/\"(\w+)\"\:/g,
(_, key) => `${key}:`
)};`,
'utf8'
);
console.log(`公式文档生成 > ${outputFile}`);
const grouped: any = {};
result.forEach((item: any) => {
const scope = item.namespace || '其他';
const arr = grouped[scope] || (grouped[scope] = []);
arr.push(item);
});
let md = '';
Object.keys(grouped).forEach(key => {
md += `## ${key}\n\n`;
grouped[key].forEach((item: any) => {
md += `### ${item.name}\n\n`;
md += `用法:\`${item.example}\`\n\n`;
if (item.params.length) {
// md += `参数:\n`;
item.params.forEach((param: any) => {
md += ` * \`${param.name}:${param.type}\` ${param.description}\n`;
});
if (item.returns) {
md += `\n返回\`${item.returns.type}\` ${
item.returns.description || ''
}\n\n`;
}
}
md += `${item.description}\n\n`;
});
});
fs.writeFileSync(outputMD, md, 'utf8');
console.log(`公式md生成 > ${outputMD}`);
}
main().catch(e => console.error(e));