import { DocumentUpdate, Hierarchy, MixinData, MixinUpdate, ModelDb } from '.' import type { Account, AttachedData, AttachedDoc, Class, Data, Doc, Mixin, Ref, Space } from './classes' import { Client } from './client' import core from './component' import type { DocumentQuery, FindOptions, FindResult, TxResult, WithLookup } from './storage' import { DocumentClassQuery, Tx, TxCUD, TxFactory } from './tx' /** * @public * * High Level operations with client, will create low level transactions. * * `notify` is not supported by TxOperations. */ export class TxOperations implements Omit { readonly txFactory: TxFactory constructor (readonly client: Client, readonly user: Ref) { this.txFactory = new TxFactory(user) } getHierarchy (): Hierarchy { return this.client.getHierarchy() } getModel (): ModelDb { return this.client.getModel() } async close (): Promise { return await this.client.close() } findAll( _class: Ref>, query: DocumentQuery, options?: FindOptions | undefined ): Promise> { return this.client.findAll(_class, query, options) } findOne( _class: Ref>, query: DocumentQuery, options?: FindOptions | undefined ): Promise | undefined> { return this.client.findOne(_class, query, options) } tx (tx: Tx): Promise { return this.client.tx(tx) } async createDoc( _class: Ref>, space: Ref, attributes: Data, id?: Ref ): Promise> { const tx = this.txFactory.createTxCreateDoc(_class, space, attributes, id) await this.client.tx(tx) return tx.objectId } async addCollection( _class: Ref>, space: Ref, attachedTo: Ref, attachedToClass: Ref>, collection: string, attributes: AttachedData

, id?: Ref

): Promise> { const tx = this.txFactory.createTxCollectionCUD( attachedToClass, attachedTo, space, collection, this.txFactory.createTxCreateDoc

(_class, space, attributes as unknown as Data

, id) ) await this.client.tx(tx) return tx.tx.objectId as unknown as Ref

} async updateCollection( _class: Ref>, space: Ref, objectId: Ref

, attachedTo: Ref, attachedToClass: Ref>, collection: string, operations: DocumentUpdate

, retrieve?: boolean ): Promise> { const tx = this.txFactory.createTxCollectionCUD( attachedToClass, attachedTo, space, collection, this.txFactory.createTxUpdateDoc(_class, space, objectId, operations, retrieve) ) await this.client.tx(tx) return tx.objectId } async removeCollection( _class: Ref>, space: Ref, objectId: Ref

, attachedTo: Ref, attachedToClass: Ref>, collection: string ): Promise> { const tx = this.txFactory.createTxCollectionCUD( attachedToClass, attachedTo, space, collection, this.txFactory.createTxRemoveDoc(_class, space, objectId) ) await this.client.tx(tx) return tx.objectId } updateDoc( _class: Ref>, space: Ref, objectId: Ref, operations: DocumentUpdate, retrieve?: boolean ): Promise { const tx = this.txFactory.createTxUpdateDoc(_class, space, objectId, operations, retrieve) return this.client.tx(tx) } removeDoc(_class: Ref>, space: Ref, objectId: Ref): Promise { const tx = this.txFactory.createTxRemoveDoc(_class, space, objectId) return this.client.tx(tx) } createMixin( objectId: Ref, objectClass: Ref>, objectSpace: Ref, mixin: Ref>, attributes: MixinData ): Promise { const tx = this.txFactory.createTxMixin(objectId, objectClass, objectSpace, mixin, attributes) return this.client.tx(tx) } updateMixin( objectId: Ref, objectClass: Ref>, objectSpace: Ref, mixin: Ref>, attributes: MixinUpdate ): Promise { const tx = this.txFactory.createTxMixin(objectId, objectClass, objectSpace, mixin, attributes) return this.client.tx(tx) } update(doc: T, update: DocumentUpdate, retrieve?: boolean): Promise { if (this.client.getHierarchy().isDerived(doc._class, core.class.AttachedDoc)) { const adoc = doc as unknown as AttachedDoc return this.updateCollection( doc._class, doc.space, adoc._id, adoc.attachedTo, adoc.attachedToClass, adoc.collection, update, retrieve ) } return this.updateDoc(doc._class, doc.space, doc._id, update, retrieve) } remove(doc: T): Promise { if (this.client.getHierarchy().isDerived(doc._class, core.class.AttachedDoc)) { const adoc = doc as unknown as AttachedDoc return this.removeCollection( doc._class, doc.space, adoc._id, adoc.attachedTo, adoc.attachedToClass, adoc.collection ) } return this.removeDoc(doc._class, doc.space, doc._id) } apply (scope: string): ApplyOperations { return new ApplyOperations(this, scope) } } /** * @public * * Builder for ApplyOperation, with same syntax as TxOperations. * * Will send real command on commit and will return boolean of operation success. */ export class ApplyOperations extends TxOperations { txes: TxCUD[] = [] matches: DocumentClassQuery[] = [] constructor (readonly ops: TxOperations, readonly scope: string) { const txClient: Client = { getHierarchy: () => ops.client.getHierarchy(), getModel: () => ops.client.getModel(), close: () => ops.client.close(), findOne: (_class, query, options?) => ops.client.findOne(_class, query, options), findAll: (_class, query, options?) => ops.client.findAll(_class, query, options), tx: async (tx): Promise => { if (ops.getHierarchy().isDerived(tx._class, core.class.TxCUD)) { this.txes.push(tx as TxCUD) } return {} } } super(txClient, ops.user) } match(_class: Ref>, query: DocumentQuery): ApplyOperations { this.matches.push({ _class, query }) return this } async commit (): Promise { return await ((await this.ops.tx( this.ops.txFactory.createTxApplyIf(core.space.Tx, this.scope, this.matches, this.txes) )) as Promise) } }