2020-10-11 19:59:46 +01:00
|
|
|
<?php
|
|
|
|
|
|
|
|
namespace Pterodactyl\Tests\Integration\Services\Databases;
|
|
|
|
|
|
|
|
use Mockery;
|
2022-10-14 17:59:20 +01:00
|
|
|
use Mockery\MockInterface;
|
2020-10-11 19:59:46 +01:00
|
|
|
use BadMethodCallException;
|
|
|
|
use InvalidArgumentException;
|
|
|
|
use Pterodactyl\Models\Database;
|
|
|
|
use Pterodactyl\Models\DatabaseHost;
|
|
|
|
use Pterodactyl\Tests\Integration\IntegrationTestCase;
|
|
|
|
use Pterodactyl\Repositories\Eloquent\DatabaseRepository;
|
|
|
|
use Pterodactyl\Services\Databases\DatabaseManagementService;
|
|
|
|
use Pterodactyl\Exceptions\Repository\DuplicateDatabaseNameException;
|
|
|
|
use Pterodactyl\Exceptions\Service\Database\TooManyDatabasesException;
|
|
|
|
use Pterodactyl\Exceptions\Service\Database\DatabaseClientFeatureNotEnabledException;
|
|
|
|
|
|
|
|
class DatabaseManagementServiceTest extends IntegrationTestCase
|
|
|
|
{
|
2022-10-14 17:59:20 +01:00
|
|
|
private MockInterface $repository;
|
2020-10-11 19:59:46 +01:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Setup tests.
|
|
|
|
*/
|
|
|
|
public function setUp(): void
|
|
|
|
{
|
|
|
|
parent::setUp();
|
|
|
|
|
|
|
|
config()->set('pterodactyl.client_features.databases.enabled', true);
|
|
|
|
|
2022-10-14 17:59:20 +01:00
|
|
|
$this->repository = $this->mock(DatabaseRepository::class);
|
2020-10-11 19:59:46 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Test that the name generated by the unique name function is what we expect.
|
|
|
|
*/
|
|
|
|
public function testUniqueDatabaseNameIsGeneratedCorrectly()
|
|
|
|
{
|
|
|
|
$this->assertSame('s1_example', DatabaseManagementService::generateUniqueDatabaseName('example', 1));
|
|
|
|
$this->assertSame('s123_something_else', DatabaseManagementService::generateUniqueDatabaseName('something_else', 123));
|
|
|
|
$this->assertSame('s123_' . str_repeat('a', 43), DatabaseManagementService::generateUniqueDatabaseName(str_repeat('a', 100), 123));
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Test that disabling the client database feature flag prevents the creation of databases.
|
|
|
|
*/
|
|
|
|
public function testExceptionIsThrownIfClientDatabasesAreNotEnabled()
|
|
|
|
{
|
|
|
|
config()->set('pterodactyl.client_features.databases.enabled', false);
|
|
|
|
|
|
|
|
$this->expectException(DatabaseClientFeatureNotEnabledException::class);
|
|
|
|
|
|
|
|
$server = $this->createServerModel();
|
|
|
|
$this->getService()->create($server, []);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Test that a server at its database limit cannot have an additional one created if
|
|
|
|
* the $validateDatabaseLimit flag is not set to false.
|
|
|
|
*/
|
|
|
|
public function testDatabaseCannotBeCreatedIfServerHasReachedLimit()
|
|
|
|
{
|
|
|
|
$server = $this->createServerModel(['database_limit' => 2]);
|
2021-01-23 20:09:16 +00:00
|
|
|
$host = DatabaseHost::factory()->create(['node_id' => $server->node_id]);
|
2020-10-11 19:59:46 +01:00
|
|
|
|
2021-01-23 20:09:16 +00:00
|
|
|
Database::factory()->times(2)->create(['server_id' => $server->id, 'database_host_id' => $host->id]);
|
2020-10-11 19:59:46 +01:00
|
|
|
|
|
|
|
$this->expectException(TooManyDatabasesException::class);
|
|
|
|
|
|
|
|
$this->getService()->create($server, []);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Test that a missing or invalid database name format causes an exception to be thrown.
|
|
|
|
*
|
|
|
|
* @dataProvider invalidDataDataProvider
|
|
|
|
*/
|
2022-10-14 17:59:20 +01:00
|
|
|
public function testEmptyDatabaseNameOrInvalidNameTriggersAnException(array $data)
|
2020-10-11 19:59:46 +01:00
|
|
|
{
|
|
|
|
$server = $this->createServerModel();
|
|
|
|
|
|
|
|
$this->expectException(InvalidArgumentException::class);
|
|
|
|
$this->expectExceptionMessage('The database name passed to DatabaseManagementService::handle MUST be prefixed with "s{server_id}_".');
|
|
|
|
|
|
|
|
$this->getService()->create($server, $data);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Test that creating a server database with an identical name triggers an exception.
|
|
|
|
*/
|
|
|
|
public function testCreatingDatabaseWithIdenticalNameTriggersAnException()
|
|
|
|
{
|
|
|
|
$server = $this->createServerModel();
|
|
|
|
$name = DatabaseManagementService::generateUniqueDatabaseName('soemthing', $server->id);
|
|
|
|
|
2021-01-23 20:09:16 +00:00
|
|
|
$host = DatabaseHost::factory()->create(['node_id' => $server->node_id]);
|
|
|
|
$host2 = DatabaseHost::factory()->create(['node_id' => $server->node_id]);
|
|
|
|
Database::factory()->create([
|
2020-10-11 19:59:46 +01:00
|
|
|
'database' => $name,
|
|
|
|
'database_host_id' => $host->id,
|
|
|
|
'server_id' => $server->id,
|
|
|
|
]);
|
|
|
|
|
|
|
|
$this->expectException(DuplicateDatabaseNameException::class);
|
|
|
|
$this->expectExceptionMessage('A database with that name already exists for this server.');
|
|
|
|
|
|
|
|
// Try to create a database with the same name as a database on a different host. We expect
|
|
|
|
// this to fail since we don't account for the specific host when checking uniqueness.
|
|
|
|
$this->getService()->create($server, [
|
|
|
|
'database' => $name,
|
|
|
|
'database_host_id' => $host2->id,
|
|
|
|
]);
|
|
|
|
|
|
|
|
$this->assertDatabaseMissing('databases', ['server_id' => $server->id]);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Test that a server database can be created successfully.
|
|
|
|
*/
|
|
|
|
public function testServerDatabaseCanBeCreated()
|
|
|
|
{
|
|
|
|
$server = $this->createServerModel();
|
|
|
|
$name = DatabaseManagementService::generateUniqueDatabaseName('soemthing', $server->id);
|
|
|
|
|
2021-01-23 20:09:16 +00:00
|
|
|
$host = DatabaseHost::factory()->create(['node_id' => $server->node_id]);
|
2020-10-11 19:59:46 +01:00
|
|
|
|
|
|
|
$this->repository->expects('createDatabase')->with($name);
|
|
|
|
|
|
|
|
$username = null;
|
|
|
|
$secondUsername = null;
|
|
|
|
$password = null;
|
|
|
|
|
|
|
|
// The value setting inside the closures if to avoid throwing an exception during the
|
|
|
|
// assertions that would get caught by the functions catcher and thus lead to the exception
|
|
|
|
// being swallowed incorrectly.
|
|
|
|
$this->repository->expects('createUser')->with(
|
|
|
|
Mockery::on(function ($value) use (&$username) {
|
|
|
|
$username = $value;
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}),
|
|
|
|
'%',
|
|
|
|
Mockery::on(function ($value) use (&$password) {
|
|
|
|
$password = $value;
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}),
|
|
|
|
null
|
|
|
|
);
|
|
|
|
|
|
|
|
$this->repository->expects('assignUserToDatabase')->with($name, Mockery::on(function ($value) use (&$secondUsername) {
|
|
|
|
$secondUsername = $value;
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}), '%');
|
|
|
|
|
|
|
|
$this->repository->expects('flush')->withNoArgs();
|
|
|
|
|
|
|
|
$response = $this->getService()->create($server, [
|
|
|
|
'remote' => '%',
|
|
|
|
'database' => $name,
|
|
|
|
'database_host_id' => $host->id,
|
|
|
|
]);
|
|
|
|
|
|
|
|
$this->assertInstanceOf(Database::class, $response);
|
|
|
|
$this->assertSame($response->server_id, $server->id);
|
2022-10-14 17:59:20 +01:00
|
|
|
$this->assertMatchesRegularExpression('/^(u\d+_)(\w){10}$/', $username);
|
2020-10-11 19:59:46 +01:00
|
|
|
$this->assertSame($username, $secondUsername);
|
|
|
|
$this->assertSame(24, strlen($password));
|
|
|
|
|
|
|
|
$this->assertDatabaseHas('databases', ['server_id' => $server->id, 'id' => $response->id]);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2022-10-14 17:59:20 +01:00
|
|
|
* Test that an exception encountered while creating the database leads to the cleanup code
|
|
|
|
* being called and any exceptions encountered while cleaning up go unreported.
|
2020-10-11 19:59:46 +01:00
|
|
|
*/
|
|
|
|
public function testExceptionEncounteredWhileCreatingDatabaseAttemptsToCleanup()
|
|
|
|
{
|
|
|
|
$server = $this->createServerModel();
|
|
|
|
$name = DatabaseManagementService::generateUniqueDatabaseName('soemthing', $server->id);
|
|
|
|
|
2021-01-23 20:09:16 +00:00
|
|
|
$host = DatabaseHost::factory()->create(['node_id' => $server->node_id]);
|
2020-10-11 19:59:46 +01:00
|
|
|
|
2021-01-23 20:33:34 +00:00
|
|
|
$this->repository->expects('createDatabase')->with($name)->andThrows(new BadMethodCallException());
|
2020-10-11 19:59:46 +01:00
|
|
|
$this->repository->expects('dropDatabase')->with($name);
|
2021-01-23 20:33:34 +00:00
|
|
|
$this->repository->expects('dropUser')->withAnyArgs()->andThrows(new InvalidArgumentException());
|
2020-10-11 19:59:46 +01:00
|
|
|
|
|
|
|
$this->expectException(BadMethodCallException::class);
|
|
|
|
|
|
|
|
$this->getService()->create($server, [
|
|
|
|
'remote' => '%',
|
|
|
|
'database' => $name,
|
|
|
|
'database_host_id' => $host->id,
|
|
|
|
]);
|
|
|
|
|
|
|
|
$this->assertDatabaseMissing('databases', ['server_id' => $server->id]);
|
|
|
|
}
|
|
|
|
|
|
|
|
public function invalidDataDataProvider(): array
|
|
|
|
{
|
|
|
|
return [
|
|
|
|
[[]],
|
|
|
|
[['database' => '']],
|
|
|
|
[['database' => 'something']],
|
|
|
|
[['database' => 's_something']],
|
|
|
|
[['database' => 's12s_something']],
|
|
|
|
[['database' => 's12something']],
|
|
|
|
];
|
|
|
|
}
|
|
|
|
|
2022-10-14 17:59:20 +01:00
|
|
|
private function getService(): DatabaseManagementService
|
2020-10-11 19:59:46 +01:00
|
|
|
{
|
|
|
|
return $this->app->make(DatabaseManagementService::class);
|
|
|
|
}
|
|
|
|
}
|