nocobase/packages/core/resourcer/src/resourcer.ts
ChengLei Shao 24ea83f0ff
Feat/create nocobase app (#273)
* create-nocobase-app template from [develop]

* change create-nocobase-app package.json config

* feat: load configuration from directory

* feat: configuration repository toObject

* feat: create application from configuration dir

* feat: application factory with plugins options

* export type

* feat: read application config &  application with plugins options

* feat: release command

* fix: database release

* chore: workflow package.json

* feat: nocobase cli package

* feat: console command

* chore: load application in command

* fix: load packages from process.cwd

* feat: cli load env file

* feat: create-nocobase-app

* fix: gitignore create-nocobase-app lib

* fix: sqlite path

* feat: create plugin

* chore: plugin files template

* chore: move cli into application

* chore: create-nocobase-app

* fix: create plugin

* chore: app-client && app-server

* chore: package.json

* feat: create-nocobase-app download template from npm

* chore: create-nocobase-app template

* fix: config of plugin-users

* fix: yarn.lock

* fix: database build error

* fix: yarn.lock

* fix: resourcer config

* chore: cross-env

* chore: app-client dependents

* fix: env

* chore: v0.6.0-alpha.1

* chore: verdaccio

* chore(versions): 😊 publish v0.6.0

* chore(versions): 😊 publish v0.6.1-alpha.0

* chore(versions): 😊 publish v0.6.2-alpha.0

* chore(versions): 😊 publish v0.6.2-alpha.1

* chore: 0.6.2-alpha.2

* feat: workspaces

* chore(versions): 😊 publish v0.6.2-alpha.3

* chore(versions): 😊 publish v0.6.2-alpha.4

* chore: create-nocobase-app

* chore: create-nocobase-app lib

* fix: update tsconfig.jest.json

* chore: .env

* chore(versions): 😊 publish v0.6.2-alpha.5

* chore(versions): 😊 publish v0.6.2-alpha.6

* feat: improve code

* chore(versions): 😊 publish v0.6.2-alpha.7

* fix: cleanup

* chore(versions): 😊 publish v0.6.2-alpha.8

* chore: tsconfig for app server package

* fix: move files

* fix: move files

Co-authored-by: chenos <chenlinxh@gmail.com>
2022-04-17 10:00:42 +08:00

356 lines
8.3 KiB
TypeScript

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, requireModule } 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 {
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 = [];
public readonly options: ResourcerOptions;
constructor(options: ResourcerOptions = {}) {
this.options = options;
}
/**
* 载入指定目录下的 resource 配置(配置的文件驱动)
*
* TODO: 配置的文件驱动现在会全部初始化,大数据时可能存在性能瓶颈,后续可以加入动态加载
*
* @param {object} [options]
* @param {string} [options.directory] 指定配置所在路径
* @param {array} [options.extensions = ['js', 'ts', 'json']] 文件后缀
*/
public import(options: ImportOptions): 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>();
files.forEach((file: string) => {
const options = requireModule(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);
}
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;
}
use(middlewares: HandlerType | HandlerType[]) {
if (typeof middlewares === 'function') {
this.middlewares.push(middlewares);
} else if (Array.isArray(middlewares)) {
this.middlewares.push(...middlewares);
}
}
restApiMiddleware(options: KoaMiddlewareOptions = {}) {
const { prefix, accessors } = options;
const restApiMiddleware = 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();
}
};
return restApiMiddleware;
}
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;