5
5
import io .netty .util .collection .IntObjectMap ;
6
6
import io .rsocket .DuplexConnection ;
7
7
import io .rsocket .RSocket ;
8
+ import io .rsocket .exceptions .CanceledException ;
8
9
import io .rsocket .frame .decoder .PayloadDecoder ;
9
10
import io .rsocket .plugins .RequestInterceptor ;
10
11
import java .util .Objects ;
11
12
import java .util .function .Function ;
13
+ import reactor .core .publisher .Sinks ;
12
14
import reactor .util .annotation .Nullable ;
13
15
14
16
class RequesterResponderSupport {
@@ -19,19 +21,24 @@ class RequesterResponderSupport {
19
21
private final PayloadDecoder payloadDecoder ;
20
22
private final ByteBufAllocator allocator ;
21
23
private final DuplexConnection connection ;
24
+ private final Sinks .Empty <Void > onGracefulShutdownSink ;
22
25
@ Nullable private final RequestInterceptor requestInterceptor ;
23
26
24
27
@ Nullable final StreamIdSupplier streamIdSupplier ;
25
28
final IntObjectMap <FrameHandler > activeStreams ;
26
29
30
+ boolean terminating ;
31
+ boolean terminated ;
32
+
27
33
public RequesterResponderSupport (
28
34
int mtu ,
29
35
int maxFrameLength ,
30
36
int maxInboundPayloadSize ,
31
37
PayloadDecoder payloadDecoder ,
32
38
DuplexConnection connection ,
33
39
@ Nullable StreamIdSupplier streamIdSupplier ,
34
- Function <RSocket , ? extends RequestInterceptor > requestInterceptorFunction ) {
40
+ Function <RSocket , ? extends RequestInterceptor > requestInterceptorFunction ,
41
+ Sinks .Empty <Void > onGracefulShutdownSink ) {
35
42
36
43
this .activeStreams = new IntObjectHashMap <>();
37
44
this .mtu = mtu ;
@@ -41,6 +48,7 @@ public RequesterResponderSupport(
41
48
this .allocator = connection .alloc ();
42
49
this .streamIdSupplier = streamIdSupplier ;
43
50
this .connection = connection ;
51
+ this .onGracefulShutdownSink = onGracefulShutdownSink ;
44
52
this .requestInterceptor = requestInterceptorFunction .apply ((RSocket ) this );
45
53
}
46
54
@@ -88,6 +96,9 @@ public int getNextStreamId() {
88
96
final StreamIdSupplier streamIdSupplier = this .streamIdSupplier ;
89
97
if (streamIdSupplier != null ) {
90
98
synchronized (this ) {
99
+ if (this .terminating ) {
100
+ throw new CanceledException ("Disposed" );
101
+ }
91
102
return streamIdSupplier .nextStreamId (this .activeStreams );
92
103
}
93
104
} else {
@@ -107,6 +118,10 @@ public int addAndGetNextStreamId(FrameHandler frameHandler) {
107
118
if (streamIdSupplier != null ) {
108
119
final IntObjectMap <FrameHandler > activeStreams = this .activeStreams ;
109
120
synchronized (this ) {
121
+ if (this .terminating ) {
122
+ throw new CanceledException ("Disposed" );
123
+ }
124
+
110
125
final int streamId = streamIdSupplier .nextStreamId (activeStreams );
111
126
112
127
activeStreams .put (streamId , frameHandler );
@@ -119,6 +134,11 @@ public int addAndGetNextStreamId(FrameHandler frameHandler) {
119
134
}
120
135
121
136
public synchronized boolean add (int streamId , FrameHandler frameHandler ) {
137
+ if (this .terminating ) {
138
+ throw new CanceledException (
139
+ "This RSocket is either disposed or disposing, and no longer accepting new requests" );
140
+ }
141
+
122
142
final IntObjectMap <FrameHandler > activeStreams = this .activeStreams ;
123
143
// copy of Map.putIfAbsent(key, value) without `streamId` boxing
124
144
final FrameHandler previousHandler = activeStreams .get (streamId );
@@ -148,14 +168,45 @@ public synchronized FrameHandler get(int streamId) {
148
168
* @return {@code true} if there is {@link FrameHandler} for the given {@code streamId} and the
149
169
* instance equals to the passed one
150
170
*/
151
- public synchronized boolean remove (int streamId , FrameHandler frameHandler ) {
152
- final IntObjectMap <FrameHandler > activeStreams = this .activeStreams ;
153
- // copy of Map.remove(key, value) without `streamId` boxing
154
- final FrameHandler curValue = activeStreams .get (streamId );
155
- if (!Objects .equals (curValue , frameHandler )) {
156
- return false ;
171
+ public boolean remove (int streamId , FrameHandler frameHandler ) {
172
+ final boolean terminated ;
173
+ synchronized (this ) {
174
+ final IntObjectMap <FrameHandler > activeStreams = this .activeStreams ;
175
+ // copy of Map.remove(key, value) without `streamId` boxing
176
+ final FrameHandler curValue = activeStreams .get (streamId );
177
+ if (!Objects .equals (curValue , frameHandler )) {
178
+ return false ;
179
+ }
180
+ activeStreams .remove (streamId );
181
+ if (this .terminating && activeStreams .size () == 0 ) {
182
+ terminated = true ;
183
+ this .terminated = true ;
184
+ } else {
185
+ terminated = false ;
186
+ }
187
+ }
188
+
189
+ if (terminated ) {
190
+ onGracefulShutdownSink .tryEmitEmpty ();
157
191
}
158
- activeStreams .remove (streamId );
159
192
return true ;
160
193
}
194
+
195
+ public void terminate () {
196
+ final boolean terminated ;
197
+ synchronized (this ) {
198
+ this .terminating = true ;
199
+
200
+ if (activeStreams .size () == 0 ) {
201
+ terminated = true ;
202
+ this .terminated = true ;
203
+ } else {
204
+ terminated = false ;
205
+ }
206
+ }
207
+
208
+ if (terminated ) {
209
+ onGracefulShutdownSink .tryEmitEmpty ();
210
+ }
211
+ }
161
212
}
0 commit comments