wiki.techinc.nl/tests/phpunit/includes/libs/objectcache/BagOStuffTestBase.php
Aaron Schulz a1cb96ee19 objectcache: remove deprecate BagOStuff::incr() method
Change-Id: I107c04e05585c975dc37ce402746a4671f2f43b5
2023-03-10 13:31:34 -08:00

531 lines
17 KiB
PHP

<?php
use Wikimedia\LightweightObjectStore\StorageAwareness;
use Wikimedia\ScopedCallback;
use Wikimedia\TestingAccessWrapper;
/**
* @author Matthias Mullie <mmullie@wikimedia.org>
* @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 ) );
}
}