@backstage/plugin-notifications-backend 0.4.4-next.1 → 0.5.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,50 @@
1
1
  # @backstage/plugin-notifications-backend
2
2
 
3
+ ## 0.5.0
4
+
5
+ ### Minor Changes
6
+
7
+ - fc15b77: **BREAKING**: Removed redundant `/health` endpoint, switch to using [the built-in endpoint instead](https://backstage.io/docs/backend-system/core-services/root-health).
8
+
9
+ ### Patch Changes
10
+
11
+ - fc15b77: Deprecated root '/' endpoints, moving them under `/notifications` instead.
12
+ - 5c9cc05: Use native fetch instead of node-fetch
13
+ - Updated dependencies
14
+ - @backstage/plugin-auth-node@0.5.5
15
+ - @backstage/backend-plugin-api@1.1.0
16
+ - @backstage/plugin-catalog-node@1.15.0
17
+ - @backstage/plugin-events-node@0.4.6
18
+ - @backstage/catalog-client@1.9.0
19
+ - @backstage/plugin-notifications-node@0.2.10
20
+ - @backstage/errors@1.2.6
21
+ - @backstage/catalog-model@1.7.2
22
+ - @backstage/config@1.3.1
23
+ - @backstage/plugin-notifications-common@0.0.7
24
+ - @backstage/plugin-signals-node@0.1.15
25
+
26
+ ## 0.5.0-next.2
27
+
28
+ ### Minor Changes
29
+
30
+ - fc15b77: **BREAKING**: Removed redundant `/health` endpoint, switch to using [the built-in endpoint instead](https://backstage.io/docs/backend-system/core-services/root-health).
31
+
32
+ ### Patch Changes
33
+
34
+ - fc15b77: Deprecated root '/' endpoints, moving them under `/notifications` instead.
35
+ - Updated dependencies
36
+ - @backstage/backend-plugin-api@1.1.0-next.2
37
+ - @backstage/errors@1.2.6-next.0
38
+ - @backstage/plugin-auth-node@0.5.5-next.2
39
+ - @backstage/plugin-catalog-node@1.15.0-next.2
40
+ - @backstage/plugin-events-node@0.4.6-next.2
41
+ - @backstage/plugin-notifications-node@0.2.10-next.2
42
+ - @backstage/plugin-signals-node@0.1.15-next.2
43
+ - @backstage/catalog-client@1.9.0-next.2
44
+ - @backstage/catalog-model@1.7.2-next.0
45
+ - @backstage/config@1.3.1-next.0
46
+ - @backstage/plugin-notifications-common@0.0.7-next.0
47
+
3
48
  ## 0.4.4-next.1
4
49
 
5
50
  ### Patch Changes
package/README.md CHANGED
@@ -29,5 +29,5 @@ To be able to do so, `external access` needs to be enabled as described in the [
29
29
  Once the API can be accessed, the request can look like:
30
30
 
31
31
  ```
32
- curl -X POST [YOUR_SERVER_URL]/api/notifications -H "Content-Type: application/json" -H "Authorization: Bearer [BASE64_ENCODED_ACCESS_TOKEN]" -d '{"recipients":{"type":"entity","entityRef":"user:development/guest"},"payload": {"title": "Title of user-targeted external message","description": "The description","link": "http://foo.com/bar","severity": "high","topic": "The topic"}}'
32
+ curl -X POST [YOUR_SERVER_URL]/api/notifications/notifications -H "Content-Type: application/json" -H "Authorization: Bearer [BASE64_ENCODED_ACCESS_TOKEN]" -d '{"recipients":{"type":"entity","entityRef":"user:development/guest"},"payload": {"title": "Title of user-targeted external message","description": "The description","link": "http://foo.com/bar","severity": "high","topic": "The topic"}}'
33
33
  ```
@@ -1,6 +1,5 @@
1
1
  'use strict';
2
2
 
3
- var backendCommon = require('@backstage/backend-common');
4
3
  var express = require('express');
5
4
  var Router = require('express-promise-router');
6
5
  var DatabaseNotificationsStore = require('../database/DatabaseNotificationsStore.cjs.js');
@@ -154,11 +153,7 @@ async function createRouter(options) {
154
153
  };
155
154
  const router = Router__default.default();
156
155
  router.use(express__default.default.json());
157
- router.get("/health", (_, response) => {
158
- logger.info("PONG!");
159
- response.json({ status: "ok" });
160
- });
161
- router.get("/", async (req, res) => {
156
+ const listNotificationsHandler = async (req, res) => {
162
157
  const user = await getUser(req);
163
158
  const opts = {
164
159
  user
@@ -208,7 +203,9 @@ async function createRouter(options) {
208
203
  totalCount,
209
204
  notifications
210
205
  });
211
- });
206
+ };
207
+ router.get("/", listNotificationsHandler);
208
+ router.get("/notifications", listNotificationsHandler);
212
209
  router.get("/status", async (req, res) => {
213
210
  const user = await getUser(req);
214
211
  const status = await store.getStatus({ user });
@@ -236,7 +233,7 @@ async function createRouter(options) {
236
233
  res.json(response);
237
234
  }
238
235
  );
239
- router.get("/:id", async (req, res) => {
236
+ const getNotificationHandler = async (req, res) => {
240
237
  const user = await getUser(req);
241
238
  const opts = {
242
239
  user,
@@ -248,8 +245,10 @@ async function createRouter(options) {
248
245
  throw new errors.NotFoundError("Not found");
249
246
  }
250
247
  res.json(notifications[0]);
251
- });
252
- router.post("/update", async (req, res) => {
248
+ };
249
+ router.get("/:id", getNotificationHandler);
250
+ router.get("/notifications/:id", getNotificationHandler);
251
+ const updateNotificationsHandler = async (req, res) => {
253
252
  const user = await getUser(req);
254
253
  const { ids, read, saved } = req.body;
255
254
  if (!ids || !Array.isArray(ids)) {
@@ -281,7 +280,9 @@ async function createRouter(options) {
281
280
  }
282
281
  const notifications = await store.getNotifications({ ids, user });
283
282
  res.json(notifications);
284
- });
283
+ };
284
+ router.post("/update", updateNotificationsHandler);
285
+ router.post("/notifications/update", updateNotificationsHandler);
285
286
  const sendBroadcastNotification = async (baseNotification, opts, origin) => {
286
287
  const { scope } = opts.payload;
287
288
  const broadcastNotification = {
@@ -374,68 +375,66 @@ async function createRouter(options) {
374
375
  }
375
376
  return notifications;
376
377
  };
377
- router.post(
378
- "/",
379
- async (req, res) => {
380
- const credentials = await httpAuth.credentials(req, {
381
- allow: ["service"]
382
- });
383
- const origin = credentials.principal.subject;
384
- const opts = await processOptions(req.body, origin);
385
- const { recipients, payload } = opts;
386
- const { title, link } = payload;
387
- const notifications = [];
388
- let users = [];
389
- if (!recipients || !title) {
390
- logger.error(`Invalid notification request received`);
391
- throw new errors.InputError(`Invalid notification request received`);
392
- }
393
- if (link) {
394
- try {
395
- validateLink(link);
396
- } catch (e) {
397
- throw new errors.InputError("Invalid link provided", e);
398
- }
378
+ const createNotificationHandler = async (req, res) => {
379
+ const credentials = await httpAuth.credentials(req, {
380
+ allow: ["service"]
381
+ });
382
+ const origin = credentials.principal.subject;
383
+ const opts = await processOptions(req.body, origin);
384
+ const { recipients, payload } = opts;
385
+ const { title, link } = payload;
386
+ const notifications = [];
387
+ let users = [];
388
+ if (!recipients || !title) {
389
+ logger.error(`Invalid notification request received`);
390
+ throw new errors.InputError(`Invalid notification request received`);
391
+ }
392
+ if (link) {
393
+ try {
394
+ validateLink(link);
395
+ } catch (e) {
396
+ throw new errors.InputError("Invalid link provided", e);
399
397
  }
400
- const baseNotification = {
401
- payload: {
402
- ...payload,
403
- severity: payload.severity ?? "normal"
404
- },
405
- origin,
406
- created: /* @__PURE__ */ new Date()
407
- };
408
- if (recipients.type === "broadcast") {
409
- const broadcast = await sendBroadcastNotification(
410
- baseNotification,
411
- opts,
412
- origin
413
- );
414
- notifications.push(broadcast);
415
- } else {
416
- const entityRef = recipients.entityRef;
417
- try {
418
- users = await getUsersForEntityRef.getUsersForEntityRef(
419
- entityRef,
420
- recipients.excludeEntityRef ?? [],
421
- { auth, catalogClient: catalog }
422
- );
423
- } catch (e) {
424
- logger.error(`Failed to resolve notification receivers: ${e}`);
425
- throw new errors.InputError("Failed to resolve notification receivers", e);
426
- }
427
- const userNotifications = await sendUserNotifications(
428
- baseNotification,
429
- users,
430
- opts,
431
- origin
398
+ }
399
+ const baseNotification = {
400
+ payload: {
401
+ ...payload,
402
+ severity: payload.severity ?? "normal"
403
+ },
404
+ origin,
405
+ created: /* @__PURE__ */ new Date()
406
+ };
407
+ if (recipients.type === "broadcast") {
408
+ const broadcast = await sendBroadcastNotification(
409
+ baseNotification,
410
+ opts,
411
+ origin
412
+ );
413
+ notifications.push(broadcast);
414
+ } else {
415
+ const entityRef = recipients.entityRef;
416
+ try {
417
+ users = await getUsersForEntityRef.getUsersForEntityRef(
418
+ entityRef,
419
+ recipients.excludeEntityRef ?? [],
420
+ { auth, catalogClient: catalog }
432
421
  );
433
- notifications.push(...userNotifications);
422
+ } catch (e) {
423
+ logger.error(`Failed to resolve notification receivers: ${e}`);
424
+ throw new errors.InputError("Failed to resolve notification receivers", e);
434
425
  }
435
- res.json(notifications);
426
+ const userNotifications = await sendUserNotifications(
427
+ baseNotification,
428
+ users,
429
+ opts,
430
+ origin
431
+ );
432
+ notifications.push(...userNotifications);
436
433
  }
437
- );
438
- router.use(backendCommon.errorHandler());
434
+ res.json(notifications);
435
+ };
436
+ router.post("/", createNotificationHandler);
437
+ router.post("/notifications", createNotificationHandler);
439
438
  return router;
440
439
  }
441
440
 
@@ -1 +1 @@
1
- {"version":3,"file":"router.cjs.js","sources":["../../src/service/router.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 */\n\nimport { errorHandler, PluginDatabaseManager } from '@backstage/backend-common';\nimport express, { Request } from 'express';\nimport Router from 'express-promise-router';\nimport {\n DatabaseNotificationsStore,\n normalizeSeverity,\n NotificationGetOptions,\n} from '../database';\nimport { v4 as uuid } from 'uuid';\nimport { CatalogApi } from '@backstage/catalog-client';\nimport {\n NotificationProcessor,\n NotificationSendOptions,\n} from '@backstage/plugin-notifications-node';\nimport { InputError, NotFoundError } from '@backstage/errors';\nimport {\n AuthService,\n HttpAuthService,\n LoggerService,\n UserInfoService,\n} from '@backstage/backend-plugin-api';\nimport { SignalsService } from '@backstage/plugin-signals-node';\nimport {\n isNotificationsEnabledFor,\n NewNotificationSignal,\n Notification,\n NotificationReadSignal,\n NotificationSettings,\n notificationSeverities,\n NotificationStatus,\n} from '@backstage/plugin-notifications-common';\nimport { parseEntityOrderFieldParams } from './parseEntityOrderFieldParams';\nimport { getUsersForEntityRef } from './getUsersForEntityRef';\nimport { Config } from '@backstage/config';\n\n/** @internal */\nexport interface RouterOptions {\n logger: LoggerService;\n config: Config;\n database: PluginDatabaseManager;\n auth: AuthService;\n httpAuth: HttpAuthService;\n userInfo: UserInfoService;\n signals?: SignalsService;\n catalog: CatalogApi;\n processors?: NotificationProcessor[];\n}\n\n/** @internal */\nexport async function createRouter(\n options: RouterOptions,\n): Promise<express.Router> {\n const {\n config,\n logger,\n database,\n auth,\n httpAuth,\n userInfo,\n catalog,\n processors = [],\n signals,\n } = options;\n\n const WEB_NOTIFICATION_CHANNEL = 'Web';\n const store = await DatabaseNotificationsStore.create({ database });\n const frontendBaseUrl = config.getString('app.baseUrl');\n\n const getUser = async (req: Request<unknown>) => {\n const credentials = await httpAuth.credentials(req, { allow: ['user'] });\n const info = await userInfo.getUserInfo(credentials);\n return info.userEntityRef;\n };\n\n const getNotificationChannels = () => {\n return [WEB_NOTIFICATION_CHANNEL, ...processors.map(p => p.getName())];\n };\n\n const getNotificationSettings = async (user: string) => {\n const { origins } = await store.getUserNotificationOrigins({ user });\n const settings = await store.getNotificationSettings({ user });\n const channels = getNotificationChannels();\n\n const response: NotificationSettings = {\n channels: channels.map(channel => {\n const channelSettings = settings.channels.find(c => c.id === channel);\n if (channelSettings) {\n return channelSettings;\n }\n return {\n id: channel,\n origins: origins.map(origin => ({\n id: origin,\n enabled: true,\n })),\n };\n }),\n };\n return response;\n };\n\n const isNotificationsEnabled = async (opts: {\n user: string;\n channel: string;\n origin: string;\n }) => {\n const settings = await getNotificationSettings(opts.user);\n return isNotificationsEnabledFor(settings, opts.channel, opts.origin);\n };\n\n const filterProcessors = async (\n notification:\n | Notification\n | ({ origin: string; user: null } & NotificationSendOptions),\n ) => {\n const result: NotificationProcessor[] = [];\n const { payload, user, origin } = notification;\n\n for (const processor of processors) {\n if (user) {\n const enabled = await isNotificationsEnabled({\n user,\n origin,\n channel: processor.getName(),\n });\n if (!enabled) {\n continue;\n }\n }\n\n if (processor.getNotificationFilters) {\n const filters = processor.getNotificationFilters();\n if (filters.minSeverity) {\n if (\n notificationSeverities.indexOf(payload.severity ?? 'normal') >\n notificationSeverities.indexOf(filters.minSeverity)\n ) {\n continue;\n }\n }\n\n if (filters.maxSeverity) {\n if (\n notificationSeverities.indexOf(payload.severity ?? 'normal') <\n notificationSeverities.indexOf(filters.maxSeverity)\n ) {\n continue;\n }\n }\n\n if (filters.excludedTopics && payload.topic) {\n if (filters.excludedTopics.includes(payload.topic)) {\n continue;\n }\n }\n }\n result.push(processor);\n }\n\n return result;\n };\n\n const processOptions = async (\n opts: NotificationSendOptions,\n origin: string,\n ) => {\n const filtered = await filterProcessors({ ...opts, origin, user: null });\n let ret = opts;\n for (const processor of filtered) {\n try {\n ret = processor.processOptions\n ? await processor.processOptions(ret)\n : ret;\n } catch (e) {\n logger.error(\n `Error while processing notification options with ${processor.getName()}: ${e}`,\n );\n }\n }\n return ret;\n };\n\n const preProcessNotification = async (\n notification: Notification,\n opts: NotificationSendOptions,\n ) => {\n const filtered = await filterProcessors(notification);\n let ret = notification;\n for (const processor of filtered) {\n try {\n ret = processor.preProcess\n ? await processor.preProcess(ret, opts)\n : ret;\n } catch (e) {\n logger.error(\n `Error while pre processing notification with ${processor.getName()}: ${e}`,\n );\n }\n }\n return ret;\n };\n\n const postProcessNotification = async (\n notification: Notification,\n opts: NotificationSendOptions,\n ) => {\n const filtered = await filterProcessors(notification);\n for (const processor of filtered) {\n if (processor.postProcess) {\n try {\n await processor.postProcess(notification, opts);\n } catch (e) {\n logger.error(\n `Error while post processing notification with ${processor.getName()}: ${e}`,\n );\n }\n }\n }\n };\n\n const validateLink = (link: string) => {\n const stripLeadingSlash = (s: string) => s.replace(/^\\//, '');\n const ensureTrailingSlash = (s: string) => s.replace(/\\/?$/, '/');\n const url = new URL(\n stripLeadingSlash(link),\n ensureTrailingSlash(frontendBaseUrl),\n );\n if (url.protocol !== 'https:' && url.protocol !== 'http:') {\n throw new Error('Only HTTP/HTTPS links are allowed');\n }\n };\n\n // TODO: Move to use OpenAPI router instead\n const router = Router();\n router.use(express.json());\n\n router.get('/health', (_, response) => {\n logger.info('PONG!');\n response.json({ status: 'ok' });\n });\n\n router.get('/', async (req, res) => {\n const user = await getUser(req);\n const opts: NotificationGetOptions = {\n user: user,\n };\n if (req.query.offset) {\n opts.offset = Number.parseInt(req.query.offset.toString(), 10);\n }\n if (req.query.limit) {\n opts.limit = Number.parseInt(req.query.limit.toString(), 10);\n }\n if (req.query.orderField) {\n opts.orderField = parseEntityOrderFieldParams(req.query);\n }\n if (req.query.search) {\n opts.search = req.query.search.toString();\n }\n if (req.query.read === 'true') {\n opts.read = true;\n } else if (req.query.read === 'false') {\n opts.read = false;\n // or keep undefined\n }\n\n if (req.query.topic) {\n opts.topic = req.query.topic.toString();\n }\n\n if (req.query.saved === 'true') {\n opts.saved = true;\n } else if (req.query.saved === 'false') {\n opts.saved = false;\n // or keep undefined\n }\n if (req.query.createdAfter) {\n const sinceEpoch = Date.parse(String(req.query.createdAfter));\n if (isNaN(sinceEpoch)) {\n throw new InputError('Unexpected date format');\n }\n opts.createdAfter = new Date(sinceEpoch);\n }\n if (req.query.minimumSeverity) {\n opts.minimumSeverity = normalizeSeverity(\n req.query.minimumSeverity.toString(),\n );\n }\n\n const [notifications, totalCount] = await Promise.all([\n store.getNotifications(opts),\n store.getNotificationsCount(opts),\n ]);\n res.json({\n totalCount,\n notifications,\n });\n });\n\n router.get('/status', async (req: Request<any, NotificationStatus>, res) => {\n const user = await getUser(req);\n const status = await store.getStatus({ user });\n res.json(status);\n });\n\n router.get(\n '/settings',\n async (req: Request<any, NotificationSettings>, res) => {\n const user = await getUser(req);\n const response = await getNotificationSettings(user);\n res.json(response);\n },\n );\n\n router.post(\n '/settings',\n async (\n req: Request<any, NotificationSettings, NotificationSettings>,\n res,\n ) => {\n const user = await getUser(req);\n const channels = getNotificationChannels();\n const settings: NotificationSettings = req.body;\n if (settings.channels.some(c => !channels.includes(c.id))) {\n throw new InputError('Invalid channel');\n }\n await store.saveNotificationSettings({ user, settings });\n const response = await getNotificationSettings(user);\n res.json(response);\n },\n );\n\n // Make sure this is the last \"GET\" handler\n router.get('/:id', async (req, res) => {\n const user = await getUser(req);\n const opts: NotificationGetOptions = {\n user: user,\n limit: 1,\n ids: [req.params.id],\n };\n const notifications = await store.getNotifications(opts);\n if (notifications.length !== 1) {\n throw new NotFoundError('Not found');\n }\n res.json(notifications[0]);\n });\n\n router.post('/update', async (req, res) => {\n const user = await getUser(req);\n const { ids, read, saved } = req.body;\n if (!ids || !Array.isArray(ids)) {\n throw new InputError();\n }\n\n if (read === true) {\n await store.markRead({ user, ids });\n\n if (signals) {\n await signals.publish<NotificationReadSignal>({\n recipients: { type: 'user', entityRef: [user] },\n message: { action: 'notification_read', notification_ids: ids },\n channel: 'notifications',\n });\n }\n } else if (read === false) {\n await store.markUnread({ user: user, ids });\n\n if (signals) {\n await signals.publish<NotificationReadSignal>({\n recipients: { type: 'user', entityRef: [user] },\n message: { action: 'notification_unread', notification_ids: ids },\n channel: 'notifications',\n });\n }\n }\n\n if (saved === true) {\n await store.markSaved({ user: user, ids });\n } else if (saved === false) {\n await store.markUnsaved({ user: user, ids });\n }\n\n const notifications = await store.getNotifications({ ids, user: user });\n res.json(notifications);\n });\n\n const sendBroadcastNotification = async (\n baseNotification: Omit<Notification, 'user' | 'id'>,\n opts: NotificationSendOptions,\n origin: string,\n ) => {\n const { scope } = opts.payload;\n const broadcastNotification = {\n ...baseNotification,\n user: null,\n id: uuid(),\n };\n const notification = await preProcessNotification(\n broadcastNotification,\n opts,\n );\n let existingNotification;\n if (scope) {\n existingNotification = await store.getExistingScopeBroadcast({\n scope,\n origin,\n });\n }\n\n let ret = notification;\n if (existingNotification) {\n const restored = await store.restoreExistingNotification({\n id: existingNotification.id,\n notification: { ...notification, user: '' },\n });\n ret = restored ?? notification;\n } else {\n await store.saveBroadcast(notification);\n }\n\n if (signals) {\n await signals.publish<NewNotificationSignal>({\n recipients: { type: 'broadcast' },\n message: {\n action: 'new_notification',\n notification_id: ret.id,\n },\n channel: 'notifications',\n });\n postProcessNotification(ret, opts);\n }\n return notification;\n };\n\n const sendUserNotifications = async (\n baseNotification: Omit<Notification, 'user' | 'id'>,\n users: string[],\n opts: NotificationSendOptions,\n origin: string,\n ) => {\n const notifications = [];\n const { scope } = opts.payload;\n const uniqueUsers = [...new Set(users)];\n for (const user of uniqueUsers) {\n const userNotification = {\n ...baseNotification,\n id: uuid(),\n user,\n };\n const notification = await preProcessNotification(userNotification, opts);\n\n const enabled = await isNotificationsEnabled({\n user,\n channel: WEB_NOTIFICATION_CHANNEL,\n origin: userNotification.origin,\n });\n\n let ret = notification;\n if (enabled) {\n let existingNotification;\n if (scope) {\n existingNotification = await store.getExistingScopeNotification({\n user,\n scope,\n origin,\n });\n }\n\n if (existingNotification) {\n const restored = await store.restoreExistingNotification({\n id: existingNotification.id,\n notification,\n });\n ret = restored ?? notification;\n } else {\n await store.saveNotification(notification);\n }\n\n notifications.push(ret);\n\n if (signals) {\n await signals.publish<NewNotificationSignal>({\n recipients: { type: 'user', entityRef: [user] },\n message: {\n action: 'new_notification',\n notification_id: ret.id,\n },\n channel: 'notifications',\n });\n }\n }\n postProcessNotification(ret, opts);\n }\n return notifications;\n };\n\n // Add new notification\n router.post(\n '/',\n async (req: Request<any, Notification[], NotificationSendOptions>, res) => {\n const credentials = await httpAuth.credentials(req, {\n allow: ['service'],\n });\n\n const origin = credentials.principal.subject;\n const opts = await processOptions(req.body, origin);\n const { recipients, payload } = opts;\n const { title, link } = payload;\n const notifications: Notification[] = [];\n let users = [];\n\n if (!recipients || !title) {\n logger.error(`Invalid notification request received`);\n throw new InputError(`Invalid notification request received`);\n }\n\n if (link) {\n try {\n validateLink(link);\n } catch (e) {\n throw new InputError('Invalid link provided', e);\n }\n }\n\n const baseNotification = {\n payload: {\n ...payload,\n severity: payload.severity ?? 'normal',\n },\n origin,\n created: new Date(),\n };\n\n if (recipients.type === 'broadcast') {\n const broadcast = await sendBroadcastNotification(\n baseNotification,\n opts,\n origin,\n );\n notifications.push(broadcast);\n } else {\n const entityRef = recipients.entityRef;\n\n try {\n users = await getUsersForEntityRef(\n entityRef,\n recipients.excludeEntityRef ?? [],\n { auth, catalogClient: catalog },\n );\n } catch (e) {\n logger.error(`Failed to resolve notification receivers: ${e}`);\n throw new InputError('Failed to resolve notification receivers', e);\n }\n\n const userNotifications = await sendUserNotifications(\n baseNotification,\n users,\n opts,\n origin,\n );\n notifications.push(...userNotifications);\n }\n\n res.json(notifications);\n },\n );\n\n router.use(errorHandler());\n return router;\n}\n"],"names":["DatabaseNotificationsStore","isNotificationsEnabledFor","notificationSeverities","Router","express","parseEntityOrderFieldParams","InputError","normalizeSeverity","NotFoundError","uuid","getUsersForEntityRef","errorHandler"],"mappings":"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"}
1
+ {"version":3,"file":"router.cjs.js","sources":["../../src/service/router.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 */\n\nimport express, { Request, Response } from 'express';\nimport Router from 'express-promise-router';\nimport {\n DatabaseNotificationsStore,\n normalizeSeverity,\n NotificationGetOptions,\n} from '../database';\nimport { v4 as uuid } from 'uuid';\nimport { CatalogApi } from '@backstage/catalog-client';\nimport {\n NotificationProcessor,\n NotificationSendOptions,\n} from '@backstage/plugin-notifications-node';\nimport { InputError, NotFoundError } from '@backstage/errors';\nimport {\n AuthService,\n DatabaseService,\n HttpAuthService,\n LoggerService,\n UserInfoService,\n} from '@backstage/backend-plugin-api';\nimport { SignalsService } from '@backstage/plugin-signals-node';\nimport {\n isNotificationsEnabledFor,\n NewNotificationSignal,\n Notification,\n NotificationReadSignal,\n NotificationSettings,\n notificationSeverities,\n NotificationStatus,\n} from '@backstage/plugin-notifications-common';\nimport { parseEntityOrderFieldParams } from './parseEntityOrderFieldParams';\nimport { getUsersForEntityRef } from './getUsersForEntityRef';\nimport { Config } from '@backstage/config';\n\n/** @internal */\nexport interface RouterOptions {\n logger: LoggerService;\n config: Config;\n database: DatabaseService;\n auth: AuthService;\n httpAuth: HttpAuthService;\n userInfo: UserInfoService;\n signals?: SignalsService;\n catalog: CatalogApi;\n processors?: NotificationProcessor[];\n}\n\n/** @internal */\nexport async function createRouter(\n options: RouterOptions,\n): Promise<express.Router> {\n const {\n config,\n logger,\n database,\n auth,\n httpAuth,\n userInfo,\n catalog,\n processors = [],\n signals,\n } = options;\n\n const WEB_NOTIFICATION_CHANNEL = 'Web';\n const store = await DatabaseNotificationsStore.create({ database });\n const frontendBaseUrl = config.getString('app.baseUrl');\n\n const getUser = async (req: Request<unknown>) => {\n const credentials = await httpAuth.credentials(req, { allow: ['user'] });\n const info = await userInfo.getUserInfo(credentials);\n return info.userEntityRef;\n };\n\n const getNotificationChannels = () => {\n return [WEB_NOTIFICATION_CHANNEL, ...processors.map(p => p.getName())];\n };\n\n const getNotificationSettings = async (user: string) => {\n const { origins } = await store.getUserNotificationOrigins({ user });\n const settings = await store.getNotificationSettings({ user });\n const channels = getNotificationChannels();\n\n const response: NotificationSettings = {\n channels: channels.map(channel => {\n const channelSettings = settings.channels.find(c => c.id === channel);\n if (channelSettings) {\n return channelSettings;\n }\n return {\n id: channel,\n origins: origins.map(origin => ({\n id: origin,\n enabled: true,\n })),\n };\n }),\n };\n return response;\n };\n\n const isNotificationsEnabled = async (opts: {\n user: string;\n channel: string;\n origin: string;\n }) => {\n const settings = await getNotificationSettings(opts.user);\n return isNotificationsEnabledFor(settings, opts.channel, opts.origin);\n };\n\n const filterProcessors = async (\n notification:\n | Notification\n | ({ origin: string; user: null } & NotificationSendOptions),\n ) => {\n const result: NotificationProcessor[] = [];\n const { payload, user, origin } = notification;\n\n for (const processor of processors) {\n if (user) {\n const enabled = await isNotificationsEnabled({\n user,\n origin,\n channel: processor.getName(),\n });\n if (!enabled) {\n continue;\n }\n }\n\n if (processor.getNotificationFilters) {\n const filters = processor.getNotificationFilters();\n if (filters.minSeverity) {\n if (\n notificationSeverities.indexOf(payload.severity ?? 'normal') >\n notificationSeverities.indexOf(filters.minSeverity)\n ) {\n continue;\n }\n }\n\n if (filters.maxSeverity) {\n if (\n notificationSeverities.indexOf(payload.severity ?? 'normal') <\n notificationSeverities.indexOf(filters.maxSeverity)\n ) {\n continue;\n }\n }\n\n if (filters.excludedTopics && payload.topic) {\n if (filters.excludedTopics.includes(payload.topic)) {\n continue;\n }\n }\n }\n result.push(processor);\n }\n\n return result;\n };\n\n const processOptions = async (\n opts: NotificationSendOptions,\n origin: string,\n ) => {\n const filtered = await filterProcessors({ ...opts, origin, user: null });\n let ret = opts;\n for (const processor of filtered) {\n try {\n ret = processor.processOptions\n ? await processor.processOptions(ret)\n : ret;\n } catch (e) {\n logger.error(\n `Error while processing notification options with ${processor.getName()}: ${e}`,\n );\n }\n }\n return ret;\n };\n\n const preProcessNotification = async (\n notification: Notification,\n opts: NotificationSendOptions,\n ) => {\n const filtered = await filterProcessors(notification);\n let ret = notification;\n for (const processor of filtered) {\n try {\n ret = processor.preProcess\n ? await processor.preProcess(ret, opts)\n : ret;\n } catch (e) {\n logger.error(\n `Error while pre processing notification with ${processor.getName()}: ${e}`,\n );\n }\n }\n return ret;\n };\n\n const postProcessNotification = async (\n notification: Notification,\n opts: NotificationSendOptions,\n ) => {\n const filtered = await filterProcessors(notification);\n for (const processor of filtered) {\n if (processor.postProcess) {\n try {\n await processor.postProcess(notification, opts);\n } catch (e) {\n logger.error(\n `Error while post processing notification with ${processor.getName()}: ${e}`,\n );\n }\n }\n }\n };\n\n const validateLink = (link: string) => {\n const stripLeadingSlash = (s: string) => s.replace(/^\\//, '');\n const ensureTrailingSlash = (s: string) => s.replace(/\\/?$/, '/');\n const url = new URL(\n stripLeadingSlash(link),\n ensureTrailingSlash(frontendBaseUrl),\n );\n if (url.protocol !== 'https:' && url.protocol !== 'http:') {\n throw new Error('Only HTTP/HTTPS links are allowed');\n }\n };\n\n // TODO: Move to use OpenAPI router instead\n const router = Router();\n router.use(express.json());\n\n const listNotificationsHandler = async (req: Request, res: Response) => {\n const user = await getUser(req);\n const opts: NotificationGetOptions = {\n user: user,\n };\n if (req.query.offset) {\n opts.offset = Number.parseInt(req.query.offset.toString(), 10);\n }\n if (req.query.limit) {\n opts.limit = Number.parseInt(req.query.limit.toString(), 10);\n }\n if (req.query.orderField) {\n opts.orderField = parseEntityOrderFieldParams(req.query);\n }\n if (req.query.search) {\n opts.search = req.query.search.toString();\n }\n if (req.query.read === 'true') {\n opts.read = true;\n } else if (req.query.read === 'false') {\n opts.read = false;\n // or keep undefined\n }\n\n if (req.query.topic) {\n opts.topic = req.query.topic.toString();\n }\n\n if (req.query.saved === 'true') {\n opts.saved = true;\n } else if (req.query.saved === 'false') {\n opts.saved = false;\n // or keep undefined\n }\n if (req.query.createdAfter) {\n const sinceEpoch = Date.parse(String(req.query.createdAfter));\n if (isNaN(sinceEpoch)) {\n throw new InputError('Unexpected date format');\n }\n opts.createdAfter = new Date(sinceEpoch);\n }\n if (req.query.minimumSeverity) {\n opts.minimumSeverity = normalizeSeverity(\n req.query.minimumSeverity.toString(),\n );\n }\n\n const [notifications, totalCount] = await Promise.all([\n store.getNotifications(opts),\n store.getNotificationsCount(opts),\n ]);\n res.json({\n totalCount,\n notifications,\n });\n };\n\n router.get('/', listNotificationsHandler); // Deprecated endpoint\n router.get('/notifications', listNotificationsHandler);\n\n router.get('/status', async (req: Request<any, NotificationStatus>, res) => {\n const user = await getUser(req);\n const status = await store.getStatus({ user });\n res.json(status);\n });\n\n router.get(\n '/settings',\n async (req: Request<any, NotificationSettings>, res) => {\n const user = await getUser(req);\n const response = await getNotificationSettings(user);\n res.json(response);\n },\n );\n\n router.post(\n '/settings',\n async (\n req: Request<any, NotificationSettings, NotificationSettings>,\n res,\n ) => {\n const user = await getUser(req);\n const channels = getNotificationChannels();\n const settings: NotificationSettings = req.body;\n if (settings.channels.some(c => !channels.includes(c.id))) {\n throw new InputError('Invalid channel');\n }\n await store.saveNotificationSettings({ user, settings });\n const response = await getNotificationSettings(user);\n res.json(response);\n },\n );\n\n const getNotificationHandler = async (req: Request, res: Response) => {\n const user = await getUser(req);\n const opts: NotificationGetOptions = {\n user: user,\n limit: 1,\n ids: [req.params.id],\n };\n const notifications = await store.getNotifications(opts);\n if (notifications.length !== 1) {\n throw new NotFoundError('Not found');\n }\n res.json(notifications[0]);\n };\n\n // Make sure this is the last \"GET\" handler\n router.get('/:id', getNotificationHandler); // Deprecated endpoint\n router.get('/notifications/:id', getNotificationHandler);\n\n const updateNotificationsHandler = async (req: Request, res: Response) => {\n const user = await getUser(req);\n const { ids, read, saved } = req.body;\n if (!ids || !Array.isArray(ids)) {\n throw new InputError();\n }\n\n if (read === true) {\n await store.markRead({ user, ids });\n\n if (signals) {\n await signals.publish<NotificationReadSignal>({\n recipients: { type: 'user', entityRef: [user] },\n message: { action: 'notification_read', notification_ids: ids },\n channel: 'notifications',\n });\n }\n } else if (read === false) {\n await store.markUnread({ user: user, ids });\n\n if (signals) {\n await signals.publish<NotificationReadSignal>({\n recipients: { type: 'user', entityRef: [user] },\n message: { action: 'notification_unread', notification_ids: ids },\n channel: 'notifications',\n });\n }\n }\n\n if (saved === true) {\n await store.markSaved({ user: user, ids });\n } else if (saved === false) {\n await store.markUnsaved({ user: user, ids });\n }\n\n const notifications = await store.getNotifications({ ids, user: user });\n res.json(notifications);\n };\n\n router.post('/update', updateNotificationsHandler); // Deprecated endpoint\n router.post('/notifications/update', updateNotificationsHandler);\n\n const sendBroadcastNotification = async (\n baseNotification: Omit<Notification, 'user' | 'id'>,\n opts: NotificationSendOptions,\n origin: string,\n ) => {\n const { scope } = opts.payload;\n const broadcastNotification = {\n ...baseNotification,\n user: null,\n id: uuid(),\n };\n const notification = await preProcessNotification(\n broadcastNotification,\n opts,\n );\n let existingNotification;\n if (scope) {\n existingNotification = await store.getExistingScopeBroadcast({\n scope,\n origin,\n });\n }\n\n let ret = notification;\n if (existingNotification) {\n const restored = await store.restoreExistingNotification({\n id: existingNotification.id,\n notification: { ...notification, user: '' },\n });\n ret = restored ?? notification;\n } else {\n await store.saveBroadcast(notification);\n }\n\n if (signals) {\n await signals.publish<NewNotificationSignal>({\n recipients: { type: 'broadcast' },\n message: {\n action: 'new_notification',\n notification_id: ret.id,\n },\n channel: 'notifications',\n });\n postProcessNotification(ret, opts);\n }\n return notification;\n };\n\n const sendUserNotifications = async (\n baseNotification: Omit<Notification, 'user' | 'id'>,\n users: string[],\n opts: NotificationSendOptions,\n origin: string,\n ) => {\n const notifications = [];\n const { scope } = opts.payload;\n const uniqueUsers = [...new Set(users)];\n for (const user of uniqueUsers) {\n const userNotification = {\n ...baseNotification,\n id: uuid(),\n user,\n };\n const notification = await preProcessNotification(userNotification, opts);\n\n const enabled = await isNotificationsEnabled({\n user,\n channel: WEB_NOTIFICATION_CHANNEL,\n origin: userNotification.origin,\n });\n\n let ret = notification;\n if (enabled) {\n let existingNotification;\n if (scope) {\n existingNotification = await store.getExistingScopeNotification({\n user,\n scope,\n origin,\n });\n }\n\n if (existingNotification) {\n const restored = await store.restoreExistingNotification({\n id: existingNotification.id,\n notification,\n });\n ret = restored ?? notification;\n } else {\n await store.saveNotification(notification);\n }\n\n notifications.push(ret);\n\n if (signals) {\n await signals.publish<NewNotificationSignal>({\n recipients: { type: 'user', entityRef: [user] },\n message: {\n action: 'new_notification',\n notification_id: ret.id,\n },\n channel: 'notifications',\n });\n }\n }\n postProcessNotification(ret, opts);\n }\n return notifications;\n };\n\n const createNotificationHandler = async (\n req: Request<any, Notification[], NotificationSendOptions>,\n res: Response,\n ) => {\n const credentials = await httpAuth.credentials(req, {\n allow: ['service'],\n });\n\n const origin = credentials.principal.subject;\n const opts = await processOptions(req.body, origin);\n const { recipients, payload } = opts;\n const { title, link } = payload;\n const notifications: Notification[] = [];\n let users = [];\n\n if (!recipients || !title) {\n logger.error(`Invalid notification request received`);\n throw new InputError(`Invalid notification request received`);\n }\n\n if (link) {\n try {\n validateLink(link);\n } catch (e) {\n throw new InputError('Invalid link provided', e);\n }\n }\n\n const baseNotification = {\n payload: {\n ...payload,\n severity: payload.severity ?? 'normal',\n },\n origin,\n created: new Date(),\n };\n\n if (recipients.type === 'broadcast') {\n const broadcast = await sendBroadcastNotification(\n baseNotification,\n opts,\n origin,\n );\n notifications.push(broadcast);\n } else {\n const entityRef = recipients.entityRef;\n\n try {\n users = await getUsersForEntityRef(\n entityRef,\n recipients.excludeEntityRef ?? [],\n { auth, catalogClient: catalog },\n );\n } catch (e) {\n logger.error(`Failed to resolve notification receivers: ${e}`);\n throw new InputError('Failed to resolve notification receivers', e);\n }\n\n const userNotifications = await sendUserNotifications(\n baseNotification,\n users,\n opts,\n origin,\n );\n notifications.push(...userNotifications);\n }\n\n res.json(notifications);\n };\n\n // Add new notification\n router.post('/', createNotificationHandler);\n router.post('/notifications', createNotificationHandler);\n\n return router;\n}\n"],"names":["DatabaseNotificationsStore","isNotificationsEnabledFor","notificationSeverities","Router","express","parseEntityOrderFieldParams","InputError","normalizeSeverity","NotFoundError","uuid","getUsersForEntityRef"],"mappings":";;;;;;;;;;;;;;;;AAiEA,eAAsB,aACpB,OACyB,EAAA;AACzB,EAAM,MAAA;AAAA,IACJ,MAAA;AAAA,IACA,MAAA;AAAA,IACA,QAAA;AAAA,IACA,IAAA;AAAA,IACA,QAAA;AAAA,IACA,QAAA;AAAA,IACA,OAAA;AAAA,IACA,aAAa,EAAC;AAAA,IACd;AAAA,GACE,GAAA,OAAA;AAEJ,EAAA,MAAM,wBAA2B,GAAA,KAAA;AACjC,EAAA,MAAM,QAAQ,MAAMA,qDAAA,CAA2B,MAAO,CAAA,EAAE,UAAU,CAAA;AAClE,EAAM,MAAA,eAAA,GAAkB,MAAO,CAAA,SAAA,CAAU,aAAa,CAAA;AAEtD,EAAM,MAAA,OAAA,GAAU,OAAO,GAA0B,KAAA;AAC/C,IAAM,MAAA,WAAA,GAAc,MAAM,QAAA,CAAS,WAAY,CAAA,GAAA,EAAK,EAAE,KAAO,EAAA,CAAC,MAAM,CAAA,EAAG,CAAA;AACvE,IAAA,MAAM,IAAO,GAAA,MAAM,QAAS,CAAA,WAAA,CAAY,WAAW,CAAA;AACnD,IAAA,OAAO,IAAK,CAAA,aAAA;AAAA,GACd;AAEA,EAAA,MAAM,0BAA0B,MAAM;AACpC,IAAO,OAAA,CAAC,0BAA0B,GAAG,UAAA,CAAW,IAAI,CAAK,CAAA,KAAA,CAAA,CAAE,OAAQ,EAAC,CAAC,CAAA;AAAA,GACvE;AAEA,EAAM,MAAA,uBAAA,GAA0B,OAAO,IAAiB,KAAA;AACtD,IAAM,MAAA,EAAE,SAAY,GAAA,MAAM,MAAM,0BAA2B,CAAA,EAAE,MAAM,CAAA;AACnE,IAAA,MAAM,WAAW,MAAM,KAAA,CAAM,uBAAwB,CAAA,EAAE,MAAM,CAAA;AAC7D,IAAA,MAAM,WAAW,uBAAwB,EAAA;AAEzC,IAAA,MAAM,QAAiC,GAAA;AAAA,MACrC,QAAA,EAAU,QAAS,CAAA,GAAA,CAAI,CAAW,OAAA,KAAA;AAChC,QAAA,MAAM,kBAAkB,QAAS,CAAA,QAAA,CAAS,KAAK,CAAK,CAAA,KAAA,CAAA,CAAE,OAAO,OAAO,CAAA;AACpE,QAAA,IAAI,eAAiB,EAAA;AACnB,UAAO,OAAA,eAAA;AAAA;AAET,QAAO,OAAA;AAAA,UACL,EAAI,EAAA,OAAA;AAAA,UACJ,OAAA,EAAS,OAAQ,CAAA,GAAA,CAAI,CAAW,MAAA,MAAA;AAAA,YAC9B,EAAI,EAAA,MAAA;AAAA,YACJ,OAAS,EAAA;AAAA,WACT,CAAA;AAAA,SACJ;AAAA,OACD;AAAA,KACH;AACA,IAAO,OAAA,QAAA;AAAA,GACT;AAEA,EAAM,MAAA,sBAAA,GAAyB,OAAO,IAIhC,KAAA;AACJ,IAAA,MAAM,QAAW,GAAA,MAAM,uBAAwB,CAAA,IAAA,CAAK,IAAI,CAAA;AACxD,IAAA,OAAOC,mDAA0B,CAAA,QAAA,EAAU,IAAK,CAAA,OAAA,EAAS,KAAK,MAAM,CAAA;AAAA,GACtE;AAEA,EAAM,MAAA,gBAAA,GAAmB,OACvB,YAGG,KAAA;AACH,IAAA,MAAM,SAAkC,EAAC;AACzC,IAAA,MAAM,EAAE,OAAA,EAAS,IAAM,EAAA,MAAA,EAAW,GAAA,YAAA;AAElC,IAAA,KAAA,MAAW,aAAa,UAAY,EAAA;AAClC,MAAA,IAAI,IAAM,EAAA;AACR,QAAM,MAAA,OAAA,GAAU,MAAM,sBAAuB,CAAA;AAAA,UAC3C,IAAA;AAAA,UACA,MAAA;AAAA,UACA,OAAA,EAAS,UAAU,OAAQ;AAAA,SAC5B,CAAA;AACD,QAAA,IAAI,CAAC,OAAS,EAAA;AACZ,UAAA;AAAA;AACF;AAGF,MAAA,IAAI,UAAU,sBAAwB,EAAA;AACpC,QAAM,MAAA,OAAA,GAAU,UAAU,sBAAuB,EAAA;AACjD,QAAA,IAAI,QAAQ,WAAa,EAAA;AACvB,UACE,IAAAC,gDAAA,CAAuB,OAAQ,CAAA,OAAA,CAAQ,QAAY,IAAA,QAAQ,IAC3DA,gDAAuB,CAAA,OAAA,CAAQ,OAAQ,CAAA,WAAW,CAClD,EAAA;AACA,YAAA;AAAA;AACF;AAGF,QAAA,IAAI,QAAQ,WAAa,EAAA;AACvB,UACE,IAAAA,gDAAA,CAAuB,OAAQ,CAAA,OAAA,CAAQ,QAAY,IAAA,QAAQ,IAC3DA,gDAAuB,CAAA,OAAA,CAAQ,OAAQ,CAAA,WAAW,CAClD,EAAA;AACA,YAAA;AAAA;AACF;AAGF,QAAI,IAAA,OAAA,CAAQ,cAAkB,IAAA,OAAA,CAAQ,KAAO,EAAA;AAC3C,UAAA,IAAI,OAAQ,CAAA,cAAA,CAAe,QAAS,CAAA,OAAA,CAAQ,KAAK,CAAG,EAAA;AAClD,YAAA;AAAA;AACF;AACF;AAEF,MAAA,MAAA,CAAO,KAAK,SAAS,CAAA;AAAA;AAGvB,IAAO,OAAA,MAAA;AAAA,GACT;AAEA,EAAM,MAAA,cAAA,GAAiB,OACrB,IAAA,EACA,MACG,KAAA;AACH,IAAM,MAAA,QAAA,GAAW,MAAM,gBAAiB,CAAA,EAAE,GAAG,IAAM,EAAA,MAAA,EAAQ,IAAM,EAAA,IAAA,EAAM,CAAA;AACvE,IAAA,IAAI,GAAM,GAAA,IAAA;AACV,IAAA,KAAA,MAAW,aAAa,QAAU,EAAA;AAChC,MAAI,IAAA;AACF,QAAA,GAAA,GAAM,UAAU,cACZ,GAAA,MAAM,SAAU,CAAA,cAAA,CAAe,GAAG,CAClC,GAAA,GAAA;AAAA,eACG,CAAG,EAAA;AACV,QAAO,MAAA,CAAA,KAAA;AAAA,UACL,CAAoD,iDAAA,EAAA,SAAA,CAAU,OAAQ,EAAC,KAAK,CAAC,CAAA;AAAA,SAC/E;AAAA;AACF;AAEF,IAAO,OAAA,GAAA;AAAA,GACT;AAEA,EAAM,MAAA,sBAAA,GAAyB,OAC7B,YAAA,EACA,IACG,KAAA;AACH,IAAM,MAAA,QAAA,GAAW,MAAM,gBAAA,CAAiB,YAAY,CAAA;AACpD,IAAA,IAAI,GAAM,GAAA,YAAA;AACV,IAAA,KAAA,MAAW,aAAa,QAAU,EAAA;AAChC,MAAI,IAAA;AACF,QAAA,GAAA,GAAM,UAAU,UACZ,GAAA,MAAM,UAAU,UAAW,CAAA,GAAA,EAAK,IAAI,CACpC,GAAA,GAAA;AAAA,eACG,CAAG,EAAA;AACV,QAAO,MAAA,CAAA,KAAA;AAAA,UACL,CAAgD,6CAAA,EAAA,SAAA,CAAU,OAAQ,EAAC,KAAK,CAAC,CAAA;AAAA,SAC3E;AAAA;AACF;AAEF,IAAO,OAAA,GAAA;AAAA,GACT;AAEA,EAAM,MAAA,uBAAA,GAA0B,OAC9B,YAAA,EACA,IACG,KAAA;AACH,IAAM,MAAA,QAAA,GAAW,MAAM,gBAAA,CAAiB,YAAY,CAAA;AACpD,IAAA,KAAA,MAAW,aAAa,QAAU,EAAA;AAChC,MAAA,IAAI,UAAU,WAAa,EAAA;AACzB,QAAI,IAAA;AACF,UAAM,MAAA,SAAA,CAAU,WAAY,CAAA,YAAA,EAAc,IAAI,CAAA;AAAA,iBACvC,CAAG,EAAA;AACV,UAAO,MAAA,CAAA,KAAA;AAAA,YACL,CAAiD,8CAAA,EAAA,SAAA,CAAU,OAAQ,EAAC,KAAK,CAAC,CAAA;AAAA,WAC5E;AAAA;AACF;AACF;AACF,GACF;AAEA,EAAM,MAAA,YAAA,GAAe,CAAC,IAAiB,KAAA;AACrC,IAAA,MAAM,oBAAoB,CAAC,CAAA,KAAc,CAAE,CAAA,OAAA,CAAQ,OAAO,EAAE,CAAA;AAC5D,IAAA,MAAM,sBAAsB,CAAC,CAAA,KAAc,CAAE,CAAA,OAAA,CAAQ,QAAQ,GAAG,CAAA;AAChE,IAAA,MAAM,MAAM,IAAI,GAAA;AAAA,MACd,kBAAkB,IAAI,CAAA;AAAA,MACtB,oBAAoB,eAAe;AAAA,KACrC;AACA,IAAA,IAAI,GAAI,CAAA,QAAA,KAAa,QAAY,IAAA,GAAA,CAAI,aAAa,OAAS,EAAA;AACzD,MAAM,MAAA,IAAI,MAAM,mCAAmC,CAAA;AAAA;AACrD,GACF;AAGA,EAAA,MAAM,SAASC,uBAAO,EAAA;AACtB,EAAO,MAAA,CAAA,GAAA,CAAIC,wBAAQ,CAAA,IAAA,EAAM,CAAA;AAEzB,EAAM,MAAA,wBAAA,GAA2B,OAAO,GAAA,EAAc,GAAkB,KAAA;AACtE,IAAM,MAAA,IAAA,GAAO,MAAM,OAAA,CAAQ,GAAG,CAAA;AAC9B,IAAA,MAAM,IAA+B,GAAA;AAAA,MACnC;AAAA,KACF;AACA,IAAI,IAAA,GAAA,CAAI,MAAM,MAAQ,EAAA;AACpB,MAAK,IAAA,CAAA,MAAA,GAAS,OAAO,QAAS,CAAA,GAAA,CAAI,MAAM,MAAO,CAAA,QAAA,IAAY,EAAE,CAAA;AAAA;AAE/D,IAAI,IAAA,GAAA,CAAI,MAAM,KAAO,EAAA;AACnB,MAAK,IAAA,CAAA,KAAA,GAAQ,OAAO,QAAS,CAAA,GAAA,CAAI,MAAM,KAAM,CAAA,QAAA,IAAY,EAAE,CAAA;AAAA;AAE7D,IAAI,IAAA,GAAA,CAAI,MAAM,UAAY,EAAA;AACxB,MAAK,IAAA,CAAA,UAAA,GAAaC,uDAA4B,CAAA,GAAA,CAAI,KAAK,CAAA;AAAA;AAEzD,IAAI,IAAA,GAAA,CAAI,MAAM,MAAQ,EAAA;AACpB,MAAA,IAAA,CAAK,MAAS,GAAA,GAAA,CAAI,KAAM,CAAA,MAAA,CAAO,QAAS,EAAA;AAAA;AAE1C,IAAI,IAAA,GAAA,CAAI,KAAM,CAAA,IAAA,KAAS,MAAQ,EAAA;AAC7B,MAAA,IAAA,CAAK,IAAO,GAAA,IAAA;AAAA,KACH,MAAA,IAAA,GAAA,CAAI,KAAM,CAAA,IAAA,KAAS,OAAS,EAAA;AACrC,MAAA,IAAA,CAAK,IAAO,GAAA,KAAA;AAAA;AAId,IAAI,IAAA,GAAA,CAAI,MAAM,KAAO,EAAA;AACnB,MAAA,IAAA,CAAK,KAAQ,GAAA,GAAA,CAAI,KAAM,CAAA,KAAA,CAAM,QAAS,EAAA;AAAA;AAGxC,IAAI,IAAA,GAAA,CAAI,KAAM,CAAA,KAAA,KAAU,MAAQ,EAAA;AAC9B,MAAA,IAAA,CAAK,KAAQ,GAAA,IAAA;AAAA,KACJ,MAAA,IAAA,GAAA,CAAI,KAAM,CAAA,KAAA,KAAU,OAAS,EAAA;AACtC,MAAA,IAAA,CAAK,KAAQ,GAAA,KAAA;AAAA;AAGf,IAAI,IAAA,GAAA,CAAI,MAAM,YAAc,EAAA;AAC1B,MAAA,MAAM,aAAa,IAAK,CAAA,KAAA,CAAM,OAAO,GAAI,CAAA,KAAA,CAAM,YAAY,CAAC,CAAA;AAC5D,MAAI,IAAA,KAAA,CAAM,UAAU,CAAG,EAAA;AACrB,QAAM,MAAA,IAAIC,kBAAW,wBAAwB,CAAA;AAAA;AAE/C,MAAK,IAAA,CAAA,YAAA,GAAe,IAAI,IAAA,CAAK,UAAU,CAAA;AAAA;AAEzC,IAAI,IAAA,GAAA,CAAI,MAAM,eAAiB,EAAA;AAC7B,MAAA,IAAA,CAAK,eAAkB,GAAAC,4CAAA;AAAA,QACrB,GAAA,CAAI,KAAM,CAAA,eAAA,CAAgB,QAAS;AAAA,OACrC;AAAA;AAGF,IAAA,MAAM,CAAC,aAAe,EAAA,UAAU,CAAI,GAAA,MAAM,QAAQ,GAAI,CAAA;AAAA,MACpD,KAAA,CAAM,iBAAiB,IAAI,CAAA;AAAA,MAC3B,KAAA,CAAM,sBAAsB,IAAI;AAAA,KACjC,CAAA;AACD,IAAA,GAAA,CAAI,IAAK,CAAA;AAAA,MACP,UAAA;AAAA,MACA;AAAA,KACD,CAAA;AAAA,GACH;AAEA,EAAO,MAAA,CAAA,GAAA,CAAI,KAAK,wBAAwB,CAAA;AACxC,EAAO,MAAA,CAAA,GAAA,CAAI,kBAAkB,wBAAwB,CAAA;AAErD,EAAA,MAAA,CAAO,GAAI,CAAA,SAAA,EAAW,OAAO,GAAA,EAAuC,GAAQ,KAAA;AAC1E,IAAM,MAAA,IAAA,GAAO,MAAM,OAAA,CAAQ,GAAG,CAAA;AAC9B,IAAA,MAAM,SAAS,MAAM,KAAA,CAAM,SAAU,CAAA,EAAE,MAAM,CAAA;AAC7C,IAAA,GAAA,CAAI,KAAK,MAAM,CAAA;AAAA,GAChB,CAAA;AAED,EAAO,MAAA,CAAA,GAAA;AAAA,IACL,WAAA;AAAA,IACA,OAAO,KAAyC,GAAQ,KAAA;AACtD,MAAM,MAAA,IAAA,GAAO,MAAM,OAAA,CAAQ,GAAG,CAAA;AAC9B,MAAM,MAAA,QAAA,GAAW,MAAM,uBAAA,CAAwB,IAAI,CAAA;AACnD,MAAA,GAAA,CAAI,KAAK,QAAQ,CAAA;AAAA;AACnB,GACF;AAEA,EAAO,MAAA,CAAA,IAAA;AAAA,IACL,WAAA;AAAA,IACA,OACE,KACA,GACG,KAAA;AACH,MAAM,MAAA,IAAA,GAAO,MAAM,OAAA,CAAQ,GAAG,CAAA;AAC9B,MAAA,MAAM,WAAW,uBAAwB,EAAA;AACzC,MAAA,MAAM,WAAiC,GAAI,CAAA,IAAA;AAC3C,MAAI,IAAA,QAAA,CAAS,QAAS,CAAA,IAAA,CAAK,CAAK,CAAA,KAAA,CAAC,SAAS,QAAS,CAAA,CAAA,CAAE,EAAE,CAAC,CAAG,EAAA;AACzD,QAAM,MAAA,IAAID,kBAAW,iBAAiB,CAAA;AAAA;AAExC,MAAA,MAAM,KAAM,CAAA,wBAAA,CAAyB,EAAE,IAAA,EAAM,UAAU,CAAA;AACvD,MAAM,MAAA,QAAA,GAAW,MAAM,uBAAA,CAAwB,IAAI,CAAA;AACnD,MAAA,GAAA,CAAI,KAAK,QAAQ,CAAA;AAAA;AACnB,GACF;AAEA,EAAM,MAAA,sBAAA,GAAyB,OAAO,GAAA,EAAc,GAAkB,KAAA;AACpE,IAAM,MAAA,IAAA,GAAO,MAAM,OAAA,CAAQ,GAAG,CAAA;AAC9B,IAAA,MAAM,IAA+B,GAAA;AAAA,MACnC,IAAA;AAAA,MACA,KAAO,EAAA,CAAA;AAAA,MACP,GAAK,EAAA,CAAC,GAAI,CAAA,MAAA,CAAO,EAAE;AAAA,KACrB;AACA,IAAA,MAAM,aAAgB,GAAA,MAAM,KAAM,CAAA,gBAAA,CAAiB,IAAI,CAAA;AACvD,IAAI,IAAA,aAAA,CAAc,WAAW,CAAG,EAAA;AAC9B,MAAM,MAAA,IAAIE,qBAAc,WAAW,CAAA;AAAA;AAErC,IAAI,GAAA,CAAA,IAAA,CAAK,aAAc,CAAA,CAAC,CAAC,CAAA;AAAA,GAC3B;AAGA,EAAO,MAAA,CAAA,GAAA,CAAI,QAAQ,sBAAsB,CAAA;AACzC,EAAO,MAAA,CAAA,GAAA,CAAI,sBAAsB,sBAAsB,CAAA;AAEvD,EAAM,MAAA,0BAAA,GAA6B,OAAO,GAAA,EAAc,GAAkB,KAAA;AACxE,IAAM,MAAA,IAAA,GAAO,MAAM,OAAA,CAAQ,GAAG,CAAA;AAC9B,IAAA,MAAM,EAAE,GAAA,EAAK,IAAM,EAAA,KAAA,KAAU,GAAI,CAAA,IAAA;AACjC,IAAA,IAAI,CAAC,GAAO,IAAA,CAAC,KAAM,CAAA,OAAA,CAAQ,GAAG,CAAG,EAAA;AAC/B,MAAA,MAAM,IAAIF,iBAAW,EAAA;AAAA;AAGvB,IAAA,IAAI,SAAS,IAAM,EAAA;AACjB,MAAA,MAAM,KAAM,CAAA,QAAA,CAAS,EAAE,IAAA,EAAM,KAAK,CAAA;AAElC,MAAA,IAAI,OAAS,EAAA;AACX,QAAA,MAAM,QAAQ,OAAgC,CAAA;AAAA,UAC5C,YAAY,EAAE,IAAA,EAAM,QAAQ,SAAW,EAAA,CAAC,IAAI,CAAE,EAAA;AAAA,UAC9C,OAAS,EAAA,EAAE,MAAQ,EAAA,mBAAA,EAAqB,kBAAkB,GAAI,EAAA;AAAA,UAC9D,OAAS,EAAA;AAAA,SACV,CAAA;AAAA;AACH,KACF,MAAA,IAAW,SAAS,KAAO,EAAA;AACzB,MAAA,MAAM,KAAM,CAAA,UAAA,CAAW,EAAE,IAAA,EAAY,KAAK,CAAA;AAE1C,MAAA,IAAI,OAAS,EAAA;AACX,QAAA,MAAM,QAAQ,OAAgC,CAAA;AAAA,UAC5C,YAAY,EAAE,IAAA,EAAM,QAAQ,SAAW,EAAA,CAAC,IAAI,CAAE,EAAA;AAAA,UAC9C,OAAS,EAAA,EAAE,MAAQ,EAAA,qBAAA,EAAuB,kBAAkB,GAAI,EAAA;AAAA,UAChE,OAAS,EAAA;AAAA,SACV,CAAA;AAAA;AACH;AAGF,IAAA,IAAI,UAAU,IAAM,EAAA;AAClB,MAAA,MAAM,KAAM,CAAA,SAAA,CAAU,EAAE,IAAA,EAAY,KAAK,CAAA;AAAA,KAC3C,MAAA,IAAW,UAAU,KAAO,EAAA;AAC1B,MAAA,MAAM,KAAM,CAAA,WAAA,CAAY,EAAE,IAAA,EAAY,KAAK,CAAA;AAAA;AAG7C,IAAA,MAAM,gBAAgB,MAAM,KAAA,CAAM,iBAAiB,EAAE,GAAA,EAAK,MAAY,CAAA;AACtE,IAAA,GAAA,CAAI,KAAK,aAAa,CAAA;AAAA,GACxB;AAEA,EAAO,MAAA,CAAA,IAAA,CAAK,WAAW,0BAA0B,CAAA;AACjD,EAAO,MAAA,CAAA,IAAA,CAAK,yBAAyB,0BAA0B,CAAA;AAE/D,EAAA,MAAM,yBAA4B,GAAA,OAChC,gBACA,EAAA,IAAA,EACA,MACG,KAAA;AACH,IAAM,MAAA,EAAE,KAAM,EAAA,GAAI,IAAK,CAAA,OAAA;AACvB,IAAA,MAAM,qBAAwB,GAAA;AAAA,MAC5B,GAAG,gBAAA;AAAA,MACH,IAAM,EAAA,IAAA;AAAA,MACN,IAAIG,OAAK;AAAA,KACX;AACA,IAAA,MAAM,eAAe,MAAM,sBAAA;AAAA,MACzB,qBAAA;AAAA,MACA;AAAA,KACF;AACA,IAAI,IAAA,oBAAA;AACJ,IAAA,IAAI,KAAO,EAAA;AACT,MAAuB,oBAAA,GAAA,MAAM,MAAM,yBAA0B,CAAA;AAAA,QAC3D,KAAA;AAAA,QACA;AAAA,OACD,CAAA;AAAA;AAGH,IAAA,IAAI,GAAM,GAAA,YAAA;AACV,IAAA,IAAI,oBAAsB,EAAA;AACxB,MAAM,MAAA,QAAA,GAAW,MAAM,KAAA,CAAM,2BAA4B,CAAA;AAAA,QACvD,IAAI,oBAAqB,CAAA,EAAA;AAAA,QACzB,YAAc,EAAA,EAAE,GAAG,YAAA,EAAc,MAAM,EAAG;AAAA,OAC3C,CAAA;AACD,MAAA,GAAA,GAAM,QAAY,IAAA,YAAA;AAAA,KACb,MAAA;AACL,MAAM,MAAA,KAAA,CAAM,cAAc,YAAY,CAAA;AAAA;AAGxC,IAAA,IAAI,OAAS,EAAA;AACX,MAAA,MAAM,QAAQ,OAA+B,CAAA;AAAA,QAC3C,UAAA,EAAY,EAAE,IAAA,EAAM,WAAY,EAAA;AAAA,QAChC,OAAS,EAAA;AAAA,UACP,MAAQ,EAAA,kBAAA;AAAA,UACR,iBAAiB,GAAI,CAAA;AAAA,SACvB;AAAA,QACA,OAAS,EAAA;AAAA,OACV,CAAA;AACD,MAAA,uBAAA,CAAwB,KAAK,IAAI,CAAA;AAAA;AAEnC,IAAO,OAAA,YAAA;AAAA,GACT;AAEA,EAAA,MAAM,qBAAwB,GAAA,OAC5B,gBACA,EAAA,KAAA,EACA,MACA,MACG,KAAA;AACH,IAAA,MAAM,gBAAgB,EAAC;AACvB,IAAM,MAAA,EAAE,KAAM,EAAA,GAAI,IAAK,CAAA,OAAA;AACvB,IAAA,MAAM,cAAc,CAAC,GAAG,IAAI,GAAA,CAAI,KAAK,CAAC,CAAA;AACtC,IAAA,KAAA,MAAW,QAAQ,WAAa,EAAA;AAC9B,MAAA,MAAM,gBAAmB,GAAA;AAAA,QACvB,GAAG,gBAAA;AAAA,QACH,IAAIA,OAAK,EAAA;AAAA,QACT;AAAA,OACF;AACA,MAAA,MAAM,YAAe,GAAA,MAAM,sBAAuB,CAAA,gBAAA,EAAkB,IAAI,CAAA;AAExE,MAAM,MAAA,OAAA,GAAU,MAAM,sBAAuB,CAAA;AAAA,QAC3C,IAAA;AAAA,QACA,OAAS,EAAA,wBAAA;AAAA,QACT,QAAQ,gBAAiB,CAAA;AAAA,OAC1B,CAAA;AAED,MAAA,IAAI,GAAM,GAAA,YAAA;AACV,MAAA,IAAI,OAAS,EAAA;AACX,QAAI,IAAA,oBAAA;AACJ,QAAA,IAAI,KAAO,EAAA;AACT,UAAuB,oBAAA,GAAA,MAAM,MAAM,4BAA6B,CAAA;AAAA,YAC9D,IAAA;AAAA,YACA,KAAA;AAAA,YACA;AAAA,WACD,CAAA;AAAA;AAGH,QAAA,IAAI,oBAAsB,EAAA;AACxB,UAAM,MAAA,QAAA,GAAW,MAAM,KAAA,CAAM,2BAA4B,CAAA;AAAA,YACvD,IAAI,oBAAqB,CAAA,EAAA;AAAA,YACzB;AAAA,WACD,CAAA;AACD,UAAA,GAAA,GAAM,QAAY,IAAA,YAAA;AAAA,SACb,MAAA;AACL,UAAM,MAAA,KAAA,CAAM,iBAAiB,YAAY,CAAA;AAAA;AAG3C,QAAA,aAAA,CAAc,KAAK,GAAG,CAAA;AAEtB,QAAA,IAAI,OAAS,EAAA;AACX,UAAA,MAAM,QAAQ,OAA+B,CAAA;AAAA,YAC3C,YAAY,EAAE,IAAA,EAAM,QAAQ,SAAW,EAAA,CAAC,IAAI,CAAE,EAAA;AAAA,YAC9C,OAAS,EAAA;AAAA,cACP,MAAQ,EAAA,kBAAA;AAAA,cACR,iBAAiB,GAAI,CAAA;AAAA,aACvB;AAAA,YACA,OAAS,EAAA;AAAA,WACV,CAAA;AAAA;AACH;AAEF,MAAA,uBAAA,CAAwB,KAAK,IAAI,CAAA;AAAA;AAEnC,IAAO,OAAA,aAAA;AAAA,GACT;AAEA,EAAM,MAAA,yBAAA,GAA4B,OAChC,GAAA,EACA,GACG,KAAA;AACH,IAAA,MAAM,WAAc,GAAA,MAAM,QAAS,CAAA,WAAA,CAAY,GAAK,EAAA;AAAA,MAClD,KAAA,EAAO,CAAC,SAAS;AAAA,KAClB,CAAA;AAED,IAAM,MAAA,MAAA,GAAS,YAAY,SAAU,CAAA,OAAA;AACrC,IAAA,MAAM,IAAO,GAAA,MAAM,cAAe,CAAA,GAAA,CAAI,MAAM,MAAM,CAAA;AAClD,IAAM,MAAA,EAAE,UAAY,EAAA,OAAA,EAAY,GAAA,IAAA;AAChC,IAAM,MAAA,EAAE,KAAO,EAAA,IAAA,EAAS,GAAA,OAAA;AACxB,IAAA,MAAM,gBAAgC,EAAC;AACvC,IAAA,IAAI,QAAQ,EAAC;AAEb,IAAI,IAAA,CAAC,UAAc,IAAA,CAAC,KAAO,EAAA;AACzB,MAAA,MAAA,CAAO,MAAM,CAAuC,qCAAA,CAAA,CAAA;AACpD,MAAM,MAAA,IAAIH,kBAAW,CAAuC,qCAAA,CAAA,CAAA;AAAA;AAG9D,IAAA,IAAI,IAAM,EAAA;AACR,MAAI,IAAA;AACF,QAAA,YAAA,CAAa,IAAI,CAAA;AAAA,eACV,CAAG,EAAA;AACV,QAAM,MAAA,IAAIA,iBAAW,CAAA,uBAAA,EAAyB,CAAC,CAAA;AAAA;AACjD;AAGF,IAAA,MAAM,gBAAmB,GAAA;AAAA,MACvB,OAAS,EAAA;AAAA,QACP,GAAG,OAAA;AAAA,QACH,QAAA,EAAU,QAAQ,QAAY,IAAA;AAAA,OAChC;AAAA,MACA,MAAA;AAAA,MACA,OAAA,sBAAa,IAAK;AAAA,KACpB;AAEA,IAAI,IAAA,UAAA,CAAW,SAAS,WAAa,EAAA;AACnC,MAAA,MAAM,YAAY,MAAM,yBAAA;AAAA,QACtB,gBAAA;AAAA,QACA,IAAA;AAAA,QACA;AAAA,OACF;AACA,MAAA,aAAA,CAAc,KAAK,SAAS,CAAA;AAAA,KACvB,MAAA;AACL,MAAA,MAAM,YAAY,UAAW,CAAA,SAAA;AAE7B,MAAI,IAAA;AACF,QAAA,KAAA,GAAQ,MAAMI,yCAAA;AAAA,UACZ,SAAA;AAAA,UACA,UAAA,CAAW,oBAAoB,EAAC;AAAA,UAChC,EAAE,IAAM,EAAA,aAAA,EAAe,OAAQ;AAAA,SACjC;AAAA,eACO,CAAG,EAAA;AACV,QAAO,MAAA,CAAA,KAAA,CAAM,CAA6C,0CAAA,EAAA,CAAC,CAAE,CAAA,CAAA;AAC7D,QAAM,MAAA,IAAIJ,iBAAW,CAAA,0CAAA,EAA4C,CAAC,CAAA;AAAA;AAGpE,MAAA,MAAM,oBAAoB,MAAM,qBAAA;AAAA,QAC9B,gBAAA;AAAA,QACA,KAAA;AAAA,QACA,IAAA;AAAA,QACA;AAAA,OACF;AACA,MAAc,aAAA,CAAA,IAAA,CAAK,GAAG,iBAAiB,CAAA;AAAA;AAGzC,IAAA,GAAA,CAAI,KAAK,aAAa,CAAA;AAAA,GACxB;AAGA,EAAO,MAAA,CAAA,IAAA,CAAK,KAAK,yBAAyB,CAAA;AAC1C,EAAO,MAAA,CAAA,IAAA,CAAK,kBAAkB,yBAAyB,CAAA;AAEvD,EAAO,OAAA,MAAA;AACT;;;;"}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@backstage/plugin-notifications-backend",
3
- "version": "0.4.4-next.1",
3
+ "version": "0.5.0",
4
4
  "backstage": {
5
5
  "role": "backend-plugin",
6
6
  "pluginId": "notifications",
@@ -39,17 +39,17 @@
39
39
  },
40
40
  "dependencies": {
41
41
  "@backstage/backend-common": "^0.25.0",
42
- "@backstage/backend-plugin-api": "1.1.0-next.1",
43
- "@backstage/catalog-client": "1.9.0-next.1",
44
- "@backstage/catalog-model": "1.7.1",
45
- "@backstage/config": "1.3.0",
46
- "@backstage/errors": "1.2.5",
47
- "@backstage/plugin-auth-node": "0.5.5-next.1",
48
- "@backstage/plugin-catalog-node": "1.15.0-next.1",
49
- "@backstage/plugin-events-node": "0.4.6-next.1",
50
- "@backstage/plugin-notifications-common": "0.0.6",
51
- "@backstage/plugin-notifications-node": "0.2.10-next.1",
52
- "@backstage/plugin-signals-node": "0.1.15-next.1",
42
+ "@backstage/backend-plugin-api": "^1.1.0",
43
+ "@backstage/catalog-client": "^1.9.0",
44
+ "@backstage/catalog-model": "^1.7.2",
45
+ "@backstage/config": "^1.3.1",
46
+ "@backstage/errors": "^1.2.6",
47
+ "@backstage/plugin-auth-node": "^0.5.5",
48
+ "@backstage/plugin-catalog-node": "^1.15.0",
49
+ "@backstage/plugin-events-node": "^0.4.6",
50
+ "@backstage/plugin-notifications-common": "^0.0.7",
51
+ "@backstage/plugin-notifications-node": "^0.2.10",
52
+ "@backstage/plugin-signals-node": "^0.1.15",
53
53
  "express": "^4.17.1",
54
54
  "express-promise-router": "^4.1.0",
55
55
  "knex": "^3.0.0",
@@ -58,13 +58,13 @@
58
58
  "yn": "^4.0.0"
59
59
  },
60
60
  "devDependencies": {
61
- "@backstage/backend-defaults": "0.6.0-next.1",
62
- "@backstage/backend-test-utils": "1.2.0-next.1",
63
- "@backstage/cli": "0.29.3-next.1",
64
- "@backstage/plugin-auth-backend": "0.24.1-next.1",
65
- "@backstage/plugin-auth-backend-module-guest-provider": "0.2.3-next.1",
66
- "@backstage/plugin-events-backend": "0.4.0-next.1",
67
- "@backstage/plugin-signals-backend": "0.2.4-next.1",
61
+ "@backstage/backend-defaults": "^0.6.0",
62
+ "@backstage/backend-test-utils": "^1.2.0",
63
+ "@backstage/cli": "^0.29.4",
64
+ "@backstage/plugin-auth-backend": "^0.24.1",
65
+ "@backstage/plugin-auth-backend-module-guest-provider": "^0.2.3",
66
+ "@backstage/plugin-events-backend": "^0.4.0",
67
+ "@backstage/plugin-signals-backend": "^0.2.4",
68
68
  "@types/express": "^4.17.6",
69
69
  "@types/supertest": "^2.0.8",
70
70
  "msw": "^1.0.0",