friendica-github/tests/src/Core/PConfig/PConfigTest.php

497 lines
13 KiB
PHP
Raw Normal View History

<?php
2020-02-09 14:45:36 +00:00
/**
* @copyright Copyright (C) 2010-2024, the Friendica project
2020-02-09 14:45:36 +00:00
*
* @license GNU AGPL version 3 or any later version
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU Affero General Public License as
* published by the Free Software Foundation, either version 3 of the
* License, or (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU Affero General Public License for more details.
*
* You should have received a copy of the GNU Affero General Public License
* along with this program. If not, see <https://www.gnu.org/licenses/>.
*
*/
namespace Friendica\Test\src\Core\PConfig;
2021-12-10 20:34:19 +00:00
use DMS\PHPUnitExtensions\ArraySubset\ArraySubsetAsserts;
2021-10-26 19:44:29 +00:00
use Friendica\Core\PConfig\Type\AbstractPConfigValues;
use Friendica\Core\PConfig\Repository\PConfig as PConfigModel;
2021-12-10 20:34:19 +00:00
use Friendica\Core\PConfig\ValueObject\Cache;
use Friendica\Test\MockedTest;
2019-07-15 18:13:53 +00:00
use Mockery;
use Mockery\MockInterface;
abstract class PConfigTest extends MockedTest
{
2021-12-10 20:34:19 +00:00
use ArraySubsetAsserts;
2019-07-15 18:13:53 +00:00
/** @var PConfigModel|MockInterface */
protected $configModel;
2021-12-10 20:34:19 +00:00
/** @var Cache */
2019-07-15 18:13:53 +00:00
protected $configCache;
2021-10-26 19:44:29 +00:00
/** @var AbstractPConfigValues */
2019-07-15 18:13:53 +00:00
protected $testedConfig;
/**
* Assert a config tree
*
* @param int $uid The uid to assert
* @param string $cat The category to assert
* @param array $data The result data array
*/
protected function assertConfig(int $uid, string $cat, array $data)
{
$result = $this->testedConfig->getCache()->getAll();
self::assertNotEmpty($result);
self::assertArrayHasKey($uid, $result);
self::assertArrayHasKey($cat, $result[$uid]);
self::assertArraySubset($data, $result[$uid][$cat]);
2019-07-15 18:13:53 +00:00
}
protected function setUp(): void
2019-07-15 18:13:53 +00:00
{
parent::setUp();
// Create the config model
$this->configModel = Mockery::mock(PConfigModel::class);
2021-12-10 20:34:19 +00:00
$this->configCache = new Cache();
2019-07-15 18:13:53 +00:00
}
/**
2021-10-26 19:44:29 +00:00
* @return \Friendica\Core\PConfig\Type\AbstractPConfigValues
2019-07-15 18:13:53 +00:00
*/
abstract public function getInstance();
2019-07-15 18:13:53 +00:00
public function dataTests()
{
return [
2019-07-15 18:13:53 +00:00
'string' => ['uid' => 1, 'data' => 'it'],
'boolTrue' => ['uid' => 2, 'data' => true],
'boolFalse' => ['uid' => 3, 'data' => false],
'integer' => ['uid' => 4, 'data' => 235],
'decimal' => ['uid' => 5, 'data' => 2.456],
'array' => ['uid' => 6, 'data' => ['1', 2, '3', true, false]],
'boolIntTrue' => ['uid' => 7, 'data' => 1],
'boolIntFalse' => ['uid' => 8, 'data' => 0],
];
}
public function dataConfigLoad()
{
$data = [
'system' => [
'key1' => 'value1',
'key2' => 'value2',
'key3' => 'value3',
],
'config' => [
'key1' => 'value1a',
'key4' => 'value4',
],
'other' => [
'key5' => 'value5',
'key6' => 'value6',
],
];
2019-07-15 18:13:53 +00:00
return [
'system' => [
'uid' => 1,
'data' => $data,
'possibleCats' => [
'system',
'config',
'other'
],
'load' => [
'system',
],
],
'other' => [
'uid' => 2,
'data' => $data,
'possibleCats' => [
'system',
'config',
'other'
],
'load' => [
'other',
],
],
'config' => [
'uid' => 3,
'data' => $data,
'possibleCats' => [
'system',
'config',
'other'
],
'load' => [
'config',
],
],
'all' => [
'uid' => 4,
'data' => $data,
'possibleCats' => [
'system',
'config',
'other'
],
'load' => [
'system',
'config',
'other'
],
],
];
}
/**
* Test the configuration initialization
*/
public function testSetUp()
2019-07-15 18:13:53 +00:00
{
$this->testedConfig = $this->getInstance();
2021-12-10 20:34:19 +00:00
self::assertInstanceOf(Cache::class, $this->testedConfig->getCache());
2019-07-15 18:13:53 +00:00
self::assertEmpty($this->testedConfig->getCache()->getAll());
}
/**
* Test the configuration load() method
*/
2019-07-15 18:13:53 +00:00
public function testLoad(int $uid, array $data, array $possibleCats, array $load)
{
2019-07-15 18:13:53 +00:00
$this->testedConfig = $this->getInstance();
2021-12-10 20:34:19 +00:00
self::assertInstanceOf(Cache::class, $this->testedConfig->getCache());
2019-07-15 18:13:53 +00:00
foreach ($load as $loadedCats) {
$this->testedConfig->load($uid, $loadedCats);
}
2019-07-15 18:13:53 +00:00
// Assert at least loaded cats are loaded
foreach ($load as $loadedCats) {
self::assertConfig($uid, $loadedCats, $data[$loadedCats]);
2019-07-15 18:13:53 +00:00
}
}
public function dataDoubleLoad()
{
return [
'config' => [
'uid' => 1,
'data1' => [
'config' => [
'key1' => 'value1',
'key2' => 'value2',
],
],
'data2' => [
'config' => [
'key1' => 'overwritten!',
'key3' => 'value3',
],
],
'expect' => [
'config' => [
// load should overwrite values everytime!
'key1' => 'overwritten!',
'key2' => 'value2',
'key3' => 'value3',
],
],
],
'other' => [
'uid' => 1,
'data1' => [
'config' => [
'key12' => 'data4',
'key45' => 7,
],
'other' => [
'key1' => 'value1',
'key2' => 'value2',
],
],
'data2' => [
'other' => [
'key1' => 'overwritten!',
'key3' => 'value3',
],
'config' => [
'key45' => 45,
'key52' => true,
]
],
'expect' => [
'other' => [
// load should overwrite values everytime!
'key1' => 'overwritten!',
'key2' => 'value2',
'key3' => 'value3',
],
'config' => [
'key12' => 'data4',
'key45' => 45,
'key52' => true,
],
],
],
];
}
/**
* Test the configuration load() method with overwrite
*/
2019-07-15 18:13:53 +00:00
public function testCacheLoadDouble(int $uid, array $data1, array $data2, array $expect)
{
2019-07-15 18:13:53 +00:00
$this->testedConfig = $this->getInstance();
2021-12-10 20:34:19 +00:00
self::assertInstanceOf(Cache::class, $this->testedConfig->getCache());
2019-07-15 18:13:53 +00:00
foreach ($data1 as $cat => $data) {
$this->testedConfig->load($uid, $cat);
}
2019-07-15 18:13:53 +00:00
// Assert at least loaded cats are loaded
foreach ($data1 as $cat => $data) {
self::assertConfig($uid, $cat, $data);
2019-07-15 18:13:53 +00:00
}
2019-07-15 18:13:53 +00:00
foreach ($data2 as $cat => $data) {
$this->testedConfig->load($uid, $cat);
}
}
/**
* Test the configuration get() and set() methods without adapter
2019-07-15 18:13:53 +00:00
*
* @dataProvider dataTests
*/
2019-07-15 18:13:53 +00:00
public function testSetGetWithoutDB(int $uid, $data)
{
2019-07-15 18:13:53 +00:00
$this->testedConfig = $this->getInstance();
2021-12-10 20:34:19 +00:00
self::assertInstanceOf(Cache::class, $this->testedConfig->getCache());
self::assertTrue($this->testedConfig->set($uid, 'test', 'it', $data));
self::assertEquals($data, $this->testedConfig->get($uid, 'test', 'it'));
self::assertEquals($data, $this->testedConfig->getCache()->get($uid, 'test', 'it'));
}
/**
2019-07-15 18:13:53 +00:00
* Test the configuration get() and set() methods with a model/db
*
* @dataProvider dataTests
*/
2019-07-15 18:13:53 +00:00
public function testSetGetWithDB(int $uid, $data)
{
2019-07-15 18:13:53 +00:00
$this->configModel->shouldReceive('set')
->with($uid, 'test', 'it', $data)
->andReturn(true)
->once();
2019-07-15 18:13:53 +00:00
$this->testedConfig = $this->getInstance();
2021-12-10 20:34:19 +00:00
self::assertInstanceOf(Cache::class, $this->testedConfig->getCache());
self::assertTrue($this->testedConfig->set($uid, 'test', 'it', $data));
self::assertEquals($data, $this->testedConfig->get($uid, 'test', 'it'));
self::assertEquals($data, $this->testedConfig->getCache()->get($uid, 'test', 'it'));
}
/**
* Test the configuration get() method with wrong value and no db
*/
public function testGetWrongWithoutDB()
{
2019-07-15 18:13:53 +00:00
$this->testedConfig = $this->getInstance();
2021-12-10 20:34:19 +00:00
self::assertInstanceOf(Cache::class, $this->testedConfig->getCache());
// without refresh
self::assertNull($this->testedConfig->get(0, 'test', 'it'));
2019-07-15 18:13:53 +00:00
/// beware that the cache returns '!<unset>!' and not null for a nonexistent value
self::assertNull($this->testedConfig->getCache()->get(0, 'test', 'it'));
// with default value
self::assertEquals('default', $this->testedConfig->get(0, 'test', 'it', 'default'));
// with default value and refresh
self::assertEquals('default', $this->testedConfig->get(0, 'test', 'it', 'default', true));
}
/**
* Test the configuration get() method with refresh
2019-07-15 18:13:53 +00:00
*
* @dataProvider dataTests
*/
2019-07-15 18:13:53 +00:00
public function testGetWithRefresh(int $uid, $data)
{
2019-07-15 18:13:53 +00:00
$this->configCache->load($uid, ['test' => ['it' => 'now']]);
2019-07-15 18:13:53 +00:00
$this->testedConfig = $this->getInstance();
2021-12-10 20:34:19 +00:00
self::assertInstanceOf(Cache::class, $this->testedConfig->getCache());
// without refresh
self::assertEquals('now', $this->testedConfig->get($uid, 'test', 'it'));
self::assertEquals('now', $this->testedConfig->getCache()->get($uid, 'test', 'it'));
2019-07-15 18:13:53 +00:00
// with refresh
self::assertEquals($data, $this->testedConfig->get($uid, 'test', 'it', null, true));
self::assertEquals($data, $this->testedConfig->getCache()->get($uid, 'test', 'it'));
// without refresh and wrong value and default
self::assertEquals('default', $this->testedConfig->get($uid, 'test', 'not', 'default'));
self::assertNull($this->testedConfig->getCache()->get($uid, 'test', 'not'));
}
/**
2019-07-15 18:13:53 +00:00
* Test the configuration delete() method without a model/db
*
* @dataProvider dataTests
*/
2019-07-15 18:13:53 +00:00
public function testDeleteWithoutDB(int $uid, $data)
{
2019-07-15 18:13:53 +00:00
$this->configCache->load($uid, ['test' => ['it' => $data]]);
2019-07-15 18:13:53 +00:00
$this->testedConfig = $this->getInstance();
2021-12-10 20:34:19 +00:00
self::assertInstanceOf(Cache::class, $this->testedConfig->getCache());
self::assertEquals($data, $this->testedConfig->get($uid, 'test', 'it'));
self::assertEquals($data, $this->testedConfig->getCache()->get($uid, 'test', 'it'));
self::assertTrue($this->testedConfig->delete($uid, 'test', 'it'));
self::assertNull($this->testedConfig->get($uid, 'test', 'it'));
self::assertNull($this->testedConfig->getCache()->get($uid, 'test', 'it'));
self::assertEmpty($this->testedConfig->getCache()->getAll());
}
/**
2019-07-15 18:13:53 +00:00
* Test the configuration delete() method with a model/db
*/
2019-07-15 18:13:53 +00:00
public function testDeleteWithDB()
{
$uid = 42;
$this->configCache->load($uid, ['test' => ['it' => 'now', 'quarter' => 'true']]);
2019-07-15 18:13:53 +00:00
$this->configModel->shouldReceive('delete')
->with($uid, 'test', 'it')
2019-07-15 18:13:53 +00:00
->andReturn(false)
->once();
$this->configModel->shouldReceive('delete')
->with($uid, 'test', 'second')
2019-07-15 18:13:53 +00:00
->andReturn(true)
->once();
$this->configModel->shouldReceive('delete')
->with($uid, 'test', 'third')
2019-07-15 18:13:53 +00:00
->andReturn(false)
->once();
$this->configModel->shouldReceive('delete')
->with($uid, 'test', 'quarter')
2019-07-15 18:13:53 +00:00
->andReturn(true)
->once();
$this->testedConfig = $this->getInstance();
2021-12-10 20:34:19 +00:00
self::assertInstanceOf(Cache::class, $this->testedConfig->getCache());
2019-07-15 18:13:53 +00:00
// directly set the value to the cache
$this->testedConfig->getCache()->set($uid, 'test', 'it', 'now');
2019-07-15 18:13:53 +00:00
self::assertEquals('now', $this->testedConfig->get($uid, 'test', 'it'));
self::assertEquals('now', $this->testedConfig->getCache()->get($uid, 'test', 'it'));
2019-07-15 18:13:53 +00:00
// delete from cache only
self::assertTrue($this->testedConfig->delete($uid, 'test', 'it'));
2019-07-15 18:13:53 +00:00
// delete from db only
self::assertTrue($this->testedConfig->delete($uid, 'test', 'second'));
2019-07-15 18:13:53 +00:00
// no delete
self::assertFalse($this->testedConfig->delete($uid, 'test', 'third'));
2019-07-15 18:13:53 +00:00
// delete both
self::assertTrue($this->testedConfig->delete($uid, 'test', 'quarter'));
self::assertEmpty($this->testedConfig->getCache()->getAll());
2019-07-15 18:13:53 +00:00
}
2019-07-15 18:13:53 +00:00
public function dataMultiUid()
{
return [
'normal' => [
'data1' => [
'uid' => 1,
'data' => [
'cat1' => [
'key1' => 'value1',
],
'cat2' => [
'key2' => 'value2',
]
],
],
'data2' => [
'uid' => 2,
'data' => [
'cat1' => [
'key1' => 'value1a',
],
'cat2' => [
'key2' => 'value2',
],
],
],
],
];
}
/**
2019-07-15 18:13:53 +00:00
* Test if multiple uids for caching are usable without errors
* @dataProvider dataMultiUid
*/
2019-07-15 18:13:53 +00:00
public function testMultipleUidsWithCache(array $data1, array $data2)
{
2019-07-15 18:13:53 +00:00
$this->configCache->load($data1['uid'], $data1['data']);
$this->configCache->load($data2['uid'], $data2['data']);
2019-07-15 18:13:53 +00:00
$this->testedConfig = $this->getInstance();
2021-12-10 20:34:19 +00:00
self::assertInstanceOf(Cache::class, $this->testedConfig->getCache());
self::assertConfig($data1['uid'], 'cat1', $data1['data']['cat1']);
self::assertConfig($data1['uid'], 'cat2', $data1['data']['cat2']);
self::assertConfig($data2['uid'], 'cat1', $data2['data']['cat1']);
self::assertConfig($data2['uid'], 'cat2', $data2['data']['cat2']);
}
2019-07-15 19:11:38 +00:00
/**
* Test when using an invalid UID
* @todo check it the clean way before using the config class
*/
public function testInvalidUid()
{
// bad UID!
$uid = 0;
2019-07-15 19:11:38 +00:00
$this->testedConfig = $this->getInstance();
self::assertNull($this->testedConfig->get($uid, 'cat1', 'cat2'));
self::assertEquals('fallback!', $this->testedConfig->get($uid, 'cat1', 'cat2', 'fallback!'));
2019-07-15 19:11:38 +00:00
self::assertFalse($this->testedConfig->set($uid, 'cat1', 'key1', 'doesn\'t matter!'));
self::assertFalse($this->testedConfig->delete($uid, 'cat1', 'key1'));
2019-07-15 19:11:38 +00:00
}
}