npm package discovery and stats viewer.

Discover Tips

  • General search

    [free text search, go nuts!]

  • Package details

    pkg:[package-name]

  • User packages

    @[username]

Sponsor

Optimize Toolset

I’ve always been into building performant and accessible sites, but lately I’ve been taking it extremely seriously. So much so that I’ve been building a tool to help me optimize and monitor the sites that I build to make sure that I’m making an attempt to offer the best experience to those who visit them. If you’re into performant, accessible and SEO friendly sites, you might like it too! You can check it out at Optimize Toolset.

About

Hi, 👋, I’m Ryan Hefner  and I built this site for me, and you! The goal of this site was to provide an easy way for me to check the stats on my npm packages, both for prioritizing issues and updates, and to give me a little kick in the pants to keep up on stuff.

As I was building it, I realized that I was actually using the tool to build the tool, and figured I might as well put this out there and hopefully others will find it to be a fast and useful way to search and browse npm packages as I have.

If you’re interested in other things I’m working on, follow me on Twitter or check out the open source projects I’ve been publishing on GitHub.

I am also working on a Twitter bot for this site to tweet the most popular, newest, random packages from npm. Please follow that account now and it will start sending out packages soon–ish.

Open Software & Tools

This site wouldn’t be possible without the immense generosity and tireless efforts from the people who make contributions to the world and share their work via open source initiatives. Thank you 🙏

© 2024 – Pkg Stats / Ryan Hefner

@janus-idp/plugin-notifications-backend

v1.6.0

Published

This Backstage backend plugin provides REST API endpoint for the notifications.

Downloads

52

Readme

Notifications

This Backstage backend plugin provides REST API endpoint for the notifications.

It's backed by a relational database, so far tested with PostgreSQL.

Deploying as a dynamic plugin

The notifications backend plugin can be loaded either as a static or a dynamic plugin.

To install it as a dynamic plugin, please follow instructions here: https://github.com/janus-idp/backstage-showcase/blob/main/showcase-docs/dynamic-plugins.md#installing-a-dynamic-plugin-package-in-the-showcase

To install it as a static plugin, several steps are required as described below for both the legacy backend and the new backend system.

In any case, do not miss the info about configuration and especially about creating entities in the Catalog as described below.

Getting started

The plugin uses a relational database to persist messages, it has been tested with the SQLite and PostgreSQL.

Upon the backend's plugin start, the backstage_plugin_notifications database and its tables are created automatically.

Optional: PostgreSQL setup

To use the Backstage's default SQLite database, no specific configuration is needed.

The following steps describe requirements for PostgreSQL:

  • Install PostgresSQL DB
  • Configure Postgres for tcp/ip Open Postgres conf file for editing:
sudo vi /var/lib/pgsql/data/pg_hba.conf

Add this line:

host   all             postgres       127.0.0.1/32                          password
  • Start Postgres server:
sudo systemctl enable --now postgresql.service

Backstage configuration for PostgreSQL

If PostgreSQL is used, additional configuration in the app-config.yaml file or app-config.local.yaml file is needed. Example:

backend:
  database:
    client: pg
    connection:
      host: 127.0.0.1
      port: 5432
      user: postgres
      password: your_secret
    knexConfig:
      pool:
        min: 3
        max: 12
        acquireTimeoutMillis: 60000
        idleTimeoutMillis: 60000
  cache:
    store: memory

Deploy as a static plugin

Add NPM dependency

cd packages/backend
yarn add @janus-idp/plugin-notifications-backend

Add the backend-plugin for the legacy backend

Create a packages/backend/src/plugins/notifications.ts file with the following content:

import { CatalogClient } from '@backstage/catalog-client';

import { Router } from 'express';

import { createRouter } from '@janus-idp/plugin-notifications-backend';

import { PluginEnvironment } from '../types';

export default async function createPlugin(
  env: PluginEnvironment,
): Promise<Router> {
  return await createRouter({
    identity: env.identity,
    logger: env.logger,
    permissions: env.permissions,
    tokenManager: env.tokenManager,
    database: env.database,
    discovery: env.discovery,
    config: env.config,
  });
}

In the packages/backend/src/index.ts, add the following:

import notifications from './plugins/notifications';
...
// Existing code for reference:
const apiRouter = Router();
...
// New code:
const notificationsEnv = useHotMemoize(module, () =>
  createEnv('notifications'),
);
apiRouter.use('/notifications', await notifications(notificationsEnv));

Installing the plugin for the new backend system

Add the following code to the packages/backend/src/index.ts:

const backend = createBackend();

/* highlight-add-next-line */
backend.add(import('@janus-idp/backstage-plugin-notifications-backend/alpha'));

backend.start();

Configuration

Optional: Plugin configuration

If you have issues to create valid JWT tokens by an external caller, use following option to bypass the service-to-service configuration for them:

notifications:
  # Workaround for issues with external caller JWT token creation.
  # When following config option is not provided and the request "authentication" header is missing, the request is ALLOWED by default
  # When following option is present, the request must contain either a valid JWT token or that provided shared secret in the "notifications-secret" header
  externalCallerSecret: your-secret-token-shared-with-external-services

We suggest using HTTPS to help prevent leaking the shared secret.

Example Request:

curl -X POST http://localhost:7007/api/notifications/notifications -H "Content-Type: application/json" -H "notifications-secret: your-secret-token-shared-with-external-services" -d '{"title":"my-title","origin":"my-origin","message":"message one","topic":"my-topic"}'

Notes:

  • The externalCallerSecret is an workaround, exclusive use of JWT tokens will probably be required in the future.
  • Sharing the same shared secret with the "auth.secret" option is not recommended.

Authentication

Please refer to https://backstage.io/docs/auth/ to set-up authentication.

The Notifications flows are based on the identity of the user.

All targetUsers, `targetGroups`` or signed-in users receiving notifications must have corresponding entities created in the Catalog. Refer to https://backstage.io/docs/auth/identity-resolver for details.

For the purpose of development, there is a users.yaml file listing example data to create a Guest user entity.

Authorization

Every service endpoint is guarded by a permission check, enabled by default.

It is up to particular deployment to provide corresponding permission policies based on https://backstage.io/docs/permissions/writing-a-policy. To register your permission policies, refer to https://backstage.io/docs/permissions/getting-started#integrating-the-permission-framework-with-your-backstage-instance.

Service-to-service and External Calls

The notifications-backend is expected to be called by frontend plugins (including the Notifications frontend plugin), other backend plugins or external services.

To configure those two flows, refer to:

  • https://backstage.io/docs/auth/service-to-service-auth.
  • https://backstage.io/docs/auth/service-to-service-auth#usage-in-external-callers

Important: User entities in Catalog

The notifications require target users or groups (as receivers) to be listed in the Catalog.

As an example how to do it, add following to the config:

catalog:
  import:
    entityFilename: catalog-info.yaml
    pullRequestBranchName: backstage-integration
  rules:
    # *** Here is a new change:
    - allow: [Component, System, API, Resource, Location, User, Group]
  locations:
    # Local example data, file locations are relative to the backend process, typically `packages/backend`
    - type: file
      # *** Here is a new change, refers to a file stored in the root of the Backstage:
      target: ../../plugins/notifications-backend/users.yaml

REST API

See src/openapi.yaml for full OpenAPI spec.

Posting a notification

A notification without target users or groups is considered a system notification. That means it is intended for all users (listed among Updates in the UI).

Request (User message and then system message):

curl -X POST http://localhost:7007/api/notifications/notifications -H "Content-Type: application/json" -d '{"title": "My message title", "message": "I have nothing to say", "origin": "my-origin", "topic":"my topic", "targetUsers": ["default/guest"], "actions": [{"title": "my-title", "url": "http://foo.bar"}, {"title": "another action", "url": "https://foo.foo.bar"}]}'

Optionally add -H "Authorization: Bearer eyJh..... with a valid JWT token if the service-to-service authorization is enabled (see above).

Response:

{ "msgid": "2daac6ff-3aaa-420d-b755-d94e54248310" }

Get notifications

Page number starts at '1'. Page number '0' along with page size '0' means no paging. User parameter is mandatory because it is needed for message status and filtering (read/unread).

Query parameters:

  • pageSize. 0 means no paging.
  • pageNumber. first page is 1. 0 means no paging.
  • orderBy.
  • orderByDirec. asc/desc
  • containsText. filter title and message containing this text (case insensitive)
  • createdAfter. fetch notifications created after this point in time
  • messageScope. all/user/system. fetch notifications intended for specific user or system notifications or both
  • read. true/false (read/unread)

Request:

curl 'http://localhost:7007/api/notifications/notifications?read=false&pageNumber=0&pageSize=0'

Response:

[
  {
    "id": "2daac6ff-3aaa-420d-b755-d94e54248310",
    "created": "2023-10-30T13:48:34.931Z",
    "isSystem": false,
    "readByUser": false,
    "origin": "my-origin",
    "title": "My title",
    "message": "I have nothing to tell",
    "topic": "my-topic",
    "actions": []
  }
]

Get count of notifications

User parameter is mandatory because it is needed for filtering (read/unread).

Important: Logged-in user:

The query requires a signed-in user whose entity is listed in the Catalog. With this condition is met, the HTTP Authorization header contains a JWT token with the user's identity.

Optionally add -H "Authorization: Bearer eyJh..... with a valid JWT token to the curl commands bellow.

Query parameters:

  • containsText. filter title and message containing this text (case insensitive)
  • createdAfter. fetch notifications created after this point in time
  • messageScope. all/user/system. fetch notifications intended for specific user or system notifications or both
  • read. true/false (read/unread)

Request:

curl http://localhost:7007/api/notifications/notifications/count

Response:

{ "count": "1" }

Set notification as read/unread

Request:

curl -X PUT 'http://localhost:7007/api/notifications/notifications/read?messageId=48bbf896-4b7c-4b68-a446-246b6a801000&read=true'

Response: A HTTP Status Code

Building a client for the API

We supply an Open API spec YAML file at src/openapi.yaml.