Files
sysPass/tests/SP/IntegrationTestCase.php
Rubén D f7aa81a176 test(IT): Test Ldap Check use cases
Signed-off-by: Rubén D <nuxsmin@syspass.org>
2024-10-16 21:40:32 +02:00

424 lines
15 KiB
PHP

<?php
/**
* sysPass
*
* @author nuxsmin
* @link https://syspass.org
* @copyright 2012-2024, Rubén Domínguez nuxsmin@$syspass.org
*
* This file is part of sysPass.
*
* sysPass is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* sysPass 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 General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with sysPass. If not, see <http://www.gnu.org/licenses/>.
*/
declare(strict_types=1);
namespace SP\Tests;
use Closure;
use DI\ContainerBuilder;
use Faker\Factory;
use Faker\Generator;
use Klein\Request;
use Klein\Response;
use PHPUnit\Framework\MockObject\Exception;
use PHPUnit\Framework\MockObject\Stub;
use PHPUnit\Framework\TestCase;
use Psr\Container\ContainerExceptionInterface;
use Psr\Container\ContainerInterface;
use Psr\Container\NotFoundExceptionInterface;
use ReflectionAttribute;
use ReflectionMethod;
use ReflectionObject;
use SP\Core\Bootstrap\Path;
use SP\Core\Bootstrap\PathsContext;
use SP\Core\Definitions\CoreDefinitions;
use SP\Core\Definitions\DomainDefinitions;
use SP\Core\UI\ThemeContext;
use SP\Domain\Account\Adapters\AccountPermission;
use SP\Domain\Account\Ports\AccountAclService;
use SP\Domain\Auth\Ports\LdapConnectionHandler;
use SP\Domain\Config\Ports\ConfigDataInterface;
use SP\Domain\Config\Ports\ConfigFileService;
use SP\Domain\Config\Ports\ConfigService;
use SP\Domain\Core\Acl\AclInterface;
use SP\Domain\Core\Bootstrap\BootstrapInterface;
use SP\Domain\Core\Bootstrap\ModuleInterface;
use SP\Domain\Core\Bootstrap\UriContextInterface;
use SP\Domain\Core\Context\Context;
use SP\Domain\Core\Context\SessionContext;
use SP\Domain\Core\Crypt\CryptInterface;
use SP\Domain\Core\Crypt\VaultInterface;
use SP\Domain\Core\Exceptions\SPException;
use SP\Domain\Core\UI\ThemeContextInterface;
use SP\Domain\Database\Ports\DatabaseInterface;
use SP\Domain\Database\Ports\DbStorageHandler;
use SP\Domain\Notification\Ports\MailService;
use SP\Domain\User\Dtos\UserDto;
use SP\Domain\User\Models\ProfileData;
use SP\Infrastructure\Database\QueryData;
use SP\Infrastructure\Database\QueryResult;
use SP\Infrastructure\File\ArchiveHandler;
use SP\Infrastructure\File\FileSystem;
use SP\Modules\Web\Bootstrap;
use SP\Tests\Generators\UserDataGenerator;
use SP\Tests\Generators\UserProfileDataGenerator;
use function DI\autowire;
use function DI\factory;
/**
* Class IntegrationTestCase
*/
abstract class IntegrationTestCase extends TestCase
{
protected static Generator $faker;
private static array $definitionsCache;
private static string $moduleFile;
protected readonly string $passwordSalt;
protected Closure|null $databaseQueryResolver = null;
protected array $definitions;
/**
* @var array<string, QueryResult> $databaseMapperResolvers
*/
private array $databaseMapperResolvers = [];
public static function setUpBeforeClass(): void
{
parent::setUpBeforeClass();
self::$faker = Factory::create();
self::$definitionsCache = array_merge(
DomainDefinitions::getDefinitions(),
CoreDefinitions::getDefinitions(REAL_APP_ROOT, 'web')
);
self::$moduleFile = FileSystem::buildPath(REAL_APP_ROOT, 'app', 'modules', 'web', 'module.php');
}
protected static function buildRequest(
string $method,
string $uri,
array $paramsGet = [],
array $paramsPost = [],
array $files = []
): Request {
$server = array_merge(
$_SERVER,
[
'REQUEST_METHOD' => strtoupper($method),
'REQUEST_URI' => $uri,
'HTTP_USER_AGENT' => self::$faker->userAgent(),
'HTTP_HOST' => 'localhost'
//'QUERY_STRING' => $query
]
);
return new Request(
array_merge($_GET, $paramsGet),
array_merge($_POST, $paramsPost),
$_COOKIE,
$server,
array_merge($_FILES, $files),
null
);
}
/**
* @param ContainerInterface $container
* @return void
* @throws ContainerExceptionInterface
* @throws NotFoundExceptionInterface
*/
protected static function runApp(ContainerInterface $container): void
{
Bootstrap::run($container->get(BootstrapInterface::class), $container->get(ModuleInterface::class));
}
/**
* @throws Exception
* @throws \Exception
*/
protected function buildContainer(Request $request, array $definitionsOverride = []): ContainerInterface
{
$definitionsOverride[Request::class] = $request;
$testDefinitions = array_merge(
FileSystem::require(self::$moduleFile),
$this->getMockedDefinitions(),
$definitionsOverride
);
$this->processAttributes(
$this->getClassAttributesMap($testDefinitions),
$this->getMethodAttributesMap($testDefinitions)
);
$containerBuilder = new ContainerBuilder();
$containerBuilder->addDefinitions(self::$definitionsCache, $testDefinitions);
return $containerBuilder->build();
}
/**
* @return array
* @throws Exception
* @throws SPException
*/
private function getMockedDefinitions(): array
{
$configData = self::createConfiguredStub(ConfigDataInterface::class, $this->getConfigData());
$configFileService = $this->createStub(ConfigFileService::class);
$configFileService->method('getConfigData')->willReturn($configData);
$databaseUtil = self::createStub(\SP\Infrastructure\Database\DatabaseUtil::class);
$databaseUtil->method('checkDatabaseConnection')->willReturn(true);
$databaseUtil->method('checkDatabaseTables')->willReturn(true);
$database = self::createStub(DatabaseInterface::class);
$database->method('runQuery')->willReturnCallback($this->getDatabaseReturn());
$database->method('beginTransaction')->willReturn(true);
$database->method('endTransaction')->willReturn(true);
$database->method('rollbackTransaction')->willReturn(true);
$acl = $this->createMock(AclInterface::class);
$acl->method('checkUserAccess')->willReturn(true);
$acl->method('getRouteFor')->willReturnCallback(static fn(int $actionId) => (string)$actionId);
$accountAcl = self::createStub(AccountAclService::class);
$accountAcl->method('getAcl')
->willReturnCallback(static function (int $actionId) {
$accountPermission = new AccountPermission($actionId);
$accountPermission->setCompiledAccountAccess(true);
$accountPermission->setCompiledShowAccess(true);
$accountPermission->setResultView(true);
$accountPermission->setResultEdit(true);
return $accountPermission;
});
return [
ConfigFileService::class => $configFileService,
LdapConnectionHandler::class => self::createStub(LdapConnectionHandler::class),
'backup.dbArchiveHandler' => self::createStub(ArchiveHandler::class),
'backup.appArchiveHandler' => self::createStub(ArchiveHandler::class),
\SP\Infrastructure\Database\DatabaseUtil::class => $databaseUtil,
Context::class => $this->getContext(),
MailService::class => self::createStub(MailService::class),
DbStorageHandler::class => self::createStub(DbStorageHandler::class),
DatabaseInterface::class => $database,
ThemeContextInterface::class => autowire(ThemeContext::class)
->constructorParameter(
'basePath',
factory(static fn(PathsContext $p) => $p[Path::VIEW])
)
->constructorParameter('baseUri', factory([UriContextInterface::class, 'getWebRoot']))
->constructorParameter('module', 'web')
->constructorParameter('name', 'material-blue'),
AclInterface::class => $acl,
AccountAclService::class => $accountAcl,
];
}
protected function getConfigData(): array
{
return [
'isInstalled' => true,
'isMaintenance' => false,
'getDbName' => self::$faker->colorName(),
'getPasswordSalt' => $this->passwordSalt
];
}
protected function getDatabaseReturn(): callable
{
return function (QueryData $queryData): QueryResult {
if ($this->databaseQueryResolver) {
return $this->databaseQueryResolver->call($this, $queryData);
}
$mapClassName = $queryData->getMapClassName();
if (isset($this->databaseMapperResolvers[$mapClassName])) {
return $this->databaseMapperResolvers[$mapClassName];
}
return new QueryResult([], 1, 100);
};
}
/**
* @throws Exception
* @throws SPException
*/
protected function getContext(): SessionContext|Stub
{
$context = self::createStub(SessionContext::class);
$context->method('isLoggedIn')->willReturn(true);
$context->method('getAuthCompleted')->willReturn(true);
$context->method('getUserData')->willReturn($this->getUserDataDto());
$context->method('getUserProfile')->willReturn($this->getUserProfile());
return $context;
}
/**
* @return UserDto
* @throws SPException
*/
protected function getUserDataDto(): UserDto
{
$properties = [
'isAdminApp' => false,
'isAdminAcc' => false,
'userGroupName' => self::$faker->colorName()
];
return UserDto::fromModel(UserDataGenerator::factory()->buildUserData())
->mutate($properties);
}
/**
* @return ProfileData
*/
protected function getUserProfile(): ProfileData
{
return UserProfileDataGenerator::factory()->buildProfileData();
}
private function processAttributes(array $classAtrributes, array $methodAttributes): void
{
$object = new ReflectionObject($this);
if (!empty($classAtrributes)) {
$this->invokeAttributes($object, $classAtrributes);
}
if (!empty($methodAttributes)) {
$methods = array_filter(
$object->getMethods(),
fn(ReflectionMethod $method) => $method->name === $this->name()
);
foreach ($methods as $method) {
$this->invokeAttributes($method, $methodAttributes);
}
}
}
private function invokeAttributes(ReflectionObject|ReflectionMethod $reflection, array $attributeMap): void
{
foreach ($reflection->getAttributes() as $attribute) {
$callable = $attributeMap[$attribute->getName()] ?? null;
if (is_callable($callable)) {
$callable($attribute);
}
}
}
private function getClassAttributesMap(array &$definitions): array
{
return [
InjectVault::class => function () use (&$definitions): void {
$vault = self::createStub(VaultInterface::class);
$vault->method('getData')->willReturn('some_data');
/** @var Context|Stub $context */
$context = $definitions[Context::class];
$context->method('getVault')->willReturn($vault);
}
];
}
/**
* @param array $definitions
* @return Closure[]
*/
private function getMethodAttributesMap(array &$definitions): array
{
return [
BodyChecker::class =>
function (ReflectionAttribute $attribute) use (&$definitions): void {
/** @var ReflectionAttribute<BodyChecker> $attribute */
$bodyChecker = (new ReflectionMethod($this, $attribute->newInstance()->target))
->getClosure($this);
$response = $this->getMockBuilder(Response::class)->onlyMethods(['body'])->getMock();
$response
->method('body')
->with(
self::callback(static function (string $output) use ($bodyChecker) {
self::assertNotEmpty($output);
$bodyChecker($output);
return true;
})
);
$definitions[Response::class] = $response;
},
InjectCrypt::class => function (ReflectionAttribute $attribute) use (&$definitions): void {
/** @var ReflectionAttribute<InjectCrypt> $attribute */
$value = $attribute->newInstance()->returnValue;
$crypt = $this->createStub(CryptInterface::class);
$crypt->method('decrypt')->willReturn($value);
$crypt->method('encrypt')->willReturn($value);
$definitions[CryptInterface::class] = $crypt;
},
InjectConfigParam::class => function (ReflectionAttribute $attribute) use (&$definitions): void {
/** @var ReflectionAttribute<InjectConfigParam> $attribute */
$parameterValueMap = $attribute->newInstance()->parameterValueMap;
$configService = self::createStub(ConfigService::class);
if (!empty($parameterValueMap)) {
$configService
->method('getByParam')
->willReturnCallback(
static function (string $parameter) use ($parameterValueMap): string {
if (array_key_exists($parameter, $parameterValueMap)) {
return $parameterValueMap[$parameter];
}
return self::$faker->colorName();
}
);
} else {
$configService
->method('getByParam')
->willReturnArgument(0);
}
$definitions[ConfigService::class] = $configService;
}
];
}
final protected function addDatabaseMapperResolver(string $className, QueryResult $queryResult): void
{
$this->databaseMapperResolvers[$className] = $queryResult;
}
protected function setUp(): void
{
parent::setUp();
$this->passwordSalt = self::$faker->sha1();
}
}