From d1c4ecf014a98ac7288be6749e71dc0b22eeb4c2 Mon Sep 17 00:00:00 2001 From: Boy132 Date: Mon, 13 Jan 2025 10:15:48 +0100 Subject: [PATCH 1/6] fix translations for activity logs --- .../ActivityResource/Pages/ListActivities.php | 6 +- .../FileResource/Pages/ListFiles.php | 3 +- .../Api/Client/Servers/FileController.php | 8 ++- app/Models/ActivityLog.php | 32 ++++++++++- .../Api/Client/ActivityLogTransformer.php | 39 +------------ lang/en/activity.php | 56 ++++++++----------- 6 files changed, 68 insertions(+), 76 deletions(-) diff --git a/app/Filament/Server/Resources/ActivityResource/Pages/ListActivities.php b/app/Filament/Server/Resources/ActivityResource/Pages/ListActivities.php index 8a19545211..8ece87f89a 100644 --- a/app/Filament/Server/Resources/ActivityResource/Pages/ListActivities.php +++ b/app/Filament/Server/Resources/ActivityResource/Pages/ListActivities.php @@ -20,7 +20,11 @@ public function table(Table $table): Table ->columns([ TextColumn::make('event') ->html() - ->formatStateUsing(fn ($state, ActivityLog $activityLog) => __('activity.'.str($state)->replace(':', '.'))) // TODO: convert properties to a format that trans likes, see ActivityLogEntry.tsx - wrapProperties + ->formatStateUsing(function ($state, ActivityLog $activityLog) { + $properties = $activityLog->wrapProperties(); + + return trans_choice('activity.'.str($state)->replace(':', '.'), array_key_exists('count', $properties) ? $properties['count'] : 1, $properties); + }) ->description(fn ($state) => $state), TextColumn::make('user') ->state(fn (ActivityLog $activityLog) => $activityLog->actor instanceof User ? $activityLog->actor->username : 'System') diff --git a/app/Filament/Server/Resources/FileResource/Pages/ListFiles.php b/app/Filament/Server/Resources/FileResource/Pages/ListFiles.php index 1f83193ba1..93fe10eee6 100644 --- a/app/Filament/Server/Resources/FileResource/Pages/ListFiles.php +++ b/app/Filament/Server/Resources/FileResource/Pages/ListFiles.php @@ -212,7 +212,8 @@ public function table(Table $table): Table Activity::event('server:file.rename') ->property('directory', $this->path) - ->property('files', [['to' => $location, 'from' => $file->name]]) + ->property('to', $location) + ->property('from', $file->name) ->log(); Notification::make() diff --git a/app/Http/Controllers/Api/Client/Servers/FileController.php b/app/Http/Controllers/Api/Client/Servers/FileController.php index 6fc443e8fd..ed40bb0acf 100644 --- a/app/Http/Controllers/Api/Client/Servers/FileController.php +++ b/app/Http/Controllers/Api/Client/Servers/FileController.php @@ -146,13 +146,17 @@ public function create(CreateFolderRequest $request, Server $server): JsonRespon */ public function rename(RenameFileRequest $request, Server $server): JsonResponse { + $files = $request->input('files'); + $this->fileRepository ->setServer($server) - ->renameFiles($request->input('root'), $request->input('files')); + ->renameFiles($request->input('root'), $files); Activity::event('server:file.rename') ->property('directory', $request->input('root')) - ->property('files', $request->input('files')) + ->property('files', $files) + ->property('to', $files['to']) + ->property('from', $files['from']) ->log(); return new JsonResponse([], Response::HTTP_NO_CONTENT); diff --git a/app/Models/ActivityLog.php b/app/Models/ActivityLog.php index 94df56848c..62fdd27cf2 100644 --- a/app/Models/ActivityLog.php +++ b/app/Models/ActivityLog.php @@ -11,6 +11,7 @@ use Illuminate\Database\Eloquent\Relations\HasMany; use Illuminate\Database\Eloquent\Relations\MorphTo; use Illuminate\Database\Eloquent\Model as IlluminateModel; +use Illuminate\Support\Str; /** * \App\Models\ActivityLog. @@ -156,8 +157,8 @@ public function htmlable(): string 'username' => 'system', ]); } - - $event = __('activity.'.str($this->event)->replace(':', '.')); + $properties = $this->wrapProperties(); + $event = trans_choice('activity.'.str($this->event)->replace(':', '.'), array_key_exists('count', $properties) ? $properties['count'] : 1, $properties); return "
@@ -171,4 +172,31 @@ public function htmlable(): string
"; } + + public function wrapProperties(): array + { + if (!$this->properties || $this->properties->isEmpty()) { + return []; + } + + $properties = $this->properties->mapWithKeys(function ($value, $key) { + if (!is_array($value)) { + // Perform some directory normalization at this point. + if ($key === 'directory') { + $value = str_replace('//', '/', '/' . trim($value, '/') . '/'); + } + + return [$key => $value]; + } + + return [Str::singular($key) => array_first($value), "{$key}_count" => count($value)]; + }); + + $keys = $properties->keys()->filter(fn ($key) => Str::endsWith($key, '_count'))->values(); + if ($keys->containsOneItem()) { + $properties = $properties->merge(['count' => $properties->get($keys[0])])->except([$keys[0]]); + } + + return $properties->toArray(); + } } diff --git a/app/Transformers/Api/Client/ActivityLogTransformer.php b/app/Transformers/Api/Client/ActivityLogTransformer.php index a54cb1f304..a5d3128d54 100644 --- a/app/Transformers/Api/Client/ActivityLogTransformer.php +++ b/app/Transformers/Api/Client/ActivityLogTransformer.php @@ -2,7 +2,6 @@ namespace App\Transformers\Api\Client; -use Illuminate\Support\Str; use App\Models\User; use App\Models\ActivityLog; use Illuminate\Database\Eloquent\Model; @@ -29,7 +28,7 @@ public function transform(ActivityLog $model): array 'is_api' => !is_null($model->api_key_id), 'ip' => $this->canViewIP($model->actor) ? $model->ip : null, 'description' => $model->description, - 'properties' => $this->properties($model), + 'properties' => $model->wrapProperties(), 'has_additional_metadata' => $this->hasAdditionalMetadata($model), 'timestamp' => $model->timestamp->toAtomString(), ]; @@ -44,42 +43,6 @@ public function includeActor(ActivityLog $model): ResourceAbstract return $this->item($model->actor, $this->makeTransformer(UserTransformer::class), User::RESOURCE_NAME); } - /** - * Transforms any array values in the properties into a countable field for easier - * use within the translation outputs. - */ - protected function properties(ActivityLog $model): object - { - if (!$model->properties || $model->properties->isEmpty()) { - return (object) []; - } - - $properties = $model->properties - ->mapWithKeys(function ($value, $key) use ($model) { - if ($key === 'ip' && $model->actor instanceof User && !$model->actor->is($this->request->user())) { - return [$key => '[hidden]']; - } - - if (!is_array($value)) { - // Perform some directory normalization at this point. - if ($key === 'directory') { - $value = str_replace('//', '/', '/' . trim($value, '/') . '/'); - } - - return [$key => $value]; - } - - return [$key => $value, "{$key}_count" => count($value)]; - }); - - $keys = $properties->keys()->filter(fn ($key) => Str::endsWith($key, '_count'))->values(); - if ($keys->containsOneItem()) { - $properties = $properties->merge(['count' => $properties->get($keys[0])])->except([$keys[0]]); - } - - return (object) $properties->toArray(); - } - /** * Determines if there are any log properties that we've not already exposed * in the response language string and that are not just the IP address or diff --git a/lang/en/activity.php b/lang/en/activity.php index 3158063b7d..eb492d4530 100644 --- a/lang/en/activity.php +++ b/lang/en/activity.php @@ -15,7 +15,7 @@ 'checkpoint' => 'Two-factor authentication requested', 'recovery-token' => 'Used two-factor recovery token', 'token' => 'Solved two-factor challenge', - 'ip-blocked' => 'Blocked request from unlisted IP address for :identifier', + 'ip-blocked' => 'Blocked request from unlisted IP address for :identifier', 'sftp' => [ 'fail' => 'Failed SFTP log in', ], @@ -26,12 +26,12 @@ 'password-changed' => 'Changed password', ], 'api-key' => [ - 'create' => 'Created new API key :identifier', - 'delete' => 'Deleted API key :identifier', + 'create' => 'Created new API key :identifier', + 'delete' => 'Deleted API key :identifier', ], 'ssh-key' => [ - 'create' => 'Added SSH key :fingerprint to account', - 'delete' => 'Removed SSH key :fingerprint from account', + 'create' => 'Added SSH key :fingerprint to account', + 'delete' => 'Removed SSH key :fingerprint from account', ], 'two-factor' => [ 'create' => 'Enabled two-factor auth', @@ -41,7 +41,7 @@ 'server' => [ 'reinstall' => 'Reinstalled server', 'console' => [ - 'command' => 'Executed ":command" on the server', + 'command' => 'Executed ":command" on the server', ], 'power' => [ 'start' => 'Started the server', @@ -67,39 +67,31 @@ 'delete' => 'Deleted database :name', ], 'file' => [ - 'compress_one' => 'Compressed :directory:file', - 'compress_other' => 'Compressed :count files in :directory', - 'read' => 'Viewed the contents of :file', - 'copy' => 'Created a copy of :file', - 'create-directory' => 'Created directory :directory:name', - 'decompress' => 'Decompressed :files in :directory', - 'delete_one' => 'Deleted :directory:files.0', - 'delete_other' => 'Deleted :count files in :directory', - 'download' => 'Downloaded :file', - 'pull' => 'Downloaded a remote file from :url to :directory', - 'rename_one' => 'Renamed :directory:files.0.from to :directory:files.0.to', - 'rename_other' => 'Renamed :count files in :directory', - 'write' => 'Wrote new content to :file', + 'compress' => 'Compressed :directory:file|Compressed :count files in :directory', + 'read' => 'Viewed the contents of :file', + 'copy' => 'Created a copy of :file', + 'create-directory' => 'Created directory :directory:name', + 'decompress' => 'Decompressed :file in :directory', + 'delete' => 'Deleted :directory:file|Deleted :count files in :directory', + 'download' => 'Downloaded :file', + 'pull' => 'Downloaded a remote file from :url to :directory', + 'rename' => 'Renamed :directory:from to :directory:to|Renamed :count files in :directory', + 'write' => 'Wrote new content to :file', 'upload' => 'Began a file upload', - 'uploaded' => 'Uploaded :directory:file', + 'uploaded' => 'Uploaded :directory:file', ], 'sftp' => [ 'denied' => 'Blocked SFTP access due to permissions', - 'create_one' => 'Created :files.0', - 'create_other' => 'Created :count new files', - 'write_one' => 'Modified the contents of :files.0', - 'write_other' => 'Modified the contents of :count files', - 'delete_one' => 'Deleted :files.0', - 'delete_other' => 'Deleted :count files', - 'create-directory_one' => 'Created the :files.0 directory', - 'create-directory_other' => 'Created :count directories', - 'rename_one' => 'Renamed :files.0.from to :files.0.to', - 'rename_other' => 'Renamed or moved :count files', + 'create' => 'Created :file|Created :count new files', + 'write' => 'Modified the contents of :file|Modified the contents of :count files', + 'delete' => 'Deleted :file|Deleted :count files', + 'create-directory' => 'Created the :file directory|Created :count directories', + 'rename' => 'Renamed :from to :to|Renamed or moved :count files', ], 'allocation' => [ 'create' => 'Added :allocation to the server', - 'notes' => 'Updated the notes for :allocation from ":old" to ":new"', - 'primary' => 'Set :allocation as the primary server allocation', + 'notes' => 'Updated the notes for :allocation from ":old" to ":new"', + 'primary' => 'Set :allocation as the primary server allocation', 'delete' => 'Deleted the :allocation allocation', ], 'schedule' => [ From f8c5afc862792317abf0b8992d59d821b39e2123 Mon Sep 17 00:00:00 2001 From: Boy132 Date: Mon, 13 Jan 2025 10:38:02 +0100 Subject: [PATCH 2/6] add backwards compatibility for old logs --- app/Models/ActivityLog.php | 9 ++++++++- 1 file changed, 8 insertions(+), 1 deletion(-) diff --git a/app/Models/ActivityLog.php b/app/Models/ActivityLog.php index 62fdd27cf2..303e079c3a 100644 --- a/app/Models/ActivityLog.php +++ b/app/Models/ActivityLog.php @@ -189,7 +189,14 @@ public function wrapProperties(): array return [$key => $value]; } - return [Str::singular($key) => array_first($value), "{$key}_count" => count($value)]; + $first = array_first($value); + + // Backwards compatibility for old logs + if (is_array($first)) { + return ["{$key}_count" => count($value)]; + } + + return [Str::singular($key) => $first, "{$key}_count" => count($value)]; }); $keys = $properties->keys()->filter(fn ($key) => Str::endsWith($key, '_count'))->values(); From 24d29889afdcf9f7a3fbacd5f0193d12503f581f Mon Sep 17 00:00:00 2001 From: Boy132 Date: Mon, 13 Jan 2025 10:41:34 +0100 Subject: [PATCH 3/6] update lang file --- lang/en/activity.php | 12 ++++++------ 1 file changed, 6 insertions(+), 6 deletions(-) diff --git a/lang/en/activity.php b/lang/en/activity.php index eb492d4530..dc49b00ece 100644 --- a/lang/en/activity.php +++ b/lang/en/activity.php @@ -52,7 +52,7 @@ 'backup' => [ 'download' => 'Downloaded the :name backup', 'delete' => 'Deleted the :name backup', - 'restore' => 'Restored the :name backup (deleted files: :truncate)', + 'restore' => 'Restored the :name backup (deleted files: :truncate)', 'restore-complete' => 'Completed restoration of the :name backup', 'restore-failed' => 'Failed to complete restoration of the :name backup', 'start' => 'Started a new backup :name', @@ -85,14 +85,14 @@ 'create' => 'Created :file|Created :count new files', 'write' => 'Modified the contents of :file|Modified the contents of :count files', 'delete' => 'Deleted :file|Deleted :count files', - 'create-directory' => 'Created the :file directory|Created :count directories', + 'create-directory' => 'Created the :file directory|Created :count directories', 'rename' => 'Renamed :from to :to|Renamed or moved :count files', ], 'allocation' => [ - 'create' => 'Added :allocation to the server', + 'create' => 'Added :allocation to the server', 'notes' => 'Updated the notes for :allocation from ":old" to ":new"', 'primary' => 'Set :allocation as the primary server allocation', - 'delete' => 'Deleted the :allocation allocation', + 'delete' => 'Deleted the :allocation allocation', ], 'schedule' => [ 'create' => 'Created the :name schedule', @@ -106,8 +106,8 @@ 'delete' => 'Deleted a task for the :name schedule', ], 'settings' => [ - 'rename' => 'Renamed the server from :old to :new', - 'description' => 'Changed the server description from :old to :new', + 'rename' => 'Renamed the server from ":old" to ":new"', + 'description' => 'Changed the server description from ":old" to ":new"', ], 'startup' => [ 'edit' => 'Changed the :variable variable from ":old" to ":new"', From 1fea0d487046466c52d1e088eb4b6b44cb63f31c Mon Sep 17 00:00:00 2001 From: Boy132 Date: Mon, 13 Jan 2025 10:47:06 +0100 Subject: [PATCH 4/6] small cleanup --- .../Resources/ActivityResource/Pages/ListActivities.php | 9 +++++---- 1 file changed, 5 insertions(+), 4 deletions(-) diff --git a/app/Filament/Server/Resources/ActivityResource/Pages/ListActivities.php b/app/Filament/Server/Resources/ActivityResource/Pages/ListActivities.php index 8ece87f89a..b32ca0c849 100644 --- a/app/Filament/Server/Resources/ActivityResource/Pages/ListActivities.php +++ b/app/Filament/Server/Resources/ActivityResource/Pages/ListActivities.php @@ -2,6 +2,7 @@ namespace App\Filament\Server\Resources\ActivityResource\Pages; +use App\Filament\Admin\Resources\UserResource\Pages\EditUser; use App\Filament\Server\Resources\ActivityResource; use App\Models\ActivityLog; use App\Models\User; @@ -20,16 +21,16 @@ public function table(Table $table): Table ->columns([ TextColumn::make('event') ->html() + ->description(fn ($state) => $state) ->formatStateUsing(function ($state, ActivityLog $activityLog) { $properties = $activityLog->wrapProperties(); - return trans_choice('activity.'.str($state)->replace(':', '.'), array_key_exists('count', $properties) ? $properties['count'] : 1, $properties); - }) - ->description(fn ($state) => $state), + return trans_choice('activity.'.str($state)->replace(':', '.'), array_get($properties, 'count', 1), $properties); + }), TextColumn::make('user') ->state(fn (ActivityLog $activityLog) => $activityLog->actor instanceof User ? $activityLog->actor->username : 'System') ->tooltip(fn (ActivityLog $activityLog) => auth()->user()->can('seeIps activityLog') ? $activityLog->ip : '') - ->url(fn (ActivityLog $activityLog): string => $activityLog->actor instanceof User ? route('filament.admin.resources.users.edit', ['record' => $activityLog->actor]) : ''), + ->url(fn (ActivityLog $activityLog): string => $activityLog->actor instanceof User ? EditUser::getUrl(['record' => $activityLog->actor], panel: 'admin', tenant: null) : ''), DateTimeColumn::make('timestamp') ->since() ->sortable(), From 9a64db6e0d739fe89ea8969fa278ae2b214d8dd0 Mon Sep 17 00:00:00 2001 From: Boy132 Date: Mon, 13 Jan 2025 20:01:18 +0100 Subject: [PATCH 5/6] fix singular/ plural for "file" --- .../Resources/FileResource/Pages/ListFiles.php | 2 +- .../Api/Client/Servers/FileController.php | 2 +- app/Models/ActivityLog.php | 2 +- lang/en/activity.php | 12 ++++++------ 4 files changed, 9 insertions(+), 9 deletions(-) diff --git a/app/Filament/Server/Resources/FileResource/Pages/ListFiles.php b/app/Filament/Server/Resources/FileResource/Pages/ListFiles.php index 93fe10eee6..fea7ab307c 100644 --- a/app/Filament/Server/Resources/FileResource/Pages/ListFiles.php +++ b/app/Filament/Server/Resources/FileResource/Pages/ListFiles.php @@ -312,7 +312,7 @@ public function table(Table $table): Table Activity::event('server:file.decompress') ->property('directory', $this->path) - ->property('files', $file->name) + ->property('file', $file->name) ->log(); Notification::make() diff --git a/app/Http/Controllers/Api/Client/Servers/FileController.php b/app/Http/Controllers/Api/Client/Servers/FileController.php index ed40bb0acf..38acc49a15 100644 --- a/app/Http/Controllers/Api/Client/Servers/FileController.php +++ b/app/Http/Controllers/Api/Client/Servers/FileController.php @@ -214,7 +214,7 @@ public function decompress(DecompressFilesRequest $request, Server $server): Jso Activity::event('server:file.decompress') ->property('directory', $request->input('root')) - ->property('files', $request->input('file')) + ->property('file', $request->input('file')) ->log(); return new JsonResponse([], JsonResponse::HTTP_NO_CONTENT); diff --git a/app/Models/ActivityLog.php b/app/Models/ActivityLog.php index 303e079c3a..bc905780a0 100644 --- a/app/Models/ActivityLog.php +++ b/app/Models/ActivityLog.php @@ -196,7 +196,7 @@ public function wrapProperties(): array return ["{$key}_count" => count($value)]; } - return [Str::singular($key) => $first, "{$key}_count" => count($value)]; + return [$key => $first, "{$key}_count" => count($value)]; }); $keys = $properties->keys()->filter(fn ($key) => Str::endsWith($key, '_count'))->values(); diff --git a/lang/en/activity.php b/lang/en/activity.php index dc49b00ece..fa712e40d5 100644 --- a/lang/en/activity.php +++ b/lang/en/activity.php @@ -67,12 +67,12 @@ 'delete' => 'Deleted database :name', ], 'file' => [ - 'compress' => 'Compressed :directory:file|Compressed :count files in :directory', + 'compress' => 'Compressed :directory:files|Compressed :count files in :directory', 'read' => 'Viewed the contents of :file', 'copy' => 'Created a copy of :file', 'create-directory' => 'Created directory :directory:name', 'decompress' => 'Decompressed :file in :directory', - 'delete' => 'Deleted :directory:file|Deleted :count files in :directory', + 'delete' => 'Deleted :directory:files|Deleted :count files in :directory', 'download' => 'Downloaded :file', 'pull' => 'Downloaded a remote file from :url to :directory', 'rename' => 'Renamed :directory:from to :directory:to|Renamed :count files in :directory', @@ -82,10 +82,10 @@ ], 'sftp' => [ 'denied' => 'Blocked SFTP access due to permissions', - 'create' => 'Created :file|Created :count new files', - 'write' => 'Modified the contents of :file|Modified the contents of :count files', - 'delete' => 'Deleted :file|Deleted :count files', - 'create-directory' => 'Created the :file directory|Created :count directories', + 'create' => 'Created :files|Created :count new files', + 'write' => 'Modified the contents of :files|Modified the contents of :count files', + 'delete' => 'Deleted :files|Deleted :count files', + 'create-directory' => 'Created the :files directory|Created :count directories', 'rename' => 'Renamed :from to :to|Renamed or moved :count files', ], 'allocation' => [ From 9eebaa3e426731e638497c52622c92acf360c242 Mon Sep 17 00:00:00 2001 From: Boy132 Date: Thu, 16 Jan 2025 13:24:42 +0100 Subject: [PATCH 6/6] fix for "rename" + disable bulk move (because it's not working) --- .../Resources/FileResource/Pages/ListFiles.php | 16 ++++++++++++---- 1 file changed, 12 insertions(+), 4 deletions(-) diff --git a/app/Filament/Server/Resources/FileResource/Pages/ListFiles.php b/app/Filament/Server/Resources/FileResource/Pages/ListFiles.php index fea7ab307c..6012599bf1 100644 --- a/app/Filament/Server/Resources/FileResource/Pages/ListFiles.php +++ b/app/Filament/Server/Resources/FileResource/Pages/ListFiles.php @@ -132,13 +132,17 @@ public function table(Table $table): Table ->required(), ]) ->action(function ($data, File $file, DaemonFileRepository $fileRepository) use ($server) { + $files = [['to' => $data['name'], 'from' => $file->name]]; + $fileRepository ->setServer($server) - ->renameFiles($this->path, [['to' => $data['name'], 'from' => $file->name]]); + ->renameFiles($this->path, $files); Activity::event('server:file.rename') ->property('directory', $this->path) - ->property('files', [['to' => $data['name'], 'from' => $file->name]]) + ->property('files', $files) + ->property('to', $data['name']) + ->property('from', $file->name) ->log(); Notification::make() @@ -206,12 +210,15 @@ public function table(Table $table): Table ->action(function ($data, File $file, DaemonFileRepository $fileRepository) use ($server) { $location = resolve_path(join_paths($this->path, $data['location'])); + $files = [['to' => $location, 'from' => $file->name]]; + $fileRepository ->setServer($server) - ->renameFiles($this->path, [['to' => $location, 'from' => $file->name]]); + ->renameFiles($this->path, $files); Activity::event('server:file.rename') ->property('directory', $this->path) + ->property('files', $files) ->property('to', $location) ->property('from', $file->name) ->log(); @@ -345,6 +352,7 @@ public function table(Table $table): Table BulkActionGroup::make([ BulkAction::make('move') ->authorize(fn () => auth()->user()->can(Permission::ACTION_FILE_UPDATE, $server)) + ->hidden() // TODO ->form([ TextInput::make('location') ->label('File name') @@ -370,7 +378,7 @@ public function table(Table $table): Table ->log(); Notification::make() - ->title(count($files) . ' Files were moved from to ' . $location) + ->title(count($files) . ' Files were moved from ' . $location) ->success() ->send(); }),