(function (global, factory) { typeof exports === 'object' && typeof module !== 'undefined' ? factory(exports, require('fs'), require('path'), require('prettier'), require('axios'), require('lodash'), require('@vue/compiler-sfc'), require('magic-string'), require('glob'), require('node:util'), require('svgo'), require('postcss-value-parser')) : typeof define === 'function' && define.amd ? define(['exports', 'fs', 'path', 'prettier', 'axios', 'lodash', '@vue/compiler-sfc', 'magic-string', 'glob', 'node:util', 'svgo', 'postcss-value-parser'], factory) : (global = typeof globalThis !== 'undefined' ? globalThis : global || self, factory(global.index = {}, global.fs, global.path, global.prettier, global.axios, global.lodash, global.compilerSfc, global.magicString, global.glob, global.util, global.svgo, global.valueParser)); })(this, (function (exports, fs, path, prettier, axios, lodash, compilerSfc, magicString, glob, util, svgo, valueParser) { 'use strict'; const config = { type: "admin", reqUrl: "", eps: { enable: true, api: "", dist: "./build/cool", mapping: [ { // 自定义匹配 custom: ({ propertyName, type }) => { // 如果没有,返回null或者不返回,则继续遍历其他匹配规则 return null; }, }, { type: "string", test: ["varchar", "text", "simple-json"], }, { type: "string[]", test: ["simple-array"], }, { type: "Date", test: ["datetime", "date"], }, { type: "number", test: ["tinyint", "int", "decimal"], }, { type: "BigInt", test: ["bigint"], }, ], }, svg: { skipNames: ["base"], }, }; // 根目录 function rootDir(path$1) { switch (config.type) { case "app": case "uniapp-x": return path.join(process.env.UNI_INPUT_DIR, path$1); default: return path.join(process.cwd(), path$1); } } // 首字母大写 function firstUpperCase(value) { return value.replace(/\b(\w)(\w*)/g, function ($0, $1, $2) { return $1.toUpperCase() + $2; }); } // 横杠转驼峰 function toCamel(str) { return str.replace(/([^-])(?:-+([^-]))/g, function ($0, $1, $2) { return $1 + $2.toUpperCase(); }); } // 创建目录 function createDir(path, recursive) { try { if (!fs.existsSync(path)) fs.mkdirSync(path, { recursive }); } catch (err) { } } // 读取文件 function readFile(path, json) { try { const content = fs.readFileSync(path, "utf8"); return json ? JSON.parse(content.replace(/\/\/.*$/gm, "").replace(/\/\*[\s\S]*?\*\//g, "")) : content; } catch (err) { } return ""; } // 写入文件 function writeFile(path, data) { try { return fs.writeFileSync(path, data); } catch (err) { } return ""; } // 解析body function parseJson(req) { return new Promise((resolve) => { let d = ""; req.on("data", function (chunk) { d += chunk; }); req.on("end", function () { try { resolve(JSON.parse(d)); } catch { resolve({}); } }); }); } // 格式化内容 function formatContent(content, options) { return prettier.format(content, { parser: "typescript", useTabs: true, tabWidth: 4, endOfLine: "lf", semi: true, ...options, }); } function error(message) { console.log("\x1B[31m%s\x1B[0m", message); } const service = {}; let list = []; // 获取请求地址 function getEpsUrl() { let url = config.eps.api; if (!url) { url = config.type; } switch (url) { case "app": case "uniapp-x": url = "/app/base/comm/eps"; break; case "admin": url = "/admin/base/open/eps"; break; } return url; } // 获取路径 function getEpsPath(filename) { return path.join(config.type == "admin" ? config.eps.dist : rootDir(config.eps.dist), filename || ""); } // 获取方法名 function getNames(v) { return Object.keys(v).filter((e) => !["namespace", "permission"].includes(e)); } // 找字段 function findColumns(sources, item) { const columns = [item.columns, item.pageColumns].flat().filter(Boolean); return (sources || []) .map((e) => columns.find((c) => c.source == e)) .filter(Boolean); } // 格式化代码 async function formatCode(text) { return prettier .format(text, { parser: "typescript", useTabs: true, tabWidth: 4, endOfLine: "lf", semi: true, singleQuote: false, printWidth: 100, trailingComma: "none", }) .catch((err) => { console.log(err); error(`[cool-eps] Failed to format /build/cool/eps.d.ts. Please delete the file and try again`); return null; }); } // 获取数据 async function getData() { // 读取本地数据 list = readFile(getEpsPath("eps.json"), true) || []; // 请求地址 const url = config.reqUrl + getEpsUrl(); // 请求数据 await axios .get(url, { timeout: 5000, }) .then((res) => { const { code, data, message } = res.data; if (code === 1000) { if (!lodash.isEmpty(data) && data) { list = lodash.values(data).flat(); } } else { error(`[cool-eps] ${message || "Failed to fetch data"}`); } }) .catch(() => { error(`[cool-eps] API service is not running → ${url}`); }); // 初始化处理 list.forEach((e) => { if (!e.namespace) { e.namespace = ""; } if (!e.api) { e.api = []; } if (!e.columns) { e.columns = []; } if (!e.search) { e.search = { fieldEq: findColumns(e.pageQueryOp?.fieldEq, e), fieldLike: findColumns(e.pageQueryOp?.fieldLike, e), keyWordLikeFields: findColumns(e.pageQueryOp?.keyWordLikeFields, e), }; } }); } // 创建 json 文件 function createJson() { const arr = list.map((e) => { return { prefix: e.prefix, name: e.name || "", api: e.api.map((e) => { return { name: e.name, method: e.method, path: e.path, }; }), search: e.search, }; }); const content = JSON.stringify(arr); const local_content = readFile(getEpsPath("eps.json")); // 是否需要更新 const isUpdate = content != local_content; if (isUpdate) { fs.createWriteStream(getEpsPath("eps.json"), { flags: "w", }).write(content); } return isUpdate; } // 创建描述文件 async function createDescribe({ list, service }) { // 获取类型 function getType({ propertyName, type }) { for (const map of config.eps.mapping) { if (map.custom) { const resType = map.custom({ propertyName, type }); if (resType) return resType; } if (map.test) { if (map.test.includes(type)) return map.type; } } return type; } // 格式化方法名 function formatName(name) { return (name || "").replace(/[:,\s,\/,-]/g, ""); } // 检查方法名,包含特殊字符则忽略 function checkName(name) { return name && !["{", "}", ":"].some((e) => name.includes(e)); } // 创建 Entity function createEntity() { const ignore = []; let t0 = ""; for (const item of list) { if (!checkName(item.name)) continue; let t = `interface ${formatName(item.name)} {`; // 合并多个列 const columns = []; [item.columns, item.pageColumns] .flat() .filter(Boolean) .forEach((e) => { const d = columns.find((c) => c.source == e.source); if (!d) { columns.push(e); } }); for (const col of columns || []) { t += ` /** * ${col.comment} */ ${col.propertyName}?: ${getType({ propertyName: col.propertyName, type: col.type, })} `; } t += ` /** * 任意键值 */ [key: string]: any; } `; if (!ignore.includes(item.name)) { ignore.push(item.name); t0 += t + "\n\n"; } } return t0; } // 创建 Controller async function createController() { let controller = ""; let chain = ""; // 处理数据 function deep(d, k) { if (!k) k = ""; for (const i in d) { const name = k + toCamel(firstUpperCase(formatName(i))); // 检查方法名 if (!checkName(name)) continue; if (d[i].namespace) { // 查找配置 const item = list.find((e) => (e.prefix || "") === `/${d[i].namespace}`); if (item) { let t = `interface ${name} {`; // 插入方法 if (item.api) { // 权限列表 const permission = []; item.api.forEach((a) => { // 方法名 const n = toCamel(formatName(a.name || lodash.last(a.path.split("/")))); // 检查方法名 if (!checkName(n)) return; if (n) { // 参数类型 let q = []; // 参数列表 const { parameters = [] } = a.dts || {}; parameters.forEach((p) => { if (p.description) { q.push(`\n/** ${p.description} */\n`); } // 检查参数名 if (!checkName(p.name)) { return false; } const a = `${p.name}${p.required ? "" : "?"}`; const b = `${p.schema.type || "string"}`; q.push(`${a}: ${b},`); }); if (lodash.isEmpty(q)) { q = ["any"]; } else { q.unshift("{"); q.push("}"); } // 返回类型 let res = ""; // 实体名 const en = item.name || "any"; switch (a.path) { case "/page": res = ` { pagination: { size: number; page: number; total: number; [key: string]: any; }; list: ${en} []; [key: string]: any; } `; break; case "/list": res = `${en} []`; break; case "/info": res = en; break; default: res = "any"; break; } // 描述 t += ` /** * ${a.summary || n} */ ${n}(data${q.length == 1 ? "?" : ""}: ${q.join("")}): Promise<${res}>; `; if (!permission.includes(n)) { permission.push(n); } } }); // 权限标识 t += ` /** * 权限标识 */ permission: { ${permission.map((e) => `${e}: string;`).join("\n")} }; `; // 权限状态 t += ` /** * 权限状态 */ _permission: { ${permission.map((e) => `${e}: boolean;`).join("\n")} }; `; t += ` request: Service['request'] `; } t += "}\n\n"; controller += t; chain += `${formatName(i)}: ${name};`; } } else { chain += `${formatName(i)}: {`; deep(d[i], name); chain += "},"; } } } // 遍历 deep(service); return ` type json = any; ${controller} interface Service { /** * 基础请求 */ request(options?: { url: string; method?: "POST" | "GET" | "PUT" | "DELETE" | "PATCH" | "HEAD" | "OPTIONS"; data?: any; params?: any; headers?: any, timeout?: number; proxy?: boolean; [key: string]: any; }): Promise; ${chain} } ${await createDict()} `; } // 文件内容 let text = ` ${createEntity()} ${await createController()} `; // 文件名 let name = "eps.d.ts"; if (config.type == "uniapp-x") { name = "eps.uts"; text = text .replaceAll("interface ", "export interface ") .replaceAll("type Dict", "export type Dict") .replaceAll("[key: string]: any;", ""); } else { text = ` declare namespace Eps { ${text} } `; } // 文本内容 const content = await formatCode(text); const local_content = readFile(getEpsPath(name)); // 是否需要更新 if (content && content != local_content) { // 创建 eps 描述文件 fs.createWriteStream(getEpsPath(name), { flags: "w", }).write(content); } } // 创建 service function createService() { // 路径第一层作为 id 标识 const id = getEpsUrl().split("/")[1]; list.forEach((e) => { // 请求地址 const path = e.prefix[0] == "/" ? e.prefix.substring(1, e.prefix.length) : e.prefix; // 分隔路径 const arr = path.replace(id, "").split("/").filter(Boolean).map(toCamel); // 遍历 function deep(d, i) { const k = arr[i]; if (k) { // 是否最后一个 if (arr[i + 1]) { if (!d[k]) { d[k] = {}; } deep(d[k], i + 1); } else { // 不存在则创建 if (!d[k]) { d[k] = { permission: {}, }; } if (!d[k].namespace) { d[k].namespace = path; } // 创建权限 if (d[k].namespace) { getNames(d[k]).forEach((i) => { d[k].permission[i] = `${d[k].namespace.replace(`${id}/`, "")}/${i}`.replace(/\//g, ":"); }); } // 创建搜索 d[k].search = e.search; // 创建方法 e.api.forEach((a) => { // 方法名 const n = a.path.replace("/", ""); if (n && !/[-:]/g.test(n)) { d[k][n] = a; } }); } } } deep(service, 0); }); } // 创建 dict async function createDict() { let p = ""; switch (config.type) { case "app": case "uniapp-x": p = "/app"; break; case "admin": p = "/admin"; break; } const url = config.reqUrl + p + "/dict/info/types"; const text = await axios .get(url) .then((res) => { const { code, data } = res.data; if (code === 1000) { let v = "string"; if (!lodash.isEmpty(data)) { v = data.map((e) => `"${e.key}"`).join(" | "); } return `type DictKey = ${v}`; } }) .catch(() => { error(`[cool-eps] Error:${url}`); }); return text || ""; } // 创建 eps async function createEps() { if (config.eps.enable) { // 获取数据 await getData(); // 创建 service createService(); // 创建目录 createDir(getEpsPath(), true); // 创建 json 文件 const isUpdate = createJson(); // 创建描述文件 createDescribe({ service, list }); return { service, list, isUpdate, }; } else { return { service: {}, list: [], }; } } function getPlugin(name) { let code = readFile(rootDir(`./src/plugins/${name}/config.ts`)); // 设置插件配置 const set = (key, value) => { const regex = new RegExp(`(return\\s*{[^}]*?\\b${key}\\b\\s*:\\s*)([^,}]+)`); if (regex.test(code)) { code = code.replace(regex, `$1${JSON.stringify(value)}`); } else { const insertPos = code.indexOf("return {") + 8; code = code.slice(0, insertPos) + `\n ${key}: ${JSON.stringify(value)},` + code.slice(insertPos); } }; // 保存插件配置 const save = async () => { const content = await formatContent(code); writeFile(rootDir(`./src/plugins/${name}/config.ts`), content); }; return { set, save, }; } // 修改插件 async function updatePlugin(options) { const plugin = getPlugin(options.name); if (options.enable !== undefined) { plugin.set("enable", options.enable); } await plugin.save(); } function getPath() { return rootDir(`.${config.type == "admin" ? "/src" : ""}/config/proxy.ts`); } async function updateProxy(data) { let code = readFile(getPath()); const regex = /const\s+value\s*=\s*['"]([^'"]+)['"]/; if (regex.test(code)) { code = code.replace(regex, `const value = '${data.name}'`); } writeFile(getPath(), code); } function getProxyTarget(proxy) { const code = readFile(getPath()); const regex = /const\s+value\s*=\s*['"]([^'"]+)['"]/; const match = code.match(regex); if (match) { const value = match[1]; try { const { target, rewrite } = proxy[`/${value}/`]; return target + rewrite(`/${value}`); } catch (err) { error(`[cool-proxy] Error:${value} → ` + getPath()); return ""; } } } // 创建文件 async function createFile(data) { const list = lodash.isArray(data) ? data : [data]; for (const item of list) { const { path: path$1, code } = item; // 格式化内容 const content = await formatContent(code, { parser: "vue", }); // 目录路径 const dir = (path$1 || "").split("/"); // 文件名 const fname = dir.pop(); // 源码路径 const srcPath = `./src/${dir.join("/")}`; // 创建目录 createDir(srcPath, true); // 创建文件 fs.createWriteStream(path.join(srcPath, fname), { flags: "w", }).write(content); } } function createTag(code, id) { if (/\.vue$/.test(id)) { let s; const str = () => s || (s = new magicString(code)); const { descriptor } = compilerSfc.parse(code); if (!descriptor.script && descriptor.scriptSetup) { const res = compilerSfc.compileScript(descriptor, { id }); const { name, lang } = res.attrs; str().appendLeft(0, `