9
9
use Predis \Client ;
10
10
use Predis \Connection \ConnectionException ;
11
11
use Predis \Connection \NodeConnectionInterface ;
12
+ use Predis \PubSub \Consumer ;
12
13
use Predis \Transaction \MultiExec ;
13
14
14
15
class PredisConnectionTest extends IntegrationTestCase
@@ -20,6 +21,16 @@ class PredisConnectionTest extends IntegrationTestCase
20
21
*/
21
22
protected $ subject ;
22
23
24
+ /**
25
+ * Messages to publish and expect for subscribe tests.
26
+ *
27
+ * @var array
28
+ */
29
+ protected $ expectedMessages = [
30
+ 'test-channel-1 ' => [ 'test message 1 ' , 'test message 2 ' , ],
31
+ 'test-channel-2 ' => [ 'test message 1 ' , 'test message 2 ' , ],
32
+ ];
33
+
23
34
/**
24
35
* Run this setup before each test
25
36
*
@@ -44,6 +55,143 @@ public function tearDown()
44
55
Mockery::close ();
45
56
}
46
57
58
+ public function testAllowsSubscriptionsOnAggregateConnection ()
59
+ {
60
+ // The Predis client itself does not currently support subscriptions on
61
+ // Sentinel connections and throws an exception that this class fixes.
62
+ $ this ->assertInstanceOf (Consumer::class, $ this ->subject ->pubSubLoop ());
63
+ }
64
+
65
+ public function testSubscribesToPubSubChannels ()
66
+ {
67
+ // Don't block the test with retries if it failed to read the expected
68
+ // number of messages from the server:
69
+ $ this ->subject ->setRetryLimit (0 );
70
+
71
+ foreach ([ 'createSubscription ' , 'subscribe ' ] as $ method ) {
72
+ $ received = [ ];
73
+
74
+ $ test = function ($ channels , $ count ) use (&$ received , $ method ) {
75
+ $ this ->subject ->$ method (
76
+ $ channels ,
77
+ function ($ message , $ channel ) use (&$ received , &$ count ) {
78
+ $ received [$ channel ][] = $ message ;
79
+
80
+ if (--$ count === 0 ) {
81
+ return false ;
82
+ }
83
+ }
84
+ );
85
+ };
86
+
87
+ $ this ->testClient ->publishForTest ($ test , $ this ->expectedMessages );
88
+
89
+ $ this ->assertEquals ($ this ->expectedMessages , $ received );
90
+ }
91
+ }
92
+
93
+ public function testSubscribesToPubSubChannelsByPattern ()
94
+ {
95
+ // Don't block the test with retries if it failed to read the expected
96
+ // number of messages from the server:
97
+ $ this ->subject ->setRetryLimit (0 );
98
+
99
+ foreach ([ 'createSubscription ' , 'psubscribe ' ] as $ method ) {
100
+ $ received = [ ];
101
+
102
+ $ test = function ($ channels , $ count ) use (&$ received , $ method ) {
103
+ $ this ->subject ->$ method (
104
+ 'test-channel-* ' ,
105
+ function ($ message , $ channel ) use (&$ received , &$ count ) {
106
+ $ received [$ channel ][] = $ message ;
107
+
108
+ if (--$ count === 0 ) {
109
+ return false ;
110
+ }
111
+ },
112
+ 'psubscribe '
113
+ );
114
+ };
115
+
116
+ $ this ->testClient ->publishForTest ($ test , $ this ->expectedMessages );
117
+
118
+ $ this ->assertEquals ($ this ->expectedMessages , $ received );
119
+ }
120
+ }
121
+
122
+ public function testSubscribesToPubSubChannelsUsingPredisApi ()
123
+ {
124
+ // Don't block the test with retries if it failed to read the expected
125
+ // number of messages from the server:
126
+ $ this ->subject ->setRetryLimit (0 );
127
+
128
+ $ received = [ ];
129
+
130
+ $ test = function ($ channels , $ count ) use (&$ received ) {
131
+ $ this ->subject ->pubSubLoop (
132
+ [ 'subscribe ' => $ channels ],
133
+ function ($ loop , $ message ) use (&$ received , &$ count ) {
134
+ if ($ message ->kind === 'message ' ) {
135
+ $ received [$ message ->channel ][] = $ message ->payload ;
136
+
137
+ if (--$ count === 0 ) {
138
+ return false ;
139
+ }
140
+ }
141
+ }
142
+ );
143
+ };
144
+
145
+ $ this ->testClient ->publishForTest ($ test , $ this ->expectedMessages );
146
+
147
+ $ this ->assertEquals ($ this ->expectedMessages , $ received );
148
+ }
149
+
150
+ public function testRetriesSubscriptionWhenConnectionFails ()
151
+ {
152
+ $ this ->switchToMinimumTimeout ();
153
+
154
+ $ expectedRetries = 2 ;
155
+ $ this ->subject = new PredisConnection ($ this ->makeClientSpy ());
156
+ $ this ->subject ->setRetryLimit ($ expectedRetries );
157
+ $ this ->subject ->setRetryWait (0 ); // retry immediately
158
+
159
+ // With a read-write timeout, Predis throws a ConnectionException if
160
+ // nothing publishes to the channel for the duration specified by the
161
+ // timeout value. We'll use this with a low timeout to simulate a real
162
+ // connection failure so we don't need to block a server manually.
163
+ try {
164
+ $ this ->subject ->subscribe ([ 'channel ' ], function () {
165
+ return false ;
166
+ });
167
+ } catch (ConnectionException $ exception ) {
168
+ // With PHPUnit, we need to wrap the throwing block to perform
169
+ // assertions afterward.
170
+ }
171
+
172
+ $ this ->subject ->client ()->getConnection () // +1 for initial attempt:
173
+ ->shouldHaveReceived ('querySentinel ' )->times ($ expectedRetries + 1 );
174
+ }
175
+
176
+ public function testSubscribesToSlaveByDefault ()
177
+ {
178
+ $ loop = $ this ->subject ->pubSubLoop ();
179
+ $ role = $ loop ->getClient ()->executeRaw ([ 'ROLE ' ]);
180
+
181
+ $ this ->assertEquals ('slave ' , $ role [0 ]);
182
+ }
183
+
184
+ public function testSubscribeFallsBackToMaster ()
185
+ {
186
+ $ this ->subject ->client ()->getConnection ()
187
+ ->shouldReceive ('getSlaves ' )->andReturn ([ ]);
188
+
189
+ $ loop = $ this ->subject ->pubSubLoop ();
190
+ $ role = $ loop ->getClient ()->executeRaw ([ 'ROLE ' ]);
191
+
192
+ $ this ->assertEquals ('master ' , $ role [0 ]);
193
+ }
194
+
47
195
public function testAllowsTransactionsOnAggregateConnection ()
48
196
{
49
197
// The Predis client itself does not currently support transactions on
@@ -121,7 +269,7 @@ public function testRetriesTransactionWhenConnectionFails()
121
269
122
270
public function testCanReconnectWhenConnectionFails ()
123
271
{
124
- $ retries = (2 / $ this ->switchToMinimumTimeout ()) + 1 ;
272
+ $ retries = ceil (2 / $ this ->switchToMinimumTimeout ()) + 1 ;
125
273
$ attempts = 0 ;
126
274
127
275
$ this ->subject = new PredisConnection ($ this ->makeClientSpy ());
0 commit comments