* @group BagOStuff * @covers BagOStuff * @covers MediumSpecificBagOStuff */ abstract class BagOStuffTestBase extends MediaWikiIntegrationTestCase { /** @var BagOStuff */ protected $cache; protected const TEST_TIME = 1563892142; protected function setUp(): void { parent::setUp(); try { $this->cache = $this->newCacheInstance(); } catch ( InvalidArgumentException $e ) { $this->markTestSkipped( "Cannot create cache instance for " . static::class . ': the configuration is presumably missing from $wgObjectCaches' ); } $this->cache->deleteMulti( [ $this->cache->makeKey( $this->testKey() ), $this->cache->makeKey( $this->testKey() ) . ':lock' ] ); } private function testKey() { return 'test-' . static::class; } /** * @return BagOStuff */ abstract protected function newCacheInstance(); protected function getCacheByClass( $className ) { $caches = $this->getConfVar( 'ObjectCaches' ); foreach ( $caches as $id => $cache ) { if ( ( $cache['class'] ?? '' ) === $className ) { return ObjectCache::getInstance( $id ); } } $this->markTestSkipped( "No $className is configured" ); } public function testMakeKey() { $cache = new HashBagOStuff( [ 'keyspace' => 'local_prefix' ] ); $localKey = $cache->makeKey( 'first', 'second', 'third' ); $globalKey = $cache->makeGlobalKey( 'first', 'second', 'third' ); $this->assertSame( 'local_prefix:first:second:third', $localKey, 'Local key interpolates parameters' ); $this->assertSame( 'global:first:second:third', $globalKey, 'Global key interpolates parameters and contains global prefix' ); $this->assertNotEquals( $localKey, $globalKey, 'Local key and global key with same parameters should not be equal' ); $this->assertNotEquals( $cache->makeKey( 'a', 'bc:', 'de' ), $cache->makeKey( 'a', 'bc', ':de' ) ); $keyEmptyCollection = $cache->makeKey( '', 'second', 'third' ); $this->assertSame( 'local_prefix::second:third', $keyEmptyCollection, 'Local key interpolates empty parameters' ); } public function testKeyIsGlobal() { $cache = new HashBagOStuff(); $localKey = $cache->makeKey( 'first', 'second', 'third' ); $globalKey = $cache->makeGlobalKey( 'first', 'second', 'third' ); $this->assertFalse( $cache->isKeyGlobal( $localKey ) ); $this->assertTrue( $cache->isKeyGlobal( $globalKey ) ); } public function testMerge() { $key = $this->cache->makeKey( $this->testKey() ); $calls = 0; $casRace = false; // emulate a race $callback = static function ( BagOStuff $cache, $key, $oldVal, &$expiry ) use ( &$calls, &$casRace ) { ++$calls; if ( $casRace ) { // Uses CAS instead? $cache->set( $key, 'conflict', 5 ); } return ( $oldVal === false ) ? 'merged' : $oldVal . 'merged'; }; // merge on non-existing value $merged = $this->cache->merge( $key, $callback, 5 ); $this->assertTrue( $merged ); $this->assertEquals( 'merged', $this->cache->get( $key ) ); // merge on existing value $merged = $this->cache->merge( $key, $callback, 5 ); $this->assertTrue( $merged ); $this->assertEquals( 'mergedmerged', $this->cache->get( $key ) ); $calls = 0; $casRace = true; $this->assertFalse( $this->cache->merge( $key, $callback, 5, 1 ), 'Non-blocking merge (CAS)' ); if ( $this->cache instanceof MultiWriteBagOStuff ) { $wrapper = TestingAccessWrapper::newFromObject( $this->cache ); $this->assertEquals( count( $wrapper->caches ), $calls ); } else { $this->assertSame( 1, $calls ); } } public function testChangeTTLRenew() { $key = $this->cache->makeKey( $this->testKey() ); $value = 'meow'; $this->cache->add( $key, $value, 60 ); $this->assertEquals( $value, $this->cache->get( $key ) ); $this->assertTrue( $this->cache->changeTTL( $key, 120 ) ); $this->assertTrue( $this->cache->changeTTL( $key, 120 ) ); $this->assertTrue( $this->cache->changeTTL( $key, 0 ) ); $this->assertEquals( $this->cache->get( $key ), $value ); $this->cache->delete( $key ); $this->assertFalse( $this->cache->changeTTL( $key, 15 ) ); } public function testChangeTTLExpireRel() { $key = $this->cache->makeKey( $this->testKey() ); $value = 'meow'; $this->cache->add( $key, $value, 5 ); $this->assertSame( $value, $this->cache->get( $key ) ); $this->assertTrue( $this->cache->changeTTL( $key, -3600 ) ); $this->assertFalse( $this->cache->get( $key ) ); $this->assertFalse( $this->cache->changeTTL( $key, -3600 ) ); } public function testChangeTTLExpireAbs() { $key = $this->cache->makeKey( $this->testKey() ); $value = 'meow'; $this->cache->add( $key, $value, 5 ); $this->assertSame( $value, $this->cache->get( $key ) ); $now = $this->cache->getCurrentTime(); $this->assertTrue( $this->cache->changeTTL( $key, (int)$now - 3600 ) ); $this->assertFalse( $this->cache->get( $key ) ); $this->assertFalse( $this->cache->changeTTL( $key, (int)$now - 3600 ) ); } public function testChangeTTLMulti() { $key1 = $this->cache->makeKey( 'test-key1' ); $key2 = $this->cache->makeKey( 'test-key2' ); $key3 = $this->cache->makeKey( 'test-key3' ); $key4 = $this->cache->makeKey( 'test-key4' ); // cleanup $this->cache->deleteMulti( [ $key1, $key2, $key3, $key4 ] ); $ok = $this->cache->changeTTLMulti( [ $key1, $key2, $key3 ], 30 ); $this->assertFalse( $ok, "No keys found" ); $this->assertFalse( $this->cache->get( $key1 ) ); $this->assertFalse( $this->cache->get( $key2 ) ); $this->assertFalse( $this->cache->get( $key3 ) ); $ok = $this->cache->setMulti( [ $key1 => 1, $key2 => 2, $key3 => 3 ] ); $this->assertTrue( $ok, "setMulti() succeeded" ); $this->assertCount( 3, $this->cache->getMulti( [ $key1, $key2, $key3 ] ), "setMulti() succeeded via getMulti() check" ); $ok = $this->cache->changeTTLMulti( [ $key1, $key2, $key3 ], 300 ); $this->assertTrue( $ok, "TTL bumped for all keys" ); $this->assertSame( 1, $this->cache->get( $key1 ) ); $this->assertEquals( 2, $this->cache->get( $key2 ) ); $this->assertEquals( 3, $this->cache->get( $key3 ) ); $ok = $this->cache->changeTTLMulti( [ $key1, $key2, $key3, $key4 ], 300 ); $this->assertFalse( $ok, "One key missing" ); $this->assertSame( 1, $this->cache->get( $key1 ), "Key still live" ); $ok = $this->cache->setMulti( [ $key1 => 1, $key2 => 2, $key3 => 3 ] ); $this->assertTrue( $ok, "setMulti() succeeded" ); $now = $this->cache->getCurrentTime(); $ok = $this->cache->changeTTLMulti( [ $key1, $key2, $key3 ], (int)$now + 86400 ); $this->assertTrue( $ok, "Expiry set for all keys" ); $this->assertSame( 1, $this->cache->get( $key1 ), "Key still live" ); // cleanup $this->cache->deleteMulti( [ $key1, $key2, $key3, $key4 ] ); } public function testAdd() { $key = $this->cache->makeKey( $this->testKey() ); $this->assertFalse( $this->cache->get( $key ) ); $this->assertTrue( $this->cache->add( $key, 'test', 5 ) ); $this->assertFalse( $this->cache->add( $key, 'test', 5 ) ); } public function testAddBackground() { $key = $this->cache->makeKey( $this->testKey() ); $this->assertFalse( $this->cache->get( $key ) ); $this->assertTrue( $this->cache->add( $key, 'test', 5, BagOStuff::WRITE_BACKGROUND ) ); for ( $i = 0; $i < 100 && $this->cache->get( $key ) !== 'test'; $i++ ) { usleep( 1000 ); } $this->assertSame( 'test', $this->cache->get( $key ) ); } public function testGet() { $value = [ 'this' => 'is', 'a' => 'test' ]; $key = $this->cache->makeKey( $this->testKey() ); $this->cache->add( $key, $value, 5 ); $this->assertSame( $this->cache->get( $key ), $value ); } public function testGetWithSetCallback() { $now = self::TEST_TIME; $cache = new HashBagOStuff( [] ); $cache->setMockTime( $now ); $key = $cache->makeKey( $this->testKey() ); $this->assertFalse( $cache->get( $key ), "No value" ); $value = $cache->getWithSetCallback( $key, 30, static function ( &$ttl ) { $ttl = 10; return 'hello kitty'; } ); $this->assertEquals( 'hello kitty', $value ); $this->assertEquals( $value, $cache->get( $key ), "Value set" ); $now += 11; $this->assertFalse( $cache->get( $key ), "Value expired" ); } public function testIncrWithInit() { $key = $this->cache->makeKey( $this->testKey() ); $val = $this->cache->get( $key ); $this->assertFalse( $val, "No value yet" ); $val = $this->cache->incrWithInit( $key, 0, 1, 3 ); $this->assertSame( 3, $val, "Correct init value" ); $val = $this->cache->incrWithInit( $key, 0, 1, 3 ); $this->assertSame( 4, $val, "Correct incremented value" ); $this->cache->delete( $key ); $val = $this->cache->incrWithInit( $key, 0, 5 ); $this->assertSame( 5, $val, "Correct incremented value" ); } public function testIncrWithInitAsync() { $key = $this->cache->makeKey( $this->testKey() ); $val = $this->cache->get( $key ); $this->assertFalse( $val, "No value yet" ); $val = $this->cache->incrWithInit( $key, 0, 1, 3, BagOStuff::WRITE_BACKGROUND ); if ( $val === true ) { $val = $this->cache->get( $key ); for ( $i = 0; $i < 1000 && $val !== 3; $i++ ) { usleep( 1000 ); $val = $this->cache->get( $key ); } } $this->assertSame( 3, $val ); $val = $this->cache->incrWithInit( $key, 0, 1, 3, BagOStuff::WRITE_BACKGROUND ); if ( $val === true ) { $val = $this->cache->get( $key ); for ( $i = 0; $i < 1000 && $val !== 4; $i++ ) { usleep( 1000 ); $val = $this->cache->get( $key ); } } $this->assertSame( 4, $val ); } public function testGetMulti() { $value1 = [ 'this' => 'is', 'a' => 'test' ]; $value2 = [ 'this' => 'is', 'another' => 'test' ]; $value3 = [ 'testing a key that may be encoded when sent to cache backend' ]; $value4 = [ 'another test where chars in key will be encoded' ]; $key1 = $this->cache->makeKey( 'test-1' ); $key2 = $this->cache->makeKey( 'test-2' ); // internally, MemcachedBagOStuffs will encode to will-%25-encode $key3 = $this->cache->makeKey( 'will-%-encode' ); $key4 = $this->cache->makeKey( 'flowdb:flow_ref:wiki:by-source:v3:Parser\'s_"broken"_+_(page)_&_grill:testwiki:1:4.7' ); // cleanup $this->cache->delete( $key1 ); $this->cache->delete( $key2 ); $this->cache->delete( $key3 ); $this->cache->delete( $key4 ); $this->cache->add( $key1, $value1, 5 ); $this->cache->add( $key2, $value2, 5 ); $this->cache->add( $key3, $value3, 5 ); $this->cache->add( $key4, $value4, 5 ); $this->assertEquals( [ $key1 => $value1, $key2 => $value2, $key3 => $value3, $key4 => $value4 ], $this->cache->getMulti( [ $key1, $key2, $key3, $key4 ] ) ); // cleanup $this->cache->delete( $key1 ); $this->cache->delete( $key2 ); $this->cache->delete( $key3 ); $this->cache->delete( $key4 ); } public function testSetDeleteMulti() { $map = [ $this->cache->makeKey( 'test-1' ) => 'Siberian', $this->cache->makeKey( 'test-2' ) => [ 'Huskies' ], $this->cache->makeKey( 'test-3' ) => [ 'are' => 'the' ], $this->cache->makeKey( 'test-4' ) => (object)[ 'greatest' => 'animal' ], $this->cache->makeKey( 'test-5' ) => 4, $this->cache->makeKey( 'test-6' ) => 'ever' ]; $this->assertTrue( $this->cache->setMulti( $map ) ); $this->assertEquals( $map, $this->cache->getMulti( array_keys( $map ) ) ); $this->assertTrue( $this->cache->deleteMulti( array_keys( $map ) ) ); $this->assertEquals( [], $this->cache->getMulti( array_keys( $map ), BagOStuff::READ_LATEST ) ); $this->assertEquals( [], $this->cache->getMulti( array_keys( $map ) ) ); } public function testDelete() { // Delete of non-existent key should return true $key = $this->cache->makeKey( 'nonexistent' ); $this->assertTrue( $this->cache->delete( $key ) ); $this->assertTrue( $this->cache->delete( $key, BagOStuff::WRITE_BACKGROUND ) ); } public function testSetSegmentable() { $key = $this->cache->makeKey( $this->testKey() ); $tiny = 418; $small = wfRandomString( 32 ); // 64 * 8 * 32768 = 16777216 bytes $big = str_repeat( wfRandomString( 32 ) . '-' . wfRandomString( 32 ), 32768 ); $callback = static function ( $cache, $key, $oldValue ) { return $oldValue . '!'; }; $cases = [ 'tiny' => $tiny, 'small' => $small, 'big' => $big ]; foreach ( $cases as $case => $value ) { $this->cache->set( $key, $value, 10, BagOStuff::WRITE_ALLOW_SEGMENTS ); $this->assertEquals( $value, $this->cache->get( $key ), "get $case" ); $this->assertEquals( [ $key => $value ], $this->cache->getMulti( [ $key ] ), "get $case" ); $this->assertTrue( $this->cache->merge( $key, $callback, 5, 1, BagOStuff::WRITE_ALLOW_SEGMENTS ), "merge $case" ); $this->assertEquals( "$value!", $this->cache->get( $key ), "merged $case" ); $this->assertEquals( "$value!", $this->cache->getMulti( [ $key ] )[$key], "merged $case" ); $this->assertTrue( $this->cache->deleteMulti( [ $key ] ), "delete $case" ); $this->assertFalse( $this->cache->get( $key ), "deleted $case" ); $this->assertEquals( [], $this->cache->getMulti( [ $key ] ), "deletd $case" ); $this->cache->set( $key, "@$value", 10, BagOStuff::WRITE_ALLOW_SEGMENTS ); $this->assertEquals( "@$value", $this->cache->get( $key ), "get $case" ); $this->assertTrue( $this->cache->delete( $key, BagOStuff::WRITE_PRUNE_SEGMENTS ), "prune $case" ); $this->assertFalse( $this->cache->get( $key ), "pruned $case" ); $this->assertEquals( [], $this->cache->getMulti( [ $key ] ), "pruned $case" ); } $this->cache->set( $key, 666, 10, BagOStuff::WRITE_ALLOW_SEGMENTS ); $this->assertEquals( 666, $this->cache->get( $key ) ); $this->assertEquals( 667, $this->cache->incrWithInit( $key, 10 ) ); $this->assertEquals( 667, $this->cache->get( $key ) ); $this->assertTrue( $this->cache->delete( $key ) ); $this->assertFalse( $this->cache->get( $key ) ); } public function testSetBackground() { $key = $this->cache->makeKey( $this->testKey() ); $this->assertTrue( $this->cache->set( $key, 'background', BagOStuff::WRITE_BACKGROUND ) ); } public function testGetScopedLock() { $key = $this->cache->makeKey( $this->testKey() ); $value1 = $this->cache->getScopedLock( $key, 0 ); $value2 = $this->cache->getScopedLock( $key, 0 ); $this->assertInstanceOf( ScopedCallback::class, $value1, 'First call returned lock' ); $this->assertNull( $value2, 'Duplicate call returned no lock' ); unset( $value1 ); $value3 = $this->cache->getScopedLock( $key, 0 ); $this->assertInstanceOf( ScopedCallback::class, $value3, 'Lock returned callback after release' ); unset( $value3 ); $value1 = $this->cache->getScopedLock( $key, 0, 5, 'reentry' ); $value2 = $this->cache->getScopedLock( $key, 0, 5, 'reentry' ); $this->assertInstanceOf( ScopedCallback::class, $value1, 'First reentrant call returned lock' ); $this->assertInstanceOf( ScopedCallback::class, $value2, 'Second reentrant call returned lock' ); } public function testReportDupes() { $logger = $this->createMock( Psr\Log\NullLogger::class ); $logger->expects( $this->once() ) ->method( 'warning' ) ->with( 'Duplicate get(): "{key}" fetched {count} times', [ 'key' => 'foo', 'count' => 2, ] ); $cache = new HashBagOStuff( [ 'reportDupes' => true, 'asyncHandler' => 'DeferredUpdates::addCallableUpdate', 'logger' => $logger, ] ); $cache->get( 'foo' ); $cache->get( 'bar' ); $cache->get( 'foo' ); DeferredUpdates::doUpdates(); } public function testLocking() { $key = $this->cache->makeKey( $this->testKey() ); $this->assertTrue( $this->cache->lock( $key ) ); $this->assertFalse( $this->cache->lock( $key ) ); $this->assertTrue( $this->cache->unlock( $key ) ); $this->assertFalse( $this->cache->unlock( $key ) ); $this->assertTrue( $this->cache->lock( $key, 5, 5, 'rclass' ) ); $this->assertTrue( $this->cache->lock( $key, 5, 5, 'rclass' ) ); $this->assertTrue( $this->cache->unlock( $key ) ); $this->assertTrue( $this->cache->unlock( $key ) ); } public function testErrorHandling() { $key = $this->cache->makeKey( $this->testKey() ); $wrapper = TestingAccessWrapper::newFromObject( $this->cache ); $wp = $this->cache->watchErrors(); $this->cache->get( $key ); $this->assertSame( StorageAwareness::ERR_NONE, $this->cache->getLastError() ); $this->assertSame( StorageAwareness::ERR_NONE, $this->cache->getLastError( $wp ) ); $wrapper->setLastError( StorageAwareness::ERR_UNREACHABLE ); $this->assertSame( StorageAwareness::ERR_UNREACHABLE, $this->cache->getLastError() ); $this->assertSame( StorageAwareness::ERR_UNREACHABLE, $this->cache->getLastError( $wp ) ); $wp = $this->cache->watchErrors(); $wrapper->setLastError( StorageAwareness::ERR_UNEXPECTED ); $wp2 = $this->cache->watchErrors(); $this->assertSame( StorageAwareness::ERR_UNEXPECTED, $this->cache->getLastError() ); $this->assertSame( StorageAwareness::ERR_UNEXPECTED, $this->cache->getLastError( $wp ) ); $this->assertSame( StorageAwareness::ERR_NONE, $this->cache->getLastError( $wp2 ) ); $this->cache->get( $key ); $this->assertSame( StorageAwareness::ERR_UNEXPECTED, $this->cache->getLastError() ); $this->assertSame( StorageAwareness::ERR_UNEXPECTED, $this->cache->getLastError( $wp ) ); $this->assertSame( StorageAwareness::ERR_NONE, $this->cache->getLastError( $wp2 ) ); } }