Skip to content

sourcefuse/loopback4-message-bus-connector

Repository files navigation

ARC By SourceFuse logo

npm version Sonar Quality Gate Synk Status GitHub contributors downloads License Powered By LoopBack 4

Overview

This is a LoopBack 4 extension for adding message queue and event based communication to your LoopBack applications. It provides a unified and extensible interface for working with different queuing systems.

Supported Connectors

  • SQSConnector – Integrates with AWS SQS using @aws-sdk/client-sqs. Supports both message sending and consumption with polling, visibility timeout, etc.

  • BullMQConnector – Integrates with BullMQ (Redis-based queue). Supports advanced job options like retries, backoff, consumer concurrency, and job polling.

  • EventBridge - Allows sending events to AWS EventBridge with support for event buses and schemas. Provides the HTTPS endpoint for receiving events.

🧩 Core Features

  • Component Based Approach Central registry for components, enabling multi-bus usage in a single application.

  • @producer() Decorator Injects a producer for sending single or multiple typed events to any configured bus.

  • @consumer Decorator Registers a service class as a consumer for a specific event and queue, handling messages automatically.

  • IProducer Interface Exposes send() and sendMultiple() methods to send messages to buses.

  • IConsumer Interface Allows you to implement a handler for a specific event type and bus, supporting strongly typed data flow.

  • Typed Event Streams Encourages defining typed contracts for all events, improving consistency and type safety between producers and consumers.

You can configure one or more of the supported queue types in your application. For each, you simply provide the required connection and queue configuration. The rest—producer/consumer setup, bindings, and event handling—is abstracted and managed by the extension.

Installation

Install EventStreamConnectorComponent using npm;

$ [npm install | yarn add] loopback4-message-bus-connector

Flow Diagram

screencapture-kzmkc5owuvsij9sl8eox-lite-vusercontent-net-2025-06-29-09_06_14

Basic Use

Configure and load EventStreamConnectorComponent in the application constructor as shown below.

import {
  EventStreamConnectorComponent
} from 'loopback4-message-bus-connector';

// ...
export class MyApplication extends BootMixin(
  ServiceMixin(RepositoryMixin(RestApplication)),
) {
  constructor(options: ApplicationConfig = {}) {
    super();
    this.component(EventStreamConnectorComponent);
    // ...
  }
  // ...
}

SQS

To use SQS as their message queue, bind its required config and connector component in your application.

import {
  SQSConnector,
  SQSBindings,
  EventStreamConnectorComponent
} from 'loopback4-message-bus-connector';

// ...
export class MyApplication extends BootMixin(
  ServiceMixin(RepositoryMixin(RestApplication)),
) {
  constructor(options: ApplicationConfig = {}) {
    super();

    this.component(EventStreamConnectorComponent);
    // SQS Config and its connector
    this.bind(SQSBindings.Config).to({
      queueConfig: {
        QueueUrl: 'http://127.0.0.1:4566/000000000000/my-test-queue',
        MessageRetentionPeriod: 60, // at least 60 seconds
        MaximumMessageSize: 262144,
        ReceiveMessageWaitTimeSeconds: 20, // typical polling time
        VisibilityTimeout: 30, // 30 seconds
      },
      Credentials: {
        region: 'us-east-1',
        accessKeyId: 'test',
        secretAccessKey: 'test',
      },
      ConsumerConfig: {
        MaxNumberOfMessages: 10,
        WaitTimeSeconds: 20,
        maxConsumers: 2,
      },
    });

    this.component(SQSConnector);

    // ...
  }
  // ...
}

to make the application as consumer, pass 'isConsumer' flag to be true in SQS config. like

const config = {
  // rest of ur config
  isConsumer: true,
};

Please follow the AWS SDK for JavaScript for more information on the configuration.

BullMQ

To use BullMq as their message queue, bind its required config and connector component in your application.

import {
  BullMQConnector,
  BullMQBindings,
  EventStreamConnectorComponent,
} from 'loopback4-message-bus-connector';

// ...
export class MyApplication extends BootMixin(
  ServiceMixin(RepositoryMixin(RestApplication)),
) {
  constructor(options: ApplicationConfig = {}) {
    super();

    this.component(EventStreamConnectorComponent);

    // Bull Mq config and connector
    this.bind(BullMQBindings.Config).to({
      QueueName: process.env.QUEUE_NAME ?? 'default-queue',
      redisConfig: {
        host: process.env.REDIS_HOST ?? 'localhost',
        port: parseInt(process.env.REDIS_PORT ?? '6379'),
        password: process.env.REDIS_PASSWORD ?? undefined,
      },
      producerConfig: {
        defaultJobOptions: {
          attempts: 3,
          backoff: 5000,
        },
      },
      consumerConfig: {
        MinConsumers: 1,
        MaxConsumers: 5,
        QueuePollInterval: 2000,
      },
    });
    this.component(BullMQConnector);
    // ...
  }
  // ...
}

to make the application as consumer, pass 'isConsumer' flag to be true in Bull config. like

const config = {
  // rest of ur config
  isConsumer: true,
};

Integration

loopback4-message-bus-connector provides a decorator '@producer()' that can be used to access the producer of each msg queue. It expects one arguement defining the type of queue, of which producer u want to use. like

@injectable({scope: BindingScope.TRANSIENT})
export class EventConnector implements IEventConnector<PublishedEvents> {
 constructor(
   @producer(QueueType.EventBridge)
   private producer: Producer,
   @producer(QueueType.SQS)
   private sqsProducer: Producer,
   @producer(QueueType.BullMQ)
   private bullMqProducer: Producer,
 ) {}

 // rest of implementation

}

Producer provider two ways of sending events - single event at a time and multiple event at a time.

export type Producer<Stream extends AnyObject = AnyObject> = {
   send: <Event extends keyof Stream>(data: Stream[Event], topic?: Event) => Promise<void>;
   sendMultiple: <Event extends keyof Stream>(data: Stream[Event][], topic?: Event) => Promise<void>;
};

It provides '@consumer' decorator to make a service as consumer. consumer needs to follow an interface.

export interface IConsumer<Stream extends AnyObject, Event extends keyof Stream> {
    event: Event;
    queue: QueueType;
    handle(data: Stream[Event]): Promise<void>;
}

and can be used as

import {
  IConsumer,
  QueueType,
  consumer,
} from 'loopback4-message-bus-connector';
import { OrchestratorStream, EventTypes, ProvisioningInputs } from '../../types';

@consumer
export class TenantProvisioningConsumerForEventSQS
  implements IConsumer<OrchestratorStream, EventTypes.TENANT_PROVISIONING>
{
  constructor(
  ) {}
  event: EventTypes.TENANT_PROVISIONING = EventTypes.TENANT_PROVISIONING;
  queue: QueueType = QueueType.SQS;
  async handle(data: ProvisioningInputs): Promise<void> {    
    console.log(`SQS: ${this.event} Event Recieved ` + JSON.stringify(data));
    return;
  }
}

About

loopback4-message-bus-queue-connector

Resources

Code of conduct

Stars

Watchers

Forks

Releases

No releases published

Packages

No packages published

Contributors 5