forked from mklooss/credis
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathSentinel.php
239 lines (239 loc) · 7.84 KB
/
Sentinel.php
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
<?php
/**
* Credis_Sentinel
*
* Implements the Sentinel API as mentioned on http://redis.io/topics/sentinel.
* Sentinel is aware of master and slave nodes in a cluster and returns instances of Credis_Client accordingly.
*
* The complexity of read/write splitting can also be abstract by calling the createCluster() method which returns a
* Credis_Cluster object that contains both the master server and a random slave. Credis_Cluster takes care of the
* read/write splitting
*
* @author Thijs Feryn <[email protected]>
* @license http://www.opensource.org/licenses/mit-license.php The MIT License
* @package Credis_Sentinel
*/
class Credis_Sentinel
{
/**
* Contains a client that connects to a Sentinel node.
* Sentinel uses the same protocol as Redis which makes using Credis_Client convenient.
* @var Credis_Client
*/
protected $_client;
/**
* Contains an active instance of Credis_Cluster per master pool
* @var Array
*/
protected $_cluster = array();
/**
* Contains an active instance of Credis_Client representing a master
* @var Array
*/
protected $_master = array();
/**
* Contains an array Credis_Client objects representing all slaves per master pool
* @var Array
*/
protected $_slaves = array();
/**
* Use the phpredis extension or the standalone implementation
* @var bool
*/
protected $_standAlone = false;
/**
* Connect with a Sentinel node. Sentinel will do the master and slave discovery
* @param Credis_Client $client
*/
public function __construct(Credis_Client $client)
{
if(!$client instanceof Credis_Client){
throw new CredisException('Sentinel client should be an instance of Credis_Client');
}
$client->forceStandalone();
$this->_client = $client;
}
/**
* @return Credis_Sentinel
*/
public function forceStandalone()
{
$this->_standAlone = true;
return $this;
}
/**
* Discover the master node automatically and return an instance of Credis_Client that connects to the master
* @param string $name
* @return Credis_Client
*/
public function createMasterClient($name)
{
$master = $this->getMasterAddressByName($name);
if(!isset($master[0]) || !isset($master[1])){
throw new CredisException('Master not found');
}
return new Credis_Client($master[0],$master[1]);
}
/**
* If a Credis_Client object exists for a master, return it. Otherwise create one and return it
* @param string $name
* @return Credis_Client
*/
public function getMasterClient($name)
{
if(!isset($this->_master[$name])){
$this->_master[$name] = $this->createMasterClient($name);
}
return $this->_master[$name];
}
/**
* Discover the slave nodes automatically and return an array of Credis_Client objects
* @param string $name
* @return Credis_Client[]
*/
public function createSlaveClients($name)
{
$slaves = $this->slaves($name);
$workingSlaves = array();
foreach($slaves as $slave) {
if(!isset($slave[9])){
throw new CredisException('Can\' retrieve slave status');
}
if(!strstr($slave[9],'s_down') && !strstr($slave[9],'disconnected')) {
$workingSlaves[] = new Credis_Client($slave[3],$slave[5]);
}
}
return $workingSlaves;
}
/**
* If an array of Credis_Client objects exist for a set of slaves, return them. Otherwise create and return them
* @param string $name
* @return Credis_Client[]
*/
public function getSlaveClients($name)
{
if(!isset($this->_slaves[$name])){
$this->_slaves[$name] = $this->createSlaveClients($name);
}
return $this->_slaves[$name];
}
/**
* Returns a Redis cluster object containing a random slave and the master
* When $selectRandomSlave is true, only one random slave is passed.
* When $selectRandomSlave is false, all clients are passed and hashing is applied in Credis_Cluster
* When $writeOnly is false, the master server will also be used for read commands.
* @param string $name
* @param int $db
* @param int $replicas
* @param bool $selectRandomSlave
* @param bool $writeOnly
* @return Credis_Cluster
*/
public function createCluster($name, $db=0, $replicas=128, $selectRandomSlave=true, $writeOnly=false)
{
$clients = array();
$workingClients = array();
$master = $this->master($name);
if(strstr($master[9],'s_down') || strstr($master[9],'disconnected')) {
throw new CredisException('The master is down');
}
$slaves = $this->slaves($name);
foreach($slaves as $slave){
if(!strstr($slave[9],'s_down') && !strstr($slave[9],'disconnected')) {
$workingClients[] = array('host'=>$slave[3],'port'=>$slave[5],'master'=>false,'db'=>$db);
}
}
if(count($workingClients)>0){
if($selectRandomSlave){
if(!$writeOnly){
$workingClients[] = array('host'=>$master[3],'port'=>$master[5],'master'=>false,'db'=>$db);
}
$clients[] = $workingClients[rand(0,count($workingClients)-1)];
} else {
$clients = $workingClients;
}
}
$clients[] = array('host'=>$master[3],'port'=>$master[5], 'db'=>$db ,'master'=>true,'write_only'=>$writeOnly);
return new Credis_Cluster($clients,$replicas,$this->_standAlone);
}
/**
* If a Credis_Cluster object exists, return it. Otherwise create one and return it.
* @param string $name
* @param int $db
* @param int $replicas
* @param bool $selectRandomSlave
* @param bool $writeOnly
* @return Credis_Cluster
*/
public function getCluster($name, $db=0, $replicas=128, $selectRandomSlave=true, $writeOnly=false)
{
if(!isset($this->_cluster[$name])){
$this->_cluster[$name] = $this->createCluster($name, $db, $replicas, $selectRandomSlave, $writeOnly);
}
return $this->_cluster[$name];
}
/**
* Catch-all method
* @param string $name
* @param array $args
* @return mixed
*/
public function __call($name, $args)
{
array_unshift($args,$name);
return call_user_func(array($this->_client,'sentinel'),$args);
}
/**
* Return information about all registered master servers
* @return mixed
*/
public function masters()
{
return $this->_client->sentinel('masters');
}
/**
* Return all information for slaves that are associated with a single master
* @param string $name
* @return mixed
*/
public function slaves($name)
{
return $this->_client->sentinel('slaves',$name);
}
/**
* Get the information for a specific master
* @param string $name
* @return mixed
*/
public function master($name)
{
return $this->_client->sentinel('master',$name);
}
/**
* Get the hostname and port for a specific master
* @param string $name
* @return mixed
*/
public function getMasterAddressByName($name)
{
return $this->_client->sentinel('get-master-addr-by-name',$name);
}
/**
* Check if the Sentinel is still responding
* @param string $name
* @return mixed
*/
public function ping()
{
return $this->_client->ping();
}
/**
* Perform an auto-failover which will re-elect another master and make the current master a slave
* @param string $name
* @return mixed
*/
public function failover($name)
{
return $this->_client->sentinel('failover',$name);
}
}