From be77d0cd9c539bf663d08f1f0c726ef7ebe2dc5d Mon Sep 17 00:00:00 2001 From: Alexander Emelin Date: Sun, 7 Apr 2024 07:21:40 +0300 Subject: [PATCH] docker compose for testing, add some tests --- .github/workflows/ci.yml | 2 +- README.md | 15 ++++ docker-compose.yml | 14 ++++ src/centrifuge.test.ts | 151 ++++++++++++++++++++++++++++++++++++--- src/centrifuge.ts | 2 + 5 files changed, 175 insertions(+), 9 deletions(-) create mode 100644 docker-compose.yml diff --git a/.github/workflows/ci.yml b/.github/workflows/ci.yml index 9f559cb3..fe7eb5ac 100644 --- a/.github/workflows/ci.yml +++ b/.github/workflows/ci.yml @@ -18,7 +18,7 @@ jobs: - name: Install dependencies run: yarn install - name: Start Centrifugo - run: docker run -d -p 8000:8000 -e CENTRIFUGO_PRESENCE=true centrifugo/centrifugo:latest centrifugo --client_insecure --http_stream --sse + run: docker compose up -d --wait - name: Lint run: yarn lint - name: Test diff --git a/README.md b/README.md index 43b8e8ec..0f927c22 100644 --- a/README.md +++ b/README.md @@ -27,6 +27,7 @@ The features implemented by this SDK can be found in [SDK feature matrix](https: * [Protobuf support](#protobuf-support) * [Using with NodeJS](#using-with-nodejs) * [Custom WebSocket constructor](#custom-websocket-constructor) +* [Run tests locally](#run-tests-locally) ## Install @@ -815,3 +816,17 @@ var centrifuge = new Centrifuge('ws://localhost:8000/connection/websocket', { ``` See a basic example with React Native where this technique is used [in this comment](https://github.com/centrifugal/centrifuge-js/issues/224#issuecomment-1538820023). + +## Run tests locally + +If you want to run `centrifuge-js` tests locally, start test Centrifugo server: + +``` +docker compose up +``` + +Then: + +``` +yarn test +``` diff --git a/docker-compose.yml b/docker-compose.yml new file mode 100644 index 00000000..268937e4 --- /dev/null +++ b/docker-compose.yml @@ -0,0 +1,14 @@ +version: '3.8' + +services: + centrifugo: + image: centrifugo/centrifugo:v5 + command: + - centrifugo + ports: + - "8000:8000" + environment: + - CENTRIFUGO_CLIENT_INSECURE=true + - CENTRIFUGO_HTTP_STREAM=true + - CENTRIFUGO_SSE=true + - CENTRIFUGO_PRESENCE=true diff --git a/src/centrifuge.test.ts b/src/centrifuge.test.ts index 4de736a1..52079fe6 100644 --- a/src/centrifuge.test.ts +++ b/src/centrifuge.test.ts @@ -1,9 +1,19 @@ import { Centrifuge } from './centrifuge' import { - DisconnectedContext, Error as CentrifugeError, - PublicationContext, TransportName, UnsubscribedContext, State, SubscriptionState + DisconnectedContext, + Error as CentrifugeError, + PublicationContext, + TransportName, + UnsubscribedContext, + State, + SubscriptionState, + SubscribedContext, } from './types'; -import { disconnectedCodes, unsubscribedCodes, connectingCodes } from './codes'; +import { + disconnectedCodes, + unsubscribedCodes, + connectingCodes, +} from './codes'; import WebSocket from 'ws'; import EventSource from 'eventsource'; @@ -15,14 +25,18 @@ test('invalid endpoint', () => { }); test('no websocket constructor', async () => { - const c = new Centrifuge('ws://localhost:8000/connection/websocket?cf_protocol_version=v2'); + const c = new Centrifuge('ws://localhost:8000/connection/websocket'); expect(() => { c.connect() }).toThrowError(); }); const transportCases = [ - ['websocket', 'ws://localhost:8000/connection/websocket?cf_protocol_version=v2'], - ['http_stream', 'http://localhost:8000/connection/http_stream?cf_protocol_version=v2'], - ['sse', 'http://localhost:8000/connection/sse?cf_protocol_version=v2'], + ['websocket', 'ws://localhost:8000/connection/websocket'], + ['http_stream', 'http://localhost:8000/connection/http_stream'], + ['sse', 'http://localhost:8000/connection/sse'], +] + +const websocketOnly = [ + ['websocket', 'ws://localhost:8000/connection/websocket'], ] test.each(transportCases)("%s: connects and disconnects", async (transport, endpoint) => { @@ -216,7 +230,7 @@ test.each(transportCases)("%s: handles offline/online events", async (transport, expect(c.state).toBe(State.Disconnected); }); -test.each(transportCases.slice(0, 1))("%s: not connecting on online in disconnected state", async (transport, endpoint) => { +test.each(transportCases)("%s: not connecting on online in disconnected state", async (transport, endpoint) => { const networkEventTarget = new EventTarget(); const c = new Centrifuge([{ @@ -364,3 +378,124 @@ test.each(transportCases)("%s: subscribe and unsubscribe loop", async (transport await disconnectedPromise; expect(c.state).toBe(State.Disconnected); }); + +// Make sure we can unsubscribe right after connect called and connect/subscribe +// frames not sent yet. +test.each(transportCases)("%s: unsubscribe right after connect", async (transport, endpoint) => { + const c = new Centrifuge([{ + transport: transport as TransportName, + endpoint: endpoint, + }], { + websocket: WebSocket, + fetch: fetch, + eventsource: EventSource, + readableStream: ReadableStream, + emulationEndpoint: 'http://localhost:8000/emulation' + }); + + c.connect(); + await c.ready(5000); + + const sub = c.newSubscription('test'); + + let unsubcribeCalled: any; + const unsubscribedPromise = new Promise((resolve, _) => { + unsubcribeCalled = resolve; + }) + let subcribeCalled: any; + const subscribedPromise = new Promise((resolve, _) => { + subcribeCalled = resolve; + }) + + sub.on('subscribed', (ctx) => { + subcribeCalled(ctx); + }) + sub.on('unsubscribed', (ctx) => { + unsubcribeCalled(ctx); + }) + + sub.subscribe(); + c.disconnect(); + c.connect(); + sub.unsubscribe(); + + expect(sub.state).toBe(SubscriptionState.Unsubscribed); + await unsubscribedPromise; + + sub.subscribe(); + await subscribedPromise; + + let disconnectCalled: any; + const disconnectedPromise = new Promise((resolve, _) => { + disconnectCalled = resolve; + }) + c.on('disconnected', (ctx) => { + disconnectCalled(ctx); + }) + + c.disconnect(); + await disconnectedPromise; + expect(c.state).toBe(State.Disconnected); +}); + +// Make sure we can unsubscribe right after connect frame sent but reply has not been yet received. +// This is important to cover bug described in https://github.com/centrifugal/centrifuge-js/pull/274. +test.each(websocketOnly)("%s: unsubscribe in between connect command and reply", async (transport, endpoint) => { + const c = new Centrifuge([{ + transport: transport as TransportName, + endpoint: endpoint, + }], { + websocket: WebSocket, + fetch: fetch, + eventsource: EventSource, + readableStream: ReadableStream, + emulationEndpoint: 'http://localhost:8000/emulation' + }); + + const sub = c.newSubscription('test'); + + let unsubcribeCalled: any; + const unsubscribedPromise = new Promise((resolve, _) => { + unsubcribeCalled = resolve; + }) + let subcribeCalled: any; + const subscribedPromise = new Promise((resolve, _) => { + subcribeCalled = resolve; + }) + + // @ts-ignore this is only for test purposes. + c.on('__centrifuge_debug:connect_frame_sent', () => { + sub.unsubscribe(); + unsubcribeCalled() + }) + + sub.on('subscribed', (ctx) => { + subcribeCalled(ctx); + }) + sub.on('unsubscribed', (ctx) => { + unsubcribeCalled(ctx); + }) + + sub.subscribe(); + c.connect(); + + await unsubscribedPromise; + await c.ready() + + await new Promise(r => setTimeout(r, 2000)); + sub.subscribe(); + + await subscribedPromise; + + let disconnectCalled: any; + const disconnectedPromise = new Promise((resolve, _) => { + disconnectCalled = resolve; + }) + c.on('disconnected', (ctx) => { + disconnectCalled(ctx); + }) + + c.disconnect(); + await disconnectedPromise; + expect(c.state).toBe(State.Disconnected); +}); diff --git a/src/centrifuge.ts b/src/centrifuge.ts index f25e927b..f6ddd389 100644 --- a/src/centrifuge.ts +++ b/src/centrifuge.ts @@ -804,6 +804,8 @@ export class Centrifuge extends (EventEmitter as new () => TypedEventEmitter