From 3cf1da35c2f0dbd558c1b9daaf2794a70f1b97f5 Mon Sep 17 00:00:00 2001 From: Jonjo McKay Date: Fri, 29 Nov 2024 14:19:51 +0000 Subject: [PATCH] Added support for enabling Nats JetStream (#877) --- docs/modules/nats.md | 4 + .../modules/nats/src/nats-container.test.ts | 39 +++++++++- packages/modules/nats/src/nats-container.ts | 75 ++++++++++++------- 3 files changed, 92 insertions(+), 26 deletions(-) diff --git a/docs/modules/nats.md b/docs/modules/nats.md index 7f8bfb497..b79f13299 100644 --- a/docs/modules/nats.md +++ b/docs/modules/nats.md @@ -21,3 +21,7 @@ npm install @testcontainers/nats --save-dev [Set credentials:](../../packages/modules/nats/src/nats-container.test.ts) inside_block:credentials + + +[Enable JetStream:](../../packages/modules/nats/src/nats-container.test.ts) inside_block:jetstream + \ No newline at end of file diff --git a/packages/modules/nats/src/nats-container.test.ts b/packages/modules/nats/src/nats-container.test.ts index 927eb24aa..2218b4448 100644 --- a/packages/modules/nats/src/nats-container.test.ts +++ b/packages/modules/nats/src/nats-container.test.ts @@ -67,6 +67,43 @@ describe("NatsContainer", () => { }); // } + // jetstream { + it("should start with JetStream ", async () => { + // enable JetStream + const container = await new NatsContainer().withJetStream().start(); + + const nc = await connect(container.getConnectionOptions()); + + // ensure JetStream is enabled, otherwise this will throw an error + await nc.jetstream().jetstreamManager(); + + // close the connection + await nc.close(); + // check if the close was OK + const err = await nc.closed(); + expect(err).toBe(undefined); + + await container.stop(); + }); + + it("should fail without JetStream ", async () => { + const container = await new NatsContainer().start(); + + const nc = await connect(container.getConnectionOptions()); + + // ensure JetStream is not enabled, as this will throw an error + await expect(nc.jetstream().jetstreamManager()).rejects.toThrow("503"); + + // close the connection + await nc.close(); + // check if the close was OK + const err = await nc.closed(); + expect(err).toBe(undefined); + + await container.stop(); + }); + // } + it("should immediately end when started with version argument ", async () => { // for the complete list of available arguments see: // See Command Line Options section inside [NATS docker image documentation](https://hub.docker.com/_/nats) @@ -75,6 +112,6 @@ describe("NatsContainer", () => { await connect(container.getConnectionOptions()); } - await expect(outputVersionAndExit()).rejects.toThrowError(); + await expect(outputVersionAndExit()).rejects.toThrow(); }); }); diff --git a/packages/modules/nats/src/nats-container.ts b/packages/modules/nats/src/nats-container.ts index bc93eaa42..6bedea76c 100755 --- a/packages/modules/nats/src/nats-container.ts +++ b/packages/modules/nats/src/nats-container.ts @@ -7,44 +7,51 @@ const HTTP_MANAGEMENT_PORT = 8222; const USER_ARGUMENT_KEY = "--user"; const PASS_ARGUMENT_KEY = "--pass"; -function buildCmdsFromArgs(args: { [p: string]: string }): string[] { - const result: string[] = []; - result.push("nats-server"); - - for (const argsKey in args) { - result.push(argsKey); - result.push(args[argsKey]); - } - return result; -} - export class NatsContainer extends GenericContainer { - private args: { [name: string]: string } = {}; + private args = new Set(); + private values = new Map(); constructor(image = "nats:2.8.4-alpine") { super(image); - this.args[USER_ARGUMENT_KEY] = "test"; - this.args[PASS_ARGUMENT_KEY] = "test"; + this.withUsername("test"); + this.withPass("test"); this.withExposedPorts(CLIENT_PORT, ROUTING_PORT_FOR_CLUSTERING, HTTP_MANAGEMENT_PORT) .withWaitStrategy(Wait.forLogMessage(/.*Server is ready.*/)) .withStartupTimeout(120_000); } + /** + * Enable JetStream + * + * @returns {this} + */ + public withJetStream(): this { + this.withArg("--jetstream"); + return this; + } + public withUsername(user: string): this { - this.args[USER_ARGUMENT_KEY] = user; + this.withArg(USER_ARGUMENT_KEY, user); return this; } public withPass(pass: string): this { - this.args[PASS_ARGUMENT_KEY] = pass; + this.withArg(PASS_ARGUMENT_KEY, pass); return this; } - public withArg(name: string, value: string) { - name = NatsContainer.ensureDashInFrontOfArgumentName(name); - this.args[name] = value; + public withArg(name: string, value: string): this; + public withArg(name: string): this; + public withArg(...args: [string, string] | [string]): this { + const [name, value] = args; + + const correctName = NatsContainer.ensureDashInFrontOfArgumentName(name); + this.args.add(correctName); + if (args.length === 2) { + this.values.set(correctName, value); + } return this; } @@ -61,23 +68,41 @@ export class NatsContainer extends GenericContainer { } public override async start(): Promise { - this.withCommand(buildCmdsFromArgs(this.args)); + this.withCommand(this.getNormalizedCommand()); return new StartedNatsContainer(await super.start(), this.getUser(), this.getPass()); } - private getUser(): string { - return this.args[USER_ARGUMENT_KEY]; + private getUser(): string | undefined { + return this.values.get(USER_ARGUMENT_KEY); } - private getPass(): string { - return this.args[PASS_ARGUMENT_KEY]; + private getPass(): string | undefined { + return this.values.get(PASS_ARGUMENT_KEY); + } + + private getNormalizedCommand(): string[] { + const result: string[] = ["nats-server"]; + for (const arg of this.args) { + result.push(arg); + if (this.values.has(arg)) { + const value = this.values.get(arg); + if (value) { + result.push(value); + } + } + } + return result; } } export class StartedNatsContainer extends AbstractStartedContainer { private readonly connectionOptions: NatsConnectionOptions; - constructor(startedTestContainer: StartedTestContainer, readonly username: string, readonly password: string) { + constructor( + startedTestContainer: StartedTestContainer, + readonly username: string | undefined, + readonly password: string | undefined + ) { super(startedTestContainer); const port = startedTestContainer.getMappedPort(CLIENT_PORT); this.connectionOptions = {