Semaphore adapters
Using semaphore adapters
MemorySemaphoreAdapter
To use the MemorySemaphoreAdapter you only need to create instance of it:
import { MemorySemaphoreAdapter } from "@daiso-tech/core/semaphore/memory-semaphore-adapter";
const memorySemaphoreAdapter = new MemorySemaphoreAdapter();
You can also provide an Map that will be used for storing the data in memory:
import { MemorySemaphoreAdapter } from "@daiso-tech/core/semaphore/memory-semaphore-adapter";
const map = new Map<any, any>();
const memorySemaphoreAdapter = new MemorySemaphoreAdapter(map);
MemorySemaphoreAdapter lets you test your app without external dependencies like Redis, ideal for local development, unit tests, integration tests and fast E2E test for the backend application.
Note the MemorySemaphoreAdapter is limited to single process usage and cannot be shared across multiple servers or processes.
MongodbSemaphoreAdapter
To use the MongodbSemaphoreAdapter, you'll need to:
- Install the required dependency:
mongodbpackage
import { MongodbSemaphoreAdapter } from "@daiso-tech/core/semaphore/mongodb-semaphore-adapter";
import { MongoClient } from "mongodb";
const client = await MongoClient.connect("YOUR_MONGODB_CONNECTION_STRING");
const database = client.db("database");
const mongodbSemaphoreAdapter = new MongodbSemaphoreAdapter({
database,
});
// You need initialize the adapter once before using it.
// During the initialization the indexes will be created
await mongodbSemaphoreAdapter.init();
You can change the collection name:
const mongodbSemaphoreAdapter = new MongodbSemaphoreAdapter({
database,
// By default "semaphore" is used as collection name
collectionName: "my-semaphore",
});
await mongodbSemaphoreAdapter.init();
You can change the collection settings:
const mongodbSemaphoreAdapter = new MongodbSemaphoreAdapter({
database,
// You configure additional collection settings
collectionSettings: {},
});
await mongodbSemaphoreAdapter.init();
To remove the semaphore collection and all stored semaphore data, use deInit method:
await mongodbSemaphoreAdapter.deInit();
Note in order to use MongodbSemaphoreAdapter correctly, ensure you use a single, consistent database across all server instances or processes.
RedisSemaphoreAdapter
To use the RedisSemaphoreAdapter, you'll need to:
- Install the required dependency:
ioredispackage
import { RedisSemaphoreAdapter } from "@daiso-tech/core/semaphore/redis-semaphore-adapter";
import Redis from "ioredis";
const database = new Redis("YOUR_REDIS_CONNECTION_STRING");
const redisSemaphoreAdapter = new RedisSemaphoreAdapter(database);
Note in order to use RedisSemaphoreAdapter correctly, ensure you use a single, consistent database across all server instances or processes.
KyselySemaphoreAdapter
To use the KyselySemaphoreAdapter, you'll need to:
- Install the required dependency:
kyselypackage
Usage with Sqlite
You will need to install better-sqlite3 package:
import { TimeSpan } from "@daiso-tech/core/utilities";
import { KyselySemaphoreAdapter } from "@daiso-tech/core/semaphore/kysely-semaphore-adapter";
import Sqlite from "better-sqlite3";
import { Kysely, SqliteDialect } from "kysely";
const database = new Sqlite("DATABASE_NAME.db");
const kysely = new Kysely({
dialect: new SqliteDialect({
database,
}),
});
const kyselySemaphoreAdapter = new KyselySemaphoreAdapter({
kysely,
});
// You need initialize the adapter once before using it.
// During the initialization the schema will be created
await kyselySemaphoreAdapter.init();
Note using KyselySemaphoreAdapter with sqlite is limited to single server usage and cannot be shared across multiple servers but it can be shared between different processes. To use it correctly, ensure all process instances access the same persisted database.
Usage with Postgres
You will need to install pg package:
import { TimeSpan } from "@daiso-tech/core/utilities";
import { KyselySemaphoreAdapter } from "@daiso-tech/core/semaphore/kysely-semaphore-adapter";
import { Pool } from "pg";
import { Kysely, PostgresDialect } from "kysely";
const database = new Pool({
database: "DATABASE_NAME",
host: "DATABASE_HOST",
user: "DATABASE_USER",
// DATABASE port
port: 5432,
password: "DATABASE_PASSWORD",
max: 10,
});
const kysely = new Kysely({
dialect: new PostgresDialect({
pool: database,
}),
});
const kyselySemaphoreAdapter = new KyselySemaphoreAdapter({
kysely,
});
// You need initialize the adapter once before using it.
// During the initialization the schema will be created
await kyselySemaphoreAdapter.init();
Note in order to use KyselySemaphoreAdapter with postgres correctly, ensure you use a single, consistent database across all server instances. This means you can't use replication.
Usage with Mysql
You will need to install mysql2 package:
import { TimeSpan } from "@daiso-tech/core/utilities";
import { KyselySemaphoreAdapter } from "@daiso-tech/core/semaphore/kysely-semaphore-adapter";
import { createPool } from "mysql2";
import { Kysely, MysqlDialect } from "kysely";
const database = createPool({
host: "DATABASE_HOST",
// Database port
port: 3306,
database: "DATABASE_NAME",
user: "DATABASE_USER",
password: "DATABASE_PASSWORD",
connectionLimit: 10,
});
const kysely = new Kysely({
dialect: new MysqlDialect({
pool: database,
}),
});
const kyselySemaphoreAdapter = new KyselySemaphoreAdapter({
kysely,
});
// You need initialize the adapter once before using it.
// During the initialization the schema will be created
await kyselySemaphoreAdapter.init();
Note in order to use KyselySemaphoreAdapter with mysql correctly, ensure you use a single, consistent database across all server instances. This means you can't use replication.
Usage with Libsql
You will need to install @libsql/kysely-libsql package:
import { TimeSpan } from "@daiso-tech/core/utilities";
import { KyselySemaphoreAdapter } from "@daiso-tech/core/semaphore/kysely-semaphore-adapter";
import { LibsqlDialect } from "@libsql/kysely-libsql";
import { Kysely } from "kysely";
const kysely = new Kysely({
dialect: new LibsqlDialect({
url: "DATABASE_URL",
}),
});
const kyselySemaphoreAdapter = new KyselySemaphoreAdapter({
kysely,
});
// You need initialize the adapter once before using it.
// During the initialization the schema will be created
await kyselySemaphoreAdapter.init();
Note in order to use KyselySemaphoreAdapter with libsql correctly, ensure you use a single, consistent database across all server instances. This means you can't use libsql embedded replicas.
Settings
Expired keys are cleared at regular intervals and you can change the interval time:
import { TimeSpan } from "@daiso-tech/core/utilities";
const kyselySemaphoreAdapter = new KyselySemaphoreAdapter({
database,
// By default, the interval is 1 minute
expiredKeysRemovalInterval: TimeSpan.fromSeconds(10),
});
await kyselySemaphoreAdapter.init();
Disabling scheduled interval cleanup of expired keys:
import { TimeSpan } from "@daiso-tech/core/utilities";
const kyselySemaphoreAdapter = new KyselySemaphoreAdapter({
database,
shouldRemoveExpiredKeys: false,
});
await kyselySemaphoreAdapter.init();
// You can remove all expired keys manually.
await kyselySemaphoreAdapter.removeAllExpired();
To remove the semaphore table and all stored semaphore data, use deInit method:
await kyselySemaphoreAdapter.deInit();
NoOpSemaphoreAdapter
The NoOpSemaphoreAdapter is a no-operation implementation, it performs no actions when called:
import { NoOpSemaphoreAdapter } from "@daiso-tech/core/semaphore/no-op-semaphore-adapter";
const noOpSemaphoreAdapter = new NoOpSemaphoreAdapter();
The NoOpSemaphoreAdapter is useful when you want to mock out or disable your SemaphoreProvider instance.
Creating semaphore adapters
Implementing your custom ISemaphoreAdapter
In order to create an adapter you need to implement the ISemaphoreAdapter contract.
Testing your custom ISemaphoreAdapter
We provide a complete test suite to verify your event bus adapter implementation. Simply use the semaphoreAdapterTestSuite function:
- Preconfigured Vitest test cases
- Standardized event bus behavior validation
- Common edge case coverage
Usage example:
// filename: MySemaphoreAdapter.test.ts
import { beforeEach, describe, expect, test } from "vitest";
import { semaphoreAdapterTestSuite } from "@daiso-tech/core/semaphore/test-utilities";
import { MemorySemaphoreAdapter } from "./MemorySemaphoreAdapter.js";
describe("class: MySemaphoreAdapter", () => {
semaphoreAdapterTestSuite({
createAdapter: () => new MemorySemaphoreAdapter(),
test,
beforeEach,
expect,
describe,
});
});
Implementing your custom IDatabaseSemaphoreAdapter
We provide an additional contract IDatabaseSemaphoreAdapter for building custom semaphore adapters tailored to databases.
Testing your custom IDatabaseSemaphoreAdapter
We provide a complete test suite to verify your event bus adapter implementation. Simply use the databaseSemaphoreAdapterTestSuite function:
- Preconfigured Vitest test cases
- Standardized event bus behavior validation
- Common edge case coverage
Usage example:
import { beforeEach, describe, expect, test } from "vitest";
import { databaseSemaphoreAdapterTestSuite } from "@daiso-tech/core/semaphore/test-utilities";
import { MyDatabaseSemaphoreAdapter } from "./MyDatabaseSemaphoreAdapter.js";
describe("class: MyDatabaseSemaphoreAdapter", () => {
databaseSemaphoreAdapterTestSuite({
createAdapter: async () => {
return new MyDatabaseSemaphoreAdapter(),
},
test,
beforeEach,
expect,
describe,
});
});
Implementing your custom ISemaphoreProvider class
In some cases, you may need to implement a custom SemaphoreProvider class to optimize performance for your specific technology stack. You can then directly implement the ISemaphoreProvider contract.
Testing your custom ISemaphoreProvider class
We provide a complete test suite to verify your custom event bus class implementation. Simply use the semaphoreProviderTestSuite function:
- Preconfigured Vitest test cases
- Standardized event bus behavior validation
- Common edge case coverage
Usage example:
// filename: MySemaphoreProvider.test.ts
import { beforeEach, describe, expect, test } from "vitest";
import { semaphoreProviderTestSuite } from "@daiso-tech/core/semaphore/test-utilities";
import { MySemaphoreProvider } from "./MySemaphoreProvider.js";
describe("class: MySemaphoreProvider", () => {
semaphoreProviderTestSuite({
createSemaphoreProvider: () => new MySemaphoreProvider(),
test,
beforeEach,
expect,
describe,
});
});
Further information
For further information refer to @daiso-tech/core/semaphore API docs.