// // Copyright © 2020, 2021 Anticrm Platform Contributors. // // 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 { getEmbeddedLabel, IntlString } from '@hcengineering/platform' import { deepEqual } from 'fast-equals' import { DOMAIN_BENCHMARK } from './benchmark' import { Account, AccountRole, AnyAttribute, AttachedDoc, Class, ClassifierKind, Collection, Doc, DocData, DOMAIN_BLOB, DOMAIN_DOC_INDEX_STATE, DOMAIN_MODEL, DOMAIN_TRANSIENT, FullTextSearchContext, IndexKind, Obj, Permission, PersonId, Ref, Role, roleOrder, SocialId, SocialIdType, SocialKey, Space, TypedSpace, WorkspaceMode, type Domain, type PluginConfiguration } from './classes' import core from './component' import { Hierarchy } from './hierarchy' import { TxOperations } from './operations' import { isPredicate } from './predicate' import { Branding, BrandingMap } from './server' import { DocumentQuery, FindResult } from './storage' import { DOMAIN_TX, TxProcessor, type Tx, type TxCreateDoc, type TxCUD, type TxUpdateDoc } from './tx' function toHex (value: number, chars: number): string { const result = value.toString(16) if (result.length < chars) { return '0'.repeat(chars - result.length) + result } return result } let counter = (Math.random() * (1 << 24)) | 0 const random = toHex((Math.random() * (1 << 24)) | 0, 6) + toHex((Math.random() * (1 << 16)) | 0, 4) function timestamp (): string { const time = (Date.now() / 1000) | 0 return toHex(time, 8) } function count (): string { const val = counter++ & 0xffffff return toHex(val, 6) } /** * @public * @returns */ export function generateId (join: string = ''): Ref { return (timestamp() + join + random + join + count()) as Ref } export function generateUuid (): string { // Consider own implementation if it will be slow return crypto.randomUUID() } /** @public */ export function isId (value: any): value is Ref { return typeof value === 'string' && /^[0-9a-f]{24,24}$/.test(value) } let currentAccount: Account /** * @public * @returns */ export function getCurrentAccount (): Account { return currentAccount } /** * @public * @param account - */ export function setCurrentAccount (account: Account): void { currentAccount = account } /** * @public */ export function escapeLikeForRegexp (value: string): string { return value.replace(/[-/\\^$*+?.()|[\]{}]/g, '\\$&') } /** * @public */ export function toFindResult (docs: T[], total?: number, lookupMap?: Record): FindResult { const length = total ?? docs.length if (Object.keys(lookupMap ?? {}).length === 0) { lookupMap = undefined } return Object.assign(docs, { total: length, lookupMap }) } export type WorkspaceUuid = string & { __workspaceUuid: true } export type WorkspaceDataId = string & { __workspaceDataId: true } export interface WorkspaceIds { uuid: WorkspaceUuid url: string dataId?: WorkspaceDataId // Old workspace identifier. E.g. Database name in Mongo, bucket in R2, etc. } /** * @public */ export function isWorkspaceCreating (mode?: WorkspaceMode): boolean { if (mode === undefined) { return false } return ['pending-creation', 'creating'].includes(mode) } /** * @public */ export function docKey (name: string, _class?: Ref>): string { return _class === undefined || _class !== core.class.Doc ? name : `${_class}%${name}` } /** * @public */ export function isFullTextAttribute (attr: AnyAttribute): boolean { return ( attr.index === IndexKind.FullText || attr.type._class === core.class.TypeBlob || attr.type._class === core.class.EnumOf || attr.type._class === core.class.TypeCollaborativeDoc ) } /** * @public */ export function isIndexedAttribute (attr: AnyAttribute): boolean { return attr.index === IndexKind.Indexed || attr.index === IndexKind.IndexedDsc } /** * @public */ export interface IdMap extends Map, T> {} /** * @public */ export function toIdMap (arr: T[]): IdMap { return new Map(arr.map((p) => [p._id, p])) } /** * @public */ export function concatLink (host: string, path: string): string { if (!host.endsWith('/') && !path.startsWith('/')) { return `${host}/${path}` } else if (host.endsWith('/') && path.startsWith('/')) { const newPath = path.slice(1) return `${host}${newPath}` } else { return `${host}${path}` } } /** * @public */ export function fillDefaults ( hierarchy: Hierarchy, object: DocData | T, _class: Ref> ): DocData | T { const baseClass = hierarchy.isDerived(_class, core.class.AttachedDoc) ? core.class.AttachedDoc : core.class.Doc const attributes = hierarchy.getAllAttributes(_class, baseClass) for (const attribute of attributes) { if (attribute[1].defaultValue !== undefined) { if ((object as any)[attribute[0]] === undefined) { // Clone default value as it might be an object (e.g. array) ;(object as any)[attribute[0]] = structuredClone(attribute[1].defaultValue) } } } return object } /** * @public */ export class AggregateValueData { constructor ( readonly name: string, readonly _id: Ref, readonly space: Ref, readonly rank?: string, readonly category?: Ref ) {} getRank (): string { return this.rank ?? '' } } /** * @public */ export class AggregateValue { constructor ( readonly name: string | undefined, readonly values: AggregateValueData[] ) {} } /** * @public */ export type CategoryType = number | string | undefined | Ref | AggregateValue export interface IDocManager { get: (ref: Ref) => T | undefined getDocs: () => T[] getIdMap: () => IdMap filter: (predicate: (value: T) => boolean) => T[] } /** * @public */ export class DocManager implements IDocManager { protected readonly byId: IdMap constructor (protected readonly docs: T[]) { this.byId = toIdMap(docs) } get (ref: Ref): T | undefined { return this.byId.get(ref) } getDocs (): T[] { return this.docs } getIdMap (): IdMap { return this.byId } filter (predicate: (value: T) => boolean): T[] { return this.docs.filter(predicate) } } /** * @public */ export class RateLimiter { idCounter: number = 0 processingQueue = new Map>() last: number = 0 rate: number queue: (() => Promise)[] = [] constructor (rate: number) { this.rate = rate } notify: (() => void)[] = [] async exec = any>(op: (args?: B) => Promise, args?: B): Promise { const processingId = this.idCounter++ while (this.processingQueue.size >= this.rate) { await new Promise((resolve) => { this.notify.push(resolve) }) } try { const p = op(args) this.processingQueue.set(processingId, p as Promise) return await p } finally { this.processingQueue.delete(processingId) const n = this.notify.shift() if (n !== undefined) { n() } } } async add = any>(op: (args?: B) => Promise, args?: B): Promise { if (this.processingQueue.size < this.rate) { void this.exec(op, args) } else { await this.exec(op, args) } } async waitProcessing (): Promise { while (this.processingQueue.size > 0) { await new Promise((resolve) => { this.notify.push(resolve) }) } } } export function mergeQueries (query1: DocumentQuery, query2: DocumentQuery): DocumentQuery { const keys1 = Object.keys(query1) const keys2 = Object.keys(query2) const query = {} for (const key of keys1) { if (!keys2.includes(key)) { Object.assign(query, { [key]: query1[key] }) } } for (const key of keys2) { if (!keys1.includes(key)) { Object.assign(query, { [key]: query2[key] }) } else { const value = mergeField(query1[key], query2[key]) if (value !== undefined) { Object.assign(query, { [key]: value }) } } } return query } function mergeField (field1: any, field2: any): any | undefined { // this is a special predicate that causes query never return any docs // it is used in cases when queries intersection is empty const never = { $in: [] } // list of ignored predicates, handled separately const ignored = ['$in', '$nin', '$ne'] const isPredicate1 = isPredicate(field1) const isPredicate2 = isPredicate(field2) if (isPredicate1 && isPredicate2) { // $in, $nin, $eq are related fields so handle them separately here const result = getInNiN(field1, field2) const keys1 = Object.keys(field1) const keys2 = Object.keys(field2) for (const key of keys1) { if (ignored.includes(key)) continue if (!keys2.includes(key)) { Object.assign(result, { [key]: field1[key] }) } else { const value = mergePredicateWithPredicate(key, field1[key], field2[key]) if (value !== undefined) { Object.assign(result, { [key]: value }) } } } for (const key of keys2) { if (ignored.includes(key)) continue if (!keys1.includes(key)) { Object.assign(result, { [key]: field2[key] }) } } return Object.keys(result).length > 0 ? result : undefined } else if (isPredicate1 || isPredicate2) { // when one field is a predicate and the other is a simple value // we need to ensure that the value matches predicate const predicate = isPredicate1 ? field1 : field2 const value = isPredicate1 ? field2 : field1 for (const x in predicate) { const result = mergePredicateWithValue(x, predicate[x], value) if ( Array.isArray(result?.$in) && result.$in.length > 0 && Array.isArray(result?.$nin) && result.$nin.length === 0 ) { delete result.$nin } if (result !== undefined) { return result } } // if we reached here, the value does not match the predicate return never } else { // both are not predicates, can filter only when values are equal return deepEqual(field1, field2) ? field1 : never } } function mergePredicateWithPredicate (predicate: string, val1: any, val2: any): any | undefined { if (val1 === undefined) return val2 if (val2 === undefined) return val1 switch (predicate) { case '$lt': return val1 < val2 ? val1 : val2 case '$lte': return val1 <= val2 ? val1 : val2 case '$gt': return val1 > val2 ? val1 : val2 case '$gte': return val1 >= val2 ? val1 : val2 } // TODO we should properly support all available predicates here // until then, fallback to the first predicate value return val1 } function mergePredicateWithValue (predicate: string, val1: any, val2: any): any | undefined { switch (predicate) { case '$in': return Array.isArray(val1) && val1.includes(val2) ? val2 : undefined case '$nin': return Array.isArray(val1) && !val1.includes(val2) ? val2 : undefined case '$lt': return val2 < val1 ? val2 : undefined case '$lte': return val2 <= val1 ? val2 : undefined case '$gt': return val2 > val1 ? val2 : undefined case '$gte': return val2 >= val1 ? val2 : undefined case '$ne': return val1 !== val2 ? val2 : undefined } // TODO we should properly support all available predicates here // until then, fallback to the non-predicate value return val2 } function getInNiN (query1: any, query2: any): any { const aIn = typeof query1 === 'object' && '$in' in query1 ? query1.$in : undefined const bIn = typeof query2 === 'object' && '$in' in query2 ? query2.$in : undefined const aNIn = (typeof query1 === 'object' && '$nin' in query1 ? query1.$nin : undefined) ?? (typeof query1 === 'object' && query1.$ne !== undefined ? [query1.$ne] : []) const bNIn = (typeof query2 === 'object' && '$nin' in query2 ? query2.$nin : undefined) ?? (typeof query1 === 'object' && query2.$ne !== undefined ? [query2.$ne] : []) const finalNin = Array.from(new Set([...aNIn, ...bNIn])) // we must keep $in if it was in the original query if (aIn !== undefined || bIn !== undefined) { const finalIn = aIn !== undefined && bIn !== undefined ? aIn.length - bIn.length < 0 ? bIn.filter((c: any) => aIn.includes(c)) : aIn.filter((c: any) => bIn.includes(c)) : aIn ?? bIn return { $in: finalIn.filter((p: any) => !finalNin.includes(p)) } } // try to preserve original $ne instead of $nin if ((typeof query1 === 'object' && '$ne' in query1) || (typeof query2 === 'object' && '$ne' in query2)) { if (finalNin.length === 1) { return { $ne: finalNin[0] } } } if (finalNin.length > 0) { return { $nin: finalNin } } return {} } export function cutObjectArray (obj: any): any { if (obj == null) { return obj } const r = {} for (const key of Object.keys(obj)) { if (Array.isArray(obj[key])) { if (obj[key].length > 3) { Object.assign(r, { [key]: [...obj[key].slice(0, 3), `... and ${obj[key].length - 3} more`] }) } else Object.assign(r, { [key]: obj[key] }) continue } if (typeof obj[key] === 'object' && obj[key] !== null) { Object.assign(r, { [key]: cutObjectArray(obj[key]) }) continue } Object.assign(r, { [key]: obj[key] }) } return r } export function includesAny (arr1: string[] | null | undefined, arr2: string[] | null | undefined): boolean { if (arr1 == null || arr1.length === 0 || arr2 == null || arr2.length === 0) { return false } return arr1.some((m) => arr2.includes(m)) } export const isEnum = (e: T) => (token: any): token is T[keyof T] => { return typeof token === 'string' && Object.values(e as Record).includes(token) } export async function checkPermission ( client: TxOperations, _id: Ref, _space: Ref, space?: TypedSpace ): Promise { space = space ?? (await client.findOne(core.class.TypedSpace, { _id: _space })) const type = await client .getModel() .findOne(core.class.SpaceType, { _id: space?.type }, { lookup: { _id: { roles: core.class.Role } } }) const mixin = type?.targetClass if (space === undefined || type === undefined || mixin === undefined) { return false } const me = getCurrentAccount() const asMixin = client.getHierarchy().as(space, mixin) const myRoles = type.$lookup?.roles?.filter((role) => includesAny((asMixin as any)[role._id], me.socialIds)) as Role[] if (myRoles === undefined) { return false } const myPermissions = new Set(myRoles.flatMap((role) => role.permissions)) return myPermissions.has(_id) } /** * @public */ export function getRoleAttributeLabel (roleName: string): IntlString { return getEmbeddedLabel(`Role: ${roleName.trim()}`) } /** * @public */ export function getFullTextIndexableAttributes ( hierarchy: Hierarchy, clazz: Ref>, skipDocs: boolean = false ): AnyAttribute[] { const allAttributes = hierarchy.getAllAttributes(clazz) const result: AnyAttribute[] = [] for (const [, attr] of allAttributes) { if (skipDocs && (attr.attributeOf === core.class.Doc || attr.attributeOf === core.class.AttachedDoc)) { continue } if (isFullTextAttribute(attr) || isIndexedAttribute(attr)) { result.push(attr) } } hierarchy .getDescendants(clazz) .filter((m) => hierarchy.getClass(m).kind === ClassifierKind.MIXIN) .forEach((m) => { for (const [, v] of hierarchy.getAllAttributes(m, clazz)) { if (skipDocs && (v.attributeOf === core.class.Doc || v.attributeOf === core.class.AttachedDoc)) { continue } if (isFullTextAttribute(v) || isIndexedAttribute(v)) { result.push(v) } } }) return result } const ctxKey = 'indexer_ftc' /** * @public */ export function getFullTextContext ( hierarchy: Hierarchy, objectClass: Ref>, contexts: Map>, FullTextSearchContext> ): Omit> { let ctx: Omit> | undefined = hierarchy.getClassifierProp(objectClass, ctxKey) if (ctx !== undefined) { return ctx } if (typeof ctx !== 'string') { const anc = hierarchy.getAncestors(objectClass) for (const oc of anc) { const ctx = contexts.get(oc) if (ctx !== undefined) { hierarchy.setClassifierProp(objectClass, ctxKey, ctx) return ctx } } } ctx = { toClass: objectClass, fullTextSummary: false, forceIndex: false, propagate: [], childProcessingAllowed: true } hierarchy.setClassifierProp(objectClass, ctxKey, ctx) return ctx } /** * @public */ export function isClassIndexable ( hierarchy: Hierarchy, c: Ref>, contexts: Map>, FullTextSearchContext> ): boolean { const indexed = hierarchy.getClassifierProp(c, 'class_indexed') if (indexed !== undefined) { return indexed as boolean } const domain = hierarchy.findDomain(c) if (domain === undefined) { hierarchy.setClassifierProp(c, 'class_indexed', false) return false } if ( domain === DOMAIN_DOC_INDEX_STATE || domain === DOMAIN_TX || domain === DOMAIN_MODEL || domain === DOMAIN_BLOB || domain === ('preference' as Domain) || domain === DOMAIN_TRANSIENT || domain === ('settings' as Domain) || domain === DOMAIN_BENCHMARK ) { hierarchy.setClassifierProp(c, 'class_indexed', false) return false } const indexMixin = hierarchy.classHierarchyMixin(c, core.mixin.IndexConfiguration) if (indexMixin?.searchDisabled !== undefined && indexMixin?.searchDisabled) { hierarchy.setClassifierProp(c, 'class_indexed', false) return false } const attrs = getFullTextIndexableAttributes(hierarchy, c, true) for (const d of hierarchy.getDescendants(c)) { if (hierarchy.isMixin(d)) { attrs.push(...getFullTextIndexableAttributes(hierarchy, d, true)) } } let result = true if (attrs.length === 0 && !(getFullTextContext(hierarchy, c, contexts)?.forceIndex ?? false)) { result = false // We need check if document has collections with indexable fields. const attrs = hierarchy.getAllAttributes(c).values() for (const attr of attrs) { if (attr.type._class === core.class.Collection) { if (isClassIndexable(hierarchy, (attr.type as Collection).of, contexts)) { result = true break } } } } hierarchy.setClassifierProp(c, 'class_indexed', result) return result } type ReduceParameters any> = T extends (...args: infer P) => any ? P : never interface NextCall { op: () => Promise } /** * Utility method to skip middle update calls, optimistically if update function is called multiple times with few different parameters, only the last variant will be executed. * The last invocation is executed after a few cycles, allowing to skip middle ones. * * This method can be used inside Svelte components to collapse complex update logic and handle interactions. */ export function reduceCalls) => Promise> ( operation: T ): (...args: ReduceParameters) => Promise { let nextCall: NextCall | undefined let currentCall: NextCall | undefined const next = (): void => { currentCall = nextCall nextCall = undefined if (currentCall !== undefined) { void currentCall.op() } } return async function (...args: ReduceParameters): Promise { const myOp = async (): Promise => { await operation(...args) next() } nextCall = { op: myOp } await Promise.resolve() if (currentCall === undefined) { next() } } } export function isOwnerOrMaintainer (): boolean { const account = getCurrentAccount() return hasAccountRole(account, AccountRole.Maintainer) } export function hasAccountRole (acc: Account, targerRole: AccountRole): boolean { return roleOrder[acc.role] >= roleOrder[targerRole] } export function getBranding (brandings: BrandingMap, key: string | undefined): Branding | null { if (key === undefined) return null return Object.values(brandings).find((branding) => branding.key === key) ?? null } export function fillConfiguration (systemTx: Tx[], configs: Map, PluginConfiguration>): void { for (const t of systemTx) { if (t._class === core.class.TxCreateDoc) { const ct = t as TxCreateDoc if (ct.objectClass === core.class.PluginConfiguration) { configs.set(ct.objectId as Ref, TxProcessor.createDoc2Doc(ct) as PluginConfiguration) } } else if (t._class === core.class.TxUpdateDoc) { const ut = t as TxUpdateDoc if (ut.objectClass === core.class.PluginConfiguration) { const c = configs.get(ut.objectId as Ref) if (c !== undefined) { TxProcessor.updateDoc2Doc(c, ut) } } } } } export function pluginFilterTx ( excludedPlugins: PluginConfiguration[], configs: Map, PluginConfiguration>, systemTx: Tx[] ): Tx[] { const stx = toIdMap(systemTx) const totalExcluded = new Set>() let msg = '' for (const a of excludedPlugins) { for (const c of configs.values()) { if (a.pluginId === c.pluginId) { for (const id of c.transactions) { if (c.classFilter !== undefined) { const filter = new Set(c.classFilter) const tx = stx.get(id as Ref) if ( tx?._class === core.class.TxCreateDoc || tx?._class === core.class.TxUpdateDoc || tx?._class === core.class.TxRemoveDoc ) { const cud = tx as TxCUD if (filter.has(cud.objectClass)) { totalExcluded.add(id as Ref) } } } else { totalExcluded.add(id as Ref) } } msg += ` ${c.pluginId}:${c.transactions.length}` } } } console.log('exclude plugin', msg) systemTx = systemTx.filter((t) => !totalExcluded.has(t._id)) return systemTx } /** * @public */ export class TimeRateLimiter { idCounter: number = 0 active: number = 0 last: number = 0 rate: number period: number executions: { time: number, running: boolean }[] = [] queue: (() => Promise)[] = [] notify: (() => void)[] = [] constructor (rate: number, period: number = 1000) { this.rate = rate this.period = period } private cleanupExecutions (): void { const now = Date.now() this.executions = this.executions.filter((time) => time.running || now - time.time < this.period) } async exec = any>(op: (args?: B) => Promise, args?: B): Promise { while (this.active >= this.rate || this.executions.length >= this.rate) { this.cleanupExecutions() if (this.executions.length < this.rate) { break } await new Promise((resolve) => { setTimeout(resolve, this.period / this.rate) }) } const v = { time: Date.now(), running: true } try { this.executions.push(v) const p = op(args) this.active++ return await p } finally { v.running = false this.active-- this.cleanupExecutions() const n = this.notify.shift() if (n !== undefined) { n() } } } async waitProcessing (): Promise { while (this.active > 0) { console.log('wait', this.active) await new Promise((resolve) => { this.notify.push(resolve) }) } } } export function combineAttributes ( attributes: any[], key: string, operator: '$push' | '$pull', arrayKey: '$each' | '$in' ): any[] { return Array.from( new Set( attributes.flatMap((attr) => Array.isArray(attr[operator]?.[key]?.[arrayKey]) ? attr[operator]?.[key]?.[arrayKey] : attr[operator]?.[key] ) ) ).filter((v) => v != null) } export function buildSocialIdString (key: SocialKey): PersonId { return `${key.type}:${key.value}` as PersonId } export function parseSocialIdString (id: PersonId): SocialKey { const [type, value] = id.split(':') return { type: type as SocialIdType, value } } export function pickPrimarySocialId (socialIds: SocialId[]): SocialId { if (socialIds.length === 0) { throw new Error('No social ids provided') } return socialIds[0] } export function notEmpty (id: T | undefined | null): id is T { return id !== undefined && id !== null && id !== '' }