@@ -55,9 +55,29 @@ export interface SocketOptions {
55
55
/**
56
56
* the authentication payload sent when connecting to the Namespace
57
57
*/
58
- auth : { [ key : string ] : any } | ( ( cb : ( data : object ) => void ) => void ) ;
58
+ auth ?: { [ key : string ] : any } | ( ( cb : ( data : object ) => void ) => void ) ;
59
+ /**
60
+ * The maximum number of retries. Above the limit, the packet will be discarded.
61
+ *
62
+ * Using `Infinity` means the delivery guarantee is "at-least-once" (instead of "at-most-once" by default), but a
63
+ * smaller value like 10 should be sufficient in practice.
64
+ */
65
+ retries ?: number ;
66
+ /**
67
+ * The default timeout in milliseconds used when waiting for an acknowledgement.
68
+ */
69
+ ackTimeout ?: number ;
59
70
}
60
71
72
+ type QueuedPacket = {
73
+ id : number ;
74
+ args : unknown [ ] ;
75
+ flags : Flags ;
76
+ pending : boolean ;
77
+ tryCount : number ;
78
+ ack ?: ( err ?: Error , ...args : unknown [ ] ) => void ;
79
+ } ;
80
+
61
81
/**
62
82
* Internal events.
63
83
* These events can't be emitted by the user.
@@ -76,6 +96,7 @@ interface Flags {
76
96
compress ?: boolean ;
77
97
volatile ?: boolean ;
78
98
timeout ?: number ;
99
+ withRetry ?: boolean ;
79
100
}
80
101
81
102
export type DisconnectDescription =
@@ -198,8 +219,16 @@ export class Socket<
198
219
* Buffer for packets that will be sent once the socket is connected
199
220
*/
200
221
public sendBuffer : Array < Packet > = [ ] ;
222
+ /**
223
+ * The queue of packets to be sent with retry in case of failure.
224
+ *
225
+ * Packets are sent one by one, each waiting for the server acknowledgement, in order to guarantee the delivery order.
226
+ * @private
227
+ */
228
+ private _queue : Array < QueuedPacket > = [ ] ;
201
229
202
230
private readonly nsp : string ;
231
+ private readonly _opts : SocketOptions ;
203
232
204
233
private ids : number = 0 ;
205
234
private acks : object = { } ;
@@ -218,6 +247,7 @@ export class Socket<
218
247
if ( opts && opts . auth ) {
219
248
this . auth = opts . auth ;
220
249
}
250
+ this . _opts = Object . assign ( { } , opts ) ;
221
251
if ( this . io . _autoConnect ) this . open ( ) ;
222
252
}
223
253
@@ -350,6 +380,24 @@ export class Socket<
350
380
}
351
381
352
382
args . unshift ( ev ) ;
383
+
384
+ if ( this . _opts . retries && ! this . flags . withRetry && ! this . flags . volatile ) {
385
+ let ack ;
386
+ if ( typeof args [ args . length - 1 ] === "function" ) {
387
+ ack = args . pop ( ) ;
388
+ }
389
+ this . _queue . push ( {
390
+ id : this . ids ++ ,
391
+ tryCount : 0 ,
392
+ pending : false ,
393
+ args,
394
+ ack,
395
+ flags : Object . assign ( { withRetry : true } , this . flags ) ,
396
+ } ) ;
397
+ this . _drainQueue ( ) ;
398
+ return this ;
399
+ }
400
+
353
401
const packet : any = {
354
402
type : PacketType . EVENT ,
355
403
data : args ,
@@ -393,7 +441,7 @@ export class Socket<
393
441
* @private
394
442
*/
395
443
private _registerAckCallback ( id : number , ack : Function ) {
396
- const timeout = this . flags . timeout ;
444
+ const timeout = this . flags . timeout ?? this . _opts . ackTimeout ;
397
445
if ( timeout === undefined ) {
398
446
this . acks [ id ] = ack ;
399
447
return ;
@@ -440,7 +488,8 @@ export class Socket<
440
488
...args : AllButLast < EventParams < EmitEvents , Ev > >
441
489
) : Promise < FirstArg < Last < EventParams < EmitEvents , Ev > > > > {
442
490
// the timeout flag is optional
443
- const withErr = this . flags . timeout !== undefined ;
491
+ const withErr =
492
+ this . flags . timeout !== undefined || this . _opts . ackTimeout !== undefined ;
444
493
return new Promise ( ( resolve , reject ) => {
445
494
args . push ( ( arg1 , arg2 ) => {
446
495
if ( withErr ) {
@@ -453,6 +502,62 @@ export class Socket<
453
502
} ) ;
454
503
}
455
504
505
+ /**
506
+ * Send the first packet of the queue, and wait for an acknowledgement from the server.
507
+ * @private
508
+ */
509
+ private _drainQueue ( ) {
510
+ debug ( "draining queue" ) ;
511
+ if ( this . _queue . length === 0 ) {
512
+ return ;
513
+ }
514
+ const packet = this . _queue [ 0 ] ;
515
+ if ( packet . pending ) {
516
+ debug (
517
+ "packet [%d] has already been sent and is waiting for an ack" ,
518
+ packet . id
519
+ ) ;
520
+ return ;
521
+ }
522
+ packet . pending = true ;
523
+ packet . tryCount ++ ;
524
+ debug ( "sending packet [%d] (try n°%d)" , packet . id , packet . tryCount ) ;
525
+ const currentId = this . ids ;
526
+ this . ids = packet . id ; // the same id is reused for consecutive retries, in order to allow deduplication on the server side
527
+ this . flags = packet . flags ;
528
+ packet . args . push ( ( err , ...responseArgs ) => {
529
+ if ( packet !== this . _queue [ 0 ] ) {
530
+ // the packet has already been acknowledged
531
+ return ;
532
+ }
533
+ const hasError = err !== null ;
534
+ if ( hasError ) {
535
+ if ( packet . tryCount > this . _opts . retries ) {
536
+ debug (
537
+ "packet [%d] is discarded after %d tries" ,
538
+ packet . id ,
539
+ packet . tryCount
540
+ ) ;
541
+ this . _queue . shift ( ) ;
542
+ if ( packet . ack ) {
543
+ packet . ack ( err ) ;
544
+ }
545
+ }
546
+ } else {
547
+ debug ( "packet [%d] was successfully sent" , packet . id ) ;
548
+ this . _queue . shift ( ) ;
549
+ if ( packet . ack ) {
550
+ packet . ack ( null , ...responseArgs ) ;
551
+ }
552
+ }
553
+ packet . pending = false ;
554
+ return this . _drainQueue ( ) ;
555
+ } ) ;
556
+
557
+ this . emit . apply ( this , packet . args ) ;
558
+ this . ids = currentId ; // restore offset
559
+ }
560
+
456
561
/**
457
562
* Sends a packet.
458
563
*
0 commit comments