1
1
/*
2
- * Copyright 2002-2016 the original author or authors.
2
+ * Copyright 2002-2017 the original author or authors.
3
3
*
4
4
* Licensed under the Apache License, Version 2.0 (the "License");
5
5
* you may not use this file except in compliance with the License.
20
20
import java .util .List ;
21
21
import java .util .Properties ;
22
22
23
+ import io .netty .channel .EventLoopGroup ;
23
24
import reactor .Environment ;
24
25
import reactor .core .config .ConfigurationReader ;
25
26
import reactor .core .config .DispatcherConfiguration ;
26
27
import reactor .core .config .DispatcherType ;
27
28
import reactor .core .config .ReactorConfiguration ;
28
- import reactor .io .net .NetStreams ;
29
+ import reactor .io .net .NetStreams . TcpClientFactory ;
29
30
import reactor .io .net .Spec .TcpClientSpec ;
31
+ import reactor .io .net .impl .netty .NettyClientSocketOptions ;
30
32
33
+ import org .springframework .context .Lifecycle ;
31
34
import org .springframework .messaging .Message ;
32
35
import org .springframework .messaging .tcp .TcpOperations ;
33
36
import org .springframework .messaging .tcp .reactor .Reactor2TcpClient ;
34
37
import org .springframework .util .concurrent .ListenableFuture ;
35
38
36
39
/**
37
- * A STOMP over TCP client that uses
38
- * {@link Reactor2TcpClient}.
40
+ * A STOMP over TCP client that uses {@link Reactor2TcpClient}.
39
41
*
40
42
* @author Rossen Stoyanchev
41
43
* @since 4.2
42
44
*/
43
- public class Reactor2TcpStompClient extends StompClientSupport {
45
+ public class Reactor2TcpStompClient extends StompClientSupport implements Lifecycle {
44
46
45
47
private final TcpOperations <byte []> tcpClient ;
46
48
49
+ private final EventLoopGroup eventLoopGroup ;
50
+
51
+ private final Environment environment ;
52
+
53
+ private volatile boolean running = false ;
54
+
47
55
48
56
/**
49
57
* Create an instance with host "127.0.0.1" and port 61613.
@@ -57,11 +65,11 @@ public Reactor2TcpStompClient() {
57
65
* @param host the host
58
66
* @param port the port
59
67
*/
60
- public Reactor2TcpStompClient (final String host , final int port ) {
61
- ConfigurationReader reader = new StompClientDispatcherConfigReader ();
62
- Environment environment = new Environment ( reader ). assignErrorJournal ();
63
- StompTcpClientSpecFactory factory = new StompTcpClientSpecFactory ( environment , host , port );
64
- this . tcpClient = new Reactor2TcpClient < byte []>( factory );
68
+ public Reactor2TcpStompClient (String host , int port ) {
69
+ this . eventLoopGroup = Reactor2TcpClient . initEventLoopGroup ();
70
+ this . environment = new Environment ();
71
+ this . tcpClient = new Reactor2TcpClient < byte []>(
72
+ new StompTcpClientSpecFactory ( host , port , this . eventLoopGroup , this . environment ) );
65
73
}
66
74
67
75
/**
@@ -70,6 +78,43 @@ public Reactor2TcpStompClient(final String host, final int port) {
70
78
*/
71
79
public Reactor2TcpStompClient (TcpOperations <byte []> tcpClient ) {
72
80
this .tcpClient = tcpClient ;
81
+ this .eventLoopGroup = null ;
82
+ this .environment = null ;
83
+ }
84
+
85
+
86
+ @ Override
87
+ public void start () {
88
+ if (!isRunning ()) {
89
+ this .running = true ;
90
+
91
+ }
92
+ }
93
+
94
+ @ Override
95
+ public void stop () {
96
+ if (isRunning ()) {
97
+ this .running = false ;
98
+ try {
99
+ if (this .eventLoopGroup != null ) {
100
+ this .eventLoopGroup .shutdownGracefully ().await (5000 );
101
+ }
102
+ if (this .environment != null ) {
103
+ this .environment .shutdown ();
104
+ }
105
+ }
106
+ catch (InterruptedException ex ) {
107
+ if (logger .isErrorEnabled ()) {
108
+ logger .error ("Failed to shutdown gracefully" , ex );
109
+ }
110
+ }
111
+
112
+ }
113
+ }
114
+
115
+ @ Override
116
+ public boolean isRunning () {
117
+ return this .running ;
73
118
}
74
119
75
120
@@ -120,30 +165,36 @@ public ReactorConfiguration read() {
120
165
}
121
166
122
167
123
- private static class StompTcpClientSpecFactory
124
- implements NetStreams .TcpClientFactory <Message <byte []>, Message <byte []>> {
125
-
126
- private final Environment environment ;
168
+ private static class StompTcpClientSpecFactory implements TcpClientFactory <Message <byte []>, Message <byte []>> {
127
169
128
170
private final String host ;
129
171
130
172
private final int port ;
131
173
132
- public StompTcpClientSpecFactory (Environment environment , String host , int port ) {
133
- this .environment = environment ;
174
+ private final EventLoopGroup eventLoopGroup ;
175
+
176
+ private final Environment environment ;
177
+
178
+
179
+ public StompTcpClientSpecFactory (String host , int port , EventLoopGroup group , Environment environment ) {
134
180
this .host = host ;
135
181
this .port = port ;
182
+ this .eventLoopGroup = group ;
183
+ this .environment = environment ;
136
184
}
137
185
138
186
@ Override
139
187
public TcpClientSpec <Message <byte []>, Message <byte []>> apply (
140
188
TcpClientSpec <Message <byte []>, Message <byte []>> tcpClientSpec ) {
141
189
190
+ final Reactor2StompCodec codec = new Reactor2StompCodec (new StompEncoder (), new StompDecoder ());
191
+
142
192
return tcpClientSpec
143
- .codec (new Reactor2StompCodec (new StompEncoder (), new StompDecoder ()))
144
193
.env (this .environment )
145
- .dispatcher (this .environment .getCachedDispatchers ("StompClient" ).get ())
146
- .connect (this .host , this .port );
194
+ .dispatcher (this .environment .getDispatcher (Environment .WORK_QUEUE ))
195
+ .connect (this .host , this .port )
196
+ .codec (codec )
197
+ .options (new NettyClientSocketOptions ().eventLoopGroup (this .eventLoopGroup ));
147
198
}
148
199
}
149
200
0 commit comments