mirror of
https://github.com/nocobase/nocobase
synced 2024-11-16 07:27:04 +00:00
24ea83f0ff
* 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>
356 lines
8.3 KiB
TypeScript
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;
|