// // Copyright © 2023 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 notification, { BaseNotificationType, CommonNotificationType, NotificationContent, NotificationProvider, NotificationType } from '@hcengineering/notification' import type { TriggerControl } from '@hcengineering/server-core' import core, { Account, Class, Doc, DocumentUpdate, Hierarchy, matchQuery, MixinUpdate, Ref, Tx, TxCreateDoc, TxCUD, TxMixin, TxProcessor, TxRemoveDoc, TxUpdateDoc } from '@hcengineering/core' import serverNotification, { getPersonAccountById, HTMLPresenter, NotificationPresenter, TextPresenter, UserInfo } from '@hcengineering/server-notification' import { getResource, IntlString, translate } from '@hcengineering/platform' import contact, { formatName, PersonAccount } from '@hcengineering/contact' import { DocUpdateMessage } from '@hcengineering/activity' import { Analytics } from '@hcengineering/analytics' import { NotifyResult } from './types' /** * @public */ export async function isUserEmployeeInFieldValue ( _: Tx, doc: Doc, user: Ref, type: NotificationType, control: TriggerControl ): Promise { if (type.field === undefined) return false const value = (doc as any)[type.field] if (value == null) return false const employee = (await control.modelDb.findAll(contact.class.PersonAccount, { _id: user as Ref }))[0] if (employee === undefined) return false if (Array.isArray(value)) { return value.includes(employee.person) } else { return value === employee.person } } /** * @public */ export async function isUserInFieldValue ( _: Tx, doc: Doc, user: Ref, type: NotificationType ): Promise { if (type.field === undefined) { return false } const value = (doc as any)[type.field] if (value === undefined) { return false } return Array.isArray(value) ? value.includes(user) : value === user } export function replaceAll (str: string, find: string, replace: string): string { return str.replace(new RegExp(escapeRegExp(find), 'g'), replace) } function escapeRegExp (str: string): string { return str.replace(/[.*+?^${}()|[\]\\]/g, '\\$&') } export async function shouldNotifyCommon ( control: TriggerControl, user: Ref, typeId: Ref ): Promise { const type = (await control.modelDb.findAll(notification.class.CommonNotificationType, { _id: typeId }))[0] if (type === undefined) { return new Map() } const result = new Map, BaseNotificationType[]>() const providers = await control.modelDb.findAll(notification.class.NotificationProvider, {}) for (const provider of providers) { const allowed = await isAllowed(control, user as Ref, type, provider) if (allowed) { const cur = result.get(provider._id) ?? [] result.set(provider._id, [...cur, type]) } } return result } export async function isAllowed ( control: TriggerControl, receiver: Ref, type: BaseNotificationType, provider: NotificationProvider ): Promise { const providersSettings = await control.queryFind(notification.class.NotificationProviderSetting, {}) const providerSetting = providersSettings.find( ({ attachedTo, modifiedBy }) => attachedTo === provider._id && modifiedBy === receiver ) if (providerSetting !== undefined && !providerSetting.enabled) { return false } if (providerSetting === undefined && !provider.defaultEnabled) { return false } const providerDefaults = await control.modelDb.findAll(notification.class.NotificationProviderDefaults, {}) if (providerDefaults.some((it) => it.provider === provider._id && it.ignoredTypes.includes(type._id))) { return false } const typesSettings = await control.queryFind(notification.class.NotificationTypeSetting, {}) const setting = typesSettings.find( (it) => it.attachedTo === provider._id && it.type === type._id && it.modifiedBy === receiver ) if (setting !== undefined) { return setting.enabled } if (providerDefaults.some((it) => it.provider === provider._id && it.enabledTypes.includes(type._id))) { return true } if (type === undefined) return false return type.defaultEnabled } export async function isShouldNotifyTx ( control: TriggerControl, tx: TxCUD, originTx: TxCUD, object: Doc, user: PersonAccount, isOwn: boolean, isSpace: boolean, docUpdateMessage?: DocUpdateMessage ): Promise { const types = await getMatchedTypes( control, tx, originTx, isOwn, isSpace, docUpdateMessage?.attributeUpdates?.attrKey ) const modifiedAccount = await getPersonAccountById(tx.modifiedBy, control) const result = new Map, BaseNotificationType[]>() const providers = await control.modelDb.findAll(notification.class.NotificationProvider, {}) for (const type of types) { if ( type.allowedForAuthor !== true && (tx.modifiedBy === user._id || // Also check if we have different account for same user. (user?.person !== undefined && user?.person === modifiedAccount?.person)) ) { continue } if (control.hierarchy.hasMixin(type, serverNotification.mixin.TypeMatch)) { const mixin = control.hierarchy.as(type, serverNotification.mixin.TypeMatch) if (mixin.func !== undefined) { const f = await getResource(mixin.func) const res = await f(tx, object, user._id, type, control) if (!res) continue } } for (const provider of providers) { const allowed = await isAllowed(control, user._id, type, provider) if (allowed) { const cur = result.get(provider._id) ?? [] result.set(provider._id, [...cur, type]) } } } return result } async function getMatchedTypes ( control: TriggerControl, tx: TxCUD, originTx: TxCUD, isOwn: boolean, isSpace: boolean, field?: string ): Promise { const allTypes = ( await control.modelDb.findAll(notification.class.NotificationType, { ...(field !== undefined ? { field } : {}) }) ).filter((p) => (isSpace ? p.spaceSubscribe === true : p.spaceSubscribe !== true)) const filtered: NotificationType[] = [] for (const type of allTypes) { if (isTypeMatched(control, type, tx, originTx, isOwn)) { filtered.push(type) } } return filtered } function isTypeMatched ( control: TriggerControl, type: NotificationType, tx: TxCUD, originTx: TxCUD, isOwn: boolean ): boolean { const h = control.hierarchy const targetClass = h.getBaseClass(type.objectClass) if (type.onlyOwn === true && !isOwn) return false if (!type.txClasses.includes(tx._class)) return false if (!control.hierarchy.isDerived(h.getBaseClass(tx.objectClass), targetClass)) return false if (originTx._class === core.class.TxCollectionCUD && type.attachedToClass !== undefined) { if (!control.hierarchy.isDerived(h.getBaseClass(originTx.objectClass), h.getBaseClass(type.attachedToClass))) { return false } } if (type.field !== undefined) { if (tx._class === core.class.TxUpdateDoc) { if (!fieldUpdated(type.field, (tx as TxUpdateDoc).operations)) return false } if (tx._class === core.class.TxMixin) { if (!fieldUpdated(type.field, (tx as TxMixin).attributes)) return false } } if (type.txMatch !== undefined) { const res = matchQuery([tx], type.txMatch, tx._class, control.hierarchy, true) if (res.length === 0) return false } return true } function fieldUpdated (field: string, ops: DocumentUpdate | MixinUpdate): boolean { if ((ops as any)[field] !== undefined) return true if ((ops.$pull as any)?.[field] !== undefined) return true if ((ops.$push as any)?.[field] !== undefined) return true return false } export async function updateNotifyContextsSpace ( control: TriggerControl, tx: TxUpdateDoc | TxMixin ): Promise { if (tx._class !== core.class.TxUpdateDoc) { return [] } const updateTx = tx as TxUpdateDoc if (updateTx.operations.space === undefined) { return [] } const notifyContexts = await control.findAll(notification.class.DocNotifyContext, { attachedTo: tx.objectId }) return notifyContexts.map((value) => control.txFactory.createTxUpdateDoc(value._class, value.space, value._id, { space: updateTx.operations.space }) ) } export function isMixinTx (tx: TxCUD): tx is TxMixin { return tx._class === core.class.TxMixin } export function getHTMLPresenter (_class: Ref>, hierarchy: Hierarchy): HTMLPresenter | undefined { return hierarchy.classHierarchyMixin(_class, serverNotification.mixin.HTMLPresenter) } export function getTextPresenter (_class: Ref>, hierarchy: Hierarchy): TextPresenter | undefined { return hierarchy.classHierarchyMixin(_class, serverNotification.mixin.TextPresenter) } async function getSenderName (control: TriggerControl, sender: UserInfo): Promise { if (sender._id === core.account.System) { return await translate(core.string.System, {}) } const { person } = sender if (person === undefined) { console.error('Cannot find person', { accountId: sender._id, person: sender.account?.person }) Analytics.handleError(new Error(`Cannot find person ${sender.account?.person}`)) return '' } return formatName(person.name, control.branding?.lastNameFirst) } async function getFallbackNotificationFullfillment ( object: Doc, originTx: TxCUD, control: TriggerControl, sender: UserInfo ): Promise { const title: IntlString = notification.string.CommonNotificationTitle let body: IntlString = notification.string.CommonNotificationBody const intlParams: Record = {} const intlParamsNotLocalized: Record = {} const textPresenter = getTextPresenter(object._class, control.hierarchy) if (textPresenter !== undefined) { const textPresenterFunc = await getResource(textPresenter.presenter) intlParams.title = await textPresenterFunc(object, control) } const tx = TxProcessor.extractTx(originTx) intlParams.senderName = await getSenderName(control, sender) if (tx._class === core.class.TxUpdateDoc) { const updateTx = tx as TxUpdateDoc const attributes = control.hierarchy.getAllAttributes(object._class) for (const attrName in updateTx.operations) { if (!Object.prototype.hasOwnProperty.call(updateTx.operations, attrName)) { continue } const attr = attributes.get(attrName) if (attr !== null && attr !== undefined) { intlParamsNotLocalized.property = attr.label if (attr.type._class === core.class.TypeString) { body = notification.string.CommonNotificationChangedProperty intlParams.newValue = (updateTx.operations as any)[attrName]?.toString() } else { body = notification.string.CommonNotificationChanged } } break } } else if (originTx._class === core.class.TxCollectionCUD && tx._class === core.class.TxCreateDoc) { const createTx = tx as TxCreateDoc const clazz = control.hierarchy.getClass(createTx.objectClass) const label = clazz.pluralLabel ?? clazz.label if (label !== undefined) { intlParamsNotLocalized.collection = clazz.pluralLabel ?? clazz.label body = notification.string.CommonNotificationCollectionAdded } } else if (originTx._class === core.class.TxCollectionCUD && tx._class === core.class.TxRemoveDoc) { const createTx = tx as TxRemoveDoc const clazz = control.hierarchy.getClass(createTx.objectClass) const label = clazz.pluralLabel ?? clazz.label if (label !== undefined) { intlParamsNotLocalized.collection = clazz.pluralLabel ?? clazz.label body = notification.string.CommonNotificationCollectionRemoved } } return { title, body, intlParams, intlParamsNotLocalized } } function getNotificationPresenter (_class: Ref>, hierarchy: Hierarchy): NotificationPresenter | undefined { return hierarchy.classHierarchyMixin(_class, serverNotification.mixin.NotificationPresenter) } export async function getNotificationContent ( originTx: TxCUD, targetUser: PersonAccount, sender: UserInfo, object: Doc, control: TriggerControl ): Promise { let { title, body, intlParams, intlParamsNotLocalized } = await getFallbackNotificationFullfillment( object, originTx, control, sender ) const actualTx = TxProcessor.extractTx(originTx) const notificationPresenter = getNotificationPresenter((actualTx as TxCUD).objectClass, control.hierarchy) if (notificationPresenter !== undefined) { const getFuillfillmentParams = await getResource(notificationPresenter.presenter) const updateIntlParams = await getFuillfillmentParams(object, originTx, targetUser._id, control) title = updateIntlParams.title body = updateIntlParams.body intlParams = { ...intlParams, ...updateIntlParams.intlParams } if (updateIntlParams.intlParamsNotLocalized != null) { intlParamsNotLocalized = { ...intlParamsNotLocalized, ...updateIntlParams.intlParamsNotLocalized } } } const content: NotificationContent = { title, body, intlParams } if (intlParamsNotLocalized !== undefined) { content.intlParamsNotLocalized = intlParamsNotLocalized } return content } export async function getUsersInfo (ids: Ref[], control: TriggerControl): Promise { const accounts = await control.modelDb.findAll(contact.class.PersonAccount, { _id: { $in: ids } }) const persons = await control.queryFind(contact.class.Person, {}) return accounts.map((account) => ({ _id: account._id, account, person: persons.find(({ _id }) => _id === account.person) })) }