Skip to content

Commit 3a0dc89

Browse files
author
ludovic
committed
Fix #2 : Handling connection lost
1 parent f5afd17 commit 3a0dc89

File tree

4 files changed

+5346
-5328
lines changed

4 files changed

+5346
-5328
lines changed

js-src/Websocket.ts

+40-15
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,6 @@
1-
import {Channel} from "./Channel";
2-
import {AxiosResponse} from "axios";
1+
import { AxiosResponse } from "axios";
2+
import axios from 'axios';
3+
import { Channel } from "./Channel";
34

45
export type Options = { authEndpoint: string, host: string };
56
export type MessageBody = { event: string, channel?: string, data: object };
@@ -19,14 +20,20 @@ export class Websocket {
1920

2021
private socketId: string;
2122

23+
private closing = false;
24+
2225
private pingInterval: NodeJS.Timeout;
2326

24-
constructor(options: Options) {
25-
this.options = options;
27+
private connect(host: string): void {
28+
console.log('Connecting');
2629

27-
this.websocket = new WebSocket(options.host)
30+
this.websocket = new WebSocket(host)
2831

2932
this.websocket.onopen = () => {
33+
this.send({
34+
event: 'whoami',
35+
})
36+
3037
while (this.buffer.length) {
3138
const message = this.buffer[0]
3239

@@ -56,8 +63,20 @@ export class Websocket {
5663
if (this.internalListeners[message.event]) {
5764
this.internalListeners[message.event](message.data)
5865
}
66+
5967
}
6068

69+
70+
this.websocket.onclose = () => {
71+
if (this.socketId && !this.closing || !this.socketId) {
72+
console.info('Connection lost, reconnecting...');
73+
setTimeout(() => {
74+
this.socketId = undefined
75+
this.connect(host)
76+
}, 1000);
77+
}
78+
};
79+
6180
this.on('whoami', ({ socket_id: socketId }) => {
6281
this.socketId = socketId
6382

@@ -72,19 +91,24 @@ export class Websocket {
7291
}
7392
})
7493

75-
this.send({
76-
event: 'whoami',
77-
})
7894

7995
// send ping every 60 seconds to keep connection alive
8096
this.pingInterval = setInterval(() => {
81-
console.log('Sending ping')
82-
83-
this.send({
84-
event: 'ping',
85-
})
97+
if (this.websocket.readyState === this.websocket.OPEN) {
98+
console.log('Sending ping')
99+
this.send({
100+
event: 'ping',
101+
})
102+
}
86103
}, 60 * 1000)
87104

105+
}
106+
107+
constructor(options: Options) {
108+
this.options = options;
109+
110+
this.connect(this.options.host);
111+
88112
return this
89113
}
90114

@@ -115,7 +139,8 @@ export class Websocket {
115139
this.buffer.push(message)
116140
}
117141

118-
close (): void {
142+
close(): void {
143+
this.closing = true
119144
this.internalListeners = {}
120145

121146
clearInterval(this.pingInterval)
@@ -146,7 +171,7 @@ export class Websocket {
146171
event: 'subscribe',
147172
data: {
148173
channel: channel.name,
149-
... response.data
174+
...response.data
150175
},
151176
})
152177
}).catch((error) => {

js-tests/Connector.test.ts

+27-4
Original file line numberDiff line numberDiff line change
@@ -2,11 +2,14 @@ import WS from "jest-websocket-mock";
22
import { Connector } from "../js-src/Connector";
33
import { Channel } from "../js-src/Channel";
44

5+
const mockedHost = 'ws://localhost:1234';
6+
57
describe('Connector', () => {
68
let server: WS;
79

810
beforeEach(() => {
9-
server = new WS("ws://localhost:1234");
11+
jest.useRealTimers();
12+
server = new WS(mockedHost);
1013
});
1114

1215
afterEach(() => {
@@ -15,7 +18,7 @@ describe('Connector', () => {
1518

1619
test('socket id is correctly set', async () => {
1720
const connector = new Connector({
18-
host: "ws://localhost:1234",
21+
host: mockedHost,
1922
})
2023

2124
await server.connected;
@@ -26,9 +29,29 @@ describe('Connector', () => {
2629
expect(connector.socketId()).toBe('test-socket-id')
2730
})
2831

32+
test('we reconnect to the server on error', async () => {
33+
const connector = new Connector({
34+
host: mockedHost,
35+
})
36+
37+
await server.connected;
38+
await expect(server).toReceiveMessage('{"event":"whoami"}');
39+
server.send('{"event":"whoami","data":{"socket_id":"test-socket-id"}}')
40+
41+
server.close();
42+
await server.closed;
43+
server.server.stop(() => (server = new WS(mockedHost)));
44+
45+
await server.connected;
46+
await expect(server).toReceiveMessage('{"event":"whoami"}');
47+
server.send('{"event":"whoami","data":{"socket_id":"test-socket-id2"}}')
48+
49+
expect(connector.socketId()).toBe('test-socket-id2')
50+
})
51+
2952
test('we can subscribe to a channel and listen to events', async () => {
3053
const connector = new Connector({
31-
host: "ws://localhost:1234",
54+
host: mockedHost,
3255
})
3356

3457
await server.connected;
@@ -57,7 +80,7 @@ describe('Connector', () => {
5780

5881
test('we can send a whisper event', async () => {
5982
const connector = new Connector({
60-
host: "ws://localhost:1234",
83+
host: mockedHost,
6184
})
6285

6386
await server.connected;

package.json

-1
Original file line numberDiff line numberDiff line change
@@ -49,7 +49,6 @@
4949
"jest": "^24.9.0",
5050
"jest-websocket-mock": "^2.2.0",
5151
"laravel-echo": "^1.10.0",
52-
"mock-socket": "^9.0.3",
5352
"rollup": "^2.10.2",
5453
"rollup-plugin-typescript2": "^0.27.1",
5554
"standard-version": "^8.0.1",

0 commit comments

Comments
 (0)