- 2.x - scripts

This commit is contained in:
Saeed Vaziry 2024-10-06 20:49:59 +02:00
parent c24b4b7333
commit 8b2338cb41
32 changed files with 936 additions and 79 deletions

View File

@ -4,7 +4,7 @@
use App\Models\NotificationChannel; use App\Models\NotificationChannel;
use App\Models\User; use App\Models\User;
use Illuminate\Support\Facades\Validator; use Illuminate\Validation\Rule;
use Illuminate\Validation\ValidationException; use Illuminate\Validation\ValidationException;
class AddChannel class AddChannel
@ -14,14 +14,12 @@ class AddChannel
*/ */
public function add(User $user, array $input): void public function add(User $user, array $input): void
{ {
$this->validate($input);
$channel = new NotificationChannel([ $channel = new NotificationChannel([
'user_id' => $user->id, 'user_id' => $user->id,
'provider' => $input['provider'], 'provider' => $input['provider'],
'label' => $input['label'], 'label' => $input['label'],
'project_id' => isset($input['global']) && $input['global'] ? null : $user->current_project_id, 'project_id' => isset($input['global']) && $input['global'] ? null : $user->current_project_id,
]); ]);
$this->validateType($channel, $input);
$channel->data = $channel->provider()->createData($input); $channel->data = $channel->provider()->createData($input);
$channel->save(); $channel->save();
@ -43,23 +41,29 @@ public function add(User $user, array $input): void
$channel->save(); $channel->save();
} }
/** public static function rules(array $input): array
* @throws ValidationException
*/
protected function validate(array $input): void
{ {
Validator::make($input, [ $rules = [
'provider' => 'required|in:'.implode(',', config('core.notification_channels_providers')), 'provider' => [
'required',
Rule::in(config('core.notification_channels_providers')),
],
'label' => 'required', 'label' => 'required',
])->validate(); ];
return array_merge($rules, static::providerRules($input));
} }
/** private static function providerRules(array $input): array
* @throws ValidationException
*/
protected function validateType(NotificationChannel $channel, array $input): void
{ {
Validator::make($input, $channel->provider()->createRules($input)) if (! isset($input['provider'])) {
->validate(); return [];
}
$notificationChannel = new NotificationChannel([
'provider' => $input['provider'],
]);
return $notificationChannel->provider()->createRules($input);
} }
} }

View File

@ -4,31 +4,22 @@
use App\Models\NotificationChannel; use App\Models\NotificationChannel;
use App\Models\User; use App\Models\User;
use Illuminate\Support\Facades\Validator;
use Illuminate\Validation\ValidationException;
class EditChannel class EditChannel
{ {
public function edit(NotificationChannel $notificationChannel, User $user, array $input): void public function edit(NotificationChannel $notificationChannel, User $user, array $input): void
{ {
$this->validate($input); $notificationChannel->fill([
'label' => $input['label'],
$notificationChannel->label = $input['label']; 'project_id' => isset($input['global']) && $input['global'] ? null : $user->current_project_id,
$notificationChannel->project_id = isset($input['global']) && $input['global'] ? null : $user->current_project_id; ]);
$notificationChannel->save(); $notificationChannel->save();
} }
/** public static function rules(array $input): array
* @throws ValidationException
*/
private function validate(array $input): void
{ {
$rules = [ return [
'label' => [ 'label' => 'required',
'required',
],
]; ];
Validator::make($input, $rules)->validate();
} }
} }

View File

@ -4,29 +4,27 @@
use App\Models\Script; use App\Models\Script;
use App\Models\User; use App\Models\User;
use Illuminate\Support\Facades\Validator;
class CreateScript class CreateScript
{ {
public function create(User $user, array $input): Script public function create(User $user, array $input): Script
{ {
$this->validate($input);
$script = new Script([ $script = new Script([
'user_id' => $user->id, 'user_id' => $user->id,
'name' => $input['name'], 'name' => $input['name'],
'content' => $input['content'], 'content' => $input['content'],
'project_id' => isset($input['global']) && $input['global'] ? null : $user->current_project_id,
]); ]);
$script->save(); $script->save();
return $script; return $script;
} }
private function validate(array $input): void public static function rules(): array
{ {
Validator::make($input, [ return [
'name' => ['required', 'string', 'max:255'], 'name' => ['required', 'string', 'max:255'],
'content' => ['required', 'string'], 'content' => ['required', 'string'],
])->validate(); ];
} }
} }

View File

@ -3,26 +3,26 @@
namespace App\Actions\Script; namespace App\Actions\Script;
use App\Models\Script; use App\Models\Script;
use Illuminate\Support\Facades\Validator; use App\Models\User;
class EditScript class EditScript
{ {
public function edit(Script $script, array $input): Script public function edit(Script $script, User $user, array $input): Script
{ {
$this->validate($input);
$script->name = $input['name']; $script->name = $input['name'];
$script->content = $input['content']; $script->content = $input['content'];
$script->project_id = isset($input['global']) && $input['global'] ? null : $user->current_project_id;
$script->save(); $script->save();
return $script; return $script;
} }
private function validate(array $input): void public static function rules(): array
{ {
Validator::make($input, [ return [
'name' => ['required', 'string', 'max:255'], 'name' => ['required', 'string', 'max:255'],
'content' => ['required', 'string'], 'content' => ['required', 'string'],
])->validate(); ];
} }
} }

View File

@ -7,30 +7,28 @@
use App\Models\ScriptExecution; use App\Models\ScriptExecution;
use App\Models\Server; use App\Models\Server;
use App\Models\ServerLog; use App\Models\ServerLog;
use Illuminate\Support\Facades\Validator;
use Illuminate\Validation\Rule; use Illuminate\Validation\Rule;
class ExecuteScript class ExecuteScript
{ {
public function execute(Script $script, Server $server, array $input): ScriptExecution public function execute(Script $script, array $input): ScriptExecution
{ {
$this->validate($server, $input);
$execution = new ScriptExecution([ $execution = new ScriptExecution([
'script_id' => $script->id, 'script_id' => $script->id,
'server_id' => $input['server'],
'user' => $input['user'], 'user' => $input['user'],
'variables' => $input['variables'] ?? [], 'variables' => $input['variables'] ?? [],
'status' => ScriptExecutionStatus::EXECUTING, 'status' => ScriptExecutionStatus::EXECUTING,
]); ]);
$execution->save(); $execution->save();
dispatch(function () use ($execution, $server, $script) { dispatch(function () use ($execution, $script) {
$content = $execution->getContent(); $content = $execution->getContent();
$log = ServerLog::make($server, 'script-'.$script->id.'-'.strtotime('now')); $log = ServerLog::make($execution->server, 'script-'.$script->id.'-'.strtotime('now'));
$log->save(); $log->save();
$execution->server_log_id = $log->id; $execution->server_log_id = $log->id;
$execution->save(); $execution->save();
$server->os()->runScript('~/', $content, $log, $execution->user); $execution->server->os()->runScript('~/', $content, $log, $execution->user);
$execution->status = ScriptExecutionStatus::COMPLETED; $execution->status = ScriptExecutionStatus::COMPLETED;
$execution->save(); $execution->save();
})->catch(function () use ($execution) { })->catch(function () use ($execution) {
@ -41,14 +39,23 @@ public function execute(Script $script, Server $server, array $input): ScriptExe
return $execution; return $execution;
} }
private function validate(Server $server, array $input): void public static function rules(array $input): array
{ {
Validator::make($input, [ if (isset($input['server'])) {
/** @var ?Server $server */
$server = Server::query()->find($input['server']);
}
return [
'server' => [
'required',
Rule::exists('servers', 'id'),
],
'user' => [ 'user' => [
'required', 'required',
Rule::in([ Rule::in([
'root', 'root',
$server->ssh_user, isset($server) ? $server?->ssh_user : null,
]), ]),
], ],
'variables' => 'array', 'variables' => 'array',
@ -57,6 +64,6 @@ private function validate(Server $server, array $input): void
'string', 'string',
'max:255', 'max:255',
], ],
])->validate(); ];
} }
} }

View File

@ -3,6 +3,7 @@
namespace App\Models; namespace App\Models;
use Carbon\Carbon; use Carbon\Carbon;
use Illuminate\Database\Eloquent\Builder;
use Illuminate\Database\Eloquent\Factories\HasFactory; use Illuminate\Database\Eloquent\Factories\HasFactory;
use Illuminate\Database\Eloquent\Relations\BelongsTo; use Illuminate\Database\Eloquent\Relations\BelongsTo;
use Illuminate\Database\Eloquent\Relations\HasMany; use Illuminate\Database\Eloquent\Relations\HasMany;
@ -18,6 +19,9 @@
* @property Carbon $updated_at * @property Carbon $updated_at
* @property Collection<ScriptExecution> $executions * @property Collection<ScriptExecution> $executions
* @property ?ScriptExecution $lastExecution * @property ?ScriptExecution $lastExecution
* @property User $user
* @property int $project_id
* @property ?Project $project
*/ */
class Script extends AbstractModel class Script extends AbstractModel
{ {
@ -27,6 +31,12 @@ class Script extends AbstractModel
'user_id', 'user_id',
'name', 'name',
'content', 'content',
'project_id',
];
protected $casts = [
'user_id' => 'int',
'project_id' => 'int',
]; ];
public static function boot(): void public static function boot(): void
@ -43,6 +53,11 @@ public function user(): BelongsTo
return $this->belongsTo(User::class); return $this->belongsTo(User::class);
} }
public function project(): BelongsTo
{
return $this->belongsTo(Project::class);
}
public function getVariables(): array public function getVariables(): array
{ {
$variables = []; $variables = [];
@ -63,4 +78,14 @@ public function lastExecution(): HasOne
{ {
return $this->hasOne(ScriptExecution::class)->latest(); return $this->hasOne(ScriptExecution::class)->latest();
} }
public static function getByProjectId(int $projectId, int $userId): Builder
{
return self::query()
->where(function (Builder $query) use ($projectId, $userId) {
$query->where('project_id', $projectId)
->orWhere('user_id', $userId)
->orWhereNull('project_id');
});
}
} }

View File

@ -2,15 +2,16 @@
namespace App\Models; namespace App\Models;
use App\Enums\ScriptExecutionStatus;
use Carbon\Carbon; use Carbon\Carbon;
use Illuminate\Database\Eloquent\Factories\HasFactory; use Illuminate\Database\Eloquent\Factories\HasFactory;
use Illuminate\Database\Eloquent\Model;
use Illuminate\Database\Eloquent\Relations\BelongsTo; use Illuminate\Database\Eloquent\Relations\BelongsTo;
/** /**
* @property int $id * @property int $id
* @property int $script_id * @property int $script_id
* @property int $server_log_id * @property int $server_log_id
* @property ?int $server_id
* @property string $user * @property string $user
* @property array $variables * @property array $variables
* @property string $status * @property string $status
@ -18,13 +19,15 @@
* @property Carbon $updated_at * @property Carbon $updated_at
* @property Script $script * @property Script $script
* @property ?ServerLog $serverLog * @property ?ServerLog $serverLog
* @property ?Server $server
*/ */
class ScriptExecution extends Model class ScriptExecution extends AbstractModel
{ {
use HasFactory; use HasFactory;
protected $fillable = [ protected $fillable = [
'script_id', 'script_id',
'server_id',
'server_log_id', 'server_log_id',
'user', 'user',
'variables', 'variables',
@ -33,10 +36,17 @@ class ScriptExecution extends Model
protected $casts = [ protected $casts = [
'script_id' => 'integer', 'script_id' => 'integer',
'server_id' => 'integer',
'server_log_id' => 'integer', 'server_log_id' => 'integer',
'variables' => 'array', 'variables' => 'array',
]; ];
public static array $statusColors = [
ScriptExecutionStatus::EXECUTING => 'warning',
ScriptExecutionStatus::COMPLETED => 'success',
ScriptExecutionStatus::FAILED => 'danger',
];
public function script(): BelongsTo public function script(): BelongsTo
{ {
return $this->belongsTo(Script::class); return $this->belongsTo(Script::class);
@ -58,4 +68,22 @@ public function serverLog(): BelongsTo
{ {
return $this->belongsTo(ServerLog::class); return $this->belongsTo(ServerLog::class);
} }
public function server(): BelongsTo
{
return $this->belongsTo(Server::class);
}
public function getServer(): ?Server
{
if ($this->server_id) {
return $this->server;
}
if ($this->server_log_id) {
return $this->serverLog?->server;
}
return null;
}
} }

View File

@ -5,6 +5,7 @@
use Illuminate\Database\Eloquent\Factories\HasFactory; use Illuminate\Database\Eloquent\Factories\HasFactory;
use Illuminate\Database\Eloquent\Relations\BelongsTo; use Illuminate\Database\Eloquent\Relations\BelongsTo;
use Illuminate\Database\Eloquent\Relations\BelongsToMany; use Illuminate\Database\Eloquent\Relations\BelongsToMany;
use Illuminate\Database\Eloquent\SoftDeletes;
/** /**
* @property int $user_id * @property int $user_id
@ -16,6 +17,7 @@
class SshKey extends AbstractModel class SshKey extends AbstractModel
{ {
use HasFactory; use HasFactory;
use SoftDeletes;
protected $fillable = [ protected $fillable = [
'user_id', 'user_id',

View File

@ -11,7 +11,10 @@ class Discord extends AbstractNotificationChannel
public function createRules(array $input): array public function createRules(array $input): array
{ {
return [ return [
'webhook_url' => 'required|url', 'webhook_url' => [
'required',
'url',
],
]; ];
} }

View File

@ -13,7 +13,10 @@ class Email extends AbstractNotificationChannel
public function createRules(array $input): array public function createRules(array $input): array
{ {
return [ return [
'email' => 'required|email', 'email' => [
'required',
'email',
],
]; ];
} }

View File

@ -11,7 +11,10 @@ class Slack extends AbstractNotificationChannel
public function createRules(array $input): array public function createRules(array $input): array
{ {
return [ return [
'webhook_url' => 'required|url', 'webhook_url' => [
'required',
'url',
],
]; ];
} }

View File

@ -14,8 +14,12 @@ class Telegram extends AbstractNotificationChannel
public function createRules(array $input): array public function createRules(array $input): array
{ {
return [ return [
'bot_token' => 'required|string', 'bot_token' => [
'chat_id' => 'required', 'required',
],
'chat_id' => [
'required',
],
]; ];
} }

View File

@ -0,0 +1,37 @@
<?php
namespace App\Policies;
use App\Models\NotificationChannel;
use App\Models\User;
use Illuminate\Auth\Access\HandlesAuthorization;
class NotificationChannelPolicy
{
use HandlesAuthorization;
public function viewAny(User $user): bool
{
return $user->isAdmin();
}
public function view(User $user, NotificationChannel $notificationChannel): bool
{
return $user->isAdmin();
}
public function create(User $user): bool
{
return $user->isAdmin();
}
public function update(User $user, NotificationChannel $notificationChannel): bool
{
return $user->isAdmin();
}
public function delete(User $user, NotificationChannel $notificationChannel): bool
{
return $user->isAdmin();
}
}

View File

@ -18,7 +18,7 @@ public function viewAny(User $user): bool
public function view(User $user, Script $script): bool public function view(User $user, Script $script): bool
{ {
return $user->id === $script->user_id; return $user->id === $script->user_id || $script->project?->users?->contains($user);
} }
public function create(User $user): bool public function create(User $user): bool

View File

@ -104,6 +104,7 @@ public function panel(Panel $panel): Panel
Authenticate::class, Authenticate::class,
]) ])
->login() ->login()
->spa()
->globalSearchKeyBindings(['command+k', 'ctrl+k']) ->globalSearchKeyBindings(['command+k', 'ctrl+k'])
->sidebarCollapsibleOnDesktop() ->sidebarCollapsibleOnDesktop()
->globalSearchFieldKeyBindingSuffix(); ->globalSearchFieldKeyBindingSuffix();

View File

@ -4,6 +4,8 @@
use App\Helpers\HtmxResponse; use App\Helpers\HtmxResponse;
use Filament\Notifications\Actions\Action; use Filament\Notifications\Actions\Action;
use Filament\Notifications\Notification; use Filament\Notifications\Notification;
use Illuminate\Http\Request;
use Illuminate\Support\Facades\Route;
function generate_public_key($privateKeyPath, $publicKeyPath): void function generate_public_key($privateKeyPath, $publicKeyPath): void
{ {
@ -146,3 +148,22 @@ function tail($filepath, $lines = 1, $adaptive = true): string
return trim($output); return trim($output);
} }
function get_from_route(string $modelName, string $routeKey): mixed
{
$model = request()->route($routeKey);
if (! $model) {
$model = Route::getRoutes()->match(Request::create(url()->previous()))->parameter($routeKey);
}
if ($model instanceof $modelName) {
return $model;
}
if ($model) {
return $modelName::query()->find($model);
}
return null;
}

View File

@ -0,0 +1,86 @@
<?php
namespace App\Web\Pages\Scripts;
use App\Actions\Script\ExecuteScript;
use App\Models\Script;
use App\Models\Server;
use App\Web\Components\Page;
use Filament\Actions\Action;
use Filament\Forms\Components\Select;
use Filament\Forms\Components\TextInput;
use Filament\Forms\Get;
use Filament\Support\Enums\MaxWidth;
use Illuminate\Contracts\Support\Htmlable;
class Executions extends Page
{
protected static bool $shouldRegisterNavigation = false;
protected static ?string $slug = 'scripts/{script}/executions';
public Script $script;
public function getTitle(): string|Htmlable
{
return $this->script->name.' - Executions';
}
public static function canAccess(): bool
{
return auth()->user()?->can('view', get_from_route(Script::class, 'script')) ?? false;
}
public function getWidgets(): array
{
return [
[Widgets\ScriptExecutionsList::class, ['script' => $this->script]],
];
}
protected function getHeaderActions(): array
{
$form = [
Select::make('server')
->options(function () {
return auth()->user()?->currentProject?->servers?->pluck('name', 'id') ?? [];
})
->rules(fn (Get $get) => ExecuteScript::rules($get())['server'])
->searchable()
->reactive(),
Select::make('user')
->rules(fn (Get $get) => ExecuteScript::rules($get())['user'])
->native(false)
->options(function (Get $get) {
$options = [
'root' => 'root',
];
$server = Server::query()->find($get('server'));
if ($server) {
$options[$server->ssh_user] = $server->ssh_user;
}
return $options;
}),
];
foreach ($this->script->getVariables() as $variable) {
$form[] = TextInput::make($variable)
->label($variable)
->rules(fn (Get $get) => ExecuteScript::rules($get())['variables.*']);
}
return [
Action::make('execute')
->icon('heroicon-o-bolt')
->modalWidth(MaxWidth::Large)
->form($form)
->action(function (array $data) {
app(ExecuteScript::class)->execute($this->script, $data);
$this->dispatch('$refresh');
}),
];
}
}

View File

@ -0,0 +1,74 @@
<?php
namespace App\Web\Pages\Scripts;
use App\Actions\Script\CreateScript;
use App\Models\Script;
use App\Web\Components\Page;
use App\Web\Fields\CodeEditorField;
use Filament\Actions\Action;
use Filament\Forms\Components\Checkbox;
use Filament\Forms\Components\TextInput;
use Filament\Support\Enums\MaxWidth;
class Index extends Page
{
protected static ?string $slug = 'scripts';
protected static ?string $navigationIcon = 'heroicon-o-bolt';
protected static ?int $navigationSort = 2;
protected static ?string $title = 'Scripts';
public static function getNavigationItemActiveRoutePattern(): string
{
return static::getRouteName().'*';
}
public static function canAccess(): bool
{
return auth()->user()?->can('viewAny', Script::class) ?? false;
}
public function getWidgets(): array
{
return [
[Widgets\ScriptsList::class],
];
}
protected function getHeaderActions(): array
{
return [
Action::make('read-the-docs')
->label('Read the Docs')
->icon('heroicon-o-document-text')
->color('gray')
->url('https://vitodeploy.com/other/scripts.html')
->openUrlInNewTab(),
Action::make('create')
->label('Create a Script')
->icon('heroicon-o-plus')
->authorize('create', Script::class)
->modalWidth(MaxWidth::ThreeExtraLarge)
->form([
TextInput::make('name')
->rules(CreateScript::rules()['name']),
CodeEditorField::make('content')
->rules(CreateScript::rules()['content'])
->helperText('You can use variables like ${VARIABLE_NAME} in the script. The variables will be asked when executing the script'),
Checkbox::make('global')
->label('Is Global (Accessible in all projects)'),
])
->modalSubmitActionLabel('Create')
->action(function (array $data) {
run_action($this, function () use ($data) {
app(CreateScript::class)->create(auth()->user(), $data);
$this->dispatch('$refresh');
});
}),
];
}
}

View File

@ -0,0 +1,79 @@
<?php
namespace App\Web\Pages\Scripts\Widgets;
use App\Models\Script;
use App\Models\ScriptExecution;
use App\Web\Pages\Servers\View;
use Filament\Tables\Actions\Action;
use Filament\Tables\Columns\TextColumn;
use Filament\Tables\Table;
use Filament\Widgets\TableWidget as Widget;
use Illuminate\Database\Eloquent\Builder;
use Illuminate\View\ComponentAttributeBag;
class ScriptExecutionsList extends Widget
{
protected $listeners = ['$refresh'];
public Script $script;
protected function getTableQuery(): Builder
{
return ScriptExecution::query()->where('script_id', $this->script->id);
}
protected function applyDefaultSortingToTableQuery(Builder $query): Builder
{
return $query->latest('created_at');
}
protected function getTableColumns(): array
{
return [
TextColumn::make('server')
->formatStateUsing(function (ScriptExecution $record) {
return $record->getServer()?->name ?? 'Unknown';
})
->url(function (ScriptExecution $record) {
$server = $record->getServer();
return $server ? View::getUrl(['server' => $server]) : null;
})
->searchable()
->sortable(),
TextColumn::make('created_at')
->label('Executed At')
->formatStateUsing(fn (ScriptExecution $record) => $record->created_at_by_timezone)
->searchable()
->sortable(),
TextColumn::make('status')
->label('Status')
->badge()
->color(fn (ScriptExecution $record) => ScriptExecution::$statusColors[$record->status])
->sortable(),
];
}
public function getTable(): Table
{
return $this->table
->heading('')
->actions([
Action::make('view')
->hiddenLabel()
->tooltip('View')
->icon('heroicon-o-eye')
->authorize(fn (ScriptExecution $record) => auth()->user()->can('view', $record->serverLog))
->modalHeading('View Log')
->modalContent(function (ScriptExecution $record) {
return view('components.console-view', [
'slot' => $record->serverLog?->getContent(),
'attributes' => new ComponentAttributeBag,
]);
})
->modalSubmitAction(false)
->modalCancelActionLabel('Close'),
]);
}
}

View File

@ -0,0 +1,89 @@
<?php
namespace App\Web\Pages\Scripts\Widgets;
use App\Actions\Script\EditScript;
use App\Models\Script;
use App\Web\Fields\CodeEditorField;
use App\Web\Pages\Scripts\Executions;
use Filament\Forms\Components\Checkbox;
use Filament\Forms\Components\TextInput;
use Filament\Support\Enums\MaxWidth;
use Filament\Tables\Actions\DeleteAction;
use Filament\Tables\Actions\EditAction;
use Filament\Tables\Columns\TextColumn;
use Filament\Tables\Table;
use Filament\Widgets\TableWidget as Widget;
use Illuminate\Database\Eloquent\Builder;
class ScriptsList extends Widget
{
protected $listeners = ['$refresh'];
protected function getTableQuery(): Builder
{
return Script::getByProjectId(auth()->user()->current_project_id, auth()->user()->id);
}
protected function getTableColumns(): array
{
return [
TextColumn::make('name')
->searchable()
->sortable(),
TextColumn::make('id')
->label('Global')
->badge()
->color(fn ($record) => $record->project_id ? 'gray' : 'success')
->formatStateUsing(function (Script $record) {
return $record->project_id ? 'No' : 'Yes';
}),
TextColumn::make('created_at')
->label('Created At')
->formatStateUsing(fn (Script $record) => $record->created_at_by_timezone)
->searchable()
->sortable(),
];
}
public function getTable(): Table
{
return $this->table
->heading('')
->recordUrl(fn (Script $record) => Executions::getUrl(['script' => $record]))
->actions([
EditAction::make('edit')
->label('Edit')
->modalHeading('Edit Script')
->mutateRecordDataUsing(function (array $data, Script $record) {
return [
'name' => $record->name,
'content' => $record->content,
'global' => $record->project_id === null,
];
})
->form([
TextInput::make('name')
->rules(EditScript::rules()['name']),
CodeEditorField::make('content')
->rules(EditScript::rules()['content'])
->helperText('You can use variables like ${VARIABLE_NAME} in the script. The variables will be asked when executing the script'),
Checkbox::make('global')
->label('Is Global (Accessible in all projects)'),
])
->authorize(fn (Script $record) => auth()->user()->can('update', $record))
->using(function (array $data, Script $record) {
app(EditScript::class)->edit($record, auth()->user(), $data);
$this->dispatch('$refresh');
})
->modalWidth(MaxWidth::ThreeExtraLarge),
DeleteAction::make('delete')
->label('Delete')
->modalHeading('Delete Script')
->authorize(fn (Script $record) => auth()->user()->can('delete', $record))
->using(function (array $data, Script $record) {
$record->delete();
}),
]);
}
}

View File

@ -7,7 +7,7 @@
use App\Models\SshKey; use App\Models\SshKey;
use Exception; use Exception;
use Filament\Notifications\Notification; use Filament\Notifications\Notification;
use Filament\Tables\Actions\Action; use Filament\Tables\Actions\DeleteAction;
use Filament\Tables\Columns\TextColumn; use Filament\Tables\Columns\TextColumn;
use Filament\Tables\Table; use Filament\Tables\Table;
use Filament\Widgets\TableWidget; use Filament\Widgets\TableWidget;
@ -21,7 +21,8 @@ class SshKeysList extends TableWidget
protected function getTableQuery(): Builder protected function getTableQuery(): Builder
{ {
return SshKey::query()->whereHas( return SshKey::withTrashed()
->whereHas(
'servers', 'servers',
fn (Builder $query) => $query->where('server_id', $this->server->id) fn (Builder $query) => $query->where('server_id', $this->server->id)
); );
@ -47,14 +48,10 @@ public function getTable(): Table
{ {
return $this->table return $this->table
->actions([ ->actions([
Action::make('delete') DeleteAction::make('delete')
->icon('heroicon-o-trash')
->tooltip('Delete')
->color('danger')
->hiddenLabel() ->hiddenLabel()
->requiresConfirmation()
->authorize(fn (SshKey $record) => auth()->user()->can('deleteServer', [SshKey::class, $this->server])) ->authorize(fn (SshKey $record) => auth()->user()->can('deleteServer', [SshKey::class, $this->server]))
->action(function (SshKey $record) { ->using(function (SshKey $record) {
try { try {
app(DeleteKeyFromServer::class)->delete($this->server, $record); app(DeleteKeyFromServer::class)->delete($this->server, $record);
} catch (Exception $e) { } catch (Exception $e) {

View File

@ -0,0 +1,65 @@
<?php
namespace App\Web\Pages\Settings\NotificationChannels\Actions;
use App\Actions\NotificationChannels\AddChannel;
use Exception;
use Filament\Forms\Components\Checkbox;
use Filament\Forms\Components\Select;
use Filament\Forms\Components\TextInput;
use Filament\Forms\Get;
use Filament\Notifications\Notification;
class Create
{
public static function form(): array
{
return [
Select::make('provider')
->options(
collect(config('core.notification_channels_providers'))
->mapWithKeys(fn ($provider) => [$provider => $provider])
)
->live()
->reactive()
->rules(fn (Get $get) => AddChannel::rules($get())['provider']),
TextInput::make('label')
->rules(fn (Get $get) => AddChannel::rules($get())['label']),
TextInput::make('webhook_url')
->label('Webhook URL')
->validationAttribute('Webhook URL')
->visible(fn (Get $get) => AddChannel::rules($get())['webhook_url'] ?? false)
->rules(fn (Get $get) => AddChannel::rules($get())['webhook_url']),
TextInput::make('email')
->visible(fn (Get $get) => AddChannel::rules($get())['email'] ?? false)
->rules(fn (Get $get) => AddChannel::rules($get())['email']),
TextInput::make('bot_token')
->label('Bot Token')
->visible(fn (Get $get) => AddChannel::rules($get())['bot_token'] ?? false)
->rules(fn (Get $get) => AddChannel::rules($get())['bot_token']),
TextInput::make('chat_id')
->label('Chat ID')
->visible(fn (Get $get) => AddChannel::rules($get())['chat_id'] ?? false)
->rules(fn (Get $get) => AddChannel::rules($get())['chat_id']),
Checkbox::make('global')
->label('Is Global (Accessible in all projects)'),
];
}
/**
* @throws Exception
*/
public static function action(array $data): void
{
try {
app(AddChannel::class)->add(auth()->user(), $data);
} catch (Exception $e) {
Notification::make()
->title($e->getMessage())
->danger()
->send();
throw $e;
}
}
}

View File

@ -0,0 +1,27 @@
<?php
namespace App\Web\Pages\Settings\NotificationChannels\Actions;
use App\Actions\NotificationChannels\EditChannel;
use App\Models\NotificationChannel;
use Filament\Forms\Components\Checkbox;
use Filament\Forms\Components\TextInput;
use Filament\Forms\Get;
class Edit
{
public static function form(): array
{
return [
TextInput::make('label')
->rules(fn (Get $get) => EditChannel::rules($get())['label']),
Checkbox::make('global')
->label('Is Global (Accessible in all projects)'),
];
}
public static function action(NotificationChannel $channel, array $data): void
{
app(EditChannel::class)->edit($channel, auth()->user(), $data);
}
}

View File

@ -0,0 +1,52 @@
<?php
namespace App\Web\Pages\Settings\NotificationChannels;
use App\Models\NotificationChannel;
use App\Web\Components\Page;
use Filament\Actions\Action;
use Filament\Support\Enums\MaxWidth;
class Index extends Page
{
protected static ?string $navigationGroup = 'Settings';
protected static ?string $slug = 'settings/notification-channels';
protected static ?string $title = 'Notification Channels';
protected static ?string $navigationIcon = 'heroicon-o-bell';
protected static ?int $navigationSort = 8;
public static function canAccess(): bool
{
return auth()->user()?->can('viewAny', NotificationChannel::class) ?? false;
}
public function getWidgets(): array
{
return [
[Widgets\NotificationChannelsList::class],
];
}
protected function getHeaderActions(): array
{
return [
Action::make('add')
->label('Add new Channel')
->icon('heroicon-o-plus')
->modalHeading('Add a new Channel')
->modalSubmitActionLabel('Add')
->form(Actions\Create::form())
->authorize('create', NotificationChannel::class)
->modalWidth(MaxWidth::Large)
->action(function (array $data) {
Actions\Create::action($data);
$this->dispatch('$refresh');
}),
];
}
}

View File

@ -0,0 +1,77 @@
<?php
namespace App\Web\Pages\Settings\NotificationChannels\Widgets;
use App\Models\NotificationChannel;
use App\Web\Pages\Settings\NotificationChannels\Actions\Edit;
use Filament\Support\Enums\MaxWidth;
use Filament\Tables\Actions\DeleteAction;
use Filament\Tables\Actions\EditAction;
use Filament\Tables\Columns\IconColumn;
use Filament\Tables\Columns\TextColumn;
use Filament\Tables\Table;
use Filament\Widgets\TableWidget as Widget;
use Illuminate\Database\Eloquent\Builder;
class NotificationChannelsList extends Widget
{
protected $listeners = [
'$refresh' => 'refreshTable',
];
protected function getTableQuery(): Builder
{
return NotificationChannel::getByProjectId(auth()->user()->current_project_id);
}
protected function getTableColumns(): array
{
return [
IconColumn::make('provider')
->icon(fn (NotificationChannel $record) => 'icon-'.$record->provider)
->width(24),
TextColumn::make('label')
->default(fn (NotificationChannel $record) => $record->label)
->searchable()
->sortable(),
TextColumn::make('id')
->label('Global')
->badge()
->color(fn (NotificationChannel $record) => $record->project_id ? 'gray' : 'success')
->formatStateUsing(function (NotificationChannel $record) {
return $record->project_id ? 'No' : 'Yes';
}),
TextColumn::make('created_at')
->label('Created At')
->formatStateUsing(fn (NotificationChannel $record) => $record->created_at_by_timezone)
->searchable()
->sortable(),
];
}
public function getTable(): Table
{
return $this->table
->heading('')
->actions([
EditAction::make('edit')
->modalHeading('Edit Notification Channel')
->mutateRecordDataUsing(function (array $data, NotificationChannel $record) {
return [
'label' => $record->label,
'global' => ! $record->project_id,
];
})
->form(Edit::form())
->authorize(fn (NotificationChannel $record) => auth()->user()->can('update', $record))
->using(fn (array $data, NotificationChannel $record) => Edit::action($record, $data))
->modalWidth(MaxWidth::Medium),
DeleteAction::make('delete')
->modalHeading('Delete Notification Channel')
->authorize(fn (NotificationChannel $record) => auth()->user()->can('delete', $record))
->using(function (array $data, NotificationChannel $record) {
//
}),
]);
}
}

View File

@ -4,7 +4,7 @@
use App\Actions\Projects\DeleteProject; use App\Actions\Projects\DeleteProject;
use App\Models\Project; use App\Models\Project;
use App\Web\Pages\Servers\Page; use App\Web\Components\Page;
use App\Web\Pages\Settings\Projects\Widgets\AddUser; use App\Web\Pages\Settings\Projects\Widgets\AddUser;
use App\Web\Pages\Settings\Projects\Widgets\ProjectUsersList; use App\Web\Pages\Settings\Projects\Widgets\ProjectUsersList;
use App\Web\Pages\Settings\Projects\Widgets\UpdateProject; use App\Web\Pages\Settings\Projects\Widgets\UpdateProject;
@ -19,9 +19,11 @@ class Settings extends Page
protected static ?string $title = 'Project Settings'; protected static ?string $title = 'Project Settings';
protected static bool $shouldRegisterNavigation = false;
public static function canAccess(): bool public static function canAccess(): bool
{ {
return auth()->user()?->can('update', request()->route('project')) ?? false; return auth()->user()?->can('update', get_from_route(Project::class, 'project')) ?? false;
} }
public Project $project; public Project $project;
@ -62,7 +64,7 @@ protected function getHeaderActions(): array
try { try {
app(DeleteProject::class)->delete(auth()->user(), $record); app(DeleteProject::class)->delete(auth()->user(), $record);
$this->redirectRoute('filament.app.resources.projects.index'); $this->redirectRoute(Index::getUrl());
} catch (Exception $e) { } catch (Exception $e) {
Notification::make() Notification::make()
->title($e->getMessage()) ->title($e->getMessage())

View File

@ -0,0 +1,63 @@
<?php
namespace App\Web\Pages\Settings\SSHKeys;
use App\Actions\SshKey\CreateSshKey;
use App\Models\SshKey;
use App\Web\Components\Page;
use Filament\Actions\CreateAction;
use Filament\Forms\Components\Textarea;
use Filament\Forms\Components\TextInput;
use Filament\Support\Enums\MaxWidth;
class Index extends Page
{
protected static ?string $navigationGroup = 'Settings';
protected static ?string $slug = 'settings/ssh-keys';
protected static ?string $title = 'SSH Keys';
protected static ?string $navigationIcon = 'heroicon-o-key';
protected static ?int $navigationSort = 9;
public static function canAccess(): bool
{
return auth()->user()?->can('viewAny', SshKey::class) ?? false;
}
public function getWidgets(): array
{
return [
[Widgets\SshKeysList::class],
];
}
protected function getHeaderActions(): array
{
return [
CreateAction::make('add')
->label('Add Key')
->icon('heroicon-o-plus')
->modalHeading('Add a new Key')
->modalSubmitActionLabel('Add')
->createAnother(false)
->form([
TextInput::make('name')
->label('Name')
->rules(CreateSshKey::rules()['name']),
Textarea::make('public_key')
->label('Public Key')
->rules(CreateSshKey::rules()['public_key']),
])
->authorize('create', SshKey::class)
->modalWidth(MaxWidth::Large)
->using(function (array $data) {
app(CreateSshKey::class)->create(auth()->user(), $data);
$this->dispatch('$refresh');
}),
];
}
}

View File

@ -0,0 +1,53 @@
<?php
namespace App\Web\Pages\Settings\SSHKeys\Widgets;
use App\Models\SshKey;
use Filament\Tables\Actions\DeleteAction;
use Filament\Tables\Columns\TextColumn;
use Filament\Tables\Table;
use Filament\Widgets\TableWidget;
use Illuminate\Database\Eloquent\Builder;
class SshKeysList extends TableWidget
{
protected $listeners = ['$refresh'];
protected function getTableQuery(): Builder
{
return SshKey::query()->where('user_id', auth()->id());
}
protected static ?string $heading = '';
protected function getTableColumns(): array
{
return [
TextColumn::make('name')
->sortable()
->searchable(),
TextColumn::make('public_key')
->tooltip('Copy')
->limit(20)
->copyable(),
TextColumn::make('created_at')
->sortable(),
];
}
public function getTable(): Table
{
return $this->table
->actions([
DeleteAction::make('delete')
->requiresConfirmation()
->authorize(fn (SshKey $record) => auth()->user()->can('delete', $record))
->action(function (SshKey $record) {
run_action($this, function () use ($record) {
$record->delete();
$this->dispatch('$refresh');
});
}),
]);
}
}

View File

@ -18,7 +18,7 @@ class Index extends Page
protected static ?string $navigationIcon = 'heroicon-o-tag'; protected static ?string $navigationIcon = 'heroicon-o-tag';
protected static ?int $navigationSort = 7; protected static ?int $navigationSort = 10;
public static function canAccess(): bool public static function canAccess(): bool
{ {

View File

@ -0,0 +1,22 @@
<?php
use Illuminate\Database\Migrations\Migration;
use Illuminate\Database\Schema\Blueprint;
use Illuminate\Support\Facades\Schema;
return new class extends Migration
{
public function up(): void
{
Schema::table('ssh_keys', function (Blueprint $table) {
$table->softDeletes();
});
}
public function down(): void
{
Schema::table('ssh_keys', function (Blueprint $table) {
$table->dropSoftDeletes();
});
}
};

View File

@ -0,0 +1,22 @@
<?php
use Illuminate\Database\Migrations\Migration;
use Illuminate\Database\Schema\Blueprint;
use Illuminate\Support\Facades\Schema;
return new class extends Migration
{
public function up(): void
{
Schema::table('scripts', function (Blueprint $table) {
$table->unsignedInteger('project_id')->nullable();
});
}
public function down(): void
{
Schema::table('scripts', function (Blueprint $table) {
$table->dropColumn('project_id');
});
}
};

View File

@ -0,0 +1,22 @@
<?php
use Illuminate\Database\Migrations\Migration;
use Illuminate\Database\Schema\Blueprint;
use Illuminate\Support\Facades\Schema;
return new class extends Migration
{
public function up(): void
{
Schema::table('script_executions', function (Blueprint $table) {
$table->unsignedInteger('server_id')->nullable();
});
}
public function down(): void
{
Schema::table('script_executions', function (Blueprint $table) {
$table->dropColumn('server_id');
});
}
};