nocobase/docs/reference/repository.md
2021-10-29 17:48:24 +08:00

13 KiB
Raw Blame History

toc
menu

Repository

repository.find()

查询数据,返回数组。无数据时为空数组,不返回 count。如果需要请使用 repository.findAndCount() 方法。

Definition
interface find<M extends Sequelize.Model> {
  (options?: FindOptions): Promise<M[]>
}

interface FindOptions extends Sequelize.FindOptions {
  // 数据过滤
  filter?: FilterOptions;
  // 输出结果显示哪些字段
  fields?: string[];
  // 输出结果不显示哪些字段
  expect?: string[];
  // 附加字段,用于控制关系字段的输出
  appends?: string[];
  // 排序,字段前面加上 “-” 表示降序
  sort?: string[];
}

// 待补充
type FilterOptions = any;
Examples
全览
await repository.find({
  // 过滤
  filter: {
    $and: [{ a: 5 }, { b: 6 }],            // (a = 5) AND (b = 6)
    $or: [{ a: 5 }, { b: 6 }],             // (a = 5) OR (b = 6)
    someAttribute: {
      // Basics
      $eq: 3,                              // = 3
      $ne: 20,                             // != 20
      $is: null,                           // IS NULL
      $not: true,                          // IS NOT TRUE
      $gt: 6,                              // > 6
      $gte: 6,
    },
    // 支持使用逗号间隔
    'someAttribute.$eq': 3,
    // 内嵌的,一般是关系数据
    nested: {
      someAttribute: {},
    },
    // 同上,也支持使用逗号间隔
    'nested.someAttribute': {
      // 同上
    },
  },
  // 字段白名单
  fields: [],
  // 附加字段,主要用于附加关系字段
  appends: [],
  // 字段黑名单
  expect: [],
  // 排序
  sort: ['-createdAt', 'updatedAt'],
});
filter 参数示例说明

以文章和标签为例,文章和标签的 collection 如下:

const Tag = db.collection({
  name: 'tags',
  fields: [
    { type: 'string', name: 'name' },
  ],
});

const Post = db.collection({
  name: 'posts',
  fields: [
    { type: 'string', name: 'name' },
    { type: 'text', name: 'content' },
    { type: 'belongsToMany', name: 'tags' },
  ],
});

最简单的筛选过滤

await Post.repository.find({
  filter: {
    'name': 'post1',
  },
});

支持多种 Operators$ 开头。更多内容,查阅 Operators 章节

await Post.repository.find({
  filter: {
    'name.$includes': 'post1',
    // 等同于
    name: {
      $includes: 'post1',
    },
  },
});

支持关系字段过滤,可以使用 dot 来表示层级结构

await Post.repository.find({
  filter: {
    'tags.name': 'tag1',
    // 等同于
    tags: {
      name: 'tag1',
    },
  },
});

多个同一关系字段的过滤可以写在一起

await Post.repository.find({
  filter: {
    'tags': {
      'name.$includes': 'tag1',
      'createdAt.$dateOn': '2020-10-28',
    },
  },
});

同时也支持 and、or 逻辑运算符

await Post.repository.find({
  filter: {
    $and: [
      // 一个 Object 只写一个条件
      { name: 'post1' },
      // 支持关系字段(非常重要)
      { 'tags.name.$includes': 'tag1' },
      { 'tags.name.$includes': 'tag2' },
    ],
  },
});
sort 参数示例说明

指定一组数据的排序,倒序时在字段前加上减号 -

await Post.repository.find({
  // 创建日期倒序
  sort: ['-createdAt'],
});

可以设置多个排序规则

await Post.repository.find({
  // 创建日期倒序ID 正序
  sort: ['-createdAt', 'id'],
});

也可以是关系表的字段

await Post.repository.find({
  // 标签名正序,文章创建日期倒序
  sort: ['tags.name', '-createdAt'],
});
fields 参数示例说明
  • fields 显示哪些字段
  • expect 不显示哪些字段
  • appends 附加哪些字段

如果并未指定 fields输出所有 AttributesAssociations 字段并不输出

await Post.repository.find();
// [{ id, name, content, createdAt, updatedAt }]

只输出指定字段时,可以用 fields

await Post.repository.find({
  fields: ['name'],
});
// [{ name }]

当 fields 里有关系字段时,按默认情况输出

await Post.repository.find({
  fields: ['id', 'name', 'tags'],
});
// 
// [{ id, name, createdAt, updatedAt, tags: [{ id, name, createdAt, updatedAt }] }]

可以只输出关系数据的某个字段

await Post.repository.find({
  fields: ['id', 'name', 'tags.name'],
});
// [{ id, name, tags: [{ name }] }]

排除某些字段时,可以使用 expect

await Post.repository.find({
  expect: ['content'],
});
// [{ id, name, createdAt, updatedAt }]

Attributes 不变,只附加 Associations 进来时,可以使用 appends

await Post.repository.find({
  appends: ['tags'],
});
// [{ id, name, content, createdAt, updatedAt, tags: [{ id, name, createdAt, updatedAt }] }]

如果某个字段只用在 filter 里,但并没有出现在 fields 里,不应该被输出

await Post.repository.find({
  filter: {
    'tags.name': 'tag1',
  },
});
// 输出所有的 Attributes但不输出 tags
// [{ id, name, content, createdAt, updatedAt }]

如果某个字段只用在 sort 里,但并没有出现在 fields 里,也不应该被输出

await Post.repository.find({
  sort: ['-tags.createdAt']
});
// 输出所有的 Attributes但不输出 tags
// [{ id, name, content, createdAt, updatedAt }]

repository.findAndCount()

按分页查询数据,并返回所有符合的数据总数。

Definition
interface findAndCount<M extends Sequelize.Model> {
  (options?: FindAndCountOptions): Promise<[ M[], number ]>
}

interface FindAndCountOptions extends Sequelize.FindAndCountOptions {
  // 数据过滤
  filter?: FilterOptions;
  // 输出结果显示哪些字段
  fields?: string[];
  // 输出结果不显示哪些字段
  expect?: string[];
  // 附加字段,用于控制关系字段的输出
  appends?: string[];
  // 排序,字段前面加上 “-” 表示降序
  sort?: string[];
  // 当前页,默认为 1
  page?: number;
  // 当前页最大数量,默认为 20
  pageSize?: number;
}
Examples

大部分参数与 repository.find() 一致,所以这里只列举 page 和 pageSize 的例子。

不填写参数时,默认 page=1pageSize=20。

await repository.findAndCount();
// [[{ id, name, content, createdAt, updatedAt }], 50]

const [models, count] = await repository.findAndCount();

指定页码和单页最大数量

await repository.findAndCount({
  page: 1,
  pageSize: 50,
});
// [[{ id, name, content, createdAt, updatedAt }], 50]

repository.findOne()

Definition
interface findOne<M extends Sequelize.Model> {
  (options?: FindOneOptions): Promise<[ M[], number ]>
}

interface FindOneOptions extends findOptions {
  // 数据过滤
  filter?: FilterOptions;
  // 输出结果显示哪些字段
  fields?: string[];
  // 输出结果不显示哪些字段
  expect?: string[];
  // 附加字段,用于控制关系字段的输出
  appends?: string[];
  // 排序,字段前面加上 “-” 表示降序
  sort?: string[];
  // 通过 pk 过滤
  filterByPk?: number | string;
}
Examples

大部分参数与 repository.find() 一致。这里只列举 filterByPk 的例子。

await repository.findOne({
  filterByPk: 1,
  // 等同于
  filter: {
    [Model.primaryKeyAttribute]: 1,
  }
});

repository.create()

创建数据

Definition
interface create<M extends Sequelize.Model> {
  (options?: CreateOptions): Promise<M>
}

interface CreateOptions {
  // 数据
  values?: any;
  // 字段白名单
  whitelist?: string[];
  // 字段黑名单
  blacklist?: string[];
  // 关系数据默认会新建并建立关联处理,如果是已存在的数据只关联,但不更新关系数据
  // 如果需要更新关联数据,可以通过 updateAssociations 指定
  updateAssociations?: string[];
}
Examples

例子之前,先定义几个 Collection 吧

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

db.collection({
  name: 'posts',
  fields: [
    { type: 'string', name: 'name' },
    { type: 'belongsTo', name: 'user' },
    { type: 'belongsToMany', name: 'tags' },
    { type: 'hasMany', name: 'comments' },
  ],
});

db.collection({
  name: 'tags',
  fields: [
    { type: 'string', name: 'name' },
    { type: 'belongsToMany', name: 'posts' },
  ],
});

db.collection({
  name: 'comments',
  fields: [
    { type: 'string', name: 'name' },
  ],
});

新建数据,可以不指定 values。

const model = await repository.create();

values 必须是 Object。

await repository.create({
  values: {
    name: 'post1',
  },
});

values 可以包含关系数据

await repository.create({
  values: {
    name: 'post1',
    // 新建并关联
    tags: [{ name: 'tag1' }],
  },
});

关联字段,可以只提供关系约束值(一般是 id

await repository.create({
  values: {
    name: 'post1',
    // 关联 id=1 的 tag
    tags: [1],
    // 也可以这样,会自动识别
    tags: 1,
  },
});

可以设置 values 的白名单和黑名单

await repository.create({
  values: {
    name: 'post1',
    // 关联 id=1 的 tag
    tags: [1],
  },
  whitelist: ['name'],
});

指定哪些关联字段在建立关联的同时可以更新数据

// 原 tag=1 的数据
// { id: 1, name: 'tag1' }

await repository.create({
  values: {
    name: 'post1',
    tags: [{
      id: 1,
      name: 'tag123', // name 与原数据不一样
    }],
  },
  // 指定了 tags建立关联时也会同步修改 tag 数据
  updateAssociations: ['tags'],
});

全览

await repository.create({
  // 待存数据
  values: {
    a: 'a',
    // 快速建立关联
    o2o: 1,    // 建立一对一关联
    m2o: 1,    // 建立多对一关联
    o2m: [1,2] // 建立一对多关联
    m2m: [1,2] // 建立多对多关联
    // 新建关联数据并建立关联
    o2o: {
      key1: 'val1',
    },
    o2m: [{key1: 'val1'}, {key2: 'val2'}],
    // 子表格数据
    subTable: [
      // 如果数据存在,更新处理
      {id: 1, key1: 'val1111'},
      // 如果数据不存在,直接创建并关联
      {key2: 'val2'},
    ],
  },
  // 字段白名单
  whitelist: [],
  // 字段黑名单
  blacklist: [],
  // 关系数据默认会新建并建立关联处理,如果是已存在的数据只关联,但不更新关系数据
  // 如果需要更新关联数据,可以通过 updateAssociations 指定
  updateAssociations: ['subTable'],
});

repository.update()

更新数据

Definition
interface update<M extends Sequelize.Model> {
  (options: UpdateOptions): Promise<M>
}

interface UpdateOptions {
  filter?: any;
  filterByPk?: number | string;
  // 数据
  values?: any;
  // 字段白名单
  whitelist?: string[];
  // 字段黑名单
  blacklist?: string[];
  // 关系数据默认会新建并建立关联处理,如果是已存在的数据只关联,但不更新关系数据
  // 如果需要更新关联数据,可以通过 updateAssociations 指定
  updateAssociations?: string[];
}
Examples

全部改

await repository.update({
  values: {
    a: 'b'
  },
});

只改某条数据

await repository.update({
  filterByPk: 1,
  values: {
    a: 'b'
  },
});

指定范围修改

await repository.update({
  filter: {
    name: 'post1.1'
  },
  values: {
    name: 'post1.2'
  },
});

repository.destroy()

删除数据

Definition
interface destroy {
  (options?: number | string | number[] | string[] | DestroyOptions): Promise<boolean | boolean[]>
}

interface DestroyOptions {
  filter?: any;
}
Examples

指定 primary key 值

repository.destroy(1);

批量 primary key 值

repository.destroy([1, 2, 3]);

复杂的 filter

repository.destroy({
  filter: {},
});

repository.relation().of()

Definition
interface relation {
  (name: string): {
    of: (parent: any): RelationRepository;
  }
}

// 关系数据的增删改查在 NocoBase 里非常重要
class RelationRepository {
  find() {}
  findOne() {}
  create() {}
  update() {}
  destroy() {}
  set() {}
  add() {}
  remove() {}
  toggle() {}
}
Examples

find、findOne、create、update 和 destroy 和常规 Repository API 层面是一致,这里重点列举关联操作的几个方法:

// user_id = 1 的 post 的 relatedQuery
const userPostsRepository = repository.relation('posts').of(1);

// 建立关联
userPostsRepository.set(1);

// 批量,仅用于 HasMany 和 BelongsToMany
userPostsRepository.set([1,2,3]);

// BelongsToMany 的中间表
userPostsRepository.set([
  [1, {/* 中间表数据 */}],
  [2, {/* 中间表数据 */}],
  [3, {/* 中间表数据 */}],
]);

// 仅用于 HasMany 和 BelongsToMany
userPostsRepository.add(1);

// BelongsToMany 的中间表
userPostsRepository.add(1, {/* 中间表数据 */});

// 删除关联
userPostsRepository.remove(1);

// 建立或解除
userPostsRepository.toggle(1);
userPostsRepository.toggle([1, 2, 3]);