🚀 Socket Launch Week Day 5:Introducing Repository Access Permissions and Custom Roles.Learn more
Sign In

@fluojs/notifications

Package Overview
Dependencies
Maintainers
1
Versions
8
Alerts
File Explorer

Advanced tools

Socket logo

Install Socket

Detect and block malicious and high-risk dependencies

Install

@fluojs/notifications

Channel-agnostic notification orchestration with optional queue and lifecycle event seams for Fluo.

Source
npmnpm
Version
1.0.0-beta.3
Version published
Weekly downloads
49
-79.58%
Maintainers
1
Weekly downloads
 
Created
Source

@fluojs/notifications

English 한국어

Channel-agnostic notification orchestration for fluo. It freezes the shared contract for notification channels, provides a Nest-like module API, and exposes optional queue-backed delivery and lifecycle event publication seams.

Table of Contents

Installation

npm install @fluojs/notifications

When to Use

  • When you want one shared dispatch contract for multiple notification channels without coupling sibling packages to each other.
  • When application code should depend on NotificationsService instead of provider-specific SDKs or transport details.
  • When bulk delivery may need to be offloaded to a queue, but direct in-process dispatch should still remain available.
  • When notification lifecycle events (requested, queued, delivered, failed) should be observable through an event publication seam.

Quick Start

1. Register the foundation module

Register notifications with NotificationsModule.forRoot(...) or NotificationsModule.forRootAsync(...).

import { Module } from '@fluojs/core';
import {
  NotificationsModule,
  type NotificationChannel,
} from '@fluojs/notifications';

const emailChannel: NotificationChannel = {
  channel: 'email',
  async send(notification) {
    console.log('sending email', notification.subject, notification.payload);

    return {
      externalId: 'email-123',
      metadata: { provider: 'demo-email' },
    };
  },
};

@Module({
  imports: [
    NotificationsModule.forRoot({
      channels: [emailChannel],
    }),
  ],
})
export class AppModule {}

2. Inject NotificationsService

import { Inject } from '@fluojs/core';
import { NotificationsService } from '@fluojs/notifications';

@Inject(NotificationsService)
export class WelcomeService {
  constructor(private readonly notifications: NotificationsService) {}

  async sendWelcomeEmail(userId: string, email: string) {
    await this.notifications.dispatch({
      channel: 'email',
      recipients: [email],
      subject: 'Welcome to fluo',
      payload: {
        template: 'welcome-email',
        userId,
      },
    });
  }
}

NotificationsModule.forRoot(...) and NotificationsModule.forRootAsync(...) export NotificationsService, NOTIFICATIONS, and NOTIFICATION_CHANNELS as global providers. Application services should declare dependencies with fluo's class-level @Inject(...) decorator so the standard-decorator DI container can resolve the service without parameter decorators.

Common Patterns

Queue-backed bulk delivery

Use the optional queue seam when many notifications should be deferred to background workers.

NotificationsModule.forRoot({
  channels: [emailChannel],
  queue: {
    adapter: {
      async enqueue(job) {
        return queue.enqueue(job);
      },
      async enqueueMany(jobs) {
        return Promise.all(jobs.map((job) => queue.enqueue(job)));
      },
    },
    bulkThreshold: 50,
  },
});

Behavioral contract notes:

  • Bulk queue delegation starts when the notification count reaches bulkThreshold.
  • dispatch() stays direct by default even when a queue adapter is configured. Use dispatch(..., { queue: true }) to opt one single notification into queue-backed delivery.
  • Use dispatch(..., { queue: false }) to force direct delivery even when a queue adapter exists.
  • Queue-backed delivery is opt-in for single dispatch and threshold-driven for dispatchMany(...).
  • dispatchMany(..., { continueOnError: true }) collects failures instead of throwing on the first failed direct delivery.
  • When queue enqueue fails, the service emits deterministic notification.dispatch.failed lifecycle events before rethrowing the enqueue error to the caller.
  • If enqueueMany(...) is unavailable, bulk queue delivery falls back to enqueueing each job individually.
  • The foundation package does not assume or import a concrete queue implementation.

Lifecycle publication through an event publisher

Publish caller-visible lifecycle events without coupling the foundation package to @fluojs/event-bus directly.

NotificationsModule.forRoot({
  channels: [emailChannel],
  events: {
    publishLifecycleEvents: true,
    publisher: {
      async publish(event) {
        await eventBus.publish(event);
      },
    },
  },
});

Published event names:

  • notification.dispatch.requested
  • notification.dispatch.queued
  • notification.dispatch.delivered
  • notification.dispatch.failed

If events.publisher is configured, lifecycle event publication defaults to on unless publishLifecycleEvents: false is set. Channel deliveries that omit externalId receive a deterministic fallback delivery id so dispatch results remain stable for callers.

Intentional limitations

The foundation package intentionally does not:

  • ship built-in email, Slack, or Discord implementations
  • inspect process.env directly
  • depend on @fluojs/queue or @fluojs/event-bus concrete runtime types
  • encode provider-specific payload semantics into the shared contract

These limitations are part of the package contract so leaf packages can evolve independently while sharing one stable orchestration layer.

Public API Overview

Core

  • NotificationsModule.forRoot(options) / NotificationsModule.forRootAsync(options)
  • NotificationsService
  • NotificationsService.createPlatformStatusSnapshot()
  • NOTIFICATIONS
  • NOTIFICATION_CHANNELS

Contracts

  • NotificationDispatchRequest
  • NotificationDispatchOptions
  • NotificationDispatchManyOptions
  • NotificationDispatchResult
  • NotificationDispatchFailure
  • NotificationDispatchStatus
  • NotificationChannel
  • NotificationChannelContext
  • NotificationChannelDelivery
  • NotificationPayload
  • NotificationsQueueAdapter
  • NotificationsQueueJob
  • NotificationsAsyncModuleOptions
  • NotificationsEventsOptions
  • NotificationsEventPublisher
  • NotificationLifecycleEvent
  • NotificationLifecycleEventName

Status and errors

  • createNotificationsPlatformStatusSnapshot(...)
  • NotificationsPlatformStatusSnapshot
  • NotificationsStatusAdapterInput
  • NotificationsConfigurationError
  • NotificationChannelNotFoundError
  • NotificationQueueNotConfiguredError

Status snapshots include operationMode, dependency diagnostics, ownership, readiness, and health fields for platform diagnostics.

  • @fluojs/queue: Recommended when bulk notification delivery should run in the background.
  • @fluojs/event-bus: Recommended when notification lifecycle events should be published to the wider app.
  • @fluojs/config: Recommended for passing provider configuration into forRootAsync() without direct environment access.

Example Sources

  • packages/notifications/src/module.test.ts: Module registration, async wiring, queue seam, and tolerant bulk dispatch examples.
  • packages/notifications/src/public-surface.test.ts: Public contract verification for root exports and TypeScript-only types.
  • packages/notifications/src/status.test.ts: Health/readiness contract examples.

Keywords

fluo

FAQs

Package last updated on 04 May 2026

Did you know?

Socket

Socket for GitHub automatically highlights issues in each pull request and monitors the health of all your open source dependencies. Discover the contents of your packages and block harmful activity before you install or update your dependencies.

Install

Related posts