
* Add new panel * Add some basic resource pages * Wip * Wip terminal * Wip * Add new panel * Add some basic resource pages * Wip * [Sub-Users] Add Invite TODO: The logic with permissions * [Sub-Users] Fix Creation * [Cron] Add basics * Add basic auth and messages * Add basic buttons * WIP on issue/353 * WIP on issue/353 * Add Database page * Update Database Page * Start of Backup Page * Composer Update * Changes * Send input * Remove this includes * Better offline handling * Consolidate top nav config * Update Backups Page * Update Backups * Change name * Add Assign All, Layout Fixes. * conflict * update schedule pages * fix phpstan * update pint.json * add cron presets to schedule * fix tests * fix task creation * schedules: disable task creation if limit is reached & disable backup action if backup limit is 0 * update activity pages * update resources * Update Edit User TODO: actually save permissions when they're changed. TODO: Figure out why Control does not update it's state... but the rest do... * .... Sure it works. TODO: Update permissions when you save editing a sub user. * user: update canAccessPanel & canAccessTenant * add helper to convert bytes into readable format * very basic file explorer * files: fix some stuff & remove dummy data * files: better error handling * files: basic file editor * files: add some actions * File manager updates * files: fix paths * Revery Composer Upgrade, Fixes SQLite * fix: Pint (#517) feat: MenuItems to and from admin * Update File Editing Updated File Editing to its own page, Added Permission checks for file manager. Co-authored-by: Boy132 <Boy132@users.noreply.github.com> * add enum for editor langs * files: add upload & pull actions * fix build * files: handle images * Update to Filament v3.2.98 * files: add remaining actions * use `authorize` instead of `hidden` * fix canAccessTenant * update date columns * files: testing & fixes * Fix File Names Co-authored-by: lancepioch <git@lance.sh> * Combine Pull/Upload * Fix BulkDelete * Uncontained tabs * Hide Lang Selection, Move Actions * Update Monaco, more custom * Add livewire config livewire limits uploads to 12MB... who knows why... Fixed uploading a single files failing * files: fix record url * basic setup for settings & startup page * make abstract class for simple app pages * Basic Startup Page * Update nav sort * small cleanup * startup: fix shouldHideComponent & getSelectOptionsFromRules * startup: fix non editable fields & set default value * startup: add todo for save button * Save Variables after update & off click Variables update when the user clicks off the input. * Notifications are cool * Add rule validation * Sort variables by sortid * pint * Settings Page + Startup Changes * settings: cleanup * refactor: use server model for ServerFormPage (formerly known as SimplePage) * Use Repeater for variables * Add Network, Remove breadcrumbs * Add paginated to file explorer * Fix updating variables * Add link to go to new client area * fix after merge * Add graphs to console page Graphs still need to get the data from the web socket. * fix pint & phpstan * fix authorizeAccess for EditFiles and Startup page * Fix rules on startup page * Update console size * Fix node name * add "global search" to files list requires https://github.com/pelican-dev/wings/pull/44 * remove debug dummy data * update view action on ListServers * enable SPA mode for app panel * remove colors from app panel they are defined globally in AppServiceProvider * update global search ui a bit (to be replaced with a custom page that is similar to the list files table) * add own page for global search untested - and route needs cleanup (if possible) * fix File getRows * remove "path" from SearchFiles (for now) * fix caching for searched files * add title and breadcrumbs to global search page * make cpu & memory charts on console page working * fix phpstan * add missing import * cleanup console views & widgets * add overview stats to console * don't be so lazy, console! * make history working * decode data to get array * add missing On * fix json_decode * change polling to 1 sec * hide "0" cpu/ memory * add data to network chart * Remove data labels * fix data on network chart * fix data on network chart (2nd try) * WIP Network Stats * Remove test * Change MaxWidth * run pint * fix phpstan * Fix storeStats cast * make $data a string this time for real * update visible check for "admin" menu item * remove account widget * rebrand "Dashboard" to "Server List" WIP - doesn't look good but is somewhat working * fix canAccessPanel * separate server list into own panel * change path to avoid conflicts with old client area (and remove sidebar width) * display correct icon and color on server list entries * show total memory if server is offline * replace custom server list page with ListRecords page * fix tests * fix namespace * remove "open" button and make whole column clickable * Update EditProfile * run pint * fix access to server list * add new login page to panels * fix next_run_at for new schedules * use new DateTimeColumn * add own column for file bytes * return to server list when clicking title * fix console loading * handle server with "conflict state" * add banner if server is in "conflict state" * fix phpstan * update docker image select * fix permission checks on Settings & Startup pages * fix query for activity log page * fix activity log not being logged * adjust ListActivities * fix phpstan * fix pint * fix profile menu item link on server panel * add ip tooltip to activity logs (and role permission) * change backup icon * update navigation sort * general code cleanup * more cleanup * Disable Restart/Stop if server is offline * Change rename notification * Remove negation on abort_unless * Add notification on save * Single disabled closure & comment unused import * Add required to Server Name & Nullable to description * mutateFormDataBeforeSave doesn't work since we use forceFill * Fix web socket connection not existing. * Fix some subuser permissions * add permission checks to resources * do not allow self-deletion * Update editing file permissions * Fix of the previous fix * add service for subuser updating * Only allow save if they have file_update * Remove unused import * Update backup delete button * Add Delete, remove bulks * Update Database page * Use Allocation Permissions * add canAccess check to startup * Add Permission checks to Settings page * add service for subuser deletion * Remove Kill permission * Updates * fix move files * add redirects * fix phpstan * activity: remove properties from tans for now * If alias, use that, else ip --------- Co-authored-by: notCharles <charles@pelican.dev> Co-authored-by: Boy132 <mail@boy132.de> Co-authored-by: Senna <62171904+Poseidon281@users.noreply.github.com> Co-authored-by: Boy132 <Boy132@users.noreply.github.com> Co-authored-by: RMartinOscar <40749467+RMartinOscar@users.noreply.github.com>
589 lines
28 KiB
PHP
589 lines
28 KiB
PHP
<?php
|
|
|
|
namespace App\Filament\Server\Resources\FileResource\Pages;
|
|
|
|
use AbdelhamidErrahmouni\FilamentMonacoEditor\MonacoEditor;
|
|
use App\Enums\EditorLanguages;
|
|
use App\Facades\Activity;
|
|
use App\Filament\Server\Resources\FileResource;
|
|
use App\Models\File;
|
|
use App\Models\Permission;
|
|
use App\Models\Server;
|
|
use App\Repositories\Daemon\DaemonFileRepository;
|
|
use App\Services\Nodes\NodeJWTService;
|
|
use App\Tables\Columns\BytesColumn;
|
|
use App\Tables\Columns\DateTimeColumn;
|
|
use Carbon\CarbonImmutable;
|
|
use Filament\Actions\Action as HeaderAction;
|
|
use Filament\Facades\Filament;
|
|
use Filament\Forms\Components\CheckboxList;
|
|
use Filament\Forms\Components\FileUpload;
|
|
use Filament\Forms\Components\Placeholder;
|
|
use Filament\Forms\Components\Select;
|
|
use Filament\Forms\Components\Tabs;
|
|
use Filament\Forms\Components\TextInput;
|
|
use Filament\Forms\Get;
|
|
use Filament\Notifications\Notification;
|
|
use Filament\Panel;
|
|
use Filament\Resources\Pages\ListRecords;
|
|
use Filament\Resources\Pages\PageRegistration;
|
|
use Filament\Tables\Actions\Action;
|
|
use Filament\Tables\Actions\ActionGroup;
|
|
use Filament\Tables\Actions\BulkAction;
|
|
use Filament\Tables\Actions\BulkActionGroup;
|
|
use Filament\Tables\Actions\DeleteAction;
|
|
use Filament\Tables\Actions\DeleteBulkAction;
|
|
use Filament\Tables\Actions\EditAction;
|
|
use Filament\Tables\Columns\TextColumn;
|
|
use Filament\Tables\Table;
|
|
use Illuminate\Database\Eloquent\Collection;
|
|
use Illuminate\Http\UploadedFile;
|
|
use Illuminate\Routing\Route;
|
|
use Illuminate\Support\Facades\Route as RouteFacade;
|
|
use Livewire\Attributes\Locked;
|
|
|
|
class ListFiles extends ListRecords
|
|
{
|
|
protected static string $resource = FileResource::class;
|
|
|
|
#[Locked]
|
|
public string $path;
|
|
|
|
public function mount(?string $path = null): void
|
|
{
|
|
parent::mount();
|
|
$this->path = $path ?? '/';
|
|
}
|
|
|
|
public function getBreadcrumbs(): array
|
|
{
|
|
$resource = static::getResource();
|
|
|
|
$breadcrumbs = [
|
|
$resource::getUrl() => $resource::getBreadcrumb(),
|
|
];
|
|
|
|
$previousParts = '';
|
|
foreach (explode('/', $this->path) as $part) {
|
|
$previousParts = $previousParts . '/' . $part;
|
|
$breadcrumbs[self::getUrl(['path' => ltrim($previousParts, '/')])] = $part;
|
|
}
|
|
|
|
return $breadcrumbs;
|
|
}
|
|
|
|
public function table(Table $table): Table
|
|
{
|
|
/** @var Server $server */
|
|
$server = Filament::getTenant();
|
|
|
|
return $table
|
|
->paginated([15, 25, 50, 100])
|
|
->defaultPaginationPageOption(15)
|
|
->query(fn () => File::get($server, $this->path)->orderByDesc('is_directory')->orderBy('name'))
|
|
->columns([
|
|
TextColumn::make('name')
|
|
->searchable()
|
|
->icon(fn (File $file) => $file->getIcon()),
|
|
BytesColumn::make('size'),
|
|
DateTimeColumn::make('modified_at')
|
|
->since()
|
|
->sortable(),
|
|
])
|
|
->recordUrl(function (File $file) use ($server) {
|
|
|
|
if ($file->is_directory) {
|
|
return self::getUrl(['path' => join_paths($this->path, $file->name)]);
|
|
}
|
|
|
|
if (!auth()->user()->can(Permission::ACTION_FILE_READ_CONTENT, $server)) {
|
|
return null;
|
|
}
|
|
|
|
return $file->canEdit() ? EditFiles::getUrl(['path' => join_paths($this->path, $file->name)]) : null;
|
|
})
|
|
->actions([
|
|
Action::make('view')
|
|
->authorize(fn () => auth()->user()->can(Permission::ACTION_FILE_READ, $server))
|
|
->label('Open')
|
|
->icon('tabler-eye')
|
|
->visible(fn (File $file) => $file->is_directory)
|
|
->url(fn (File $file) => self::getUrl(['path' => join_paths($this->path, $file->name)])),
|
|
EditAction::make('edit')
|
|
->authorize(fn () => auth()->user()->can(Permission::ACTION_FILE_READ_CONTENT, $server))
|
|
->label('Edit')
|
|
->icon('tabler-edit')
|
|
->visible(fn (File $file) => $file->canEdit())
|
|
->url(fn (File $file) => EditFiles::getUrl(['path' => join_paths($this->path, $file->name)])),
|
|
ActionGroup::make([
|
|
Action::make('rename')
|
|
->authorize(fn () => auth()->user()->can(Permission::ACTION_FILE_UPDATE, $server))
|
|
->label('Rename')
|
|
->icon('tabler-forms')
|
|
->form([
|
|
TextInput::make('name')
|
|
->label('File name')
|
|
->default(fn (File $file) => $file->name)
|
|
->required(),
|
|
])
|
|
->action(function ($data, File $file) use ($server) {
|
|
// @phpstan-ignore-next-line
|
|
app(DaemonFileRepository::class)
|
|
->setServer($server)
|
|
->renameFiles($this->path, [['to' => $data['name'], 'from' => $file->name]]);
|
|
|
|
Activity::event('server:file.rename')
|
|
->property('directory', $this->path)
|
|
->property('files', [['to' => $data['name'], 'from' => $file->name]])
|
|
->log();
|
|
|
|
Notification::make()
|
|
->title('File Renamed')
|
|
->body(fn () => $file->name . ' -> ' . $data['name'])
|
|
->success()
|
|
->send();
|
|
}),
|
|
Action::make('copy')
|
|
->authorize(fn () => auth()->user()->can(Permission::ACTION_FILE_CREATE, $server))
|
|
->label('Copy')
|
|
->icon('tabler-copy')
|
|
->visible(fn (File $file) => $file->is_file)
|
|
->action(function (File $file) use ($server) {
|
|
// @phpstan-ignore-next-line
|
|
app(DaemonFileRepository::class)
|
|
->setServer($server)
|
|
->copyFile(join_paths($this->path, $file->name));
|
|
|
|
Activity::event('server:file.copy')
|
|
->property('file', join_paths($this->path, $file->name))
|
|
->log();
|
|
|
|
Notification::make()
|
|
->title('File copied')
|
|
->success()
|
|
->send();
|
|
|
|
return redirect(ListFiles::getUrl(['path' => $this->path]));
|
|
}),
|
|
Action::make('download')
|
|
->authorize(fn () => auth()->user()->can(Permission::ACTION_FILE_READ_CONTENT, $server))
|
|
->label('Download')
|
|
->icon('tabler-download')
|
|
->visible(fn (File $file) => $file->is_file)
|
|
->action(function (File $file) use ($server) {
|
|
// @phpstan-ignore-next-line
|
|
$token = app(NodeJWTService::class)
|
|
->setExpiresAt(CarbonImmutable::now()->addMinutes(15))
|
|
->setUser(auth()->user())
|
|
->setClaims([
|
|
'file_path' => rawurldecode(join_paths($this->path, $file->name)),
|
|
'server_uuid' => $server->uuid,
|
|
])
|
|
->handle($server->node, auth()->user()->id . $server->uuid);
|
|
|
|
Activity::event('server:file.download')
|
|
->property('file', join_paths($this->path, $file->name))
|
|
->log();
|
|
|
|
return redirect()->away(sprintf('%s/download/file?token=%s', $server->node->getConnectionAddress(), $token->toString())); // TODO: download works, but breaks modals
|
|
}),
|
|
Action::make('move')
|
|
->authorize(fn () => auth()->user()->can(Permission::ACTION_FILE_UPDATE, $server))
|
|
->label('Move')
|
|
->icon('tabler-replace')
|
|
->form([
|
|
TextInput::make('location')
|
|
->label('File name')
|
|
->hint('Enter the new name and directory of this file or folder, relative to the current directory.')
|
|
->default(fn (File $file) => $file->name)
|
|
->required()
|
|
->live(),
|
|
Placeholder::make('new_location')
|
|
->content(fn (Get $get) => resolve_path('./' . join_paths($this->path, $get('location')))),
|
|
])
|
|
->action(function ($data, File $file) use ($server) {
|
|
$location = resolve_path(join_paths($this->path, $data['location']));
|
|
|
|
// @phpstan-ignore-next-line
|
|
app(DaemonFileRepository::class)
|
|
->setServer($server)
|
|
->renameFiles($this->path, [['to' => $location, 'from' => $file->name]]);
|
|
|
|
Activity::event('server:file.rename')
|
|
->property('directory', $this->path)
|
|
->property('files', [['to' => $location, 'from' => $file->name]])
|
|
->log();
|
|
|
|
Notification::make()
|
|
->title(join_paths($this->path, $file->name) . ' was moved to ' . $location)
|
|
->success()
|
|
->send();
|
|
}),
|
|
Action::make('permissions')
|
|
->authorize(fn () => auth()->user()->can(Permission::ACTION_FILE_UPDATE, $server))
|
|
->label('Permissions')
|
|
->icon('tabler-license')
|
|
->form([
|
|
CheckboxList::make('owner')
|
|
->bulkToggleable()
|
|
->options([
|
|
'read' => 'Read',
|
|
'write' => 'Write',
|
|
'execute' => 'Execute',
|
|
])
|
|
->formatStateUsing(function ($state, File $file) {
|
|
$mode = (int) substr((string) $file->mode_bits, 0, 1);
|
|
|
|
return $this->getPermissionsFromModeBit($mode);
|
|
}),
|
|
CheckboxList::make('group')
|
|
->bulkToggleable()
|
|
->options([
|
|
'read' => 'Read',
|
|
'write' => 'Write',
|
|
'execute' => 'Execute',
|
|
])
|
|
->formatStateUsing(function ($state, File $file) {
|
|
$mode = (int) substr((string) $file->mode_bits, 1, 1);
|
|
|
|
return $this->getPermissionsFromModeBit($mode);
|
|
}),
|
|
CheckboxList::make('public')
|
|
->bulkToggleable()
|
|
->options([
|
|
'read' => 'Read',
|
|
'write' => 'Write',
|
|
'execute' => 'Execute',
|
|
])
|
|
->formatStateUsing(function ($state, File $file) {
|
|
$mode = (int) substr((string) $file->mode_bits, 2, 1);
|
|
|
|
return $this->getPermissionsFromModeBit($mode);
|
|
}),
|
|
])
|
|
->action(function ($data, File $file) use ($server) {
|
|
$owner = (in_array('read', $data['owner']) ? 4 : 0) | (in_array('write', $data['owner']) ? 2 : 0) | (in_array('execute', $data['owner']) ? 1 : 0);
|
|
$group = (in_array('read', $data['group']) ? 4 : 0) | (in_array('write', $data['group']) ? 2 : 0) | (in_array('execute', $data['group']) ? 1 : 0);
|
|
$public = (in_array('read', $data['public']) ? 4 : 0) | (in_array('write', $data['public']) ? 2 : 0) | (in_array('execute', $data['public']) ? 1 : 0);
|
|
|
|
$mode = $owner . $group . $public;
|
|
|
|
// @phpstan-ignore-next-line
|
|
app(DaemonFileRepository::class)
|
|
->setServer($server)
|
|
->chmodFiles($this->path, [['file' => $file->name, 'mode' => $mode]]);
|
|
|
|
Notification::make()
|
|
->title('Permissions changed to ' . $mode)
|
|
->success()
|
|
->send();
|
|
}),
|
|
Action::make('archive')
|
|
->authorize(fn () => auth()->user()->can(Permission::ACTION_FILE_ARCHIVE, $server))
|
|
->label('Archive')
|
|
->icon('tabler-archive')
|
|
->action(function (File $file) use ($server) {
|
|
// @phpstan-ignore-next-line
|
|
app(DaemonFileRepository::class)
|
|
->setServer($server)
|
|
->compressFiles($this->path, [$file->name]);
|
|
|
|
Activity::event('server:file.compress')
|
|
->property('directory', $this->path)
|
|
->property('files', [$file->name])
|
|
->log();
|
|
|
|
Notification::make()
|
|
->title('Archive created')
|
|
->success()
|
|
->send();
|
|
|
|
return redirect(ListFiles::getUrl(['path' => $this->path]));
|
|
}),
|
|
Action::make('unarchive')
|
|
->authorize(fn () => auth()->user()->can(Permission::ACTION_FILE_ARCHIVE, $server))
|
|
->label('Unarchive')
|
|
->icon('tabler-archive')
|
|
->visible(fn (File $file) => $file->isArchive())
|
|
->action(function (File $file) use ($server) {
|
|
// @phpstan-ignore-next-line
|
|
app(DaemonFileRepository::class)
|
|
->setServer($server)
|
|
->decompressFile($this->path, $file->name);
|
|
|
|
Activity::event('server:file.decompress')
|
|
->property('directory', $this->path)
|
|
->property('files', $file->name)
|
|
->log();
|
|
|
|
Notification::make()
|
|
->title('Unarchive completed')
|
|
->success()
|
|
->send();
|
|
|
|
return redirect(ListFiles::getUrl(['path' => $this->path]));
|
|
}),
|
|
]),
|
|
DeleteAction::make()
|
|
->authorize(fn () => auth()->user()->can(Permission::ACTION_FILE_DELETE, $server))
|
|
->label('')
|
|
->icon('tabler-trash')
|
|
->requiresConfirmation()
|
|
->modalDescription(fn (File $file) => $file->name)
|
|
->modalHeading('Delete file?')
|
|
->action(function (File $file) use ($server) {
|
|
// @phpstan-ignore-next-line
|
|
app(DaemonFileRepository::class)
|
|
->setServer($server)
|
|
->deleteFiles($this->path, [$file->name]);
|
|
|
|
Activity::event('server:file.delete')
|
|
->property('directory', $this->path)
|
|
->property('files', $file->name)
|
|
->log();
|
|
}),
|
|
])
|
|
->bulkActions([
|
|
BulkActionGroup::make([
|
|
BulkAction::make('move')
|
|
->authorize(fn () => auth()->user()->can(Permission::ACTION_FILE_UPDATE, $server))
|
|
->form([
|
|
TextInput::make('location')
|
|
->label('File name')
|
|
->hint('Enter the new name and directory of this file or folder, relative to the current directory.')
|
|
->default(fn (File $file) => $file->name)
|
|
->required()
|
|
->live(),
|
|
Placeholder::make('new_location')
|
|
->content(fn (Get $get) => resolve_path('./' . join_paths($this->path, $get('location') ?? ''))),
|
|
])
|
|
->action(function (Collection $files, $data) use ($server) {
|
|
$location = resolve_path(join_paths($this->path, $data['location']));
|
|
|
|
// @phpstan-ignore-next-line
|
|
$files = $files->map(fn ($file) => ['to' => $location, 'from' => $file->name])->toArray();
|
|
|
|
// @phpstan-ignore-next-line
|
|
app(DaemonFileRepository::class)
|
|
->setServer($server)
|
|
->renameFiles($this->path, $files);
|
|
|
|
Activity::event('server:file.rename')
|
|
->property('directory', $this->path)
|
|
->property('files', $files)
|
|
->log();
|
|
|
|
Notification::make()
|
|
->title(count($files) . ' Files were moved from to ' . $location)
|
|
->success()
|
|
->send();
|
|
}),
|
|
BulkAction::make('archive')
|
|
->authorize(fn () => auth()->user()->can(Permission::ACTION_FILE_ARCHIVE, $server))
|
|
->action(function (Collection $files) use ($server) {
|
|
// @phpstan-ignore-next-line
|
|
$files = $files->map(fn ($file) => $file->name)->toArray();
|
|
|
|
// @phpstan-ignore-next-line
|
|
app(DaemonFileRepository::class)
|
|
->setServer($server)
|
|
->compressFiles($this->path, $files);
|
|
|
|
Activity::event('server:file.compress')
|
|
->property('directory', $this->path)
|
|
->property('files', $files)
|
|
->log();
|
|
|
|
Notification::make()
|
|
->title('Archive created')
|
|
->success()
|
|
->send();
|
|
|
|
return redirect(ListFiles::getUrl(['path' => $this->path]));
|
|
}),
|
|
DeleteBulkAction::make()
|
|
->authorize(fn () => auth()->user()->can(Permission::ACTION_FILE_DELETE, $server))
|
|
->action(function (Collection $files) use ($server) {
|
|
// @phpstan-ignore-next-line
|
|
$files = $files->map(fn ($file) => $file->name)->toArray();
|
|
|
|
// @phpstan-ignore-next-line
|
|
app(DaemonFileRepository::class)
|
|
->setServer($server)
|
|
->deleteFiles($this->path, $files);
|
|
|
|
Activity::event('server:file.delete')
|
|
->property('directory', $this->path)
|
|
->property('files', $files)
|
|
->log();
|
|
|
|
Notification::make()
|
|
->title(count($files) . ' Files deleted.')
|
|
->success()
|
|
->send();
|
|
}),
|
|
]),
|
|
]);
|
|
}
|
|
|
|
protected function getHeaderActions(): array
|
|
{
|
|
/** @var Server $server */
|
|
$server = Filament::getTenant();
|
|
|
|
return [
|
|
HeaderAction::make('new_file')
|
|
->authorize(fn () => auth()->user()->can(Permission::ACTION_FILE_CREATE, $server))
|
|
->label('New File')
|
|
->color('gray')
|
|
->keyBindings('')
|
|
->modalSubmitActionLabel('Create')
|
|
->action(function ($data) use ($server) {
|
|
// @phpstan-ignore-next-line
|
|
app(DaemonFileRepository::class)
|
|
->setServer($server)
|
|
->putContent(join_paths($this->path, $data['name']), $data['editor'] ?? '');
|
|
|
|
Activity::event('server:file.write')
|
|
->property('file', join_paths($this->path, $data['name']))
|
|
->log();
|
|
})
|
|
->form([
|
|
TextInput::make('name')
|
|
->label('File Name')
|
|
->required(),
|
|
Select::make('lang')
|
|
->live()
|
|
->hidden() //TODO: Make file language selection work
|
|
->label('Language')
|
|
->placeholder('File Language')
|
|
->options(EditorLanguages::class),
|
|
MonacoEditor::make('editor')
|
|
->label('')
|
|
->view('filament.plugins.monaco-editor')
|
|
->language(fn (Get $get) => $get('lang')),
|
|
]),
|
|
HeaderAction::make('new_folder')
|
|
->authorize(fn () => auth()->user()->can(Permission::ACTION_FILE_CREATE, $server))
|
|
->label('New Folder')
|
|
->color('gray')
|
|
->action(function ($data) use ($server) {
|
|
// @phpstan-ignore-next-line
|
|
app(DaemonFileRepository::class)
|
|
->setServer($server)
|
|
->createDirectory($data['name'], $this->path);
|
|
|
|
Activity::event('server:file.write')
|
|
->property('file', join_paths($this->path, $data['name']))
|
|
->log();
|
|
})
|
|
->form([
|
|
TextInput::make('name')
|
|
->label('Folder Name')
|
|
->required(),
|
|
]),
|
|
HeaderAction::make('upload')
|
|
->authorize(fn () => auth()->user()->can(Permission::ACTION_FILE_CREATE, $server))
|
|
->label('Upload')
|
|
->action(function ($data) use ($server) {
|
|
if (count($data['files']) > 0 && !isset($data['url'])) {
|
|
/** @var UploadedFile $file */
|
|
foreach ($data['files'] as $file) {
|
|
// @phpstan-ignore-next-line
|
|
app(DaemonFileRepository::class)
|
|
->setServer($server)
|
|
->putContent(join_paths($this->path, $file->getClientOriginalName()), $file->getContent());
|
|
|
|
Activity::event('server:file.uploaded')
|
|
->property('directory', $this->path)
|
|
->property('file', $file->getFilename())
|
|
->log();
|
|
}
|
|
} elseif ($data['url'] !== null) {
|
|
// @phpstan-ignore-next-line
|
|
app(DaemonFileRepository::class)
|
|
->setServer($server)
|
|
->pull($data['url'], $this->path);
|
|
|
|
Activity::event('server:file.pull')
|
|
->property('url', $data['url'])
|
|
->property('directory', $this->path)
|
|
->log();
|
|
}
|
|
|
|
return redirect(ListFiles::getUrl(['path' => $this->path]));
|
|
|
|
})
|
|
->form([
|
|
Tabs::make()
|
|
->contained(false)
|
|
->schema([
|
|
Tabs\Tab::make('Upload Files')
|
|
->live()
|
|
->schema([
|
|
FileUpload::make('files')
|
|
->label('File(s)')
|
|
->storeFiles(false)
|
|
->previewable(false)
|
|
->preserveFilenames()
|
|
->multiple(),
|
|
]),
|
|
Tabs\Tab::make('Upload From URL')
|
|
->live()
|
|
->disabled(fn (Get $get) => count($get('files')) > 0)
|
|
->schema([
|
|
TextInput::make('url')
|
|
->label('URL')
|
|
->url(),
|
|
]),
|
|
]),
|
|
]),
|
|
HeaderAction::make('search')
|
|
->authorize(fn () => auth()->user()->can(Permission::ACTION_FILE_READ, $server))
|
|
->label('Global Search')
|
|
->modalSubmitActionLabel('Search')
|
|
->form([
|
|
TextInput::make('searchTerm')
|
|
->placeholder('Enter a search term, e.g. *.txt')
|
|
->minLength(3),
|
|
])
|
|
->action(fn ($data) => redirect(SearchFiles::getUrl([
|
|
'searchTerm' => $data['searchTerm'],
|
|
'path' => $this->path,
|
|
]))),
|
|
];
|
|
}
|
|
|
|
public static function route(string $path): PageRegistration
|
|
{
|
|
return new PageRegistration(
|
|
page: static::class,
|
|
route: fn (Panel $panel): Route => RouteFacade::get($path, static::class)
|
|
->middleware(static::getRouteMiddleware($panel))
|
|
->withoutMiddleware(static::getWithoutRouteMiddleware($panel))
|
|
->where('path', '.*'),
|
|
);
|
|
}
|
|
|
|
private function getPermissionsFromModeBit(int $mode): array
|
|
{
|
|
if ($mode === 1) {
|
|
return ['execute'];
|
|
} elseif ($mode === 2) {
|
|
return ['write'];
|
|
} elseif ($mode === 3) {
|
|
return ['write', 'execute'];
|
|
} elseif ($mode === 4) {
|
|
return ['read'];
|
|
} elseif ($mode === 5) {
|
|
return ['read', 'execute'];
|
|
} elseif ($mode === 6) {
|
|
return ['read', 'write'];
|
|
} elseif ($mode === 7) {
|
|
return ['read', 'write', 'execute'];
|
|
}
|
|
|
|
return [];
|
|
}
|
|
}
|