nukak
Version:
flexible and efficient ORM, with declarative JSON syntax and smart type-safety
278 lines • 41.1 kB
JavaScript
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"]}