140 lines
5.3 KiB
PHP
Raw Normal View History

2024-01-06 15:24:57 +10:00
<?php
namespace App\Livewire\Project\Database;
use Exception;
use Livewire\Component;
use Livewire\WithFileUploads;
use App\Models\Server;
use App\Models\StandaloneMariadb;
use App\Models\StandaloneMongodb;
use App\Models\StandaloneMysql;
use App\Models\StandalonePostgresql;
use App\Models\StandaloneRedis;
2024-01-10 15:42:54 +01:00
use Illuminate\Support\Facades\Storage;
2024-01-06 15:24:57 +10:00
class Import extends Component
{
use WithFileUploads;
public $file;
public $resource;
public $parameters;
2024-01-10 15:42:54 +01:00
public $containers;
2024-01-06 15:24:57 +10:00
public bool $validated = true;
public bool $scpInProgress = false;
public bool $importRunning = false;
public string $validationMsg = '';
public Server $server;
public string $container;
public array $importCommands = [];
2024-01-10 15:42:54 +01:00
public string $postgresqlRestoreCommand = 'pg_restore -U $POSTGRES_USER -d $POSTGRES_DB';
2024-02-24 21:12:34 +01:00
public string $mysqlRestoreCommand = 'mysql -u $MYSQL_USER -p$MYSQL_PASSWORD $MYSQL_DATABASE';
public string $mariadbRestoreCommand = 'mariadb -u $MARIADB_USER -p$MARIADB_PASSWORD $MARIADB_DATABASE';
2024-01-06 15:24:57 +10:00
2024-01-10 15:42:54 +01:00
public function getListeners()
{
$userId = auth()->user()->id;
return [
"echo-private:user.{$userId},DatabaseStatusChanged" => '$refresh',
];
}
2024-01-06 15:24:57 +10:00
public function mount()
{
$this->parameters = get_route_parameters();
$this->getContainers();
}
public function getContainers()
{
$this->containers = collect();
if (!data_get($this->parameters, 'database_uuid')) {
abort(404);
}
$resource = StandalonePostgresql::where('uuid', $this->parameters['database_uuid'])->first();
if (is_null($resource)) {
$resource = StandaloneRedis::where('uuid', $this->parameters['database_uuid'])->first();
if (is_null($resource)) {
$resource = StandaloneMongodb::where('uuid', $this->parameters['database_uuid'])->first();
if (is_null($resource)) {
$resource = StandaloneMysql::where('uuid', $this->parameters['database_uuid'])->first();
if (is_null($resource)) {
$resource = StandaloneMariadb::where('uuid', $this->parameters['database_uuid'])->first();
if (is_null($resource)) {
abort(404);
}
}
}
}
}
$this->resource = $resource;
$this->server = $this->resource->destination->server;
$this->container = $this->resource->uuid;
2024-01-10 15:42:54 +01:00
if (str(data_get($this, 'resource.status'))->startsWith('running')) {
2024-01-06 15:24:57 +10:00
$this->containers->push($this->container);
}
if ($this->containers->count() > 1) {
$this->validated = false;
$this->validationMsg = 'The database service has more than one container running. Cannot import.';
}
2024-01-10 15:42:54 +01:00
if (
$this->resource->getMorphClass() == 'App\Models\StandaloneRedis'
|| $this->resource->getMorphClass() == 'App\Models\StandaloneMongodb'
) {
2024-01-06 15:24:57 +10:00
$this->validated = false;
$this->validationMsg = 'This database type is not currently supported.';
}
}
2024-01-10 15:42:54 +01:00
2024-01-06 15:24:57 +10:00
public function runImport()
{
$this->validate([
'file' => 'required|file|max:102400'
]);
$this->importRunning = true;
$this->scpInProgress = true;
try {
$uploadedFilename = $this->file->store('backup-import');
2024-01-10 15:42:54 +01:00
$path = Storage::path($uploadedFilename);
2024-01-06 15:24:57 +10:00
$tmpPath = '/tmp/' . basename($uploadedFilename);
// SCP the backup file to the server.
instant_scp($path, $tmpPath, $this->server);
$this->scpInProgress = false;
$this->importCommands[] = "docker cp {$tmpPath} {$this->container}:{$tmpPath}";
switch ($this->resource->getMorphClass()) {
case 'App\Models\StandaloneMariadb':
2024-01-10 15:42:54 +01:00
$this->importCommands[] = "docker exec {$this->container} sh -c '{$this->mariadbRestoreCommand} < {$tmpPath}'";
2024-01-06 15:24:57 +10:00
$this->importCommands[] = "rm {$tmpPath}";
2024-01-10 15:42:54 +01:00
break;
2024-01-06 15:24:57 +10:00
case 'App\Models\StandaloneMysql':
2024-01-10 15:42:54 +01:00
$this->importCommands[] = "docker exec {$this->container} sh -c '{$this->mysqlRestoreCommand} < {$tmpPath}'";
2024-01-06 15:24:57 +10:00
$this->importCommands[] = "rm {$tmpPath}";
2024-01-10 15:42:54 +01:00
break;
2024-01-06 15:24:57 +10:00
case 'App\Models\StandalonePostgresql':
2024-01-10 15:42:54 +01:00
$this->importCommands[] = "docker exec {$this->container} sh -c '{$this->postgresqlRestoreCommand} {$tmpPath}'";
2024-01-06 15:24:57 +10:00
$this->importCommands[] = "rm {$tmpPath}";
2024-01-10 15:42:54 +01:00
break;
2024-01-06 15:24:57 +10:00
}
$this->importCommands[] = "docker exec {$this->container} sh -c 'rm {$tmpPath}'";
$this->importCommands[] = "docker exec {$this->container} sh -c 'echo \"Import finished with exit code $?\"'";
if (!empty($this->importCommands)) {
$activity = remote_process($this->importCommands, $this->server, ignore_errors: true);
$this->dispatch('activityMonitor', $activity->id);
2024-01-06 15:24:57 +10:00
}
} catch (\Throwable $e) {
$this->validated = false;
$this->validationMsg = $e->getMessage();
}
}
}