wait for server to connect, then provision

This commit is contained in:
2025-03-30 12:30:34 +00:00
parent 20e0a3995f
commit 435a7ac1e3
6 changed files with 107 additions and 10 deletions

View File

@@ -4,8 +4,11 @@ namespace App\Enums;
enum ServerStatus: string enum ServerStatus: string
{ {
case PENDING = 'pending'; case WAITING_FOR_PROVIDER = 'waiting-for-provider';
case PROVIDER_TIMEOUT = 'provider-timeout';
case UNPROVISIONED = 'unprovisioned';
case PROVISIONING = 'provisioning'; case PROVISIONING = 'provisioning';
case PROVISIONING_FAILED = 'provisioning-failed';
case UPDATING = 'updating'; case UPDATING = 'updating';
case ACTIVE = 'active'; case ACTIVE = 'active';
case DELETING = 'deleting'; case DELETING = 'deleting';

View File

@@ -6,6 +6,7 @@ use App\Actions\GenerateRandomSlug;
use App\Actions\GetProviderService; use App\Actions\GetProviderService;
use App\Enums\ServerProvider; use App\Enums\ServerProvider;
use App\Enums\ServerStatus; use App\Enums\ServerStatus;
use App\Jobs\Servers\WaitForServerToConnect;
use App\Models\Organisation; use App\Models\Organisation;
use App\Services\ServerProviders\HetznerService; use App\Services\ServerProviders\HetznerService;
use Illuminate\Http\Request; use Illuminate\Http\Request;
@@ -56,6 +57,7 @@ class ServerController extends Controller
public function store(Request $request) public function store(Request $request)
{ {
$rootPassword = Str::random(32); $rootPassword = Str::random(32);
$sudoPassword = Str::random(32);
$providerService = app(GetProviderService::class)->execute($request->provider); $providerService = app(GetProviderService::class)->execute($request->provider);
if (!$providerService) { if (!$providerService) {
@@ -79,13 +81,19 @@ class ServerController extends Controller
'ipv4' => $createdServer->ipv4, 'ipv4' => $createdServer->ipv4,
'ipv6' => $createdServer->ipv6, 'ipv6' => $createdServer->ipv6,
'provider_status' => $createdServer->status, 'provider_status' => $createdServer->status,
'status' => ServerStatus::PENDING, 'status' => ServerStatus::WAITING_FOR_PROVIDER,
'region' => $request->location, 'region' => $request->location,
'os' => $request->image, 'os' => $request->image,
'plan' => $request->server_type, 'plan' => $request->server_type,
'user' => '', 'user' => '',
]); ]);
dispatch(new WaitForServerToConnect(
server: $server,
rootPassword: $rootPassword,
sudoPassword: $sudoPassword,
))->delay(now()->addSeconds(30));
return redirect()->route('servers.show', ['organisation' => $organisation->id, 'server' => $server->id]); return redirect()->route('servers.show', ['organisation' => $organisation->id, 'server' => $server->id]);
} }

View File

@@ -2,10 +2,12 @@
namespace App\Jobs\Servers; namespace App\Jobs\Servers;
use App\Enums\ServerStatus;
use App\Models\Server; use App\Models\Server;
use Illuminate\Contracts\Queue\ShouldBeEncrypted; use Illuminate\Contracts\Queue\ShouldBeEncrypted;
use Illuminate\Contracts\Queue\ShouldQueue; use Illuminate\Contracts\Queue\ShouldQueue;
use Illuminate\Foundation\Queue\Queueable; use Illuminate\Foundation\Queue\Queueable;
use Spatie\Ssh\Ssh;
class ProvisionServer implements ShouldQueue, ShouldBeEncrypted class ProvisionServer implements ShouldQueue, ShouldBeEncrypted
{ {
@@ -14,13 +16,29 @@ class ProvisionServer implements ShouldQueue, ShouldBeEncrypted
public function __construct( public function __construct(
protected Server $server, protected Server $server,
protected string $rootPassword, protected string $rootPassword,
) protected string $sudoPassword,
{ ) {
// //
} }
public function handle(): void public function handle(): void
{ {
// $ssh = Ssh::create('root', $this->server->ipv4 ?? $this->server->ipv6)
->usePassword($this->rootPassword)
->setTimeout(10);
$provisionScriptUrl = route('provision-script', [
'sudo_password' => $this->sudoPassword,
'hostname' => $this->server->hostname,
'server_id' => $this->server->id,
]);
// Download the provision script and execute it
// The script will run in the background
$ssh->execute("cd ~ && wget --quiet --output-document=provision.sh \"{$provisionScriptUrl}\" && chmod +x provision.sh && ./provision.sh &");
$this->server->update([
'status' => ServerStatus::PROVISIONING,
]);
} }
} }

View File

@@ -0,0 +1,49 @@
<?php
namespace App\Jobs\Servers;
use App\Enums\ServerStatus;
use App\Models\Server;
use Illuminate\Contracts\Queue\ShouldBeEncrypted;
use Illuminate\Contracts\Queue\ShouldQueue;
use Illuminate\Foundation\Queue\Queueable;
use Spatie\Ssh\Ssh;
class WaitForServerToConnect implements ShouldQueue, ShouldBeEncrypted
{
use Queueable;
public function __construct(
protected Server $server,
protected string $rootPassword,
protected string $sudoPassword,
)
{
//
}
public function handle(): void
{
try {
Ssh::create('root', $this->server->ipv4 ?? $this->server->ipv6)
->usePassword($this->rootPassword)
->setTimeout(10)
->execute('echo "Connected"');
$this->server->update([
'status' => ServerStatus::UNPROVISIONED,
]);
dispatch(new ProvisionServer($this->server, $this->rootPassword, $this->sudoPassword));
} catch (\Throwable $e) {
return;
}
}
public function failed(\Throwable $exception): void
{
$this->server->update([
'status' => ServerStatus::PROVIDER_TIMEOUT,
]);
}
}

View File

@@ -1,6 +1,7 @@
#!/bin/bash #!/bin/bash
# [!server!] # [!hostname!] - server hostname
# [!sudo_password!] # [!sudo_password!] - the sudo password to set
# [!server_id!] - the servers id
apt_wait() { apt_wait() {
while fuser /var/lib/dpkg/lock >/dev/null 2>&1; do while fuser /var/lib/dpkg/lock >/dev/null 2>&1; do
@@ -52,8 +53,8 @@ if [ ! -d /root/.ssh ]; then
fi fi
# Set The Hostname If Necessary # Set The Hostname If Necessary
echo "[!server!]" > /etc/hostname sed -i 's/127\.0\.0\.1.*localhost/127.0.0.1 [!server!].localdomain [!server!] localhost/' /etc/hosts echo "[!hostname!]" > /etc/hostname sed -i 's/127\.0\.0\.1.*localhost/127.0.0.1 [!hostname!].localdomain [!hostname!] localhost/' /etc/hosts
hostname [!server!] hostname [!hostname!]
# Setup Keystone User # Setup Keystone User
useradd keystone useradd keystone
@@ -124,4 +125,4 @@ EOF
# Callback that the server is installed # Callback that the server is installed
curl --insecure --data "event_id=878&server_id=390&sudo_password=[!sudo_password!]&recipe_id=" https://keystone.test/provisioning/callback/app curl --insecure --data "server_id=[!server_id!]&sudo_password=[!sudo_password!] https://keystone.test/provisioning/callback/app

View File

@@ -4,6 +4,7 @@ use App\Http\Controllers\ApplicationController;
use App\Http\Controllers\EnvironmentController; use App\Http\Controllers\EnvironmentController;
use App\Http\Controllers\OrganisationController; use App\Http\Controllers\OrganisationController;
use App\Http\Controllers\ServerController; use App\Http\Controllers\ServerController;
use Illuminate\Http\Request;
use Illuminate\Support\Facades\Route; use Illuminate\Support\Facades\Route;
use Inertia\Inertia; use Inertia\Inertia;
@@ -36,5 +37,22 @@ Route::middleware(['auth', 'verified'])->group(function () {
}); });
}); });
Route::get('/provision-script', function (Request $request) {
$validated = $request->validate([
'sudo_password' => ['required', 'string'],
'hostname' => ['required', 'string'],
'server_id' => ['required', 'integer', 'exists:servers,id'],
]);
$script = file_get_contents(base_path('provision.sh'));
$script = str_replace('[!hostname!]', $validated['hostname'], $script);
$script = str_replace('[!sudo_password!]', $validated['sudo_password'], $script);
$script = str_replace('[!server_id!]', $validated['server_id'], $script);
return response($script)
->header('Content-Type', 'text/plain');
})->name('provision-script');
require __DIR__ . '/settings.php'; require __DIR__ . '/settings.php';
require __DIR__ . '/auth.php'; require __DIR__ . '/auth.php';