Merge "Http::getProxy() method to get proxy configuration"
[lhc/web/wiklou.git] / includes / libs / objectcache / APCBagOStuff.php
index b5419b7..8f70fc7 100644 (file)
  * @ingroup Cache
  */
 class APCBagOStuff extends BagOStuff {
+
+       /**
+        * @var bool If true, trust the APC implementation to serialize and
+        * deserialize objects correctly. If false, (de-)serialize in PHP.
+        */
+       protected $nativeSerialize;
+
        /**
         * @var string String to append to each APC key. This may be changed
         *  whenever the handling of values is changed, to prevent existing code
         *  from encountering older values which it cannot handle.
-        **/
-       const KEY_SUFFIX = ':1';
+        */
+       const KEY_SUFFIX = ':2';
+
+       /**
+        * Constructor
+        *
+        * Available parameters are:
+        *   - nativeSerialize:     If true, pass objects to apc_store(), and trust it
+        *                          to serialize them correctly. If false, serialize
+        *                          all values in PHP.
+        *
+        * @param array $params
+        */
+       public function __construct( array $params = [] ) {
+               parent::__construct( $params );
+
+               if ( isset( $params['nativeSerialize'] ) ) {
+                       $this->nativeSerialize = $params['nativeSerialize'];
+               } elseif ( extension_loaded( 'apcu' ) && ini_get( 'apc.serializer' ) === 'default' ) {
+                       // APCu has a memory corruption bug when the serializer is set to 'default'.
+                       // See T120267, and upstream bug reports:
+                       //  - https://github.com/krakjoe/apcu/issues/38
+                       //  - https://github.com/krakjoe/apcu/issues/35
+                       //  - https://github.com/krakjoe/apcu/issues/111
+                       $this->logger->warning(
+                               'The APCu extension is loaded and the apc.serializer INI setting ' .
+                               'is set to "default". This can cause memory corruption! ' .
+                               'You should change apc.serializer to "php" instead. ' .
+                               'See <https://github.com/krakjoe/apcu/issues/38>.'
+                       );
+                       $this->nativeSerialize = false;
+               } else {
+                       $this->nativeSerialize = true;
+               }
+       }
 
        protected function doGet( $key, $flags = 0 ) {
                $val = apc_fetch( $key . self::KEY_SUFFIX );
 
+               if ( is_string( $val ) && !$this->nativeSerialize ) {
+                       $val = $this->isInteger( $val )
+                               ? intval( $val )
+                               : unserialize( $val );
+               }
+
                return $val;
        }
 
        public function set( $key, $value, $exptime = 0, $flags = 0 ) {
+               if ( !$this->nativeSerialize && !$this->isInteger( $value ) ) {
+                       $value = serialize( $value );
+               }
+
                apc_store( $key . self::KEY_SUFFIX, $value, $exptime );
 
                return true;