diff --git a/app/Actions/Database/LinkUser.php b/app/Actions/Database/LinkUser.php index 166afc2..43026e8 100755 --- a/app/Actions/Database/LinkUser.php +++ b/app/Actions/Database/LinkUser.php @@ -27,8 +27,7 @@ public function link(DatabaseUser $databaseUser, array $input): void ->whereIn('name', $input['databases']) ->count(); if (count($input['databases']) !== $dbs) { - throw ValidationException::withMessages(['databases' => __('Databases not found!')]) - ->errorBag('linkUser'); + throw ValidationException::withMessages(['databases' => __('Databases not found!')]); } $databaseUser->databases = $input['databases']; diff --git a/app/Actions/Database/RunBackup.php b/app/Actions/Database/RunBackup.php index 8f37bf2..7ea949d 100644 --- a/app/Actions/Database/RunBackup.php +++ b/app/Actions/Database/RunBackup.php @@ -9,7 +9,7 @@ class RunBackup { - public function run(Backup $backup): void + public function run(Backup $backup): BackupFile { $file = new BackupFile([ 'backup_id' => $backup->id, @@ -26,5 +26,7 @@ public function run(Backup $backup): void $file->status = BackupFileStatus::FAILED; $file->save(); })->onConnection('ssh'); + + return $file; } } diff --git a/app/Actions/Queue/ManageQueue.php b/app/Actions/Queue/ManageQueue.php index b73bf5f..4a89fb3 100644 --- a/app/Actions/Queue/ManageQueue.php +++ b/app/Actions/Queue/ManageQueue.php @@ -3,7 +3,6 @@ namespace App\Actions\Queue; use App\Enums\QueueStatus; -use App\Enums\ServiceStatus; use App\Models\Queue; class ManageQueue @@ -14,10 +13,7 @@ public function start(Queue $queue): void $queue->save(); dispatch(function () use ($queue) { $queue->server->processManager()->handler()->start($queue->id, $queue->site_id); - $queue->status = ServiceStatus::READY; - $queue->save(); - })->catch(function () use ($queue) { - $queue->status = ServiceStatus::FAILED; + $queue->status = QueueStatus::RUNNING; $queue->save(); })->onConnection('ssh'); } @@ -28,10 +24,7 @@ public function stop(Queue $queue): void $queue->save(); dispatch(function () use ($queue) { $queue->server->processManager()->handler()->stop($queue->id, $queue->site_id); - $queue->status = ServiceStatus::STOPPED; - $queue->save(); - })->catch(function () use ($queue) { - $queue->status = ServiceStatus::FAILED; + $queue->status = QueueStatus::STOPPED; $queue->save(); })->onConnection('ssh'); } @@ -42,10 +35,7 @@ public function restart(Queue $queue): void $queue->save(); dispatch(function () use ($queue) { $queue->server->processManager()->handler()->restart($queue->id, $queue->site_id); - $queue->status = ServiceStatus::READY; - $queue->save(); - })->catch(function () use ($queue) { - $queue->status = ServiceStatus::FAILED; + $queue->status = QueueStatus::RUNNING; $queue->save(); })->onConnection('ssh'); } diff --git a/app/Actions/Script/CreateScript.php b/app/Actions/Script/CreateScript.php deleted file mode 100644 index a469a6c..0000000 --- a/app/Actions/Script/CreateScript.php +++ /dev/null @@ -1,41 +0,0 @@ -validateInputs($input); - - $script = new Script([ - 'user_id' => $creator->id, - 'name' => $input['name'], - 'content' => $input['content'], - ]); - $script->save(); - - return $script; - } - - /** - * @throws ValidationException - */ - private function validateInputs(array $input): void - { - $rules = [ - 'name' => 'required', - 'content' => 'required', - ]; - - Validator::make($input, $rules)->validateWithBag('createScript'); - } -} diff --git a/app/Actions/Script/GetScripts.php b/app/Actions/Script/GetScripts.php deleted file mode 100755 index 5aeac76..0000000 --- a/app/Actions/Script/GetScripts.php +++ /dev/null @@ -1,17 +0,0 @@ -scripts() - ->orderBy('id', 'desc') - ->paginate(6) - ->onEachSide(1); - } -} diff --git a/app/Actions/Script/UpdateScript.php b/app/Actions/Script/UpdateScript.php deleted file mode 100644 index 267ca35..0000000 --- a/app/Actions/Script/UpdateScript.php +++ /dev/null @@ -1,37 +0,0 @@ -validateInputs($input); - - $script->name = $input['name']; - $script->content = $input['content']; - $script->save(); - - return $script; - } - - /** - * @throws ValidationException - */ - private function validateInputs(array $input): void - { - $rules = [ - 'name' => 'required', - 'content' => 'required', - ]; - - Validator::make($input, $rules)->validateWithBag('updateScript'); - } -} diff --git a/app/Actions/Server/GetServers.php b/app/Actions/Server/GetServers.php deleted file mode 100755 index d7d33a3..0000000 --- a/app/Actions/Server/GetServers.php +++ /dev/null @@ -1,14 +0,0 @@ -latest()->get(); - } -} diff --git a/app/Actions/ServerProvider/CreateServerProvider.php b/app/Actions/ServerProvider/CreateServerProvider.php index 06bb7f3..bdcb453 100644 --- a/app/Actions/ServerProvider/CreateServerProvider.php +++ b/app/Actions/ServerProvider/CreateServerProvider.php @@ -28,7 +28,7 @@ public function create(User $user, array $input): ServerProvider } catch (Exception) { throw ValidationException::withMessages([ 'provider' => [ - __("Couldn't connect to provider. Please check your credentials and try again later."), + sprintf("Couldn't connect to %s. Please check your credentials.", $input['provider']), ], ]); } diff --git a/app/Actions/ServerProvider/DeleteServerProvider.php b/app/Actions/ServerProvider/DeleteServerProvider.php new file mode 100644 index 0000000..873ddb0 --- /dev/null +++ b/app/Actions/ServerProvider/DeleteServerProvider.php @@ -0,0 +1,21 @@ +servers()->exists()) { + throw new Exception('This server provider is being used by a server.'); + } + + $serverProvider->delete(); + } +} diff --git a/app/Actions/Service/Manage.php b/app/Actions/Service/Manage.php index 761c536..a7a9a90 100644 --- a/app/Actions/Service/Manage.php +++ b/app/Actions/Service/Manage.php @@ -19,9 +19,6 @@ public function start(Service $service): void $service->status = ServiceStatus::FAILED; } $service->save(); - })->catch(function () use ($service) { - $service->status = ServiceStatus::FAILED; - $service->save(); })->onConnection('ssh'); } @@ -37,9 +34,6 @@ public function stop(Service $service): void $service->status = ServiceStatus::FAILED; } $service->save(); - })->catch(function () use ($service) { - $service->status = ServiceStatus::FAILED; - $service->save(); })->onConnection('ssh'); } @@ -55,9 +49,6 @@ public function restart(Service $service): void $service->status = ServiceStatus::FAILED; } $service->save(); - })->catch(function () use ($service) { - $service->status = ServiceStatus::FAILED; - $service->save(); })->onConnection('ssh'); } } diff --git a/app/Actions/SourceControl/DeleteSourceControl.php b/app/Actions/SourceControl/DeleteSourceControl.php new file mode 100644 index 0000000..e084056 --- /dev/null +++ b/app/Actions/SourceControl/DeleteSourceControl.php @@ -0,0 +1,17 @@ +sites()->exists()) { + throw new \Exception('This source control is being used by a site.'); + } + + $sourceControl->delete(); + } +} diff --git a/app/Actions/StorageProvider/DeleteStorageProvider.php b/app/Actions/StorageProvider/DeleteStorageProvider.php new file mode 100644 index 0000000..d8cd5cf --- /dev/null +++ b/app/Actions/StorageProvider/DeleteStorageProvider.php @@ -0,0 +1,21 @@ +backups()->exists()) { + throw new Exception('This storage provider is being used by a backup.'); + } + + $storageProvider->delete(); + } +} diff --git a/app/Console/Commands/RunBackupCommand.php b/app/Console/Commands/RunBackupCommand.php index d349a7b..c6d3dfe 100644 --- a/app/Console/Commands/RunBackupCommand.php +++ b/app/Console/Commands/RunBackupCommand.php @@ -15,14 +15,19 @@ class RunBackupCommand extends Command public function handle(): void { + $total = 0; + Backup::query() ->where('interval', $this->argument('interval')) ->where('status', BackupStatus::RUNNING) - ->chunk(100, function ($backups) { + ->chunk(100, function ($backups) use (&$total) { /** @var Backup $backup */ foreach ($backups as $backup) { app(RunBackup::class)->run($backup); + $total++; } }); + + $this->info("{$total} backups started"); } } diff --git a/app/Exceptions/SourceControlIsNotConnected.php b/app/Exceptions/SourceControlIsNotConnected.php index e93deaf..7b0f1aa 100755 --- a/app/Exceptions/SourceControlIsNotConnected.php +++ b/app/Exceptions/SourceControlIsNotConnected.php @@ -2,13 +2,8 @@ namespace App\Exceptions; -use App\Models\SourceControl; use Exception; class SourceControlIsNotConnected extends Exception { - public function __construct(protected SourceControl|string|null $sourceControl, ?string $message = null) - { - parent::__construct($message ?? 'Source control is not connected'); - } } diff --git a/app/Helpers/SSH.php b/app/Helpers/SSH.php index 86e6f5e..d2d91b4 100755 --- a/app/Helpers/SSH.php +++ b/app/Helpers/SSH.php @@ -88,7 +88,7 @@ public function connect(bool $sftp = false): void Log::error('Error connecting', [ 'msg' => $e->getMessage(), ]); - throw $e; + throw new SSHConnectionError($e->getMessage()); } } diff --git a/app/Http/Controllers/Settings/ProjectController.php b/app/Http/Controllers/Settings/ProjectController.php index 085e1f1..3d1f5c2 100644 --- a/app/Http/Controllers/Settings/ProjectController.php +++ b/app/Http/Controllers/Settings/ProjectController.php @@ -47,11 +47,14 @@ public function update(Request $request, Project $project): HtmxResponse public function delete(Project $project): RedirectResponse { + /** @var User $user */ + $user = auth()->user(); + /** @var Project $project */ - $project = auth()->user()->projects()->findOrFail($project->id); + $project = $user->projects()->findOrFail($project->id); try { - app(DeleteProject::class)->delete(auth()->user(), $project); + app(DeleteProject::class)->delete($user, $project); } catch (ValidationException $e) { Toast::error($e->getMessage()); diff --git a/app/Http/Controllers/Settings/ServerProviderController.php b/app/Http/Controllers/Settings/ServerProviderController.php index 9d7b76e..bae2092 100644 --- a/app/Http/Controllers/Settings/ServerProviderController.php +++ b/app/Http/Controllers/Settings/ServerProviderController.php @@ -3,6 +3,7 @@ namespace App\Http\Controllers\Settings; use App\Actions\ServerProvider\CreateServerProvider; +use App\Actions\ServerProvider\DeleteServerProvider; use App\Facades\Toast; use App\Helpers\HtmxResponse; use App\Http\Controllers\Controller; @@ -32,14 +33,15 @@ public function connect(Request $request): HtmxResponse return htmx()->redirect(route('server-providers')); } - /** - * @TODO Update servers using this provider - */ - public function delete(int $id): RedirectResponse + public function delete(ServerProvider $serverProvider): RedirectResponse { - $serverProvider = ServerProvider::query()->findOrFail($id); + try { + app(DeleteServerProvider::class)->delete($serverProvider); + } catch (\Exception $e) { + Toast::error($e->getMessage()); - $serverProvider->delete(); + return back(); + } Toast::success('Server provider deleted.'); diff --git a/app/Http/Controllers/Settings/SourceControlController.php b/app/Http/Controllers/Settings/SourceControlController.php index c17ae98..f3bb097 100644 --- a/app/Http/Controllers/Settings/SourceControlController.php +++ b/app/Http/Controllers/Settings/SourceControlController.php @@ -3,6 +3,7 @@ namespace App\Http\Controllers\Settings; use App\Actions\SourceControl\ConnectSourceControl; +use App\Actions\SourceControl\DeleteSourceControl; use App\Facades\Toast; use App\Helpers\HtmxResponse; use App\Http\Controllers\Controller; @@ -11,9 +12,6 @@ use Illuminate\Http\RedirectResponse; use Illuminate\Http\Request; -/** - * @TODO Assign user to source control - */ class SourceControlController extends Controller { public function index(): View @@ -34,11 +32,15 @@ public function connect(Request $request): HtmxResponse return htmx()->redirect(route('source-controls')); } - public function delete(int $id): RedirectResponse + public function delete(SourceControl $sourceControl): RedirectResponse { - $sourceControl = SourceControl::query()->findOrFail($id); + try { + app(DeleteSourceControl::class)->delete($sourceControl); + } catch (\Exception $e) { + Toast::error($e->getMessage()); - $sourceControl->delete(); + return back(); + } Toast::success('Source control deleted.'); diff --git a/app/Http/Controllers/Settings/StorageProviderController.php b/app/Http/Controllers/Settings/StorageProviderController.php index bc57676..939c56b 100644 --- a/app/Http/Controllers/Settings/StorageProviderController.php +++ b/app/Http/Controllers/Settings/StorageProviderController.php @@ -3,6 +3,7 @@ namespace App\Http\Controllers\Settings; use App\Actions\StorageProvider\CreateStorageProvider; +use App\Actions\StorageProvider\DeleteStorageProvider; use App\Facades\Toast; use App\Helpers\HtmxResponse; use App\Http\Controllers\Controller; @@ -32,14 +33,15 @@ public function connect(Request $request): HtmxResponse return htmx()->redirect(route('storage-providers')); } - /** - * @TODO Update servers using this provider - */ - public function delete(int $id): RedirectResponse + public function delete(StorageProvider $storageProvider): RedirectResponse { - $storageProvider = StorageProvider::query()->findOrFail($id); + try { + app(DeleteStorageProvider::class)->delete($storageProvider); + } catch (\Exception $e) { + Toast::error($e->getMessage()); - $storageProvider->delete(); + return back(); + } Toast::success('Storage provider deleted.'); diff --git a/app/Http/Controllers/SiteSettingController.php b/app/Http/Controllers/SiteSettingController.php index 6382af3..b2b940f 100644 --- a/app/Http/Controllers/SiteSettingController.php +++ b/app/Http/Controllers/SiteSettingController.php @@ -3,6 +3,7 @@ namespace App\Http\Controllers; use App\Facades\Toast; +use App\Helpers\HtmxResponse; use App\Models\Server; use App\Models\Site; use Illuminate\Contracts\View\View; @@ -43,7 +44,7 @@ public function updateVhost(Server $server, Site $site, Request $request): Redir return back(); } - public function updatePHPVersion(Server $server, Site $site, Request $request): RedirectResponse + public function updatePHPVersion(Server $server, Site $site, Request $request): HtmxResponse { $this->validate($request, [ 'version' => [ @@ -60,6 +61,6 @@ public function updatePHPVersion(Server $server, Site $site, Request $request): Toast::error($e->getMessage()); } - return back(); + return htmx()->back(); } } diff --git a/app/Models/Redirect.php b/app/Models/Redirect.php deleted file mode 100644 index 6221029..0000000 --- a/app/Models/Redirect.php +++ /dev/null @@ -1,36 +0,0 @@ - 'integer', - 'mode' => 'integer', - ]; - - public function site(): BelongsTo - { - return $this->belongsTo(Site::class); - } -} diff --git a/app/Models/Script.php b/app/Models/Script.php deleted file mode 100644 index 8b085ae..0000000 --- a/app/Models/Script.php +++ /dev/null @@ -1,38 +0,0 @@ - 'integer', - ]; - - public function creator(): BelongsTo - { - return $this->belongsTo(User::class); - } - - public function executions(): HasMany - { - return $this->hasMany(ScriptExecution::class, 'script_id'); - } -} diff --git a/app/Models/ScriptExecution.php b/app/Models/ScriptExecution.php deleted file mode 100644 index f0aa0cd..0000000 --- a/app/Models/ScriptExecution.php +++ /dev/null @@ -1,42 +0,0 @@ - 'integer', - 'server_id' => 'integer', - ]; - - public function script(): BelongsTo - { - return $this->belongsTo(Script::class); - } - - public function server(): BelongsTo - { - return $this->belongsTo(Server::class); - } -} diff --git a/app/Models/Server.php b/app/Models/Server.php index e6dfea4..7e63f01 100755 --- a/app/Models/Server.php +++ b/app/Models/Server.php @@ -51,7 +51,6 @@ * @property FirewallRule[] $firewallRules * @property CronJob[] $cronJobs * @property Queue[] $queues - * @property ScriptExecution[] $scriptExecutions * @property Backup[] $backups * @property Queue[] $daemons * @property SshKey[] $sshKeys @@ -121,7 +120,6 @@ public static function boot(): void $server->cronJobs()->delete(); $server->queues()->delete(); $server->daemons()->delete(); - $server->scriptExecutions()->delete(); $server->sshKeys()->detach(); if (File::exists($server->sshKey()['public_key_path'])) { File::delete($server->sshKey()['public_key_path']); @@ -187,11 +185,6 @@ public function queues(): HasMany return $this->hasMany(Queue::class); } - public function scriptExecutions(): HasMany - { - return $this->hasMany(ScriptExecution::class); - } - public function backups(): HasMany { return $this->hasMany(Backup::class); diff --git a/app/Models/ServerLog.php b/app/Models/ServerLog.php index 79c8895..2efee2c 100755 --- a/app/Models/ServerLog.php +++ b/app/Models/ServerLog.php @@ -4,6 +4,7 @@ use Illuminate\Database\Eloquent\Factories\HasFactory; use Illuminate\Database\Eloquent\Relations\BelongsTo; +use Illuminate\Support\Facades\Log; use Illuminate\Support\Facades\Storage; use Illuminate\Support\Str; @@ -38,8 +39,12 @@ public static function boot(): void parent::boot(); static::deleting(function (ServerLog $log) { - if (Storage::disk($log->disk)->exists($log->name)) { - Storage::disk($log->disk)->delete($log->name); + try { + if (Storage::disk($log->disk)->exists($log->name)) { + Storage::disk($log->disk)->delete($log->name); + } + } catch (\Exception $e) { + Log::error($e->getMessage(), ['exception' => $e]); } }); } diff --git a/app/Models/ServerProvider.php b/app/Models/ServerProvider.php index 366663c..130c39d 100644 --- a/app/Models/ServerProvider.php +++ b/app/Models/ServerProvider.php @@ -4,6 +4,7 @@ use Illuminate\Database\Eloquent\Factories\HasFactory; use Illuminate\Database\Eloquent\Relations\BelongsTo; +use Illuminate\Database\Eloquent\Relations\HasMany; /** * @property int $user_id @@ -40,4 +41,9 @@ public function getCredentials(): array { return $this->credentials; } + + public function servers(): HasMany + { + return $this->hasMany(Server::class, 'provider_id'); + } } diff --git a/app/Models/Site.php b/app/Models/Site.php index b1355de..66bd533 100755 --- a/app/Models/Site.php +++ b/app/Models/Site.php @@ -33,7 +33,6 @@ * @property Deployment[] $deployments * @property ?GitHook $gitHook * @property DeploymentScript $deploymentScript - * @property Redirect[] $redirects * @property Queue[] $queues * @property Ssl[] $ssls * @property ?Ssl $activeSsl @@ -76,7 +75,6 @@ public static function boot(): void parent::boot(); static::deleting(function (Site $site) { - $site->redirects()->delete(); $site->queues()->delete(); $site->ssls()->delete(); $site->deployments()->delete(); @@ -116,11 +114,6 @@ public function deploymentScript(): HasOne return $this->hasOne(DeploymentScript::class); } - public function redirects(): HasMany - { - return $this->hasMany(Redirect::class); - } - public function queues(): HasMany { return $this->hasMany(Queue::class); @@ -192,8 +185,8 @@ public function php(): ?Service public function changePHPVersion($version): void { - $this->php_version = $version; $this->server->webserver()->handler()->changePHPVersion($this, $version); + $this->php_version = $version; $this->save(); } diff --git a/app/Models/SourceControl.php b/app/Models/SourceControl.php index 00f16d2..0abf2d1 100755 --- a/app/Models/SourceControl.php +++ b/app/Models/SourceControl.php @@ -4,6 +4,7 @@ use App\SourceControlProviders\SourceControlProvider; use Illuminate\Database\Eloquent\Factories\HasFactory; +use Illuminate\Database\Eloquent\Relations\HasMany; /** * @property string $provider @@ -37,4 +38,9 @@ public function getRepo(?string $repo = null): ?array { return $this->provider()->getRepo($repo); } + + public function sites(): HasMany + { + return $this->hasMany(Site::class); + } } diff --git a/app/Models/StorageProvider.php b/app/Models/StorageProvider.php index bd69d9e..7fe9b4c 100644 --- a/app/Models/StorageProvider.php +++ b/app/Models/StorageProvider.php @@ -4,6 +4,7 @@ use Illuminate\Database\Eloquent\Factories\HasFactory; use Illuminate\Database\Eloquent\Relations\BelongsTo; +use Illuminate\Database\Eloquent\Relations\HasMany; /** * @property int $user_id @@ -39,4 +40,9 @@ public function provider(): \App\StorageProviders\StorageProvider return new $providerClass($this); } + + public function backups(): HasMany + { + return $this->hasMany(Backup::class, 'storage_id'); + } } diff --git a/app/Models/User.php b/app/Models/User.php index 5d18ddb..219e4b7 100755 --- a/app/Models/User.php +++ b/app/Models/User.php @@ -86,11 +86,6 @@ public function serverProviders(): HasMany return $this->hasMany(ServerProvider::class); } - public function scripts(): HasMany - { - return $this->hasMany(Script::class, 'user_id'); - } - public function sourceControl(string $provider): HasOne { return $this->hasOne(SourceControl::class)->where('provider', $provider); diff --git a/app/NotificationChannels/Telegram.php b/app/NotificationChannels/Telegram.php index b692323..017399b 100644 --- a/app/NotificationChannels/Telegram.php +++ b/app/NotificationChannels/Telegram.php @@ -60,6 +60,6 @@ private function sendToTelegram(string $text): void 'text' => $text, 'parse_mode' => 'markdown', 'disable_web_page_preview' => true, - ]); + ])->throw(); } } diff --git a/app/Providers/AppServiceProvider.php b/app/Providers/AppServiceProvider.php index d4e8133..321ca5d 100644 --- a/app/Providers/AppServiceProvider.php +++ b/app/Providers/AppServiceProvider.php @@ -5,7 +5,6 @@ use App\Helpers\Notifier; use App\Helpers\SSH; use App\Helpers\Toast; -use App\Support\SocialiteProviders\DropboxProvider; use Illuminate\Contracts\Container\BindingResolutionException; use Illuminate\Http\Resources\Json\ResourceCollection; use Illuminate\Support\Facades\URL; @@ -39,26 +38,8 @@ public function boot(): void return new Toast; }); - $this->extendSocialite(); - if (str(config('app.url'))->startsWith('https://')) { URL::forceScheme('https'); } } - - /** - * @throws BindingResolutionException - */ - private function extendSocialite(): void - { - $socialite = $this->app->make('Laravel\Socialite\Contracts\Factory'); - $socialite->extend( - 'dropbox', - function ($app) use ($socialite) { - $config = $app['config']['services.dropbox']; - - return $socialite->buildProvider(DropboxProvider::class, $config); - } - ); - } } diff --git a/app/Support/SocialiteProviders/DropboxProvider.php b/app/Support/SocialiteProviders/DropboxProvider.php deleted file mode 100644 index a0253d9..0000000 --- a/app/Support/SocialiteProviders/DropboxProvider.php +++ /dev/null @@ -1,78 +0,0 @@ -buildAuthUrlFromBase('https://www.dropbox.com/oauth2/authorize', $state); - } - - /** - * {@inheritdoc} - */ - protected function getTokenUrl(): string - { - return 'https://api.dropboxapi.com/oauth2/token'; - } - - /** - * {@inheritdoc} - */ - protected function getTokenFields($code): array - { - return array_merge(parent::getTokenFields($code), [ - 'grant_type' => 'authorization_code', - ]); - } - - /** - * {@inheritdoc} - * - * @throws GuzzleException - */ - protected function getUserByToken($token) - { - $response = $this->getHttpClient()->post('https://api.dropboxapi.com/2/users/get_current_account', [ - 'headers' => [ - 'Authorization' => 'Bearer '.$token, - ], - ]); - - return json_decode($response->getBody(), true); - } - - /** - * {@inheritdoc} - */ - protected function mapUserToObject(array $user): User - { - return (new User)->setRaw($user)->map([ - 'id' => $user['account_id'], - 'nickname' => null, - 'name' => $user['name']['display_name'], - 'email' => $user['email'], - 'avatar' => Arr::get($user, 'profile_photo_url'), - ]); - } -} diff --git a/app/Support/Testing/SSHFake.php b/app/Support/Testing/SSHFake.php index a2213f7..def89d1 100644 --- a/app/Support/Testing/SSHFake.php +++ b/app/Support/Testing/SSHFake.php @@ -2,6 +2,7 @@ namespace App\Support\Testing; +use App\Exceptions\SSHConnectionError; use App\Helpers\SSH; use Illuminate\Support\Traits\ReflectsClosures; use PHPUnit\Framework\Assert; @@ -14,11 +15,25 @@ class SSHFake extends SSH protected ?string $output; + protected bool $connectionWillFail = false; + public function __construct(?string $output = null) { $this->output = $output; } + public function connectionWillFail(): void + { + $this->connectionWillFail = true; + } + + public function connect(bool $sftp = false): void + { + if ($this->connectionWillFail) { + throw new SSHConnectionError('Connection failed'); + } + } + public function exec(string|array $commands, string $log = '', ?int $siteId = null): string { if ($log) { @@ -45,6 +60,11 @@ public function exec(string|array $commands, string $log = '', ?int $siteId = nu return $output; } + public function upload(string $local, string $remote): void + { + $this->log = null; + } + public function assertExecuted(array|string $commands): void { if (! $this->commands) { diff --git a/composer.json b/composer.json index a7a334e..0d9ae80 100644 --- a/composer.json +++ b/composer.json @@ -17,7 +17,6 @@ "laravel/fortify": "^1.17", "laravel/framework": "^10.0", "laravel/sanctum": "^3.2", - "laravel/socialite": "^5.2", "laravel/tinker": "^2.8", "opcodesio/log-viewer": "^3.0", "phpseclib/phpseclib": "~3.0" diff --git a/composer.lock b/composer.lock index ff190b1..22920d5 100644 --- a/composer.lock +++ b/composer.lock @@ -4,7 +4,7 @@ "Read more about it at https://getcomposer.org/doc/01-basic-usage.md#installing-dependencies", "This file is @generated automatically" ], - "content-hash": "56d10a007fc0e676b85e8559984d1867", + "content-hash": "3e51f672a4139d2a840cdcd2e48ccaaf", "packages": [ { "name": "aws/aws-crt-php", @@ -2200,76 +2200,6 @@ }, "time": "2023-11-08T14:08:06+00:00" }, - { - "name": "laravel/socialite", - "version": "v5.12.1", - "source": { - "type": "git", - "url": "https://github.com/laravel/socialite.git", - "reference": "7dae1b072573809f32ab6dcf4aebb57c8b3e8acf" - }, - "dist": { - "type": "zip", - "url": "https://api.github.com/repos/laravel/socialite/zipball/7dae1b072573809f32ab6dcf4aebb57c8b3e8acf", - "reference": "7dae1b072573809f32ab6dcf4aebb57c8b3e8acf", - "shasum": "" - }, - "require": { - "ext-json": "*", - "guzzlehttp/guzzle": "^6.0|^7.0", - "illuminate/contracts": "^6.0|^7.0|^8.0|^9.0|^10.0|^11.0", - "illuminate/http": "^6.0|^7.0|^8.0|^9.0|^10.0|^11.0", - "illuminate/support": "^6.0|^7.0|^8.0|^9.0|^10.0|^11.0", - "league/oauth1-client": "^1.10.1", - "php": "^7.2|^8.0" - }, - "require-dev": { - "mockery/mockery": "^1.0", - "orchestra/testbench": "^4.0|^5.0|^6.0|^7.0|^8.0|^9.0", - "phpstan/phpstan": "^1.10", - "phpunit/phpunit": "^8.0|^9.3|^10.4" - }, - "type": "library", - "extra": { - "branch-alias": { - "dev-master": "5.x-dev" - }, - "laravel": { - "providers": [ - "Laravel\\Socialite\\SocialiteServiceProvider" - ], - "aliases": { - "Socialite": "Laravel\\Socialite\\Facades\\Socialite" - } - } - }, - "autoload": { - "psr-4": { - "Laravel\\Socialite\\": "src/" - } - }, - "notification-url": "https://packagist.org/downloads/", - "license": [ - "MIT" - ], - "authors": [ - { - "name": "Taylor Otwell", - "email": "taylor@laravel.com" - } - ], - "description": "Laravel wrapper around OAuth 1 & OAuth 2 libraries.", - "homepage": "https://laravel.com", - "keywords": [ - "laravel", - "oauth" - ], - "support": { - "issues": "https://github.com/laravel/socialite/issues", - "source": "https://github.com/laravel/socialite" - }, - "time": "2024-02-16T08:58:20+00:00" - }, { "name": "laravel/tinker", "version": "v2.9.0", @@ -2730,82 +2660,6 @@ ], "time": "2024-01-28T23:22:08+00:00" }, - { - "name": "league/oauth1-client", - "version": "v1.10.1", - "source": { - "type": "git", - "url": "https://github.com/thephpleague/oauth1-client.git", - "reference": "d6365b901b5c287dd41f143033315e2f777e1167" - }, - "dist": { - "type": "zip", - "url": "https://api.github.com/repos/thephpleague/oauth1-client/zipball/d6365b901b5c287dd41f143033315e2f777e1167", - "reference": "d6365b901b5c287dd41f143033315e2f777e1167", - "shasum": "" - }, - "require": { - "ext-json": "*", - "ext-openssl": "*", - "guzzlehttp/guzzle": "^6.0|^7.0", - "guzzlehttp/psr7": "^1.7|^2.0", - "php": ">=7.1||>=8.0" - }, - "require-dev": { - "ext-simplexml": "*", - "friendsofphp/php-cs-fixer": "^2.17", - "mockery/mockery": "^1.3.3", - "phpstan/phpstan": "^0.12.42", - "phpunit/phpunit": "^7.5||9.5" - }, - "suggest": { - "ext-simplexml": "For decoding XML-based responses." - }, - "type": "library", - "extra": { - "branch-alias": { - "dev-master": "1.0-dev", - "dev-develop": "2.0-dev" - } - }, - "autoload": { - "psr-4": { - "League\\OAuth1\\Client\\": "src/" - } - }, - "notification-url": "https://packagist.org/downloads/", - "license": [ - "MIT" - ], - "authors": [ - { - "name": "Ben Corlett", - "email": "bencorlett@me.com", - "homepage": "http://www.webcomm.com.au", - "role": "Developer" - } - ], - "description": "OAuth 1.0 Client Library", - "keywords": [ - "Authentication", - "SSO", - "authorization", - "bitbucket", - "identity", - "idp", - "oauth", - "oauth1", - "single sign on", - "trello", - "tumblr", - "twitter" - ], - "support": { - "issues": "https://github.com/thephpleague/oauth1-client/issues", - "source": "https://github.com/thephpleague/oauth1-client/tree/v1.10.1" - }, - "time": "2022-04-15T14:02:14+00:00" - }, { "name": "monolog/monolog", "version": "3.5.0", @@ -9529,5 +9383,5 @@ "ext-ftp": "*" }, "platform-dev": [], - "plugin-api-version": "2.6.0" + "plugin-api-version": "2.2.0" } diff --git a/database/factories/RedirectFactory.php b/database/factories/RedirectFactory.php deleted file mode 100644 index b2ab1cf..0000000 --- a/database/factories/RedirectFactory.php +++ /dev/null @@ -1,25 +0,0 @@ - $this->faker->randomNumber(), - 'mode' => $this->faker->randomNumber(), - 'from' => $this->faker->word(), - 'to' => $this->faker->word(), - 'status' => $this->faker->word(), - 'created_at' => Carbon::now(), - 'updated_at' => Carbon::now(), - ]; - } -} diff --git a/database/factories/ScriptExecutionFactory.php b/database/factories/ScriptExecutionFactory.php deleted file mode 100644 index 7140f1a..0000000 --- a/database/factories/ScriptExecutionFactory.php +++ /dev/null @@ -1,26 +0,0 @@ - $this->faker->word(), - 'finished_at' => Carbon::now(), - 'created_at' => Carbon::now(), - 'updated_at' => Carbon::now(), - 'script_id' => Script::factory(), - 'server_id' => Server::factory(), - ]; - } -} diff --git a/database/factories/ScriptFactory.php b/database/factories/ScriptFactory.php deleted file mode 100644 index 983669a..0000000 --- a/database/factories/ScriptFactory.php +++ /dev/null @@ -1,15 +0,0 @@ -|<=|=>|=|!=" }, { token: "paren.lparen", regex: "[\\[\\(\\{]" }, { token: "paren.rparen", regex: "[\\]\\)\\}]" }, { token: "text", regex: "\\s+" }] } }; r.inherits(u, i), t.ShHighlightRules = u }) diff --git a/public/static/libs/ace/theme-github.js b/public/static/libs/ace/theme-github.js new file mode 100644 index 0000000..e5dd93d --- /dev/null +++ b/public/static/libs/ace/theme-github.js @@ -0,0 +1,5 @@ +ace.define("ace/theme/github", ["require", "exports", "module", "ace/lib/dom"], function (e, t, n) { + t.isDark = !1, t.cssClass = "ace-github", t.cssText = '/* CSS style content from github\'s default pygments highlighter template.Cursor and selection styles from textmate.css. */.ace-github .ace_gutter {background: #e8e8e8;color: #AAA;}.ace-github .ace_scroller {background: #fff;}.ace-github .ace_keyword {font-weight: bold;}.ace-github .ace_string {color: #D14;}.ace-github .ace_variable.ace_class {color: teal;}.ace-github .ace_constant.ace_numeric {color: #099;}.ace-github .ace_constant.ace_buildin {color: #0086B3;}.ace-github .ace_support.ace_function {color: #0086B3;}.ace-github .ace_comment {color: #998;font-style: italic;}.ace-github .ace_variable.ace_language {color: #0086B3;}.ace-github .ace_paren {font-weight: bold;}.ace-github .ace_boolean {font-weight: bold;}.ace-github .ace_string.ace_regexp {color: #009926;font-weight: normal;}.ace-github .ace_variable.ace_instance {color: teal;}.ace-github .ace_constant.ace_language {font-weight: bold;}.ace-github .ace_text-layer {}.ace-github .ace_cursor {border-left: 2px solid black;}.ace-github .ace_overwrite-cursors .ace_cursor {border-left: 0px;border-bottom: 1px solid black;}.ace-github .ace_marker-layer .ace_active-line {background: rgb(255, 255, 204);}.ace-github .ace_marker-layer .ace_selection {background: rgb(181, 213, 255);}.ace-github.ace_multiselect .ace_selection.ace_start {box-shadow: 0 0 3px 0px white;border-radius: 2px;}/* bold keywords cause cursor issues for some fonts *//* this disables bold style for editor and keeps for static highlighter */.ace-github.ace_nobold .ace_line > span {font-weight: normal !important;}.ace-github .ace_marker-layer .ace_step {background: rgb(252, 255, 0);}.ace-github .ace_marker-layer .ace_stack {background: rgb(164, 229, 101);}.ace-github .ace_marker-layer .ace_bracket {margin: -1px 0 0 -1px;border: 1px solid rgb(192, 192, 192);}.ace-github .ace_gutter-active-line {background-color : rgba(0, 0, 0, 0.07);}.ace-github .ace_marker-layer .ace_selected-word {background: rgb(250, 250, 255);border: 1px solid rgb(200, 200, 250);}.ace-github .ace_print-margin {width: 1px;background: #e8e8e8;}.ace-github .ace_indent-guide {background: url("") right repeat-y;}'; + var r = e("../lib/dom"); + r.importCssString(t.cssText, t.cssClass) +}) diff --git a/public/static/libs/ace/theme-one-dark.js b/public/static/libs/ace/theme-one-dark.js new file mode 100644 index 0000000..eb2ec1d --- /dev/null +++ b/public/static/libs/ace/theme-one-dark.js @@ -0,0 +1,5 @@ +ace.define("ace/theme/one-dark", ["require", "exports", "module", "ace/lib/dom"], function (e, t, n) { + t.isDark = !1, t.cssClass = "ace-one-dark", t.cssText = '/* CSS style content from one-dark\'s default pygments highlighter template.Cursor and selection styles from textmate.css. */.ace-one-dark .ace_gutter{background:#282c34;color:#6a6f7a}.ace-one-dark .ace_print-margin{width:1px;background:#e8e8e8}.ace-one-dark{background-color:#282c34;color:#abb2bf}.ace-one-dark .ace_cursor{color:#528bff}.ace-one-dark .ace_marker-layer .ace_selection{background:#3d4350}.ace-one-dark.ace_multiselect .ace_selection.ace_start{box-shadow:0 0 3px 0 #282c34;border-radius:2px}.ace-one-dark .ace_marker-layer .ace_step{background:#c6dbae}.ace-one-dark .ace_marker-layer .ace_bracket{margin:-1px 0 0 -1px;border:1px solid #747369}.ace-one-dark .ace_marker-layer .ace_active-line{background:rgba(76,87,103,.19)}.ace-one-dark .ace_gutter-active-line{background-color:rgba(76,87,103,.19)}.ace-one-dark .ace_marker-layer .ace_selected-word{border:1px solid #3d4350}.ace-one-dark .ace_fold{background-color:#61afef;border-color:#abb2bf}.ace-one-dark .ace_keyword{color:#c678dd}.ace-one-dark .ace_keyword.ace_operator{color:#c678dd}.ace-one-dark .ace_keyword.ace_other.ace_unit{color:#d19a66}.ace-one-dark .ace_constant.ace_language{color:#d19a66}.ace-one-dark .ace_constant.ace_numeric{color:#d19a66}.ace-one-dark .ace_constant.ace_character{color:#56b6c2}.ace-one-dark .ace_constant.ace_other{color:#56b6c2}.ace-one-dark .ace_support.ace_function{color:#61afef}.ace-one-dark .ace_support.ace_constant{color:#d19a66}.ace-one-dark .ace_support.ace_class{color:#e5c07b}.ace-one-dark .ace_support.ace_type{color:#e5c07b}.ace-one-dark .ace_storage{color:#c678dd}.ace-one-dark .ace_storage.ace_type{color:#c678dd}.ace-one-dark .ace_invalid{color:#fff;background-color:#f2777a}.ace-one-dark .ace_invalid.ace_deprecated{color:#272b33;background-color:#d27b53}.ace-one-dark .ace_string{color:#98c379}.ace-one-dark .ace_string.ace_regexp{color:#e06c75}.ace-one-dark .ace_comment{font-style:italic;color:#5c6370}.ace-one-dark .ace_variable{color:#e06c75}.ace-one-dark .ace_variable.ace_parameter{color:#d19a66}.ace-one-dark .ace_meta.ace_tag{color:#e06c75}.ace-one-dark .ace_entity.ace_other.ace_attribute-name{color:#e06c75}.ace-one-dark .ace_entity.ace_name.ace_function{color:#61afef}.ace-one-dark .ace_entity.ace_name.ace_tag{color:#e06c75}.ace-one-dark .ace_markup.ace_heading{color:#98c379}.ace-one-dark .ace_indent-guide{background:url() right repeat-y}'; + var r = e("../lib/dom"); + r.importCssString(t.cssText, t.cssClass) +}) diff --git a/resources/views/databases/partials/database-backups.blade.php b/resources/views/databases/partials/database-backups.blade.php index 5eaf828..c2c650e 100644 --- a/resources/views/databases/partials/database-backups.blade.php +++ b/resources/views/databases/partials/database-backups.blade.php @@ -16,6 +16,7 @@