nocobase/packages/core/resourcer/src/resourcer.ts
ChengLei Shao 261d4c4137
refactor: establish a sound testing system (#3179)
* chore: use vitest to replace jest

* chore: support vitest

* feat: vitest 1.0

* fix: test

* chore: yarn.lock

* chore: github actions

* fix: test

* fix: test

* fix: test

* fix: test

* fix: jest.fn

* fix: require

* fix: test

* fix: build

* fix: test

* fix: test

* fix: test

* fix: test

* fix: test

* fix: test

* fix: test

* fix: dynamic import

* fix: bug

* chore: yarn run test command

* chore: package.json

* chore: package.json

* chore: vite 5

* fix: fix variable test

* fix: import json

* feat: initEnv

* fix: env.APP_ENV_PATH

* chore: get package json

* fix: remove GlobalThmeProvider

* chore: update snap

* chore: test env

* chore: test env

* chore: import module

* chore: jest

* fix: load package json

* chore: test

* fix: bug

* chore: test

* chore: test

* chore: test

* chore: test

* chore: test

* fix: import file in windows

* chore: import module with absolute file path

* fix: test error

* test: update snapshot

* chore: update yarn.lock

* fix: front-end tests do not include utils folder

* refactor: use vitest-dom

* fix: fix build

* fix: test error

* fix: change to vitest.config.mts

* fix: types error

* fix: types error

* fix: types error

* fix: error

* fix: test

* chore: test

* fix: test package

* feat: update dependencies

* refactor: test

* fix: error

* fix: error

* fix: __dirname is not defined in ES module scope

* fix: allow only

* fix: error

* fix: error

* fix: error

* fix: create-app

* fix: install-deps

* feat: update docs

---------

Co-authored-by: chenos <chenlinxh@gmail.com>
Co-authored-by: dream2023 <1098626505@qq.com>
Co-authored-by: Zeke Zhang <958414905@qq.com>
2023-12-21 20:39:11 +08:00

353 lines
8.3 KiB
TypeScript

import { importModule, Toposort, ToposortOptions } from '@nocobase/utils';
import glob from 'glob';
import compose from 'koa-compose';
import _ from 'lodash';
import { pathToRegexp } from 'path-to-regexp';
import Action, { ActionName } from './action';
import Resource, { ResourceOptions } from './resource';
import { getNameByParams, ParsedParams, parseQuery, parseRequest } from './utils';
export interface ResourcerContext {
resourcer?: Resourcer;
action?: Action;
[key: string]: any;
}
export interface KoaMiddlewareOptions {
/**
* 前缀
*/
prefix?: string;
/**
* 自定义 resource name 的获取规则
*
* 默认规则 relatedTable ? relatedTable.table : table
*/
nameRule?: (params: ParsedParams) => string;
/**
* 自定义 action name
*
* 默认为
*
* - list 查看列表
* - create 新增数据
* - get 查看数据详情
* - update 更新数据
* - delete 删除数据
*/
accessors?: {
/**
* 查看列表
*/
list?: string;
/**
* 新增数据
*/
create?: string;
/**
* 查看数据详情
*/
get?: string;
/**
* 更新数据
*/
update?: string;
/**
* 删除数据
*/
delete?: string;
};
}
export interface ResourcerOptions {
/**
* 前缀
*/
prefix?: string;
/**
* 自定义 action name
*
* 默认为
*
* - list 查看列表
* - create 新增数据
* - get 查看数据详情
* - update 更新数据
* - delete 删除数据
*/
accessors?: {
/**
* 查看列表
*/
list?: string;
/**
* 新增数据
*/
create?: string;
/**
* 查看数据详情
*/
get?: string;
/**
* 更新数据
*/
update?: string;
/**
* 删除数据
*/
delete?: string;
};
}
export interface ExecuteOptions {
/**
* 资源名称
*/
resource: string;
/**
* 自定义 action name
*
* 默认
* - list 查看列表
* - create 新增数据
* - get 查看数据详情
* - update 更新数据
* - delete 删除数据
*/
action: ActionName;
}
export type HandlerType = (ctx: ResourcerContext, next: () => Promise<any>) => any;
export interface Handlers {
[key: string]: HandlerType;
}
export interface ImportOptions {
/**
* 指定配置所在路径
*/
directory: string;
/**
* 文件后缀,默认值 ['js', 'ts', 'json']
*/
extensions?: string[];
}
export class Resourcer {
public readonly options: ResourcerOptions;
protected resources = new Map<string, Resource>();
/**
* 全局定义的 action handlers
*/
protected handlers = new Map<ActionName, any>();
protected actionHandlers = new Map<ActionName, any>();
protected middlewareHandlers = new Map<string, any>();
protected middlewares: Toposort<any>;
constructor(options: ResourcerOptions = {}) {
this.options = options;
this.middlewares = new Toposort<any>();
}
/**
* 载入指定目录下的 resource 配置(配置的文件驱动)
*
* TODO: 配置的文件驱动现在会全部初始化,大数据时可能存在性能瓶颈,后续可以加入动态加载
*
* @param {object} [options]
* @param {string} [options.directory] 指定配置所在路径
* @param {array} [options.extensions = ['js', 'ts', 'json']] 文件后缀
*/
public async import(options: ImportOptions): Promise<Map<string, Resource>> {
const { extensions = ['js', 'ts', 'json'], directory } = options;
const patten = `${directory}/*.{${extensions.join(',')}}`;
const files = glob.sync(patten, {
ignore: ['**/*.d.ts'],
});
const resources = new Map<string, Resource>();
for (const file of files) {
const options = await importModule(file);
const table = this.define(typeof options === 'function' ? options(this) : options);
resources.set(table.getName(), table);
}
return resources;
}
/**
* resource 配置
*
* @param name
* @param options
*/
define(options: ResourceOptions) {
const { name } = options;
const resource = new Resource(options, this);
this.resources.set(name, resource);
return resource;
}
isDefined(name: string) {
return this.resources.has(name);
}
removeResource(name) {
return this.resources.delete(name);
}
registerAction(name: ActionName, handler: HandlerType) {
this.registerActionHandler(name, handler);
}
registerActions(handlers: Handlers) {
this.registerActionHandlers(handlers);
}
/**
* 注册全局的 action handlers
*
* @param handlers
*/
registerActionHandlers(handlers: Handlers) {
for (const [name, handler] of Object.entries(handlers)) {
this.registerActionHandler(name, handler);
}
}
registerActionHandler(name: ActionName, handler: HandlerType) {
this.actionHandlers.set(name, handler);
}
getRegisteredHandler(name: ActionName) {
return this.actionHandlers.get(name);
}
getRegisteredHandlers() {
return this.actionHandlers;
}
getResource(name: string): Resource {
if (!this.resources.has(name)) {
throw new Error(`${name} resource does not exist`);
}
return this.resources.get(name);
}
getAction(name: string, action: ActionName): Action {
// 支持注册局部 action
if (this.actionHandlers.has(`${name}:${action}`)) {
return this.getResource(name).getAction(`${name}:${action}`);
}
return this.getResource(name).getAction(action);
}
getMiddlewares() {
return this.middlewares.nodes;
}
use(middlewares: HandlerType | HandlerType[], options: ToposortOptions = {}) {
this.middlewares.add(middlewares, options);
}
restApiMiddleware({ prefix, accessors }: KoaMiddlewareOptions = {}) {
return async (ctx: ResourcerContext, next: () => Promise<any>) => {
ctx.resourcer = this;
let params = parseRequest(
{
path: ctx.request.path,
method: ctx.request.method,
},
{
prefix: this.options.prefix || prefix,
accessors: this.options.accessors || accessors,
},
);
if (!params) {
return next();
}
try {
const resource = this.getResource(getNameByParams(params));
// 为关系资源时,暂时需要再执行一遍 parseRequest
if (resource.options.type && resource.options.type !== 'single') {
params = parseRequest(
{
path: ctx.request.path,
method: ctx.request.method,
type: resource.options.type,
},
{
prefix: this.options.prefix || prefix,
accessors: this.options.accessors || accessors,
},
);
if (!params) {
return next();
}
}
// action 需要 clone 之后再赋给 ctx
ctx.action = this.getAction(getNameByParams(params), params.actionName).clone();
ctx.action.setContext(ctx);
ctx.action.actionName = params.actionName;
ctx.action.resourceOf = params.associatedIndex;
ctx.action.resourceName = params.associatedName
? `${params.associatedName}.${params.resourceName}`
: params.resourceName;
ctx.action.params.filterByTk = params.resourceIndex;
const query = parseQuery(ctx.request.querystring);
if (pathToRegexp('/resourcer/{:associatedName.}?:resourceName{\\::actionName}').test(ctx.request.path)) {
ctx.action.mergeParams({
...query,
...params,
...ctx.request.body,
});
} else {
ctx.action.mergeParams({
...query,
...params,
...(_.isEmpty(ctx.request.body) ? {} : { values: ctx.request.body }),
});
}
return compose(ctx.action.getHandlers())(ctx, next);
} catch (error) {
return next();
}
};
}
middleware(options: KoaMiddlewareOptions = {}) {
return this.restApiMiddleware(options);
}
/**
* 实验性 API
*
* @param options
* @param context
* @param next
*/
async execute(options: ExecuteOptions, context: ResourcerContext = {}, next?: any) {
const { resource, action } = options;
context.resourcer = this;
context.action = this.getAction(resource, action);
return await context.action.execute(context, next);
}
}
export default Resourcer;