nocobase/docs/index.md
2021-09-12 22:33:19 +08:00

15 KiB
Raw Blame History

title toc
基础概念 menu

NocoBase

考虑到大家是初次接触 NocoBase开发文档的第一篇从宏观的角度带大家了解 NocoBase 的基础概念。NocoBase 采用微内核的架构,框架只保留核心概念,各类功能都以插件形式扩展。

微服务 - Microservices

先来个例子一睹为快,新建一个 server.js 文件,代码如下:

const Server = require('@nocobase/server');

const server = new Server();

server.collection({
  name: 'users',
  fields: [
    { type: 'string', name: 'username' },
    { type: 'password', name: 'password' },
  ],
});

server.start(process.argv);

终端运行

# 根据配置生成数据库表结构
node server.js db sync
# 启动应用
node server.js start --port=3000

相关 users 表的 REST API 就生成了

GET     http://localhost:3000/api/users
POST    http://localhost:3000/api/users
GET     http://localhost:3000/api/users/1
PUT     http://localhost:3000/api/users/1
DELETE  http://localhost:3000/api/users/1

除了内置的 REST API 以外,还可以自定义其他操作,如登录、注册、注销等。

server.registerActions({
  async login(ctx, next) {},
  async register(ctx, next) {},
  async logout(ctx, next) {},
}, {
  resourceName: 'users',
});

以上操作的 HTTP API 为:

POST    http://localhost:3000/api/users:login
POST    http://localhost:3000/api/users:register
POST    http://localhost:3000/api/users:logout

自定义的 HTTP API 依旧保持 REST API 的风格,以 <resourceName>:<actionName> 格式表示。REST API 也可以显式指定 actionName当指定了 actionName 时,无所谓使用什么 Request Method

GET     http://localhost:3000/api/users:list
POST    http://localhost:3000/api/users:create
GET     http://localhost:3000/api/users:get/1
POST    http://localhost:3000/api/users:update/1
POST    http://localhost:3000/api/users:destroy/1

结合客户端 SDK 是这样的:

const { ClientSDK } = require('@nocobase/client');

const client = new ClientSDK();

await client.resource('users').list();
await client.resource('users').create();
await client.resource('users').get();
await client.resource('users').update();
await client.resource('users').destroy();
await client.resource('users').login();
await client.resource('users').register();
await client.resource('users').logout();

数据集 - Collection

上述例子,通过 server.collection() 方法定义数据的 SchemaSchema 的核心为字段配置,字段类型包括:

Attribute 属性

  • Boolean 布尔型
  • String 字符串
  • Text 长文本
  • Integer 整数型
  • Float 浮点型
  • Decimal 货币
  • Json/Jsonb/Array 不同数据库的 JSON 类型不一致,存在兼容性问题
  • Time 时间
  • Date 日期
  • Virtual 虚拟字段
  • Reference 引用
  • Formula 计算公式
  • Context 上下文
  • Password 密码
  • Sort 排序

Association/Realtion 关系

  • HasOne 一对一
  • HasMany 一对多
  • BelongsTo 多对一
  • BelongsToMany 多对多
  • Polymorphic 多态

比如一个微型博客的表结构可以这样设计:

// 用户
server.collection({
  name: 'users',
  fields: [
    { type: 'string', name: 'username', unique: true },
    { type: 'password', name: 'password', unique: true },
    { type: 'hasMany', name: 'posts' },
  ],
});

// 文章
server.collection({
  name: 'posts',
  fields: [
    { type: 'string', name: 'title' },
    { type: 'text', name: 'content' },
    { type: 'belongsToMany', name: 'tags' },
    { type: 'hasMany', name: 'comments' },
    { type: 'belongsTo', name: 'author', target: 'users' },
  ],
});

// 标签
server.collection({
  name: 'tags',
  fields: [
    { type: 'string', name: 'name' },
    { type: 'belongsToMany', name: 'posts' },
  ],
});

// 评论
server.collection({
  name: 'comments',
  fields: [
    { type: 'text', name: 'content' },
    { type: 'belongsTo', name: 'user' },
  ],
});

存在外键关联时,也无需顾虑建表和字段的顺序,db sync 时会自动处理。为了方便开发,提供了一些有用的属性或方法:

  • server.db 数据库实例
  • server.db.getModel() 获取 Model
  • server.db.getTable() 获取 Schema Table

资源 & 操作 - Resource & Action

不同于常规的 MVC + RouterNocoBase 的路由Resourcer基于资源Resource和操作Action设计将 REST 和 RPC 结合起来,提供更为灵活且统一的 Resource Action APIAction 不局限于增删改查。资源可以通过 server.resource 方法定义,如:

server.resource({
  name: 'users',
  actions: {
    list: {
      fields: ['id', 'username'], // 只输出 id 和 username 字段
      filter: {
        'username.$ne': 'admin', // 数据范围筛选过滤 username != admin
      },
      sort: ['-created_at'], // 创建时间倒序
      perPage: 50,
    },
    get: {
      fields: ['id', 'username'], // 只输出 id 和 username 字段
      filter: {
        'username.$ne': 'admin', // 数据范围筛选过滤 username != admin
      },
    },
    create: {
      fields: ['username'], // 白名单
    },
    update: {
      fields: ['username'], // 白名单
    },
    destroy: {
      filter: { // 不能删除 admin
        'username.$ne': 'admin',
      },
    },
  },
});

server.collection()server.resource() 的区别?

  • server.collection() 定义数据的 Schema结构和关系
  • server.resource() 定义数据的 Action操作方法

一般情况无需显式声明 collection 的 resource因为已定义的 collection 会自动同步给 resource。

事件 - Event

在操作执行前、后都放置了相关的事件监听器,可以通过 server.db.onserver.on 添加。区别在于:

  • server.db.on 添加数据库的监听器
  • server.on 添加服务器的监听器

users:login 为例,在数据库里为「查询」操作,在服务器里为「登录」操作。如果需要记录登录操作日志,需要在 server.on 里处理。

// 创建数据时,执行 User.create() 时触发
server.db.on('users:beforeCreate', async (model) => {});

// 客户端 `POST /api/users:login` 时触发
server.on('users:beforeLogin', async (ctx, next) => {});

// 客户端 `POST /api/users` 时触发
server.on('users:beforeCreate', async (ctx, next) => {});

中间件 - Middleware

Server 基于 Koa所有 Koa 的插件(中间件)都可以直接使用,可以通过 server.use 添加。如:

server.use(async (ctx, next) => {
  const token = ctx.get('Authorization').replace(/^Bearer\s+/gi, '');
  if (token !== '123456') {
    return ctx.throw(401, 'Unauthorized');
  }
  return next();
});

弥补 koa.use 的不足,提供了更完善的 middleware 适配器

import { middleware } from '@nocobase/server';

server.use(middleware(async (ctx, next) => {}, {
  name: 'middlewareName1',
  resourceNames: [],
  actionNames: [],
  insertBefore: '',
  insertAfter: '',
}));

命令行 - CLI

除此之外Server 还集成了 commander可用于 cli 场景。目前内置的有:

  • db sync --force 用于配置与数据库表结构同步
  • start --port 启动应用

自定义:

server.command('foo').action(async () => {
  console.log('foo...');
});

插件 - Plugin

上文,讲述了核心的扩展接口,包括但不局限于:

  • Database/Collection
    • server.db database 实例
    • server.collection() 等同于 server.db.table()
  • Resource/Action
    • server.resource() 等同于 server.resourcer.define()
    • server.registerActions() 等同于 server.resourcer.registerActions()
  • Hook/Event
    • server.on() 添加服务器监听器
    • server.db.on() 添加数据库监听器
  • Middleware
    • server.use() 添加中间件
  • CLI
    • server.cli commander 实例
    • server.command() 等同于 server.cli.command()
  • Plugin
    • server.pluginManager 插件管理器
    • server.plugin 等同于 server.pluginManager.add()

基于以上扩展接口,提供模块化、可插拔的插件,可以通过 server.plugin() 添加。

最简单的插件

server.plugin(function pluginName1() {

});

JSON 风格

包括安装、激活、载入、禁用、卸载流程的配置

server.plugin({
  async install() {},
  async activate() {},
  async bootstrap() {},
  async deactivate() {},
  async unstall() {},
}, {
  name: 'pluginName1',
  displayName: '插件名称',
  version: '1.2.3',
  dependencies: {
    pluginName2: '1.x', 
    pluginName3: '1.x',
  },
});

OOP 风格

class MyPlugin extends Plugin {
  async bootstrap() {}
  async install() {}
  async unstall() {}
  async activate() {}
  async deactivate() {}
}

server.plugin(MyPlugin, {
  name: 'pluginName1',
  displayName: '插件名称',
  version: '1.2.3',
  dependencies: {
    pluginName2: '1.x', 
    pluginName3: '1.x',
  },
});

引用独立的 Package

server.plugin('@nocobase/plugin-action-logs');

插件信息也可以直接写在 package.json

{
  name: 'pluginName1',
  displayName: '插件名称',
  version: '1.2.3',
  dependencies: {
    pluginName2: '1.x', 
    pluginName3: '1.x',
  },
}

通过 server.plugin() 添加的插件需要激活才能使用

await server.pluginManager.activate(['pluginName1', 'pluginName2']);

插件 CLI

plugin install pluginName1
plugin unstall pluginName1
plugin activate pluginName1
plugin deactivate pluginName1

目前已有的插件:

  • @nocobase/plugin-collections 提供数据表配置接口,可通过 HTTP API 管理数据表。
  • @nocobase/plugin-action-logs 操作日志
  • @nocobase/plugin-automations 自动化(未升级 v0.5,暂不能使用)
  • @nocobase/plugin-china-region 中国行政区
  • @nocobase/plugin-client 提供客户端,无代码的可视化配置界面,需要与 @nocobase/client 配合使用
  • @nocobase/plugin-export 导出
  • @nocobase/plugin-file-manager 文件管理器
  • @nocobase/plugin-permissions 角色和权限
  • @nocobase/plugin-system-settings 系统配置
  • @nocobase/plugin-ui-router 前端路由配置
  • @nocobase/plugin-ui-schema ui 配置
  • @nocobase/plugin-users 用户模块

测试 - Testing

有代码就需要测试,@nocobase/test 提供了 mockDatabase 和 mockServer 用于数据库和服务器的测试,如:

import { mockServer, MockServer } from '@nocobase/test';

describe('mock server', () => {
  let api: MockServer;

  beforeEach(() => {
    api = mockServer({
      dataWrapping: false,
    });
    api.registerActions({
      list: async (ctx, next) => {
        ctx.body = [1, 2];
        await next();
      },
    });
    api.resource({
      name: 'test',
    });
  });

  afterEach(async () => {
    return api.destroy();
  });

  it('agent.get', async () => {
    const response = await api.agent().get('/test');
    expect(response.body).toEqual([1, 2]);
  });

  it('agent.resource', async () => {
    const response = await api.agent().resource('test').list();
    expect(response.body).toEqual([1, 2]);
  });
});

客户端 - Client

为了让更多非开发人员也能参与进来NocoBase 提供了配套的客户端插件 —— 无代码的可视化配置界面。客户端插件需要与 @nocobase/client 配合使用,可以直接使用,也可以自行改造。

插件配置

server.plugin('@nocobase/plugin-client', {
  // 自定义 dist 路径
  dist: path.resolve(__dirname, './node_modules/@nocobase/client/app'),
});

为了满足各类场景需求,客户端 @nocobase/client 提供了丰富的基础组件:

  • Action - 操作
    • Action.Window 当前浏览器窗口/标签里打开
    • Action.Drawer 打开抽屉(默认右侧划出)
    • Action.Modal 打开对话框
    • Action.Dropdown 下拉菜单
    • Action.Popover 气泡卡片
    • Action.Group 按钮分组
    • Action.Bar 操作栏
  • AddNew 「添加」模块
    • AddNew.CardItem - 添加区块
    • AddNew.PaneItem - 添加区块(查看面板,与当前查看的数据相关)
    • AddNew.FormItem - 添加字段
  • BlockItem/CardItem/FormItem - 装饰器
    • BlockItem - 普通装饰器(无包装效果)
    • CardItem - 卡片装饰器
    • FormItem - 字段装饰器
  • Calendar - 日历
  • Cascader - 级联选择
  • Chart - 图表
  • Checkbox - 勾选
  • Checkbox.Group - 多选框
  • Collection - 数据表配置
  • Collection.Field - 数据表字段
  • ColorSelect - 颜色选择器
  • DatePicker - 日期选择器
  • DesignableBar - 配置工具栏
  • Filter - 筛选器
  • Form - 表单
  • Grid - 栅格布局
  • IconPicker - 图标选择器
  • Input - 输入框
  • Input.TextArea - 多行输入框
  • InputNumber - 数字框
  • Kanban - 看板
  • ListPicker - 列表选择器(用于选择、展示关联数据)
  • Markdown 编辑器
  • Menu - 菜单
  • Password - 密码
  • Radio - 单选框
  • Select - 选择器
  • Table - 表格
  • Tabs - 标签页
  • TimePicker - 时间选择器
  • Upload - 上传

开发们也可以自行扩展,以上组件基于 Formily 构建,怎么自定义组件大家查看相关组件源码或 Formily 文档吧,这里说点不一样的。

  • 如何扩展数据库字段?
  • 如何将第三方区块添加到 AddNew 模块中?
  • 如何在操作栏里添加更多的内置操作?
  • 如何自定义配置工具栏?

除了组件具备灵活的扩展以外,客户端也可以在任意前端框架中使用,可以自定义 Request 和 Router

import React from 'react';
import { MemoryRouter } from 'react-router-dom';
import { ClientSDK, Application } from '@nocobase/client';

// 初始化 client 实例
const client = new ClientSDK({
  request: (options) => Promise.resolve({}),
});

// 适配 Route Component
const RouteSwitch = createRouteSwitch({
  components: {
    AdminLayout,
    AuthLayout,
    RouteSchemaRenderer,
  },
});

ReactDOM.render(
  
    
      
    
  ,
  document.getElementById('root'),
);

更多细节,可以通过 create-nocobase-app 初始化项目脚手架并体验。

yarn create nocobase-app my-nocobase-project

nocobase-app 默认使用 umijs 作为项目构建工具,并集成了 Server 作数据接口,初始化的目录结构如下:

|- src
  |- pages
  |- apis
|- .env
|- .umirc.ts
|- package.json

应用场景

常规管理系统(包括后台数据管理)

一套完整的 WEB 服务,包括前后端

API 服务

无需客户端,提供纯后端接口,也同样适用于 Open API

小程序 + 后台管理

只需要一套数据库,但有两套用户和权限,一套是用于后台用户,一套用于小程序用户

SaaS 服务(共享用户)

每个应用有自己配套的数据库,各应用数据完全隔离。应用不需要用户和权限模块,因为 SaaS 主站全局共享了。

SaaS 服务(独立用户)

每个应用有自己的独立用户模块和权限