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.
Install EventStreamConnectorComponent using npm
;
$ [npm install | yarn add] loopback4-message-bus-connector
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);
// ...
}
// ...
}
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.
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,
};
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;
}
}