Files
sysPass/tests/SP/IntegrationTestCase.php
2024-08-21 14:32:51 +02:00

310 lines
11 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 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 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\LdapConnectionInterface;
use SP\Domain\Config\Ports\ConfigDataInterface;
use SP\Domain\Config\Ports\ConfigFileService;
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\Exceptions\InvalidClassException;
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\Domain\User\Models\User;
use SP\Domain\User\Models\UserPreferences;
use SP\Infrastructure\Database\QueryData;
use SP\Infrastructure\Database\QueryResult;
use SP\Infrastructure\File\ArchiveHandler;
use SP\Infrastructure\File\FileException;
use SP\Infrastructure\File\FileSystem;
use SP\Modules\Web\Bootstrap;
use SP\Mvc\View\OutputHandlerInterface;
use SP\Tests\Generators\UserDataGenerator;
use SP\Tests\Generators\UserProfileDataGenerator;
use SP\Tests\Stubs\OutputHandlerStub;
use function DI\autowire;
use function DI\factory;
/**
* Class IntegrationTestCase
*/
abstract class IntegrationTestCase extends TestCase
{
protected static Generator $faker;
protected readonly string $passwordSalt;
protected Closure|null $databaseQueryResolver = null;
/**
* @var array<string, QueryResult> $databaseMapperResolvers
*/
private array $databaseMapperResolvers = [];
public static function setUpBeforeClass(): void
{
parent::setUpBeforeClass();
self::$faker = Factory::create();
}
/**
* @throws Exception
* @throws \Exception
*/
protected function buildContainer(array $definitions = [], Request $request = null): ContainerInterface
{
$configData = $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 = self::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;
});
$mockedDefinitions = [
ConfigFileService::class => $configFileService,
LdapConnectionInterface::class => self::createStub(LdapConnectionInterface::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
];
if ($request) {
$definitions += [Request::class => $request];
}
$containerBuilder = new ContainerBuilder();
$containerBuilder->addDefinitions(
DomainDefinitions::getDefinitions(),
CoreDefinitions::getDefinitions(REAL_APP_ROOT, 'web'),
$definitions,
$mockedDefinitions
);
return $containerBuilder->build();
}
/**
* @throws Exception
*/
protected function getConfigData(): ConfigDataInterface|Stub
{
$configData = self::createStub(ConfigDataInterface::class);
$configData->method('isInstalled')->willReturn(true);
$configData->method('isMaintenance')->willReturn(false);
$configData->method('getDbName')->willReturn(self::$faker->colorName());
$configData->method('getPasswordSalt')->willReturn($this->passwordSalt);
return $configData;
}
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
{
$user = UserDataGenerator::factory()->buildUserData()->mutate(['isAdminApp' => false, 'isAdminAcc' => false]);
return UserDto::fromModel($user, User::class)
->mutate(['preferences', $user->hydrate(UserPreferences::class)]);
}
/**
* @return ProfileData
*/
protected function getUserProfile(): ProfileData
{
return UserProfileDataGenerator::factory()->buildProfileData();
}
final protected function addDatabaseMapperResolver(string $className, QueryResult $queryResult): void
{
$this->databaseMapperResolvers[$className] = $queryResult;
}
protected 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 callable $outputChecker
* @return OutputHandlerInterface
*/
protected function setupOutputHandler(callable $outputChecker): OutputHandlerInterface
{
return new OutputHandlerStub($outputChecker(...)->bindTo($this));
}
/**
* @param ContainerInterface $container
* @return void
* @throws ContainerExceptionInterface
* @throws NotFoundExceptionInterface
*/
protected function runApp(ContainerInterface $container): void
{
Bootstrap::run($container->get(BootstrapInterface::class), $container->get(ModuleInterface::class));
}
/**
* @throws FileException
* @throws InvalidClassException
*/
protected function getModuleDefinitions(): array
{
return FileSystem::require(FileSystem::buildPath(REAL_APP_ROOT, 'app', 'modules', 'web', 'module.php'));
}
protected function setUp(): void
{
parent::setUp();
$this->passwordSalt = self::$faker->sha1();
}
}