// // Copyright © 2020, 2021 Anticrm Platform Contributors. // Copyright © 2021 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 type { Doc, PropertyType } from './classes' import type { Position, PullArray, QueryUpdate } from './tx' /** * @internal */ export type _OperatorFunc = (doc: Doc, op: any) => void function $push (document: Doc, keyval: Record): void { const doc = document as any for (const key in keyval) { if (doc[key] === undefined) { doc[key] = [] } const val = keyval[key] if (typeof val === 'object') { const arr = doc[key] as Array const desc = val as Position if ('$each' in desc) { arr.splice(desc.$position ?? 0, 0, ...desc.$each) } else { arr.push(val) } } else { doc[key].push(val) } } } function $pull (document: Doc, keyval: Record): void { const doc = document as any for (const key in keyval) { if (doc[key] === undefined) { doc[key] = [] } const arr = doc[key] as Array if (typeof keyval[key] === 'object' && keyval[key] !== null) { const { $in } = keyval[key] as PullArray doc[key] = arr.filter((val) => { if ($in !== undefined) { return !$in.includes(val) } else { // We need to match all fields for (const [kk, kv] of Object.entries(keyval[key])) { if (val[kk] !== kv) { return true } } return false } }) } else { doc[key] = arr.filter((val) => val !== keyval[key]) } } } function matchArrayElement (docs: any[], query: Partial): any[] { let result = [...docs] for (const key in query) { const value = (query as any)[key] const tresult: any[] = [] for (const object of result) { const val = object[key] if (val === value) { tresult.push(object) } } result = tresult if (tresult.length === 0) { break } } return result } function $update (document: Doc, keyval: Record): void { const doc = document as any for (const key in keyval) { if (doc[key] === undefined) { doc[key] = [] } const val = keyval[key] if (typeof val === 'object') { const arr = doc[key] as Array const desc = val as QueryUpdate for (const m of matchArrayElement(arr, desc.$query)) { for (const [k, v] of Object.entries(desc.$update)) { m[k] = v } } } } } function $move (document: Doc, keyval: Record): void { const doc = document as any for (const key in keyval) { if (doc[key] === undefined) { doc[key] = [] } const arr = doc[key] as Array const desc = keyval[key] doc[key] = arr.filter((val) => val !== desc.$value) doc[key].splice(desc.$position, 0, desc.$value) } } function $pushMixin (document: Doc, options: any): void { const doc = document as any const mixinId = options.$mixin if (mixinId === undefined) { throw new Error('$mixin must be specified for $push_mixin operation') } const mixin = doc[mixinId] const keyval = options.values for (const key in keyval) { const arr = mixin[key] if (arr === undefined) { mixin[key] = [keyval[key]] } else { arr.push(keyval[key]) } } } function $inc (document: Doc, keyval: Record): void { const doc = document as unknown as Record for (const key in keyval) { const cur = doc[key] ?? 0 doc[key] = cur + keyval[key] } } function $unset (document: Doc, keyval: Record): void { const doc = document as any for (const key in keyval) { if (doc[key] !== undefined) { // eslint-disable-next-line @typescript-eslint/no-dynamic-delete delete doc[key] } } } const operators: Record = { $push, $pull, $update, $move, $pushMixin, $inc, $unset } /** * @public */ export function isOperator (o: Record): boolean { if (o === null || typeof o !== 'object') { return false } const keys = Object.keys(o) return keys.length > 0 && keys.every((key) => key.startsWith('$')) } /** * @internal * @param name - * @returns */ export function _getOperator (name: string): _OperatorFunc { const operator = operators[name] if (operator === undefined) throw new Error('unknown operator: ' + name) return operator }