UNPKG

nukak

Version:

flexible and efficient ORM, with declarative JSON syntax and smart type-safety

278 lines 41.1 kB
import { __decorate, __metadata } from "tslib"; import { getMeta } from '../entity/decorator/index.js'; import { GenericRepository } from '../repository/index.js'; import { augmentWhere, clone, filterPersistableRelationKeys, filterRelationKeys, getKeys } from '../util/index.js'; import { Serialized } from './decorator/index.js'; /** * Base class for all database queriers. * It provides a standardized way to execute tasks serially to prevent race conditions on database connections. */ export class AbstractQuerier { /** * Internal promise used to queue database operations. * This ensures that each operation is executed serially, preventing race conditions * and ensuring that the database connection is used safely across concurrent calls. */ taskQueue = Promise.resolve(); findOneById(entity, id, q = {}) { const meta = getMeta(entity); q.$where = augmentWhere(meta, q.$where, id); return this.findOne(entity, q); } async findOne(entity, q) { const rows = await this.findMany(entity, { ...q, $limit: 1 }); return rows[0]; } findManyAndCount(entity, q) { const qCount = { ...q, }; delete qCount.$sort; delete qCount.$limit; delete qCount.$skip; return Promise.all([this.findMany(entity, q), this.count(entity, qCount)]); } async insertOne(entity, payload) { const [id] = await this.insertMany(entity, [payload]); return id; } updateOneById(entity, id, payload) { return this.updateMany(entity, { $where: id }, payload); } deleteOneById(entity, id, opts) { return this.deleteMany(entity, { $where: id }, opts); } async saveOne(entity, payload) { const [id] = await this.saveMany(entity, [payload]); return id; } async saveMany(entity, payload) { const meta = getMeta(entity); const ids = []; const updates = []; const inserts = []; for (const it of payload) { if (it[meta.id]) { if (getKeys(it).length === 1) { ids.push(it[meta.id]); } else { updates.push(it); } } else { inserts.push(it); } } return Promise.all([ ...ids, ...(inserts.length ? await this.insertMany(entity, inserts) : []), ...updates.map(async (it) => { const { [meta.id]: id, ...data } = it; await this.updateOneById(entity, id, data); return id; }), ]); } async fillToManyRelations(entity, payload, select) { if (!payload.length) { return; } const meta = getMeta(entity); const relKeys = filterRelationKeys(meta, select); for (const relKey of relKeys) { const relOpts = meta.relations[relKey]; const relEntity = relOpts.entity(); const relSelect = clone(select[relKey]); const relQuery = relSelect === true || relSelect === undefined ? {} : Array.isArray(relSelect) ? { $select: relSelect } : relSelect; const ids = payload.map((it) => it[meta.id]); if (relOpts.through) { const localField = relOpts.references[0].local; const throughEntity = relOpts.through(); const throughMeta = getMeta(throughEntity); const targetRelKey = getKeys(throughMeta.relations).find((key) => throughMeta.relations[key].references.some(({ local }) => local === relOpts.references[1].local)); const throughFounds = await this.findMany(throughEntity, { ...relQuery, $select: { [localField]: true, [targetRelKey]: { ...relQuery, $required: true, }, }, $where: { ...relQuery.$where, [localField]: ids, }, }); const founds = throughFounds.map((it) => ({ ...it[targetRelKey], [localField]: it[localField] })); this.putChildrenInParents(payload, founds, meta.id, localField, relKey); } else if (relOpts.cardinality === '1m') { const foreignField = relOpts.references[0].foreign; if (relQuery.$select) { if (Array.isArray(relQuery.$select)) { if (!relQuery.$select.includes(foreignField)) { relQuery.$select.push(foreignField); } } else if (!relQuery.$select[foreignField]) { relQuery.$select[foreignField] = true; } } relQuery.$where = { ...relQuery.$where, [foreignField]: ids }; const founds = await this.findMany(relEntity, relQuery); this.putChildrenInParents(payload, founds, meta.id, foreignField, relKey); } } } putChildrenInParents(parents, children, parentIdKey, referenceKey, relKey) { const childrenByParentMap = children.reduce((acc, child) => { const parenId = child[referenceKey]; if (!acc[parenId]) { acc[parenId] = []; } acc[parenId].push(child); return acc; }, {}); for (const parent of parents) { const parentId = parent[parentIdKey]; parent[relKey] = childrenByParentMap[parentId]; } } async insertRelations(entity, payload) { const meta = getMeta(entity); await Promise.all(payload.map((it) => { const relKeys = filterPersistableRelationKeys(meta, it, 'persist'); if (!relKeys.length) { return Promise.resolve(); } return Promise.all(relKeys.map((relKey) => this.saveRelation(entity, it, relKey))); })); } async updateRelations(entity, q, payload) { const meta = getMeta(entity); const relKeys = filterPersistableRelationKeys(meta, payload, 'persist'); if (!relKeys.length) { return; } const founds = await this.findMany(entity, { ...q, $select: [meta.id] }); const ids = founds.map((found) => found[meta.id]); await Promise.all(ids.map((id) => Promise.all(relKeys.map((relKey) => this.saveRelation(entity, { ...payload, [meta.id]: id }, relKey, true))))); } async deleteRelations(entity, ids, opts) { const meta = getMeta(entity); const relKeys = filterPersistableRelationKeys(meta, meta.relations, 'delete'); for (const relKey of relKeys) { const relOpts = meta.relations[relKey]; const relEntity = relOpts.entity(); const localField = relOpts.references[0].local; if (relOpts.through) { const throughEntity = relOpts.through(); await this.deleteMany(throughEntity, { $where: { [localField]: ids } }, opts); return; } await this.deleteMany(relEntity, { [localField]: ids }, opts); } } async saveRelation(entity, payload, relKey, isUpdate) { const meta = getMeta(entity); const id = payload[meta.id]; const { entity: entityGetter, cardinality, references, through } = meta.relations[relKey]; const relEntity = entityGetter(); const relPayload = payload[relKey]; if (cardinality === '1m' || cardinality === 'mm') { if (through) { const localField = references[0].local; const throughEntity = through(); if (isUpdate) { await this.deleteMany(throughEntity, { $where: { [localField]: id } }); } if (relPayload) { const savedIds = await this.saveMany(relEntity, relPayload); const throughBodies = savedIds.map((relId) => ({ [references[0].local]: id, [references[1].local]: relId, })); await this.insertMany(throughEntity, throughBodies); } return; } const foreignField = references[0].foreign; if (isUpdate) { await this.deleteMany(relEntity, { $where: { [foreignField]: id } }); } if (relPayload) { for (const it of relPayload) { it[foreignField] = id; } await this.saveMany(relEntity, relPayload); } return; } if (cardinality === '11') { const foreignField = references[0].foreign; if (relPayload === null) { await this.deleteMany(relEntity, { $where: { [foreignField]: id } }); return; } await this.saveOne(relEntity, { ...relPayload, [foreignField]: id }); return; } if (cardinality === 'm1' && relPayload) { const localField = references[0].local; const referenceId = await this.insertOne(relEntity, relPayload); await this.updateOneById(entity, id, { [localField]: referenceId }); return; } } getRepository(entity) { return new GenericRepository(entity, this); } async transaction(callback) { try { await this.beginTransaction(); const res = await callback(); await this.commitTransaction(); return res; } catch (err) { await this.rollbackTransaction(); throw err; } finally { await this.release(); } } async releaseIfFree() { if (!this.hasOpenTransaction) { await this.internalRelease(); } } /** * Schedules a task to be executed serially in the querier instance. * This is used by the @Serialized decorator to protect database-level operations. * * @param task - The async task to execute. * @returns A promise that resolves with the task's result. */ async serialize(task) { const res = this.taskQueue.then(task); this.taskQueue = res.catch(() => { }); return res; } async release() { return this.internalRelease(); } } __decorate([ Serialized(), __metadata("design:type", Function), __metadata("design:paramtypes", []), __metadata("design:returntype", Promise) ], AbstractQuerier.prototype, "release", null); //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"abstractQuerier.js","sourceRoot":"","sources":["../../src/querier/abstractQuerier.ts"],"names":[],"mappings":"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","sourcesContent":["import { getMeta } from '../entity/decorator/index.js';\nimport { GenericRepository } from '../repository/index.js';\nimport type {\n  IdValue,\n  Key,\n  Querier,\n  Query,\n  QueryConflictPaths,\n  QueryOne,\n  QueryOptions,\n  QuerySearch,\n  QuerySelect,\n  QueryUpdateResult,\n  RelationKey,\n  RelationValue,\n  Repository,\n  Type,\n} from '../type/index.js';\nimport { augmentWhere, clone, filterPersistableRelationKeys, filterRelationKeys, getKeys } from '../util/index.js';\nimport { Serialized } from './decorator/index.js';\n\n/**\n * Base class for all database queriers.\n * It provides a standardized way to execute tasks serially to prevent race conditions on database connections.\n */\nexport abstract class AbstractQuerier implements Querier {\n  /**\n   * Internal promise used to queue database operations.\n   * This ensures that each operation is executed serially, preventing race conditions\n   * and ensuring that the database connection is used safely across concurrent calls.\n   */\n  private taskQueue: Promise<unknown> = Promise.resolve();\n\n  findOneById<E>(entity: Type<E>, id: IdValue<E>, q: QueryOne<E> = {}) {\n    const meta = getMeta(entity);\n    q.$where = augmentWhere(meta, q.$where, id);\n    return this.findOne(entity, q);\n  }\n\n  async findOne<E>(entity: Type<E>, q: QueryOne<E>) {\n    const rows = await this.findMany(entity, { ...q, $limit: 1 });\n    return rows[0];\n  }\n\n  abstract findMany<E>(entity: Type<E>, q: Query<E>): Promise<E[]>;\n\n  findManyAndCount<E>(entity: Type<E>, q: Query<E>) {\n    const qCount = {\n      ...q,\n    } satisfies QuerySearch<E>;\n    delete qCount.$sort;\n    delete qCount.$limit;\n    delete qCount.$skip;\n    return Promise.all([this.findMany(entity, q), this.count(entity, qCount)]);\n  }\n\n  abstract count<E>(entity: Type<E>, q: QuerySearch<E>): Promise<number>;\n\n  async insertOne<E>(entity: Type<E>, payload: E) {\n    const [id] = await this.insertMany(entity, [payload]);\n    return id;\n  }\n\n  abstract insertMany<E>(entity: Type<E>, payload: E[]): Promise<IdValue<E>[]>;\n\n  updateOneById<E>(entity: Type<E>, id: IdValue<E>, payload: E) {\n    return this.updateMany(entity, { $where: id }, payload);\n  }\n\n  abstract updateMany<E>(entity: Type<E>, q: QuerySearch<E>, payload: E): Promise<number>;\n\n  abstract upsertOne<E>(entity: Type<E>, conflictPaths: QueryConflictPaths<E>, payload: E): Promise<QueryUpdateResult>;\n\n  deleteOneById<E>(entity: Type<E>, id: IdValue<E>, opts?: QueryOptions) {\n    return this.deleteMany(entity, { $where: id }, opts);\n  }\n\n  abstract deleteMany<E>(entity: Type<E>, q: QuerySearch<E>, opts?: QueryOptions): Promise<number>;\n\n  async saveOne<E>(entity: Type<E>, payload: E) {\n    const [id] = await this.saveMany(entity, [payload]);\n    return id;\n  }\n\n  async saveMany<E>(entity: Type<E>, payload: E[]) {\n    const meta = getMeta(entity);\n    const ids: IdValue<E>[] = [];\n    const updates: E[] = [];\n    const inserts: E[] = [];\n\n    for (const it of payload) {\n      if (it[meta.id]) {\n        if (getKeys(it).length === 1) {\n          ids.push(it[meta.id]);\n        } else {\n          updates.push(it);\n        }\n      } else {\n        inserts.push(it);\n      }\n    }\n\n    return Promise.all([\n      ...ids,\n      ...(inserts.length ? await this.insertMany(entity, inserts) : []),\n      ...updates.map(async (it) => {\n        const { [meta.id]: id, ...data } = it;\n        await this.updateOneById(entity, id, data as E);\n        return id;\n      }),\n    ]);\n  }\n\n  protected async fillToManyRelations<E>(entity: Type<E>, payload: E[], select: QuerySelect<E>) {\n    if (!payload.length) {\n      return;\n    }\n\n    const meta = getMeta(entity);\n    const relKeys = filterRelationKeys(meta, select);\n\n    for (const relKey of relKeys) {\n      const relOpts = meta.relations[relKey];\n      const relEntity = relOpts.entity();\n      type RelEntity = typeof relEntity;\n      const relSelect = clone(select[relKey as string]);\n      const relQuery: Query<RelEntity> =\n        relSelect === true || relSelect === undefined\n          ? {}\n          : Array.isArray(relSelect)\n            ? { $select: relSelect }\n            : relSelect;\n      const ids = payload.map((it) => it[meta.id]);\n\n      if (relOpts.through) {\n        const localField = relOpts.references[0].local;\n        const throughEntity = relOpts.through();\n        const throughMeta = getMeta(throughEntity);\n        const targetRelKey = getKeys(throughMeta.relations).find((key) =>\n          throughMeta.relations[key].references.some(({ local }) => local === relOpts.references[1].local),\n        );\n        const throughFounds = await this.findMany(throughEntity, {\n          ...relQuery,\n          $select: {\n            [localField]: true,\n            [targetRelKey]: {\n              ...relQuery,\n              $required: true,\n            },\n          },\n          $where: {\n            ...relQuery.$where,\n            [localField]: ids,\n          },\n        });\n        const founds = throughFounds.map((it) => ({ ...it[targetRelKey], [localField]: it[localField] }));\n        this.putChildrenInParents(payload, founds, meta.id, localField, relKey);\n      } else if (relOpts.cardinality === '1m') {\n        const foreignField = relOpts.references[0].foreign;\n        if (relQuery.$select) {\n          if (Array.isArray(relQuery.$select)) {\n            if (!relQuery.$select.includes(foreignField as Key<RelEntity>)) {\n              relQuery.$select.push(foreignField as Key<RelEntity>);\n            }\n          } else if (!relQuery.$select[foreignField]) {\n            relQuery.$select[foreignField] = true;\n          }\n        }\n        relQuery.$where = { ...relQuery.$where, [foreignField]: ids };\n        const founds = await this.findMany(relEntity, relQuery);\n        this.putChildrenInParents(payload, founds, meta.id, foreignField, relKey);\n      }\n    }\n  }\n\n  protected putChildrenInParents<E>(\n    parents: E[],\n    children: E[],\n    parentIdKey: string,\n    referenceKey: string,\n    relKey: string,\n  ): void {\n    const childrenByParentMap = children.reduce((acc, child) => {\n      const parenId = child[referenceKey];\n      if (!acc[parenId]) {\n        acc[parenId] = [];\n      }\n      acc[parenId].push(child);\n      return acc;\n    }, {});\n\n    for (const parent of parents) {\n      const parentId = parent[parentIdKey];\n      parent[relKey] = childrenByParentMap[parentId];\n    }\n  }\n\n  protected async insertRelations<E>(entity: Type<E>, payload: E[]) {\n    const meta = getMeta(entity);\n    await Promise.all(\n      payload.map((it) => {\n        const relKeys = filterPersistableRelationKeys(meta, it, 'persist');\n        if (!relKeys.length) {\n          return Promise.resolve();\n        }\n        return Promise.all(relKeys.map((relKey) => this.saveRelation(entity, it, relKey)));\n      }),\n    );\n  }\n\n  protected async updateRelations<E>(entity: Type<E>, q: QuerySearch<E>, payload: E) {\n    const meta = getMeta(entity);\n    const relKeys = filterPersistableRelationKeys(meta, payload, 'persist');\n\n    if (!relKeys.length) {\n      return;\n    }\n\n    const founds = await this.findMany(entity, { ...q, $select: [meta.id] });\n    const ids = founds.map((found) => found[meta.id]);\n\n    await Promise.all(\n      ids.map((id) =>\n        Promise.all(relKeys.map((relKey) => this.saveRelation(entity, { ...payload, [meta.id]: id }, relKey, true))),\n      ),\n    );\n  }\n\n  protected async deleteRelations<E>(entity: Type<E>, ids: IdValue<E>[], opts?: QueryOptions) {\n    const meta = getMeta(entity);\n    const relKeys = filterPersistableRelationKeys(meta, meta.relations as E, 'delete');\n\n    for (const relKey of relKeys) {\n      const relOpts = meta.relations[relKey];\n      const relEntity = relOpts.entity();\n      const localField = relOpts.references[0].local;\n      if (relOpts.through) {\n        const throughEntity = relOpts.through();\n        await this.deleteMany(throughEntity, { $where: { [localField]: ids } }, opts);\n        return;\n      }\n      await this.deleteMany(relEntity, { [localField]: ids }, opts);\n    }\n  }\n\n  protected async saveRelation<E>(entity: Type<E>, payload: E, relKey: RelationKey<E>, isUpdate?: boolean) {\n    const meta = getMeta(entity);\n    const id = payload[meta.id];\n    const { entity: entityGetter, cardinality, references, through } = meta.relations[relKey];\n    const relEntity = entityGetter();\n    const relPayload = payload[relKey] as unknown as RelationValue<E>[];\n\n    if (cardinality === '1m' || cardinality === 'mm') {\n      if (through) {\n        const localField = references[0].local;\n\n        const throughEntity = through();\n        if (isUpdate) {\n          await this.deleteMany(throughEntity, { $where: { [localField]: id } });\n        }\n        if (relPayload) {\n          const savedIds = await this.saveMany(relEntity, relPayload);\n          const throughBodies = savedIds.map((relId) => ({\n            [references[0].local]: id,\n            [references[1].local]: relId,\n          }));\n          await this.insertMany(throughEntity, throughBodies);\n        }\n        return;\n      }\n      const foreignField = references[0].foreign;\n      if (isUpdate) {\n        await this.deleteMany(relEntity, { $where: { [foreignField]: id } });\n      }\n      if (relPayload) {\n        for (const it of relPayload) {\n          it[foreignField] = id;\n        }\n        await this.saveMany(relEntity, relPayload);\n      }\n      return;\n    }\n\n    if (cardinality === '11') {\n      const foreignField = references[0].foreign;\n      if (relPayload === null) {\n        await this.deleteMany(relEntity, { $where: { [foreignField]: id } });\n        return;\n      }\n      await this.saveOne(relEntity, { ...relPayload, [foreignField]: id });\n      return;\n    }\n\n    if (cardinality === 'm1' && relPayload) {\n      const localField = references[0].local;\n      const referenceId = await this.insertOne(relEntity, relPayload);\n      await this.updateOneById(entity, id, { [localField]: referenceId });\n      return;\n    }\n  }\n\n  getRepository<E>(entity: Type<E>): Repository<E> {\n    return new GenericRepository(entity, this);\n  }\n\n  abstract readonly hasOpenTransaction: boolean;\n\n  async transaction<T>(callback: () => Promise<T>) {\n    try {\n      await this.beginTransaction();\n      const res = await callback();\n      await this.commitTransaction();\n      return res;\n    } catch (err) {\n      await this.rollbackTransaction();\n      throw err;\n    } finally {\n      await this.release();\n    }\n  }\n\n  async releaseIfFree() {\n    if (!this.hasOpenTransaction) {\n      await this.internalRelease();\n    }\n  }\n\n  /**\n   * Schedules a task to be executed serially in the querier instance.\n   * This is used by the @Serialized decorator to protect database-level operations.\n   *\n   * @param task - The async task to execute.\n   * @returns A promise that resolves with the task's result.\n   */\n  protected async serialize<T>(task: () => Promise<T>): Promise<T> {\n    const res = this.taskQueue.then(task);\n    this.taskQueue = res.catch(() => {});\n    return res;\n  }\n\n  abstract beginTransaction(): Promise<void>;\n\n  abstract commitTransaction(): Promise<void>;\n\n  abstract rollbackTransaction(): Promise<void>;\n\n  protected abstract internalRelease(): Promise<void>;\n\n  @Serialized()\n  async release(): Promise<void> {\n    return this.internalRelease();\n  }\n}\n"]}