@backstage/plugin-notifications-backend 0.5.11 → 0.5.12-next.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/CHANGELOG.md CHANGED
@@ -1,5 +1,21 @@
1
1
  # @backstage/plugin-notifications-backend
2
2
 
3
+ ## 0.5.12-next.0
4
+
5
+ ### Patch Changes
6
+
7
+ - 05f60e1: Refactored constructor parameter properties to explicit property declarations for compatibility with TypeScript's `erasableSyntaxOnly` setting. This internal refactoring maintains all existing functionality while ensuring TypeScript compilation compatibility.
8
+ - Updated dependencies
9
+ - @backstage/plugin-notifications-node@0.2.21-next.0
10
+ - @backstage/config@1.3.6-next.0
11
+ - @backstage/catalog-model@1.7.6-next.0
12
+ - @backstage/backend-plugin-api@1.4.5-next.0
13
+ - @backstage/errors@1.2.7
14
+ - @backstage/types@1.2.2
15
+ - @backstage/plugin-catalog-node@1.19.2-next.0
16
+ - @backstage/plugin-notifications-common@0.1.2-next.0
17
+ - @backstage/plugin-signals-node@0.1.26-next.0
18
+
3
19
  ## 0.5.11
4
20
 
5
21
  ### Patch Changes
@@ -41,11 +41,12 @@ const generateSettingsHash = (user, channel, origin, topic) => {
41
41
  return crypto__default.default.createHash("sha256").update(rawKey).digest("hex");
42
42
  };
43
43
  class DatabaseNotificationsStore {
44
+ isSQLite = false;
45
+ db;
44
46
  constructor(db) {
45
47
  this.db = db;
46
48
  this.isSQLite = this.db.client.config.client.includes("sqlite3");
47
49
  }
48
- isSQLite = false;
49
50
  static async create({
50
51
  database,
51
52
  skipMigrations
@@ -1 +1 @@
1
- {"version":3,"file":"DatabaseNotificationsStore.cjs.js","sources":["../../src/database/DatabaseNotificationsStore.ts"],"sourcesContent":["/*\n * Copyright 2023 The Backstage Authors\n *\n * Licensed under the Apache License, Version 2.0 (the \"License\");\n * you may not use this file except in compliance with the License.\n * You may obtain a copy of the License at\n *\n * http://www.apache.org/licenses/LICENSE-2.0\n *\n * Unless required by applicable law or agreed to in writing, software\n * distributed under the License is distributed on an \"AS IS\" BASIS,\n * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n * See the License for the specific language governing permissions and\n * limitations under the License.\n */\nimport {\n DatabaseService,\n resolvePackagePath,\n} from '@backstage/backend-plugin-api';\nimport {\n NotificationGetOptions,\n NotificationModifyOptions,\n NotificationsStore,\n TopicGetOptions,\n} from './NotificationsStore';\nimport {\n Notification,\n NotificationSettings,\n notificationSeverities,\n NotificationSeverity,\n} from '@backstage/plugin-notifications-common';\nimport { Knex } from 'knex';\nimport crypto from 'crypto';\nimport { durationToMilliseconds, HumanDuration } from '@backstage/types';\n\nconst migrationsDir = resolvePackagePath(\n '@backstage/plugin-notifications-backend',\n 'migrations',\n);\n\nconst NOTIFICATION_COLUMNS = [\n 'id',\n 'title',\n 'description',\n 'severity',\n 'link',\n 'origin',\n 'scope',\n 'topic',\n 'icon',\n 'created',\n 'updated',\n 'user',\n 'read',\n 'saved',\n];\n\ntype NotificationRowType = {\n id: string;\n user: string;\n title: string;\n description?: string | null;\n severity: string;\n link: string | null;\n origin: string;\n scope: string | null;\n topic: string | null;\n created: Date;\n updated: Date | null;\n read: Date | null;\n saved: Date | null;\n icon: string | null;\n};\n\ntype BroadcastRowType = {\n id: string;\n title: string;\n description: string | null;\n link: string | null;\n origin: string;\n scope: string | null;\n topic: string | null;\n created: Date;\n updated: Date | null;\n icon: string | null;\n};\n\ntype BroadcastUserStatusRowType = {\n broadcast_id: string;\n user: string;\n read: Date | null;\n saved: Date | null;\n};\n\ntype UserSettingsRowType = {\n user: string;\n channel: string;\n origin: string;\n enabled: boolean;\n};\n\nexport const normalizeSeverity = (input?: string): NotificationSeverity => {\n let lower = (input ?? 'normal').toLowerCase() as NotificationSeverity;\n if (notificationSeverities.indexOf(lower) < 0) {\n lower = 'normal';\n }\n return lower;\n};\n\nexport const generateSettingsHash = (\n user: string,\n channel: string,\n origin: string,\n topic: string | null,\n): string => {\n const rawKey = `${user}|${channel}|${origin}|${topic ?? ''}`;\n return crypto.createHash('sha256').update(rawKey).digest('hex');\n};\n\n/** @internal */\nexport class DatabaseNotificationsStore implements NotificationsStore {\n private readonly isSQLite = false;\n\n private constructor(private readonly db: Knex) {\n this.isSQLite = this.db.client.config.client.includes('sqlite3');\n }\n\n static async create({\n database,\n skipMigrations,\n }: {\n database: DatabaseService;\n skipMigrations?: boolean;\n }): Promise<DatabaseNotificationsStore> {\n const client = await database.getClient();\n\n if (!database.migrations?.skip && !skipMigrations) {\n await client.migrate.latest({\n directory: migrationsDir,\n });\n }\n\n return new DatabaseNotificationsStore(client);\n }\n\n private mapToInteger = (val: string | number | undefined): number => {\n return typeof val === 'string' ? Number.parseInt(val, 10) : val ?? 0;\n };\n\n private mapToNotifications = (rows: any[]): Notification[] => {\n return rows.map(row => ({\n id: row.id,\n user: row.type === 'broadcast' ? null : row.user,\n created: new Date(row.created),\n saved: row.saved,\n read: row.read,\n updated: row.updated,\n origin: row.origin,\n payload: {\n title: row.title,\n description: row.description,\n link: row.link,\n topic: row.topic,\n severity: row.severity,\n scope: row.scope,\n icon: row.icon,\n },\n }));\n };\n\n private mapToNotificationSettings = (rows: any[]): NotificationSettings => {\n return rows.reduce(\n (acc, row) => {\n let chan = acc.channels.find(\n (channel: { id: string }) => channel.id === row.channel,\n );\n if (!chan) {\n acc.channels.push({\n id: row.channel,\n origins: [],\n });\n chan = acc.channels[acc.channels.length - 1];\n }\n let origin = chan.origins.find(\n (ori: { id: string }) => ori.id === row.origin,\n );\n if (!origin) {\n origin = {\n id: row.origin,\n enabled: true,\n topics: [],\n };\n chan.origins.push(origin);\n }\n if (row.topic === null) {\n origin.enabled = Boolean(row.enabled);\n } else {\n let topic = origin.topics.find(\n (top: { id: string }) => top.id === row.topic,\n );\n if (!topic) {\n topic = {\n id: row.topic,\n enabled: Boolean(row.enabled),\n };\n origin.topics.push(topic);\n }\n }\n return acc;\n },\n { channels: [] },\n );\n };\n\n private mapNotificationToDbRow = (notification: Notification) => {\n return {\n id: notification.id,\n user: notification.user,\n origin: notification.origin,\n created: notification.created,\n topic: notification.payload?.topic,\n link: notification.payload?.link,\n title: notification.payload?.title,\n description: notification.payload?.description,\n severity: normalizeSeverity(notification.payload?.severity),\n scope: notification.payload?.scope,\n icon: notification.payload.icon,\n saved: notification.saved,\n read: notification.read,\n };\n };\n\n private mapBroadcastToDbRow = (notification: Notification) => {\n return {\n id: notification.id,\n origin: notification.origin,\n created: notification.created,\n topic: notification.payload?.topic,\n link: notification.payload?.link,\n title: notification.payload?.title,\n description: notification.payload?.description,\n severity: normalizeSeverity(notification.payload?.severity),\n icon: notification.payload.icon,\n scope: notification.payload?.scope,\n };\n };\n\n private getBroadcastUnion = (user?: string | null) => {\n return this.db<BroadcastRowType>('broadcast')\n .leftJoin('broadcast_user_status', function clause() {\n const join = this.on('id', '=', 'broadcast_user_status.broadcast_id');\n if (user !== null && user !== undefined) {\n join.andOnVal('user', '=', user);\n }\n })\n .select([...NOTIFICATION_COLUMNS, this.db.raw(\"'broadcast' as type\")]);\n };\n\n private getNotificationsBaseQuery = (\n options: NotificationGetOptions | NotificationModifyOptions,\n ) => {\n const { user, orderField } = options;\n\n const subQuery = this.db<NotificationRowType>('notification')\n .select([...NOTIFICATION_COLUMNS, this.db.raw(\"'entity' as type\")])\n .unionAll([this.getBroadcastUnion(user)])\n .as('notifications');\n\n const query = this.db.from(subQuery).where(q => {\n q.where('user', user).orWhereNull('user');\n });\n\n if (orderField && orderField.length > 0) {\n orderField.forEach(orderBy => {\n query.orderBy(orderBy.field, orderBy.order);\n });\n } else if (!orderField) {\n query.orderBy('created', 'desc');\n }\n\n if (options.createdAfter) {\n if (this.isSQLite) {\n query.where('created', '>=', options.createdAfter.valueOf());\n } else {\n query.where('created', '>=', options.createdAfter.toISOString());\n }\n }\n\n if (options.limit) {\n query.limit(options.limit);\n }\n\n if (options.offset) {\n query.offset(options.offset);\n }\n\n if (options.search) {\n query.whereRaw(\n `(LOWER(title) LIKE LOWER(?) OR LOWER(description) LIKE LOWER(?))`,\n [`%${options.search}%`, `%${options.search}%`],\n );\n }\n\n if (options.ids) {\n query.whereIn('id', options.ids);\n }\n\n if (options.read) {\n query.whereNotNull('read');\n } else if (options.read === false) {\n query.whereNull('read');\n } // or match both if undefined\n\n if (options.topic) {\n query.where('topic', '=', options.topic);\n }\n\n if (options.saved) {\n query.whereNotNull('saved');\n } else if (options.saved === false) {\n query.whereNull('saved');\n } // or match both if undefined\n\n if (options.minimumSeverity !== undefined) {\n const idx = notificationSeverities.indexOf(options.minimumSeverity);\n const equalOrHigher = notificationSeverities.slice(0, idx + 1);\n query.whereIn('severity', equalOrHigher);\n }\n\n return query;\n };\n\n async getNotifications(options: NotificationGetOptions) {\n const notificationQuery = this.getNotificationsBaseQuery(options);\n const notifications = await notificationQuery.select([\n ...NOTIFICATION_COLUMNS,\n 'type',\n ]);\n return this.mapToNotifications(notifications);\n }\n\n async getNotificationsCount(options: NotificationGetOptions) {\n const countOptions: NotificationGetOptions = { ...options };\n countOptions.limit = undefined;\n countOptions.offset = undefined;\n countOptions.orderField = [];\n const notificationQuery = this.getNotificationsBaseQuery(countOptions);\n const response = await notificationQuery.count('id as CNT');\n return Number(response[0].CNT);\n }\n\n async saveNotification(notification: Notification) {\n await this.db\n .insert(this.mapNotificationToDbRow(notification))\n .into('notification');\n }\n\n async saveBroadcast(notification: Notification) {\n await this.db\n .insert(this.mapBroadcastToDbRow(notification))\n .into('broadcast');\n if (notification.saved || notification.read) {\n await this.db\n .insert({\n user: notification.user,\n broadcast_id: notification.id,\n saved: notification.saved,\n read: notification.read,\n })\n .into('broadcast_user_status');\n }\n }\n\n async getStatus(options: NotificationGetOptions) {\n const notificationQuery = this.getNotificationsBaseQuery({\n ...options,\n orderField: [],\n });\n const readSubQuery = notificationQuery\n .clone()\n .count('id')\n .whereNotNull('read')\n .as('READ');\n const unreadSubQuery = notificationQuery\n .clone()\n .count('id')\n .whereNull('read')\n .as('UNREAD');\n\n const query = await notificationQuery\n .select(readSubQuery, unreadSubQuery)\n .first();\n\n return {\n unread: this.mapToInteger((query as any)?.UNREAD),\n read: this.mapToInteger((query as any)?.READ),\n };\n }\n\n async getExistingScopeNotification(options: {\n user: string;\n scope: string;\n origin: string;\n }) {\n const query = this.db<NotificationRowType>('notification')\n .where('user', options.user)\n .where('scope', options.scope)\n .where('origin', options.origin)\n .limit(1);\n\n const rows = await query;\n if (!rows || rows.length === 0) {\n return null;\n }\n return this.mapToNotifications(rows)[0];\n }\n\n async getExistingScopeBroadcast(options: { scope: string; origin: string }) {\n const query = this.db<BroadcastRowType>('broadcast')\n .where('scope', options.scope)\n .where('origin', options.origin)\n .limit(1);\n\n const rows = await query;\n if (!rows || rows.length === 0) {\n return null;\n }\n return this.mapToNotifications(rows)[0];\n }\n\n async restoreExistingNotification({\n id,\n notification,\n }: {\n id: string;\n notification: Notification;\n }) {\n const updateColumns = {\n title: notification.payload.title,\n description: notification.payload.description,\n link: notification.payload.link,\n topic: notification.payload.topic,\n updated: new Date(),\n severity: normalizeSeverity(notification.payload?.severity),\n read: null,\n };\n\n const notificationQuery = this.db('notification')\n .where('id', id)\n .where('user', notification.user);\n const broadcastQuery = this.db('broadcast').where('id', id);\n\n await Promise.all([\n notificationQuery.update(updateColumns),\n broadcastQuery.update({ ...updateColumns, read: undefined }),\n ]);\n\n return await this.getNotification({ id, user: notification.user });\n }\n\n async getNotification(options: {\n id: string;\n user?: string | null;\n }): Promise<Notification | null> {\n const rows = await this.db\n .select('*')\n .from(\n this.db<NotificationRowType>('notification')\n .select([...NOTIFICATION_COLUMNS, this.db.raw(\"'entity' as type\")])\n .unionAll([this.getBroadcastUnion(options.user)])\n .as('notifications'),\n )\n .where('id', options.id)\n .limit(1);\n if (!rows || rows.length === 0) {\n return null;\n }\n return this.mapToNotifications(rows)[0];\n }\n\n private markReadSaved = async (\n ids: string[],\n user: string,\n read?: Date | null,\n saved?: Date | null,\n ) => {\n await this.db<NotificationRowType>('notification')\n .whereIn('id', ids)\n .where('user', user)\n .update({ read, saved });\n\n const broadcasts = this.mapToNotifications(\n await this.db('broadcast').whereIn('id', ids).select(),\n );\n\n if (broadcasts.length > 0)\n if (!this.isSQLite) {\n await this.db<BroadcastUserStatusRowType>('broadcast_user_status')\n .insert(\n broadcasts.map(b => ({\n broadcast_id: b.id,\n user,\n read,\n saved,\n })),\n )\n .onConflict(['broadcast_id', 'user'])\n .merge(['read', 'saved']);\n } else {\n // SQLite does not support upsert so fall back to this (mostly for tests and local dev)\n for (const b of broadcasts) {\n const baseQuery = this.db<BroadcastUserStatusRowType>(\n 'broadcast_user_status',\n )\n .where('broadcast_id', b.id)\n .where('user', user);\n const exists = await baseQuery.clone().limit(1).select().first();\n if (exists) {\n await baseQuery.clone().update({ read, saved });\n } else {\n await baseQuery\n .clone()\n .insert({ broadcast_id: b.id, user, read, saved });\n }\n }\n }\n };\n\n async markRead(options: NotificationModifyOptions): Promise<void> {\n await this.markReadSaved(options.ids, options.user, new Date(), undefined);\n }\n\n async markUnread(options: NotificationModifyOptions): Promise<void> {\n await this.markReadSaved(options.ids, options.user, null, undefined);\n }\n\n async markSaved(options: NotificationModifyOptions): Promise<void> {\n await this.markReadSaved(options.ids, options.user, undefined, new Date());\n }\n\n async markUnsaved(options: NotificationModifyOptions): Promise<void> {\n await this.markReadSaved(options.ids, options.user, undefined, null);\n }\n\n async getUserNotificationOrigins(options: {\n user: string;\n }): Promise<{ origins: string[] }> {\n const rows: { origin: string }[] = await this.db<NotificationRowType>(\n 'notification',\n )\n .where('user', options.user)\n .select('origin')\n .distinct();\n return { origins: rows.map(row => row.origin) };\n }\n\n async getUserNotificationTopics(options: {\n user: string;\n }): Promise<{ topics: { origin: string; topic: string }[] }> {\n const rows: { topic: string; origin: string }[] =\n await this.db<NotificationRowType>('notification')\n .where('user', options.user)\n .select('topic', 'origin')\n .whereNotNull('topic')\n .distinct();\n\n return {\n topics: rows.map(row => ({ origin: row.origin, topic: row.topic })),\n };\n }\n\n async getNotificationSettings(options: {\n user: string;\n origin?: string;\n channel?: string;\n topic?: string;\n }): Promise<NotificationSettings> {\n const settingsQuery = this.db<UserSettingsRowType>('user_settings').where(\n 'user',\n options.user,\n );\n if (options.origin) {\n settingsQuery.where('origin', options.origin);\n }\n\n if (options.channel) {\n settingsQuery.where('channel', options.channel);\n }\n\n if (options.topic) {\n settingsQuery.where('topic', options.topic);\n }\n const settings = await settingsQuery.select();\n return this.mapToNotificationSettings(settings);\n }\n\n async saveNotificationSettings(options: {\n user: string;\n settings: NotificationSettings;\n }): Promise<void> {\n const rows: {\n settings_key_hash: string;\n user: string;\n channel: string;\n origin: string;\n topic: string | null;\n enabled: boolean;\n }[] = [];\n\n options.settings.channels.forEach(channel => {\n channel.origins.forEach(origin => {\n rows.push({\n settings_key_hash: generateSettingsHash(\n options.user,\n channel.id,\n origin.id,\n null,\n ),\n user: options.user,\n channel: channel.id,\n origin: origin.id,\n topic: null,\n enabled: origin.enabled,\n });\n\n origin.topics?.forEach(topic => {\n rows.push({\n settings_key_hash: generateSettingsHash(\n options.user,\n channel.id,\n origin.id,\n topic.id,\n ),\n user: options.user,\n channel: channel.id,\n origin: origin.id,\n topic: topic.id,\n enabled: origin.enabled && topic.enabled,\n });\n });\n });\n });\n\n await this.db<UserSettingsRowType>('user_settings')\n .where('user', options.user)\n .delete();\n await this.db<UserSettingsRowType>('user_settings').insert(rows);\n }\n\n async getTopics(options: TopicGetOptions): Promise<{ topics: string[] }> {\n const notificationQuery = this.getNotificationsBaseQuery({\n ...options,\n orderField: [{ field: 'topic', order: 'asc' }],\n });\n const topics = await notificationQuery\n .whereNotNull('topic')\n .distinct(['topic']);\n return { topics: topics.map(row => row.topic) };\n }\n\n async clearNotifications(options: {\n maxAge: HumanDuration;\n }): Promise<{ deletedCount: number }> {\n const ms = durationToMilliseconds(options.maxAge);\n const now = new Date(new Date().getTime() - ms);\n const notificationsCount = await this.db('notification')\n .where(builder => {\n builder.where('created', '<=', now).whereNull('updated');\n })\n .orWhere('updated', '<=', now)\n .delete();\n const broadcastsCount = await this.db('broadcast')\n .where(builder => {\n builder.where('created', '<=', now).whereNull('updated');\n })\n .orWhere('updated', '<=', now)\n .delete();\n return { deletedCount: notificationsCount + broadcastsCount };\n }\n}\n"],"names":["resolvePackagePath","notificationSeverities","crypto","durationToMilliseconds"],"mappings":"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"}
1
+ {"version":3,"file":"DatabaseNotificationsStore.cjs.js","sources":["../../src/database/DatabaseNotificationsStore.ts"],"sourcesContent":["/*\n * Copyright 2023 The Backstage Authors\n *\n * Licensed under the Apache License, Version 2.0 (the \"License\");\n * you may not use this file except in compliance with the License.\n * You may obtain a copy of the License at\n *\n * http://www.apache.org/licenses/LICENSE-2.0\n *\n * Unless required by applicable law or agreed to in writing, software\n * distributed under the License is distributed on an \"AS IS\" BASIS,\n * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n * See the License for the specific language governing permissions and\n * limitations under the License.\n */\nimport {\n DatabaseService,\n resolvePackagePath,\n} from '@backstage/backend-plugin-api';\nimport {\n NotificationGetOptions,\n NotificationModifyOptions,\n NotificationsStore,\n TopicGetOptions,\n} from './NotificationsStore';\nimport {\n Notification,\n NotificationSettings,\n notificationSeverities,\n NotificationSeverity,\n} from '@backstage/plugin-notifications-common';\nimport { Knex } from 'knex';\nimport crypto from 'crypto';\nimport { durationToMilliseconds, HumanDuration } from '@backstage/types';\n\nconst migrationsDir = resolvePackagePath(\n '@backstage/plugin-notifications-backend',\n 'migrations',\n);\n\nconst NOTIFICATION_COLUMNS = [\n 'id',\n 'title',\n 'description',\n 'severity',\n 'link',\n 'origin',\n 'scope',\n 'topic',\n 'icon',\n 'created',\n 'updated',\n 'user',\n 'read',\n 'saved',\n];\n\ntype NotificationRowType = {\n id: string;\n user: string;\n title: string;\n description?: string | null;\n severity: string;\n link: string | null;\n origin: string;\n scope: string | null;\n topic: string | null;\n created: Date;\n updated: Date | null;\n read: Date | null;\n saved: Date | null;\n icon: string | null;\n};\n\ntype BroadcastRowType = {\n id: string;\n title: string;\n description: string | null;\n link: string | null;\n origin: string;\n scope: string | null;\n topic: string | null;\n created: Date;\n updated: Date | null;\n icon: string | null;\n};\n\ntype BroadcastUserStatusRowType = {\n broadcast_id: string;\n user: string;\n read: Date | null;\n saved: Date | null;\n};\n\ntype UserSettingsRowType = {\n user: string;\n channel: string;\n origin: string;\n enabled: boolean;\n};\n\nexport const normalizeSeverity = (input?: string): NotificationSeverity => {\n let lower = (input ?? 'normal').toLowerCase() as NotificationSeverity;\n if (notificationSeverities.indexOf(lower) < 0) {\n lower = 'normal';\n }\n return lower;\n};\n\nexport const generateSettingsHash = (\n user: string,\n channel: string,\n origin: string,\n topic: string | null,\n): string => {\n const rawKey = `${user}|${channel}|${origin}|${topic ?? ''}`;\n return crypto.createHash('sha256').update(rawKey).digest('hex');\n};\n\n/** @internal */\nexport class DatabaseNotificationsStore implements NotificationsStore {\n private readonly isSQLite = false;\n\n private readonly db: Knex;\n\n private constructor(db: Knex) {\n this.db = db;\n this.isSQLite = this.db.client.config.client.includes('sqlite3');\n }\n\n static async create({\n database,\n skipMigrations,\n }: {\n database: DatabaseService;\n skipMigrations?: boolean;\n }): Promise<DatabaseNotificationsStore> {\n const client = await database.getClient();\n\n if (!database.migrations?.skip && !skipMigrations) {\n await client.migrate.latest({\n directory: migrationsDir,\n });\n }\n\n return new DatabaseNotificationsStore(client);\n }\n\n private mapToInteger = (val: string | number | undefined): number => {\n return typeof val === 'string' ? Number.parseInt(val, 10) : val ?? 0;\n };\n\n private mapToNotifications = (rows: any[]): Notification[] => {\n return rows.map(row => ({\n id: row.id,\n user: row.type === 'broadcast' ? null : row.user,\n created: new Date(row.created),\n saved: row.saved,\n read: row.read,\n updated: row.updated,\n origin: row.origin,\n payload: {\n title: row.title,\n description: row.description,\n link: row.link,\n topic: row.topic,\n severity: row.severity,\n scope: row.scope,\n icon: row.icon,\n },\n }));\n };\n\n private mapToNotificationSettings = (rows: any[]): NotificationSettings => {\n return rows.reduce(\n (acc, row) => {\n let chan = acc.channels.find(\n (channel: { id: string }) => channel.id === row.channel,\n );\n if (!chan) {\n acc.channels.push({\n id: row.channel,\n origins: [],\n });\n chan = acc.channels[acc.channels.length - 1];\n }\n let origin = chan.origins.find(\n (ori: { id: string }) => ori.id === row.origin,\n );\n if (!origin) {\n origin = {\n id: row.origin,\n enabled: true,\n topics: [],\n };\n chan.origins.push(origin);\n }\n if (row.topic === null) {\n origin.enabled = Boolean(row.enabled);\n } else {\n let topic = origin.topics.find(\n (top: { id: string }) => top.id === row.topic,\n );\n if (!topic) {\n topic = {\n id: row.topic,\n enabled: Boolean(row.enabled),\n };\n origin.topics.push(topic);\n }\n }\n return acc;\n },\n { channels: [] },\n );\n };\n\n private mapNotificationToDbRow = (notification: Notification) => {\n return {\n id: notification.id,\n user: notification.user,\n origin: notification.origin,\n created: notification.created,\n topic: notification.payload?.topic,\n link: notification.payload?.link,\n title: notification.payload?.title,\n description: notification.payload?.description,\n severity: normalizeSeverity(notification.payload?.severity),\n scope: notification.payload?.scope,\n icon: notification.payload.icon,\n saved: notification.saved,\n read: notification.read,\n };\n };\n\n private mapBroadcastToDbRow = (notification: Notification) => {\n return {\n id: notification.id,\n origin: notification.origin,\n created: notification.created,\n topic: notification.payload?.topic,\n link: notification.payload?.link,\n title: notification.payload?.title,\n description: notification.payload?.description,\n severity: normalizeSeverity(notification.payload?.severity),\n icon: notification.payload.icon,\n scope: notification.payload?.scope,\n };\n };\n\n private getBroadcastUnion = (user?: string | null) => {\n return this.db<BroadcastRowType>('broadcast')\n .leftJoin('broadcast_user_status', function clause() {\n const join = this.on('id', '=', 'broadcast_user_status.broadcast_id');\n if (user !== null && user !== undefined) {\n join.andOnVal('user', '=', user);\n }\n })\n .select([...NOTIFICATION_COLUMNS, this.db.raw(\"'broadcast' as type\")]);\n };\n\n private getNotificationsBaseQuery = (\n options: NotificationGetOptions | NotificationModifyOptions,\n ) => {\n const { user, orderField } = options;\n\n const subQuery = this.db<NotificationRowType>('notification')\n .select([...NOTIFICATION_COLUMNS, this.db.raw(\"'entity' as type\")])\n .unionAll([this.getBroadcastUnion(user)])\n .as('notifications');\n\n const query = this.db.from(subQuery).where(q => {\n q.where('user', user).orWhereNull('user');\n });\n\n if (orderField && orderField.length > 0) {\n orderField.forEach(orderBy => {\n query.orderBy(orderBy.field, orderBy.order);\n });\n } else if (!orderField) {\n query.orderBy('created', 'desc');\n }\n\n if (options.createdAfter) {\n if (this.isSQLite) {\n query.where('created', '>=', options.createdAfter.valueOf());\n } else {\n query.where('created', '>=', options.createdAfter.toISOString());\n }\n }\n\n if (options.limit) {\n query.limit(options.limit);\n }\n\n if (options.offset) {\n query.offset(options.offset);\n }\n\n if (options.search) {\n query.whereRaw(\n `(LOWER(title) LIKE LOWER(?) OR LOWER(description) LIKE LOWER(?))`,\n [`%${options.search}%`, `%${options.search}%`],\n );\n }\n\n if (options.ids) {\n query.whereIn('id', options.ids);\n }\n\n if (options.read) {\n query.whereNotNull('read');\n } else if (options.read === false) {\n query.whereNull('read');\n } // or match both if undefined\n\n if (options.topic) {\n query.where('topic', '=', options.topic);\n }\n\n if (options.saved) {\n query.whereNotNull('saved');\n } else if (options.saved === false) {\n query.whereNull('saved');\n } // or match both if undefined\n\n if (options.minimumSeverity !== undefined) {\n const idx = notificationSeverities.indexOf(options.minimumSeverity);\n const equalOrHigher = notificationSeverities.slice(0, idx + 1);\n query.whereIn('severity', equalOrHigher);\n }\n\n return query;\n };\n\n async getNotifications(options: NotificationGetOptions) {\n const notificationQuery = this.getNotificationsBaseQuery(options);\n const notifications = await notificationQuery.select([\n ...NOTIFICATION_COLUMNS,\n 'type',\n ]);\n return this.mapToNotifications(notifications);\n }\n\n async getNotificationsCount(options: NotificationGetOptions) {\n const countOptions: NotificationGetOptions = { ...options };\n countOptions.limit = undefined;\n countOptions.offset = undefined;\n countOptions.orderField = [];\n const notificationQuery = this.getNotificationsBaseQuery(countOptions);\n const response = await notificationQuery.count('id as CNT');\n return Number(response[0].CNT);\n }\n\n async saveNotification(notification: Notification) {\n await this.db\n .insert(this.mapNotificationToDbRow(notification))\n .into('notification');\n }\n\n async saveBroadcast(notification: Notification) {\n await this.db\n .insert(this.mapBroadcastToDbRow(notification))\n .into('broadcast');\n if (notification.saved || notification.read) {\n await this.db\n .insert({\n user: notification.user,\n broadcast_id: notification.id,\n saved: notification.saved,\n read: notification.read,\n })\n .into('broadcast_user_status');\n }\n }\n\n async getStatus(options: NotificationGetOptions) {\n const notificationQuery = this.getNotificationsBaseQuery({\n ...options,\n orderField: [],\n });\n const readSubQuery = notificationQuery\n .clone()\n .count('id')\n .whereNotNull('read')\n .as('READ');\n const unreadSubQuery = notificationQuery\n .clone()\n .count('id')\n .whereNull('read')\n .as('UNREAD');\n\n const query = await notificationQuery\n .select(readSubQuery, unreadSubQuery)\n .first();\n\n return {\n unread: this.mapToInteger((query as any)?.UNREAD),\n read: this.mapToInteger((query as any)?.READ),\n };\n }\n\n async getExistingScopeNotification(options: {\n user: string;\n scope: string;\n origin: string;\n }) {\n const query = this.db<NotificationRowType>('notification')\n .where('user', options.user)\n .where('scope', options.scope)\n .where('origin', options.origin)\n .limit(1);\n\n const rows = await query;\n if (!rows || rows.length === 0) {\n return null;\n }\n return this.mapToNotifications(rows)[0];\n }\n\n async getExistingScopeBroadcast(options: { scope: string; origin: string }) {\n const query = this.db<BroadcastRowType>('broadcast')\n .where('scope', options.scope)\n .where('origin', options.origin)\n .limit(1);\n\n const rows = await query;\n if (!rows || rows.length === 0) {\n return null;\n }\n return this.mapToNotifications(rows)[0];\n }\n\n async restoreExistingNotification({\n id,\n notification,\n }: {\n id: string;\n notification: Notification;\n }) {\n const updateColumns = {\n title: notification.payload.title,\n description: notification.payload.description,\n link: notification.payload.link,\n topic: notification.payload.topic,\n updated: new Date(),\n severity: normalizeSeverity(notification.payload?.severity),\n read: null,\n };\n\n const notificationQuery = this.db('notification')\n .where('id', id)\n .where('user', notification.user);\n const broadcastQuery = this.db('broadcast').where('id', id);\n\n await Promise.all([\n notificationQuery.update(updateColumns),\n broadcastQuery.update({ ...updateColumns, read: undefined }),\n ]);\n\n return await this.getNotification({ id, user: notification.user });\n }\n\n async getNotification(options: {\n id: string;\n user?: string | null;\n }): Promise<Notification | null> {\n const rows = await this.db\n .select('*')\n .from(\n this.db<NotificationRowType>('notification')\n .select([...NOTIFICATION_COLUMNS, this.db.raw(\"'entity' as type\")])\n .unionAll([this.getBroadcastUnion(options.user)])\n .as('notifications'),\n )\n .where('id', options.id)\n .limit(1);\n if (!rows || rows.length === 0) {\n return null;\n }\n return this.mapToNotifications(rows)[0];\n }\n\n private markReadSaved = async (\n ids: string[],\n user: string,\n read?: Date | null,\n saved?: Date | null,\n ) => {\n await this.db<NotificationRowType>('notification')\n .whereIn('id', ids)\n .where('user', user)\n .update({ read, saved });\n\n const broadcasts = this.mapToNotifications(\n await this.db('broadcast').whereIn('id', ids).select(),\n );\n\n if (broadcasts.length > 0)\n if (!this.isSQLite) {\n await this.db<BroadcastUserStatusRowType>('broadcast_user_status')\n .insert(\n broadcasts.map(b => ({\n broadcast_id: b.id,\n user,\n read,\n saved,\n })),\n )\n .onConflict(['broadcast_id', 'user'])\n .merge(['read', 'saved']);\n } else {\n // SQLite does not support upsert so fall back to this (mostly for tests and local dev)\n for (const b of broadcasts) {\n const baseQuery = this.db<BroadcastUserStatusRowType>(\n 'broadcast_user_status',\n )\n .where('broadcast_id', b.id)\n .where('user', user);\n const exists = await baseQuery.clone().limit(1).select().first();\n if (exists) {\n await baseQuery.clone().update({ read, saved });\n } else {\n await baseQuery\n .clone()\n .insert({ broadcast_id: b.id, user, read, saved });\n }\n }\n }\n };\n\n async markRead(options: NotificationModifyOptions): Promise<void> {\n await this.markReadSaved(options.ids, options.user, new Date(), undefined);\n }\n\n async markUnread(options: NotificationModifyOptions): Promise<void> {\n await this.markReadSaved(options.ids, options.user, null, undefined);\n }\n\n async markSaved(options: NotificationModifyOptions): Promise<void> {\n await this.markReadSaved(options.ids, options.user, undefined, new Date());\n }\n\n async markUnsaved(options: NotificationModifyOptions): Promise<void> {\n await this.markReadSaved(options.ids, options.user, undefined, null);\n }\n\n async getUserNotificationOrigins(options: {\n user: string;\n }): Promise<{ origins: string[] }> {\n const rows: { origin: string }[] = await this.db<NotificationRowType>(\n 'notification',\n )\n .where('user', options.user)\n .select('origin')\n .distinct();\n return { origins: rows.map(row => row.origin) };\n }\n\n async getUserNotificationTopics(options: {\n user: string;\n }): Promise<{ topics: { origin: string; topic: string }[] }> {\n const rows: { topic: string; origin: string }[] =\n await this.db<NotificationRowType>('notification')\n .where('user', options.user)\n .select('topic', 'origin')\n .whereNotNull('topic')\n .distinct();\n\n return {\n topics: rows.map(row => ({ origin: row.origin, topic: row.topic })),\n };\n }\n\n async getNotificationSettings(options: {\n user: string;\n origin?: string;\n channel?: string;\n topic?: string;\n }): Promise<NotificationSettings> {\n const settingsQuery = this.db<UserSettingsRowType>('user_settings').where(\n 'user',\n options.user,\n );\n if (options.origin) {\n settingsQuery.where('origin', options.origin);\n }\n\n if (options.channel) {\n settingsQuery.where('channel', options.channel);\n }\n\n if (options.topic) {\n settingsQuery.where('topic', options.topic);\n }\n const settings = await settingsQuery.select();\n return this.mapToNotificationSettings(settings);\n }\n\n async saveNotificationSettings(options: {\n user: string;\n settings: NotificationSettings;\n }): Promise<void> {\n const rows: {\n settings_key_hash: string;\n user: string;\n channel: string;\n origin: string;\n topic: string | null;\n enabled: boolean;\n }[] = [];\n\n options.settings.channels.forEach(channel => {\n channel.origins.forEach(origin => {\n rows.push({\n settings_key_hash: generateSettingsHash(\n options.user,\n channel.id,\n origin.id,\n null,\n ),\n user: options.user,\n channel: channel.id,\n origin: origin.id,\n topic: null,\n enabled: origin.enabled,\n });\n\n origin.topics?.forEach(topic => {\n rows.push({\n settings_key_hash: generateSettingsHash(\n options.user,\n channel.id,\n origin.id,\n topic.id,\n ),\n user: options.user,\n channel: channel.id,\n origin: origin.id,\n topic: topic.id,\n enabled: origin.enabled && topic.enabled,\n });\n });\n });\n });\n\n await this.db<UserSettingsRowType>('user_settings')\n .where('user', options.user)\n .delete();\n await this.db<UserSettingsRowType>('user_settings').insert(rows);\n }\n\n async getTopics(options: TopicGetOptions): Promise<{ topics: string[] }> {\n const notificationQuery = this.getNotificationsBaseQuery({\n ...options,\n orderField: [{ field: 'topic', order: 'asc' }],\n });\n const topics = await notificationQuery\n .whereNotNull('topic')\n .distinct(['topic']);\n return { topics: topics.map(row => row.topic) };\n }\n\n async clearNotifications(options: {\n maxAge: HumanDuration;\n }): Promise<{ deletedCount: number }> {\n const ms = durationToMilliseconds(options.maxAge);\n const now = new Date(new Date().getTime() - ms);\n const notificationsCount = await this.db('notification')\n .where(builder => {\n builder.where('created', '<=', now).whereNull('updated');\n })\n .orWhere('updated', '<=', now)\n .delete();\n const broadcastsCount = await this.db('broadcast')\n .where(builder => {\n builder.where('created', '<=', now).whereNull('updated');\n })\n .orWhere('updated', '<=', now)\n .delete();\n return { deletedCount: notificationsCount + broadcastsCount };\n }\n}\n"],"names":["resolvePackagePath","notificationSeverities","crypto","durationToMilliseconds"],"mappings":"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"}
@@ -15,6 +15,8 @@ const partitionEntityRefs = (refs) => {
15
15
  return ret;
16
16
  };
17
17
  class DefaultNotificationRecipientResolver {
18
+ auth;
19
+ catalog;
18
20
  constructor(auth, catalog) {
19
21
  this.auth = auth;
20
22
  this.catalog = catalog;
@@ -1 +1 @@
1
- {"version":3,"file":"DefaultNotificationRecipientResolver.cjs.js","sources":["../../src/service/DefaultNotificationRecipientResolver.ts"],"sourcesContent":["/*\n * Copyright 2024 The Backstage Authors\n *\n * Licensed under the Apache License, Version 2.0 (the \"License\");\n * you may not use this file except in compliance with the License.\n * You may obtain a copy of the License at\n *\n * http://www.apache.org/licenses/LICENSE-2.0\n *\n * Unless required by applicable law or agreed to in writing, software\n * distributed under the License is distributed on an \"AS IS\" BASIS,\n * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n * See the License for the specific language governing permissions and\n * limitations under the License.\n */\n\nimport {\n Entity,\n isGroupEntity,\n isUserEntity,\n parseEntityRef,\n RELATION_HAS_MEMBER,\n RELATION_OWNED_BY,\n RELATION_PARENT_OF,\n stringifyEntityRef,\n} from '@backstage/catalog-model';\nimport { AuthService } from '@backstage/backend-plugin-api';\nimport { CatalogService } from '@backstage/plugin-catalog-node';\nimport { NotificationRecipientResolver } from '@backstage/plugin-notifications-node';\n\nconst isUserEntityRef = (ref: string) =>\n parseEntityRef(ref).kind.toLocaleLowerCase() === 'user';\n\n// Partitions array of entity references to two arrays; user entity refs and other entity refs\nconst partitionEntityRefs = (refs: string[]): string[][] => {\n const ret = [[], []] as string[][];\n for (const ref of refs) {\n if (isUserEntityRef(ref)) {\n ret[0].push(ref);\n } else {\n ret[1].push(ref);\n }\n }\n return ret;\n};\n\nexport class DefaultNotificationRecipientResolver\n implements NotificationRecipientResolver\n{\n constructor(\n private readonly auth: AuthService,\n private readonly catalog: CatalogService,\n ) {}\n\n async resolveNotificationRecipients(options: {\n entityRefs: string[];\n excludedEntityRefs?: string[];\n }): Promise<{ userEntityRefs: string[] }> {\n const { entityRefs, excludedEntityRefs = [] } = options;\n\n const [userEntityRefs, otherEntityRefs] = partitionEntityRefs(entityRefs);\n const users: string[] = userEntityRefs.filter(\n ref => !excludedEntityRefs.includes(ref),\n );\n const filtered = otherEntityRefs.filter(\n ref => !excludedEntityRefs.includes(ref),\n );\n\n const fields = ['kind', 'metadata.name', 'metadata.namespace', 'relations'];\n let entities: Array<Entity | undefined> = [];\n if (filtered.length > 0) {\n const fetchedEntities = await this.catalog.getEntitiesByRefs(\n {\n entityRefs: filtered,\n fields,\n },\n { credentials: await this.auth.getOwnServiceCredentials() },\n );\n entities = fetchedEntities.items;\n }\n\n const cachedEntityRefs = new Map<string, string[]>();\n\n const mapEntity = async (entity: Entity | undefined): Promise<string[]> => {\n if (!entity) {\n return [];\n }\n\n const currentEntityRef = stringifyEntityRef(entity);\n if (excludedEntityRefs.includes(currentEntityRef)) {\n return [];\n }\n\n if (cachedEntityRefs.has(currentEntityRef)) {\n return cachedEntityRefs.get(currentEntityRef)!;\n }\n\n if (isUserEntity(entity)) {\n return [currentEntityRef];\n }\n\n if (isGroupEntity(entity)) {\n if (!entity.relations?.length) {\n return [];\n }\n\n const groupUsers = entity.relations\n .filter(\n relation =>\n relation.type === RELATION_HAS_MEMBER &&\n isUserEntityRef(relation.targetRef),\n )\n .map(r => r.targetRef);\n\n const childGroupRefs = entity.relations\n .filter(relation => relation.type === RELATION_PARENT_OF)\n .map(r => r.targetRef);\n\n let childGroupUsers: string[][] = [];\n if (childGroupRefs.length > 0) {\n const childGroups = await this.catalog.getEntitiesByRefs(\n {\n entityRefs: childGroupRefs,\n fields,\n },\n { credentials: await this.auth.getOwnServiceCredentials() },\n );\n childGroupUsers = await Promise.all(childGroups.items.map(mapEntity));\n }\n\n const ret = [\n ...new Set([...groupUsers, ...childGroupUsers.flat(2)]),\n ].filter(ref => !excludedEntityRefs.includes(ref));\n cachedEntityRefs.set(currentEntityRef, ret);\n return ret;\n }\n\n if (entity.relations?.length) {\n const ownerRef = entity.relations.find(\n relation => relation.type === RELATION_OWNED_BY,\n )?.targetRef;\n\n if (!ownerRef) {\n return [];\n }\n\n if (isUserEntityRef(ownerRef)) {\n if (excludedEntityRefs.includes(ownerRef)) {\n return [];\n }\n return [ownerRef];\n }\n\n const owner = await this.catalog.getEntityByRef(ownerRef, {\n credentials: await this.auth.getOwnServiceCredentials(),\n });\n const ret = await mapEntity(owner);\n cachedEntityRefs.set(currentEntityRef, ret);\n return ret;\n }\n\n return [];\n };\n\n for (const entity of entities) {\n const u = await mapEntity(entity);\n users.push(...u);\n }\n\n return {\n userEntityRefs: [...new Set(users)]\n .filter(Boolean)\n // Need to filter again after resolving users\n .filter(ref => !excludedEntityRefs.includes(ref)),\n };\n }\n}\n"],"names":["parseEntityRef","stringifyEntityRef","isUserEntity","isGroupEntity","RELATION_HAS_MEMBER","RELATION_PARENT_OF","RELATION_OWNED_BY"],"mappings":"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"}
1
+ {"version":3,"file":"DefaultNotificationRecipientResolver.cjs.js","sources":["../../src/service/DefaultNotificationRecipientResolver.ts"],"sourcesContent":["/*\n * Copyright 2024 The Backstage Authors\n *\n * Licensed under the Apache License, Version 2.0 (the \"License\");\n * you may not use this file except in compliance with the License.\n * You may obtain a copy of the License at\n *\n * http://www.apache.org/licenses/LICENSE-2.0\n *\n * Unless required by applicable law or agreed to in writing, software\n * distributed under the License is distributed on an \"AS IS\" BASIS,\n * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n * See the License for the specific language governing permissions and\n * limitations under the License.\n */\n\nimport {\n Entity,\n isGroupEntity,\n isUserEntity,\n parseEntityRef,\n RELATION_HAS_MEMBER,\n RELATION_OWNED_BY,\n RELATION_PARENT_OF,\n stringifyEntityRef,\n} from '@backstage/catalog-model';\nimport { AuthService } from '@backstage/backend-plugin-api';\nimport { CatalogService } from '@backstage/plugin-catalog-node';\nimport { NotificationRecipientResolver } from '@backstage/plugin-notifications-node';\n\nconst isUserEntityRef = (ref: string) =>\n parseEntityRef(ref).kind.toLocaleLowerCase() === 'user';\n\n// Partitions array of entity references to two arrays; user entity refs and other entity refs\nconst partitionEntityRefs = (refs: string[]): string[][] => {\n const ret = [[], []] as string[][];\n for (const ref of refs) {\n if (isUserEntityRef(ref)) {\n ret[0].push(ref);\n } else {\n ret[1].push(ref);\n }\n }\n return ret;\n};\n\nexport class DefaultNotificationRecipientResolver\n implements NotificationRecipientResolver\n{\n private readonly auth: AuthService;\n private readonly catalog: CatalogService;\n\n constructor(auth: AuthService, catalog: CatalogService) {\n this.auth = auth;\n this.catalog = catalog;\n }\n\n async resolveNotificationRecipients(options: {\n entityRefs: string[];\n excludedEntityRefs?: string[];\n }): Promise<{ userEntityRefs: string[] }> {\n const { entityRefs, excludedEntityRefs = [] } = options;\n\n const [userEntityRefs, otherEntityRefs] = partitionEntityRefs(entityRefs);\n const users: string[] = userEntityRefs.filter(\n ref => !excludedEntityRefs.includes(ref),\n );\n const filtered = otherEntityRefs.filter(\n ref => !excludedEntityRefs.includes(ref),\n );\n\n const fields = ['kind', 'metadata.name', 'metadata.namespace', 'relations'];\n let entities: Array<Entity | undefined> = [];\n if (filtered.length > 0) {\n const fetchedEntities = await this.catalog.getEntitiesByRefs(\n {\n entityRefs: filtered,\n fields,\n },\n { credentials: await this.auth.getOwnServiceCredentials() },\n );\n entities = fetchedEntities.items;\n }\n\n const cachedEntityRefs = new Map<string, string[]>();\n\n const mapEntity = async (entity: Entity | undefined): Promise<string[]> => {\n if (!entity) {\n return [];\n }\n\n const currentEntityRef = stringifyEntityRef(entity);\n if (excludedEntityRefs.includes(currentEntityRef)) {\n return [];\n }\n\n if (cachedEntityRefs.has(currentEntityRef)) {\n return cachedEntityRefs.get(currentEntityRef)!;\n }\n\n if (isUserEntity(entity)) {\n return [currentEntityRef];\n }\n\n if (isGroupEntity(entity)) {\n if (!entity.relations?.length) {\n return [];\n }\n\n const groupUsers = entity.relations\n .filter(\n relation =>\n relation.type === RELATION_HAS_MEMBER &&\n isUserEntityRef(relation.targetRef),\n )\n .map(r => r.targetRef);\n\n const childGroupRefs = entity.relations\n .filter(relation => relation.type === RELATION_PARENT_OF)\n .map(r => r.targetRef);\n\n let childGroupUsers: string[][] = [];\n if (childGroupRefs.length > 0) {\n const childGroups = await this.catalog.getEntitiesByRefs(\n {\n entityRefs: childGroupRefs,\n fields,\n },\n { credentials: await this.auth.getOwnServiceCredentials() },\n );\n childGroupUsers = await Promise.all(childGroups.items.map(mapEntity));\n }\n\n const ret = [\n ...new Set([...groupUsers, ...childGroupUsers.flat(2)]),\n ].filter(ref => !excludedEntityRefs.includes(ref));\n cachedEntityRefs.set(currentEntityRef, ret);\n return ret;\n }\n\n if (entity.relations?.length) {\n const ownerRef = entity.relations.find(\n relation => relation.type === RELATION_OWNED_BY,\n )?.targetRef;\n\n if (!ownerRef) {\n return [];\n }\n\n if (isUserEntityRef(ownerRef)) {\n if (excludedEntityRefs.includes(ownerRef)) {\n return [];\n }\n return [ownerRef];\n }\n\n const owner = await this.catalog.getEntityByRef(ownerRef, {\n credentials: await this.auth.getOwnServiceCredentials(),\n });\n const ret = await mapEntity(owner);\n cachedEntityRefs.set(currentEntityRef, ret);\n return ret;\n }\n\n return [];\n };\n\n for (const entity of entities) {\n const u = await mapEntity(entity);\n users.push(...u);\n }\n\n return {\n userEntityRefs: [...new Set(users)]\n .filter(Boolean)\n // Need to filter again after resolving users\n .filter(ref => !excludedEntityRefs.includes(ref)),\n };\n }\n}\n"],"names":["parseEntityRef","stringifyEntityRef","isUserEntity","isGroupEntity","RELATION_HAS_MEMBER","RELATION_PARENT_OF","RELATION_OWNED_BY"],"mappings":"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"}
@@ -4,6 +4,11 @@ var config = require('@backstage/config');
4
4
  var errors = require('@backstage/errors');
5
5
 
6
6
  class NotificationCleaner {
7
+ retention = { years: 1 };
8
+ enabled = true;
9
+ scheduler;
10
+ logger;
11
+ database;
7
12
  constructor(config$1, scheduler, logger, database) {
8
13
  this.scheduler = scheduler;
9
14
  this.logger = logger;
@@ -22,8 +27,6 @@ class NotificationCleaner {
22
27
  });
23
28
  }
24
29
  }
25
- retention = { years: 1 };
26
- enabled = true;
27
30
  async initTaskRunner() {
28
31
  if (!this.enabled) {
29
32
  return;
@@ -1 +1 @@
1
- {"version":3,"file":"NotificationCleaner.cjs.js","sources":["../../src/service/NotificationCleaner.ts"],"sourcesContent":["/*\n * Copyright 2025 The Backstage Authors\n *\n * Licensed under the Apache License, Version 2.0 (the \"License\");\n * you may not use this file except in compliance with the License.\n * You may obtain a copy of the License at\n *\n * http://www.apache.org/licenses/LICENSE-2.0\n *\n * Unless required by applicable law or agreed to in writing, software\n * distributed under the License is distributed on an \"AS IS\" BASIS,\n * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n * See the License for the specific language governing permissions and\n * limitations under the License.\n */\nimport {\n LoggerService,\n SchedulerService,\n SchedulerServiceTaskScheduleDefinition,\n} from '@backstage/backend-plugin-api';\nimport { Config, readDurationFromConfig } from '@backstage/config';\nimport { NotificationsStore } from '../database';\nimport { HumanDuration } from '@backstage/types';\nimport { ForwardedError } from '@backstage/errors';\n\nexport class NotificationCleaner {\n private readonly retention: HumanDuration = { years: 1 };\n private readonly enabled: boolean = true;\n\n constructor(\n config: Config,\n private readonly scheduler: SchedulerService,\n private readonly logger: LoggerService,\n private readonly database: NotificationsStore,\n ) {\n if (config.has('notifications.retention')) {\n const retentionConfig = config.get('notifications.retention');\n if (typeof retentionConfig === 'boolean' && !retentionConfig) {\n logger.info(\n 'Notification retention is disabled, skipping notification cleaner task',\n );\n this.enabled = false;\n return;\n }\n this.retention = readDurationFromConfig(config, {\n key: 'notifications.retention',\n });\n }\n }\n\n async initTaskRunner() {\n if (!this.enabled) {\n return;\n }\n\n const schedule: SchedulerServiceTaskScheduleDefinition = {\n frequency: { cron: '0 0 * * *' },\n timeout: { hours: 1 },\n initialDelay: { hours: 1 },\n scope: 'global',\n };\n\n const taskRunner = this.scheduler.createScheduledTaskRunner(schedule);\n await taskRunner.run({\n id: 'notification-cleaner',\n fn: async () => {\n await this.clearNotifications(\n this.logger,\n this.database,\n this.retention,\n );\n },\n });\n }\n\n private async clearNotifications(\n logger: LoggerService,\n database: NotificationsStore,\n retention: HumanDuration,\n ) {\n logger.info('Starting notification cleaner task');\n try {\n const result = await database.clearNotifications({ maxAge: retention });\n logger.info(\n `Notification cleaner task completed successfully, deleted ${result.deletedCount} notifications`,\n );\n } catch (error) {\n throw new ForwardedError('Notification cleaner task failed', error);\n }\n }\n}\n"],"names":["config","readDurationFromConfig","ForwardedError"],"mappings":";;;;;AAyBO,MAAM,mBAAA,CAAoB;AAAA,EAI/B,WAAA,CACEA,QAAA,EACiB,SAAA,EACA,MAAA,EACA,QAAA,EACjB;AAHiB,IAAA,IAAA,CAAA,SAAA,GAAA,SAAA;AACA,IAAA,IAAA,CAAA,MAAA,GAAA,MAAA;AACA,IAAA,IAAA,CAAA,QAAA,GAAA,QAAA;AAEjB,IAAA,IAAIA,QAAA,CAAO,GAAA,CAAI,yBAAyB,CAAA,EAAG;AACzC,MAAA,MAAM,eAAA,GAAkBA,QAAA,CAAO,GAAA,CAAI,yBAAyB,CAAA;AAC5D,MAAA,IAAI,OAAO,eAAA,KAAoB,SAAA,IAAa,CAAC,eAAA,EAAiB;AAC5D,QAAA,MAAA,CAAO,IAAA;AAAA,UACL;AAAA,SACF;AACA,QAAA,IAAA,CAAK,OAAA,GAAU,KAAA;AACf,QAAA;AAAA,MACF;AACA,MAAA,IAAA,CAAK,SAAA,GAAYC,8BAAuBD,QAAA,EAAQ;AAAA,QAC9C,GAAA,EAAK;AAAA,OACN,CAAA;AAAA,IACH;AAAA,EACF;AAAA,EAtBiB,SAAA,GAA2B,EAAE,KAAA,EAAO,CAAA,EAAE;AAAA,EACtC,OAAA,GAAmB,IAAA;AAAA,EAuBpC,MAAM,cAAA,GAAiB;AACrB,IAAA,IAAI,CAAC,KAAK,OAAA,EAAS;AACjB,MAAA;AAAA,IACF;AAEA,IAAA,MAAM,QAAA,GAAmD;AAAA,MACvD,SAAA,EAAW,EAAE,IAAA,EAAM,WAAA,EAAY;AAAA,MAC/B,OAAA,EAAS,EAAE,KAAA,EAAO,CAAA,EAAE;AAAA,MACpB,YAAA,EAAc,EAAE,KAAA,EAAO,CAAA,EAAE;AAAA,MACzB,KAAA,EAAO;AAAA,KACT;AAEA,IAAA,MAAM,UAAA,GAAa,IAAA,CAAK,SAAA,CAAU,yBAAA,CAA0B,QAAQ,CAAA;AACpE,IAAA,MAAM,WAAW,GAAA,CAAI;AAAA,MACnB,EAAA,EAAI,sBAAA;AAAA,MACJ,IAAI,YAAY;AACd,QAAA,MAAM,IAAA,CAAK,kBAAA;AAAA,UACT,IAAA,CAAK,MAAA;AAAA,UACL,IAAA,CAAK,QAAA;AAAA,UACL,IAAA,CAAK;AAAA,SACP;AAAA,MACF;AAAA,KACD,CAAA;AAAA,EACH;AAAA,EAEA,MAAc,kBAAA,CACZ,MAAA,EACA,QAAA,EACA,SAAA,EACA;AACA,IAAA,MAAA,CAAO,KAAK,oCAAoC,CAAA;AAChD,IAAA,IAAI;AACF,MAAA,MAAM,SAAS,MAAM,QAAA,CAAS,mBAAmB,EAAE,MAAA,EAAQ,WAAW,CAAA;AACtE,MAAA,MAAA,CAAO,IAAA;AAAA,QACL,CAAA,0DAAA,EAA6D,OAAO,YAAY,CAAA,cAAA;AAAA,OAClF;AAAA,IACF,SAAS,KAAA,EAAO;AACd,MAAA,MAAM,IAAIE,qBAAA,CAAe,kCAAA,EAAoC,KAAK,CAAA;AAAA,IACpE;AAAA,EACF;AACF;;;;"}
1
+ {"version":3,"file":"NotificationCleaner.cjs.js","sources":["../../src/service/NotificationCleaner.ts"],"sourcesContent":["/*\n * Copyright 2025 The Backstage Authors\n *\n * Licensed under the Apache License, Version 2.0 (the \"License\");\n * you may not use this file except in compliance with the License.\n * You may obtain a copy of the License at\n *\n * http://www.apache.org/licenses/LICENSE-2.0\n *\n * Unless required by applicable law or agreed to in writing, software\n * distributed under the License is distributed on an \"AS IS\" BASIS,\n * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n * See the License for the specific language governing permissions and\n * limitations under the License.\n */\nimport {\n LoggerService,\n SchedulerService,\n SchedulerServiceTaskScheduleDefinition,\n} from '@backstage/backend-plugin-api';\nimport { Config, readDurationFromConfig } from '@backstage/config';\nimport { NotificationsStore } from '../database';\nimport { HumanDuration } from '@backstage/types';\nimport { ForwardedError } from '@backstage/errors';\n\nexport class NotificationCleaner {\n private readonly retention: HumanDuration = { years: 1 };\n private readonly enabled: boolean = true;\n private readonly scheduler: SchedulerService;\n private readonly logger: LoggerService;\n private readonly database: NotificationsStore;\n\n constructor(\n config: Config,\n scheduler: SchedulerService,\n logger: LoggerService,\n database: NotificationsStore,\n ) {\n this.scheduler = scheduler;\n this.logger = logger;\n this.database = database;\n if (config.has('notifications.retention')) {\n const retentionConfig = config.get('notifications.retention');\n if (typeof retentionConfig === 'boolean' && !retentionConfig) {\n logger.info(\n 'Notification retention is disabled, skipping notification cleaner task',\n );\n this.enabled = false;\n return;\n }\n this.retention = readDurationFromConfig(config, {\n key: 'notifications.retention',\n });\n }\n }\n\n async initTaskRunner() {\n if (!this.enabled) {\n return;\n }\n\n const schedule: SchedulerServiceTaskScheduleDefinition = {\n frequency: { cron: '0 0 * * *' },\n timeout: { hours: 1 },\n initialDelay: { hours: 1 },\n scope: 'global',\n };\n\n const taskRunner = this.scheduler.createScheduledTaskRunner(schedule);\n await taskRunner.run({\n id: 'notification-cleaner',\n fn: async () => {\n await this.clearNotifications(\n this.logger,\n this.database,\n this.retention,\n );\n },\n });\n }\n\n private async clearNotifications(\n logger: LoggerService,\n database: NotificationsStore,\n retention: HumanDuration,\n ) {\n logger.info('Starting notification cleaner task');\n try {\n const result = await database.clearNotifications({ maxAge: retention });\n logger.info(\n `Notification cleaner task completed successfully, deleted ${result.deletedCount} notifications`,\n );\n } catch (error) {\n throw new ForwardedError('Notification cleaner task failed', error);\n }\n }\n}\n"],"names":["config","readDurationFromConfig","ForwardedError"],"mappings":";;;;;AAyBO,MAAM,mBAAA,CAAoB;AAAA,EACd,SAAA,GAA2B,EAAE,KAAA,EAAO,CAAA,EAAE;AAAA,EACtC,OAAA,GAAmB,IAAA;AAAA,EACnB,SAAA;AAAA,EACA,MAAA;AAAA,EACA,QAAA;AAAA,EAEjB,WAAA,CACEA,QAAA,EACA,SAAA,EACA,MAAA,EACA,QAAA,EACA;AACA,IAAA,IAAA,CAAK,SAAA,GAAY,SAAA;AACjB,IAAA,IAAA,CAAK,MAAA,GAAS,MAAA;AACd,IAAA,IAAA,CAAK,QAAA,GAAW,QAAA;AAChB,IAAA,IAAIA,QAAA,CAAO,GAAA,CAAI,yBAAyB,CAAA,EAAG;AACzC,MAAA,MAAM,eAAA,GAAkBA,QAAA,CAAO,GAAA,CAAI,yBAAyB,CAAA;AAC5D,MAAA,IAAI,OAAO,eAAA,KAAoB,SAAA,IAAa,CAAC,eAAA,EAAiB;AAC5D,QAAA,MAAA,CAAO,IAAA;AAAA,UACL;AAAA,SACF;AACA,QAAA,IAAA,CAAK,OAAA,GAAU,KAAA;AACf,QAAA;AAAA,MACF;AACA,MAAA,IAAA,CAAK,SAAA,GAAYC,8BAAuBD,QAAA,EAAQ;AAAA,QAC9C,GAAA,EAAK;AAAA,OACN,CAAA;AAAA,IACH;AAAA,EACF;AAAA,EAEA,MAAM,cAAA,GAAiB;AACrB,IAAA,IAAI,CAAC,KAAK,OAAA,EAAS;AACjB,MAAA;AAAA,IACF;AAEA,IAAA,MAAM,QAAA,GAAmD;AAAA,MACvD,SAAA,EAAW,EAAE,IAAA,EAAM,WAAA,EAAY;AAAA,MAC/B,OAAA,EAAS,EAAE,KAAA,EAAO,CAAA,EAAE;AAAA,MACpB,YAAA,EAAc,EAAE,KAAA,EAAO,CAAA,EAAE;AAAA,MACzB,KAAA,EAAO;AAAA,KACT;AAEA,IAAA,MAAM,UAAA,GAAa,IAAA,CAAK,SAAA,CAAU,yBAAA,CAA0B,QAAQ,CAAA;AACpE,IAAA,MAAM,WAAW,GAAA,CAAI;AAAA,MACnB,EAAA,EAAI,sBAAA;AAAA,MACJ,IAAI,YAAY;AACd,QAAA,MAAM,IAAA,CAAK,kBAAA;AAAA,UACT,IAAA,CAAK,MAAA;AAAA,UACL,IAAA,CAAK,QAAA;AAAA,UACL,IAAA,CAAK;AAAA,SACP;AAAA,MACF;AAAA,KACD,CAAA;AAAA,EACH;AAAA,EAEA,MAAc,kBAAA,CACZ,MAAA,EACA,QAAA,EACA,SAAA,EACA;AACA,IAAA,MAAA,CAAO,KAAK,oCAAoC,CAAA;AAChD,IAAA,IAAI;AACF,MAAA,MAAM,SAAS,MAAM,QAAA,CAAS,mBAAmB,EAAE,MAAA,EAAQ,WAAW,CAAA;AACtE,MAAA,MAAA,CAAO,IAAA;AAAA,QACL,CAAA,0DAAA,EAA6D,OAAO,YAAY,CAAA,cAAA;AAAA,OAClF;AAAA,IACF,SAAS,KAAA,EAAO;AACd,MAAA,MAAM,IAAIE,qBAAA,CAAe,kCAAA,EAAoC,KAAK,CAAA;AAAA,IACpE;AAAA,EACF;AACF;;;;"}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@backstage/plugin-notifications-backend",
3
- "version": "0.5.11",
3
+ "version": "0.5.12-next.0",
4
4
  "backstage": {
5
5
  "role": "backend-plugin",
6
6
  "pluginId": "notifications",
@@ -42,15 +42,15 @@
42
42
  "test": "backstage-cli package test"
43
43
  },
44
44
  "dependencies": {
45
- "@backstage/backend-plugin-api": "^1.4.4",
46
- "@backstage/catalog-model": "^1.7.5",
47
- "@backstage/config": "^1.3.5",
48
- "@backstage/errors": "^1.2.7",
49
- "@backstage/plugin-catalog-node": "^1.19.1",
50
- "@backstage/plugin-notifications-common": "^0.1.1",
51
- "@backstage/plugin-notifications-node": "^0.2.20",
52
- "@backstage/plugin-signals-node": "^0.1.25",
53
- "@backstage/types": "^1.2.2",
45
+ "@backstage/backend-plugin-api": "1.4.5-next.0",
46
+ "@backstage/catalog-model": "1.7.6-next.0",
47
+ "@backstage/config": "1.3.6-next.0",
48
+ "@backstage/errors": "1.2.7",
49
+ "@backstage/plugin-catalog-node": "1.19.2-next.0",
50
+ "@backstage/plugin-notifications-common": "0.1.2-next.0",
51
+ "@backstage/plugin-notifications-node": "0.2.21-next.0",
52
+ "@backstage/plugin-signals-node": "0.1.26-next.0",
53
+ "@backstage/types": "1.2.2",
54
54
  "express": "^4.17.1",
55
55
  "express-promise-router": "^4.1.0",
56
56
  "knex": "^3.0.0",
@@ -58,13 +58,13 @@
58
58
  "uuid": "^11.0.0"
59
59
  },
60
60
  "devDependencies": {
61
- "@backstage/backend-defaults": "^0.13.0",
62
- "@backstage/backend-test-utils": "^1.9.1",
63
- "@backstage/cli": "^0.34.4",
64
- "@backstage/plugin-auth-backend": "^0.25.5",
65
- "@backstage/plugin-auth-backend-module-guest-provider": "^0.2.13",
66
- "@backstage/plugin-events-backend": "^0.5.7",
67
- "@backstage/plugin-signals-backend": "^0.3.9",
61
+ "@backstage/backend-defaults": "0.13.1-next.0",
62
+ "@backstage/backend-test-utils": "1.10.0-next.0",
63
+ "@backstage/cli": "0.34.5-next.0",
64
+ "@backstage/plugin-auth-backend": "0.25.6-next.0",
65
+ "@backstage/plugin-auth-backend-module-guest-provider": "0.2.14-next.0",
66
+ "@backstage/plugin-events-backend": "0.5.8-next.0",
67
+ "@backstage/plugin-signals-backend": "0.3.10-next.0",
68
68
  "@types/express": "^4.17.6",
69
69
  "@types/supertest": "^2.0.8",
70
70
  "supertest": "^7.0.0"