// // Copyright © 2022 Hardcore Engineering Inc. // // Licensed under the Eclipse Public License, Version 2.0 (the "License"); // you may not use this file except in compliance with the License. You may // obtain a copy of the License at https://www.eclipse.org/legal/epl-2.0 // // Unless required by applicable law or agreed to in writing, software // distributed under the License is distributed on an "AS IS" BASIS, // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. // // See the License for the specific language governing permissions and // limitations under the License. // import { Class, Doc, DocumentQuery, FindOptions, FindResult, MeasureContext, Ref, ServerStorage, Tx } from '@hcengineering/core' import { BroadcastFunc, Middleware, SessionContext, TxMiddlewareResult } from '@hcengineering/server-core' import { BaseMiddleware } from './base' import { deepEqual } from 'fast-equals' interface Query { _class: Ref> query: DocumentQuery result: FindResult | Promise> | undefined options?: FindOptions callbacks: number max: number } /** * @public */ export class QueryJoinMiddleware extends BaseMiddleware implements Middleware { private readonly queries: Map>, Query[]> = new Map>, Query[]>() private constructor (storage: ServerStorage, next?: Middleware) { super(storage, next) } static async create ( ctx: MeasureContext, broadcast: BroadcastFunc, storage: ServerStorage, next?: Middleware ): Promise { return new QueryJoinMiddleware(storage, next) } async tx (ctx: SessionContext, tx: Tx): Promise { return await this.provideTx(ctx, tx) } handleBroadcast (tx: Tx[], targets?: string[]): Tx[] { return this.provideHandleBroadcast(tx, targets) } override async findAll( ctx: SessionContext, _class: Ref>, query: DocumentQuery, options?: FindOptions ): Promise> { // Will find a query or add + 1 to callbacks const q = this.findQuery(_class, query, options) ?? this.createQuery(_class, query, options) if (q.result === undefined) { q.result = this.provideFindAll(ctx, _class, query, options) } if (q.result instanceof Promise) { q.result = await q.result q.callbacks-- } this.removeFromQueue(q) return q.result as FindResult } private findQuery( _class: Ref>, query: DocumentQuery, options?: FindOptions ): Query | undefined { const queries = this.queries.get(_class) if (queries === undefined) return for (const q of queries) { if (!deepEqual(query, q.query) || !deepEqual(options, q.options)) { continue } q.callbacks++ q.max++ return q } } private createQuery(_class: Ref>, query: DocumentQuery, options?: FindOptions): Query { const queries = this.queries.get(_class) ?? [] const q: Query = { _class, query, result: undefined, options: options as FindOptions, callbacks: 1, max: 1 } queries.push(q) this.queries.set(_class, queries) return q } private removeFromQueue (q: Query): void { if (q.callbacks === 0) { const queries = this.queries.get(q._class) ?? [] this.queries.set( q._class, queries.filter((it) => it !== q) ) } } }