154 lines
		
	
	
		
			4.5 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
			
		
		
	
	
			154 lines
		
	
	
		
			4.5 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
<?php
 | 
						|
 | 
						|
namespace App\Jobs;
 | 
						|
 | 
						|
use App\Models\Server;
 | 
						|
use App\Services\ConfigurationRepository;
 | 
						|
use Illuminate\Bus\Queueable;
 | 
						|
use Illuminate\Contracts\Queue\ShouldBeEncrypted;
 | 
						|
use Illuminate\Contracts\Queue\ShouldQueue;
 | 
						|
use Illuminate\Foundation\Bus\Dispatchable;
 | 
						|
use Illuminate\Queue\InteractsWithQueue;
 | 
						|
use Illuminate\Queue\Middleware\WithoutOverlapping;
 | 
						|
use Illuminate\Queue\SerializesModels;
 | 
						|
use Illuminate\Support\Facades\Log;
 | 
						|
 | 
						|
class ServerConnectionCheckJob implements ShouldBeEncrypted, ShouldQueue
 | 
						|
{
 | 
						|
    use Dispatchable, InteractsWithQueue, Queueable, SerializesModels;
 | 
						|
 | 
						|
    public $tries = 1;
 | 
						|
 | 
						|
    public $timeout = 30;
 | 
						|
 | 
						|
    public function __construct(
 | 
						|
        public Server $server,
 | 
						|
        public bool $disableMux = true
 | 
						|
    ) {}
 | 
						|
 | 
						|
    public function middleware(): array
 | 
						|
    {
 | 
						|
        return [(new WithoutOverlapping('server-connection-check-'.$this->server->uuid))->expireAfter(45)->dontRelease()];
 | 
						|
    }
 | 
						|
 | 
						|
    private function disableSshMux(): void
 | 
						|
    {
 | 
						|
        $configRepository = app(ConfigurationRepository::class);
 | 
						|
        $configRepository->disableSshMux();
 | 
						|
    }
 | 
						|
 | 
						|
    public function handle()
 | 
						|
    {
 | 
						|
        try {
 | 
						|
            // Check if server is disabled
 | 
						|
            if ($this->server->settings->force_disabled) {
 | 
						|
                $this->server->settings->update([
 | 
						|
                    'is_reachable' => false,
 | 
						|
                    'is_usable' => false,
 | 
						|
                ]);
 | 
						|
                Log::debug('ServerConnectionCheck: Server is disabled', [
 | 
						|
                    'server_id' => $this->server->id,
 | 
						|
                    'server_name' => $this->server->name,
 | 
						|
                ]);
 | 
						|
 | 
						|
                return;
 | 
						|
            }
 | 
						|
 | 
						|
            // Temporarily disable mux if requested
 | 
						|
            if ($this->disableMux) {
 | 
						|
                $this->disableSshMux();
 | 
						|
            }
 | 
						|
 | 
						|
            // Check basic connectivity first
 | 
						|
            $isReachable = $this->checkConnection();
 | 
						|
 | 
						|
            if (! $isReachable) {
 | 
						|
                $this->server->settings->update([
 | 
						|
                    'is_reachable' => false,
 | 
						|
                    'is_usable' => false,
 | 
						|
                ]);
 | 
						|
 | 
						|
                Log::warning('ServerConnectionCheck: Server not reachable', [
 | 
						|
                    'server_id' => $this->server->id,
 | 
						|
                    'server_name' => $this->server->name,
 | 
						|
                    'server_ip' => $this->server->ip,
 | 
						|
                ]);
 | 
						|
 | 
						|
                return;
 | 
						|
            }
 | 
						|
 | 
						|
            // Server is reachable, check if Docker is available
 | 
						|
            $isUsable = $this->checkDockerAvailability();
 | 
						|
 | 
						|
            $this->server->settings->update([
 | 
						|
                'is_reachable' => true,
 | 
						|
                'is_usable' => $isUsable,
 | 
						|
            ]);
 | 
						|
 | 
						|
        } catch (\Throwable $e) {
 | 
						|
            $this->server->settings->update([
 | 
						|
                'is_reachable' => false,
 | 
						|
                'is_usable' => false,
 | 
						|
            ]);
 | 
						|
 | 
						|
            throw $e;
 | 
						|
        }
 | 
						|
    }
 | 
						|
 | 
						|
    private function checkConnection(): bool
 | 
						|
    {
 | 
						|
        try {
 | 
						|
            // Use instant_remote_process with a simple command
 | 
						|
            // This will automatically handle mux, sudo, IPv6, Cloudflare tunnel, etc.
 | 
						|
            $output = instant_remote_process_with_timeout(
 | 
						|
                ['ls -la /'],
 | 
						|
                $this->server,
 | 
						|
                false // don't throw error
 | 
						|
            );
 | 
						|
 | 
						|
            return $output !== null;
 | 
						|
        } catch (\Throwable $e) {
 | 
						|
            Log::debug('ServerConnectionCheck: Connection check failed', [
 | 
						|
                'server_id' => $this->server->id,
 | 
						|
                'error' => $e->getMessage(),
 | 
						|
            ]);
 | 
						|
 | 
						|
            return false;
 | 
						|
        }
 | 
						|
    }
 | 
						|
 | 
						|
    private function checkDockerAvailability(): bool
 | 
						|
    {
 | 
						|
        try {
 | 
						|
            // Use instant_remote_process to check Docker
 | 
						|
            // The function will automatically handle sudo for non-root users
 | 
						|
            $output = instant_remote_process_with_timeout(
 | 
						|
                ['docker version --format json'],
 | 
						|
                $this->server,
 | 
						|
                false // don't throw error
 | 
						|
            );
 | 
						|
 | 
						|
            if ($output === null) {
 | 
						|
                return false;
 | 
						|
            }
 | 
						|
 | 
						|
            // Try to parse the JSON output to ensure Docker is really working
 | 
						|
            $output = trim($output);
 | 
						|
            if (! empty($output)) {
 | 
						|
                $dockerInfo = json_decode($output, true);
 | 
						|
 | 
						|
                return isset($dockerInfo['Server']['Version']);
 | 
						|
            }
 | 
						|
 | 
						|
            return false;
 | 
						|
        } catch (\Throwable $e) {
 | 
						|
            Log::debug('ServerConnectionCheck: Docker check failed', [
 | 
						|
                'server_id' => $this->server->id,
 | 
						|
                'error' => $e->getMessage(),
 | 
						|
            ]);
 | 
						|
 | 
						|
            return false;
 | 
						|
        }
 | 
						|
    }
 | 
						|
}
 |