24use Wikimedia\ScopedCallback;
52 parent::__construct( $params );
56 'compress_threshold' => 1500,
57 'connect_timeout' => 0.5,
59 'serializer' =>
'php',
60 'use_binary_protocol' =>
false,
61 'allow_tcp_nagle_delay' =>
true
64 if ( $params[
'persistent'] ) {
68 $connectionPoolId = md5(
serialize( $params ) );
69 $client =
new Memcached( $connectionPoolId );
71 $client =
new Memcached();
74 $this->initializeClient( $client, $params );
76 $this->client = $client;
80 ini_set(
'memcached.compression_threshold', $params[
'compress_threshold'] );
91 private function initializeClient( Memcached $client, array $params ) {
92 if ( $client->getServerList() ) {
93 $this->logger->debug( __METHOD__ .
": pre-initialized client instance." );
98 $this->logger->debug( __METHOD__ .
": initializing new client instance." );
101 Memcached::OPT_NO_BLOCK =>
false,
102 Memcached::OPT_BUFFER_WRITES =>
false,
103 Memcached::OPT_NOREPLY =>
false,
105 Memcached::OPT_BINARY_PROTOCOL => $params[
'use_binary_protocol'],
107 Memcached::OPT_CONNECT_TIMEOUT => $params[
'connect_timeout'] * 1000,
108 Memcached::OPT_SEND_TIMEOUT => $params[
'timeout'],
109 Memcached::OPT_RECV_TIMEOUT => $params[
'timeout'],
110 Memcached::OPT_POLL_TIMEOUT => $params[
'timeout'] / 1000,
112 Memcached::OPT_TCP_NODELAY => !$params[
'allow_tcp_nagle_delay'],
114 Memcached::OPT_LIBKETAMA_COMPATIBLE =>
true
116 if ( isset( $params[
'retry_timeout'] ) ) {
117 $options[Memcached::OPT_RETRY_TIMEOUT] = $params[
'retry_timeout'];
119 if ( isset( $params[
'server_failure_limit'] ) ) {
120 $options[Memcached::OPT_SERVER_FAILURE_LIMIT] = $params[
'server_failure_limit'];
122 if ( $params[
'serializer'] ===
'php' ) {
123 $options[Memcached::OPT_SERIALIZER] = Memcached::SERIALIZER_PHP;
124 } elseif ( $params[
'serializer'] ===
'igbinary' ) {
126 if ( !Memcached::HAVE_IGBINARY ) {
127 throw new RuntimeException(
128 __CLASS__ .
': the igbinary extension is not available ' .
129 'but igbinary serialization was requested.'
132 $options[Memcached::OPT_SERIALIZER] = Memcached::SERIALIZER_IGBINARY;
135 if ( !$client->setOptions( $options ) ) {
136 throw new RuntimeException(
137 "Invalid options: " . json_encode( $options, JSON_PRETTY_PRINT )
142 foreach ( $params[
'servers'] as $host ) {
143 if ( preg_match(
'/^\[(.+)\]:(\d+)$/', $host, $m ) ) {
144 $servers[] = [ $m[1], (int)$m[2] ];
145 } elseif ( preg_match(
'/^([^:]+):(\d+)$/', $host, $m ) ) {
146 $servers[] = [ $m[1], (int)$m[2] ];
148 $servers[] = [ $host, false ];
152 if ( !$client->addServers( $servers ) ) {
153 throw new RuntimeException(
"Failed to inject server address list" );
165 private function noReplyScope( $flags ) {
166 if ( $flags !==
true && !( $flags & self::WRITE_BACKGROUND ) ) {
169 $client = $this->client;
170 $client->setOption( Memcached::OPT_NOREPLY,
true );
171 return new ScopedCallback(
static function () use ( $client ) {
172 $client->setOption( Memcached::OPT_NOREPLY,
false );
176 protected function doGet( $key, $flags = 0, &$casToken =
null ) {
177 $getToken = ( $casToken === self::PASS_BY_REF );
180 $this->
debug(
"get($key)" );
187 $flags = Memcached::GET_EXTENDED;
188 $res = $this->client->get( $routeKey,
null, $flags );
189 if ( is_array(
$res ) ) {
190 $result =
$res[
'value'];
191 $casToken =
$res[
'cas'];
196 $result = $this->client->get( $routeKey );
202 protected function doSet( $key, $value, $exptime = 0, $flags = 0 ) {
203 $this->
debug(
"set($key)" );
207 $noReplyScope = $this->noReplyScope( $flags );
208 $result = $this->client->set( $routeKey, $value, $this->
fixExpiry( $exptime ) );
209 ScopedCallback::consume( $noReplyScope );
211 return ( !$result && $this->client->getResultCode() === Memcached::RES_NOTSTORED )
217 protected function doCas( $casToken, $key, $value, $exptime = 0, $flags = 0 ) {
218 $this->
debug(
"cas($key)" );
221 $result = $this->client->cas(
231 $this->
debug(
"delete($key)" );
234 $noReplyScope = $this->noReplyScope( $flags );
235 $result = $this->client->delete( $routeKey );
236 ScopedCallback::consume( $noReplyScope );
238 return ( !$result && $this->client->getResultCode() === Memcached::RES_NOTFOUND )
244 protected function doAdd( $key, $value, $exptime = 0, $flags = 0 ) {
245 $this->
debug(
"add($key)" );
248 $noReplyScope = $this->noReplyScope( $flags );
249 $result = $this->client->add(
254 ScopedCallback::consume( $noReplyScope );
259 public function incr( $key, $value = 1, $flags = 0 ) {
260 $this->
debug(
"incr($key)" );
263 $noReplyScope = $this->noReplyScope( $flags );
264 $result = $this->client->increment( $routeKey, $value );
265 ScopedCallback::consume( $noReplyScope );
270 public function decr( $key, $value = 1, $flags = 0 ) {
271 $this->
debug(
"decr($key)" );
274 $noReplyScope = $this->noReplyScope( $flags );
275 $result = $this->client->decrement( $routeKey, $value );
276 ScopedCallback::consume( $noReplyScope );
282 $this->
debug(
"incrWithInit($key)" );
285 $scope = $this->noReplyScope(
true );
286 $this->
checkResult( $key, $this->client->add( $routeKey, $init - $step, $this->fixExpiry( $exptime ) ) );
287 $this->
checkResult( $key, $this->client->increment( $routeKey, $step ) );
288 ScopedCallback::consume( $scope );
294 $this->
debug(
"incrWithInit($key)" );
297 $result = $this->client->increment( $routeKey, $step );
299 if ( $newValue ===
false && !$this->
getLastError( $watchPoint ) ) {
301 $result = $this->client->add( $routeKey, $init, $this->
fixExpiry( $exptime ) );
302 $newValue = $this->
checkResult( $key, $result ) ? $init :
false;
303 if ( $newValue ===
false && !$this->
getLastError( $watchPoint ) ) {
305 $result = $this->client->increment( $routeKey, $step );
330 static $statusByCode = [
331 Memcached::RES_HOST_LOOKUP_FAILURE => self::ERR_UNREACHABLE,
332 Memcached::RES_SERVER_MARKED_DEAD => self::ERR_UNREACHABLE,
333 Memcached::RES_SERVER_TEMPORARILY_DISABLED => self::ERR_UNREACHABLE,
334 Memcached::RES_UNKNOWN_READ_FAILURE => self::ERR_NO_RESPONSE,
335 Memcached::RES_WRITE_FAILURE => self::ERR_NO_RESPONSE,
336 Memcached::RES_PARTIAL_READ => self::ERR_NO_RESPONSE,
339 3 => self::ERR_UNREACHABLE,
340 27 => self::ERR_UNREACHABLE,
341 6 => self::ERR_NO_RESPONSE
344 if ( $result !==
false ) {
348 $client = $this->client;
349 $code = $client->getResultCode();
351 case Memcached::RES_SUCCESS:
353 case Memcached::RES_DATA_EXISTS:
354 case Memcached::RES_NOTSTORED:
355 case Memcached::RES_NOTFOUND:
356 $this->
debug(
"result: " . $client->getResultMessage() );
359 $msg = $client->getResultMessage();
361 if ( $key !==
false ) {
362 $server = $client->getServerByKey( $key );
363 $logCtx[
'memcached-server'] =
"{$server['host']}:{$server['port']}";
364 $logCtx[
'memcached-key'] = $key;
365 $msg =
"Memcached error for key \"{memcached-key}\" " .
366 "on server \"{memcached-server}\": $msg";
368 $msg =
"Memcached error: $msg";
370 $this->logger->error( $msg, $logCtx );
371 $this->
setLastError( $statusByCode[$code] ?? self::ERR_UNEXPECTED );
377 $this->
debug(
'getMulti(' . implode(
', ',
$keys ) .
')' );
380 foreach (
$keys as $key ) {
387 $resByRouteKey = $this->client->getMulti( $routeKeys );
389 if ( is_array( $resByRouteKey ) ) {
391 foreach ( $resByRouteKey as $routeKey => $value ) {
402 protected function doSetMulti( array $data, $exptime = 0, $flags = 0 ) {
403 $this->
debug(
'setMulti(' . implode(
', ', array_keys( $data ) ) .
')' );
406 $dataByRouteKey = [];
407 foreach ( $data as $key => $value ) {
411 $noReplyScope = $this->noReplyScope( $flags );
415 $result = @$this->client->setMulti( $dataByRouteKey, $exptime );
416 ScopedCallback::consume( $noReplyScope );
421 $this->
debug(
'deleteMulti(' . implode(
', ',
$keys ) .
')' );
424 foreach (
$keys as $key ) {
428 $noReplyScope = $this->noReplyScope( $flags );
429 $resultArray = $this->client->deleteMulti( $routeKeys ) ?: [];
430 ScopedCallback::consume( $noReplyScope );
433 foreach ( $resultArray as $code ) {
434 if ( !in_array( $code, [
true, Memcached::RES_NOTFOUND ],
true ) ) {
444 $this->
debug(
"touch($key)" );
449 $result = $this->client->touch( $routeKey, $this->
fixExpiry( $exptime ) );
455 if ( is_int( $value ) ) {
459 $serializer = $this->client->getOption( Memcached::OPT_SERIALIZER );
460 if ( $serializer === Memcached::SERIALIZER_PHP ) {
462 } elseif ( $serializer === Memcached::SERIALIZER_IGBINARY ) {
463 return igbinary_serialize( $value );
466 throw new UnexpectedValueException( __METHOD__ .
": got serializer '$serializer'." );
474 $serializer = $this->client->getOption( Memcached::OPT_SERIALIZER );
475 if ( $serializer === Memcached::SERIALIZER_PHP ) {
477 } elseif ( $serializer === Memcached::SERIALIZER_IGBINARY ) {
478 return igbinary_unserialize( $value );
481 throw new UnexpectedValueException( __METHOD__ .
": got serializer '$serializer'." );
unserialize( $serialized)
getLastError( $watchPoint=0)
Get the "last error" registry.
setLastError( $error)
Set the "last error" registry due to a problem encountered during an attempted operation.
watchErrors()
Get a "watch point" token that can be used to get the "last error" to occur after now.
int $lastError
BagOStuff:ERR_* constant of the last error that occurred.
isInteger( $value)
Check if a value is an integer.
guessSerialSizeOfValues(array $values)
Estimate the size of a each variable once serialized.
Base class for memcached clients.
validateKeyAndPrependRoute( $key)
A wrapper class for the PECL memcached client.
doAdd( $key, $value, $exptime=0, $flags=0)
Insert an item if it does not already exist.
doGet( $key, $flags=0, &$casToken=null)
Get an item.
doGetMulti(array $keys, $flags=0)
Get an associative array containing the item for each of the keys that have items.
doCas( $casToken, $key, $value, $exptime=0, $flags=0)
Set an item if the current CAS token matches the provided CAS token.
doSet( $key, $value, $exptime=0, $flags=0)
Set an item.
incr( $key, $value=1, $flags=0)
Increase stored value of $key by $value while preserving its TTL.
doChangeTTL( $key, $exptime, $flags)
doDeleteMulti(array $keys, $flags=0)
decr( $key, $value=1, $flags=0)
Decrease stored value of $key by $value while preserving its TTL.
setNewPreparedValues(array $valueByKey)
Stage a set of new key values for storage and estimate the amount of bytes needed.
doSetMulti(array $data, $exptime=0, $flags=0)
checkResult( $key, $result)
Check the return value from a client method call and take any necessary action.
doDelete( $key, $flags=0)
Delete an item.
__construct( $params)
Available parameters are:
doIncrWithInitAsync( $key, $exptime, $step, $init)
doIncrWithInitSync( $key, $exptime, $step, $init)