@@ -95,12 +95,14 @@ class TransportWsEvent {
95
95
received = null ,
96
96
message = null ,
97
97
event = null ;
98
+
98
99
const TransportWsEvent .opened (WebSocketChannel this .socket)
99
100
: type = TransportWsEventType .opened,
100
101
payload = null ,
101
102
received = null ,
102
103
message = null ,
103
104
event = null ;
105
+
104
106
const TransportWsEvent .connected (
105
107
WebSocketChannel this .socket,
106
108
this .payload,
@@ -116,6 +118,7 @@ class TransportWsEvent {
116
118
socket = null ,
117
119
message = null ,
118
120
event = null ;
121
+
119
122
const TransportWsEvent .pong (
120
123
this .payload, {
121
124
required bool this .received,
@@ -130,12 +133,14 @@ class TransportWsEvent {
130
133
received = null ,
131
134
socket = null ,
132
135
event = null ;
136
+
133
137
const TransportWsEvent .closed (Object this .event)
134
138
: type = TransportWsEventType .closed,
135
139
payload = null ,
136
140
received = null ,
137
141
message = null ,
138
142
socket = null ;
143
+
139
144
const TransportWsEvent .error (Object this .event)
140
145
: type = TransportWsEventType .error,
141
146
payload = null ,
@@ -382,6 +387,17 @@ class TransportWsClientOptions {
382
387
/// @default [randomizedExponentialBackoff]
383
388
final Future <void > Function (int retries) retryWait;
384
389
390
+ /// Check if the close event or connection error is fatal. If you return `false` ,
391
+ /// the client will fail immediately without additional retries; however, if you
392
+ /// return `true` , the client will keep retrying until the `retryAttempts` have
393
+ /// been exceeded.
394
+ /// The argument is whatever has been thrown during the connection phase.
395
+ /// Beware, the library classifies a few close events as fatal regardless of
396
+ /// what is returned here. They are listed in the documentation of the
397
+ /// `retryAttempts` option.
398
+ /// @default [shouldRetryDefault]
399
+ final bool Function (Object errOrCloseEvent) shouldRetry;
400
+
385
401
/// Check if the close event or connection error is fatal. If you return `true` ,
386
402
/// the client will fail immediately without additional retries; however, if you
387
403
/// return `false` , the client will keep retrying until the `retryAttempts` have
@@ -464,6 +480,7 @@ class TransportWsClientOptions {
464
480
this .disablePong = false ,
465
481
this .retryAttempts = 0 ,
466
482
this .retryWait = randomizedExponentialBackoff,
483
+ this .shouldRetry = shouldRetryDefault,
467
484
this .isFatalConnectionProblem = isFatalConnectionProblemDefault,
468
485
this .eventHandlers,
469
486
this .generateID = generateUUID,
@@ -506,6 +523,29 @@ class TransportWsClientOptions {
506
523
final v = c.group (0 ) == "x" ? r : (r & 0x3 ) | 0x8 ;
507
524
return v.toRadixString (16 );
508
525
});
526
+
527
+ /// By default, connection should not retry on fatal errors
528
+ static bool shouldRetryDefault (Object errOrCloseEvent) {
529
+ if (errOrCloseEvent is LikeCloseEvent &&
530
+ (_isFatalInternalCloseCode (errOrCloseEvent.code) ||
531
+ const [
532
+ CloseCode .internalServerError,
533
+ CloseCode .internalClientError,
534
+ CloseCode .badRequest,
535
+ CloseCode .badResponse,
536
+ CloseCode .unauthorized,
537
+ // CloseCode.Forbidden, might grant access out after retry
538
+ CloseCode .subprotocolNotAcceptable,
539
+ // CloseCode.ConnectionInitialisationTimeout, might not time out after retry
540
+ // CloseCode.ConnectionAcknowledgementTimeout, might not time out after retry
541
+ CloseCode .subscriberAlreadyExists,
542
+ CloseCode .tooManyInitialisationRequests,
543
+ // 4499, // Terminated, probably because the socket froze, we want to retry
544
+ ].contains (errOrCloseEvent.code))) {
545
+ return false ;
546
+ }
547
+ return true ;
548
+ }
509
549
}
510
550
511
551
class _Connected {
@@ -588,29 +628,15 @@ class _ConnectionState {
588
628
/// Checks the `connect` problem and evaluates if the client should retry.
589
629
bool shouldRetryConnectOrThrow (Object errOrCloseEvent) {
590
630
options.log? .call ("shouldRetryConnectOrThrow $errOrCloseEvent " );
591
- // some close codes are worth reporting immediately
592
- if (errOrCloseEvent is LikeCloseEvent &&
593
- (_isFatalInternalCloseCode (errOrCloseEvent.code) ||
594
- const [
595
- CloseCode .internalServerError,
596
- CloseCode .internalClientError,
597
- CloseCode .badRequest,
598
- CloseCode .badResponse,
599
- CloseCode .unauthorized,
600
- // CloseCode.Forbidden, might grant access out after retry
601
- CloseCode .subprotocolNotAcceptable,
602
- // CloseCode.ConnectionInitialisationTimeout, might not time out after retry
603
- // CloseCode.ConnectionAcknowledgementTimeout, might not time out after retry
604
- CloseCode .subscriberAlreadyExists,
605
- CloseCode .tooManyInitialisationRequests,
606
- // 4499, // Terminated, probably because the socket froze, we want to retry
607
- ].contains (errOrCloseEvent.code))) {
608
- throw errOrCloseEvent;
609
- }
610
631
611
632
// client was disposed, no retries should proceed regardless
612
633
if (disposed) return false ;
613
634
635
+ // some close codes are worth reporting immediately
636
+ if (! options.shouldRetry (errOrCloseEvent)) {
637
+ throw errOrCloseEvent;
638
+ }
639
+
614
640
// normal closure (possibly all subscriptions have completed)
615
641
// if no locks were acquired in the meantime, shouldnt try again
616
642
if (errOrCloseEvent is LikeCloseEvent && errOrCloseEvent.code == 1000 ) {
@@ -850,7 +876,7 @@ class _ConnectionState {
850
876
? "DONE"
851
877
: LikeCloseEvent (
852
878
code: socket.closeCode! ,
853
- reason: socket.closeReason! ,
879
+ reason: socket.closeReason,
854
880
),
855
881
),
856
882
onError: (Object err) => onError? .call (err),
@@ -929,7 +955,9 @@ class _Client extends TransportWsClient {
929
955
_Client ({required this .state});
930
956
931
957
final _ConnectionState state;
958
+
932
959
_Emitter get emitter => state.emitter;
960
+
933
961
@override
934
962
TransportWsClientOptions get options => state.options;
935
963
@@ -1066,6 +1094,7 @@ class _Emitter {
1066
1094
void Function (TransportWsMessage ) listener,
1067
1095
) onMessage;
1068
1096
final void Function (String logMessage)? log;
1097
+
1069
1098
_Emitter ({
1070
1099
required this .listeners,
1071
1100
required this .onMessage,
@@ -1153,7 +1182,7 @@ class LikeCloseEvent {
1153
1182
final int code;
1154
1183
1155
1184
/// Returns the WebSocket connection close reason provided by the server. */
1156
- final String reason;
1185
+ final String ? reason;
1157
1186
1158
1187
final bool ? wasClean;
1159
1188
0 commit comments