mirror of
				https://github.com/pelican-dev/panel.git
				synced 2025-11-04 11:26:52 +01:00 
			
		
		
		
	Add Egg Images (#1849)
This commit is contained in:
		
							parent
							
								
									b2aff5445b
								
							
						
					
					
						commit
						21f9f259d0
					
				@ -16,6 +16,7 @@ use Filament\Actions\ActionGroup;
 | 
			
		||||
use Filament\Actions\DeleteAction;
 | 
			
		||||
use Filament\Forms\Components\Checkbox;
 | 
			
		||||
use Filament\Forms\Components\CodeEditor;
 | 
			
		||||
use Filament\Forms\Components\FileUpload;
 | 
			
		||||
use Filament\Forms\Components\Hidden;
 | 
			
		||||
use Filament\Forms\Components\KeyValue;
 | 
			
		||||
use Filament\Forms\Components\Repeater;
 | 
			
		||||
@ -24,13 +25,19 @@ use Filament\Forms\Components\TagsInput;
 | 
			
		||||
use Filament\Forms\Components\Textarea;
 | 
			
		||||
use Filament\Forms\Components\TextInput;
 | 
			
		||||
use Filament\Forms\Components\Toggle;
 | 
			
		||||
use Filament\Infolists\Components\TextEntry;
 | 
			
		||||
use Filament\Notifications\Notification;
 | 
			
		||||
use Filament\Resources\Pages\EditRecord;
 | 
			
		||||
use Filament\Schemas\Components\Fieldset;
 | 
			
		||||
use Filament\Schemas\Components\Flex;
 | 
			
		||||
use Filament\Schemas\Components\Grid;
 | 
			
		||||
use Filament\Schemas\Components\Image;
 | 
			
		||||
use Filament\Schemas\Components\Tabs;
 | 
			
		||||
use Filament\Schemas\Components\Tabs\Tab;
 | 
			
		||||
use Filament\Schemas\Components\Utilities\Get;
 | 
			
		||||
use Filament\Schemas\Components\Utilities\Set;
 | 
			
		||||
use Filament\Schemas\Schema;
 | 
			
		||||
use Filament\Support\Enums\IconSize;
 | 
			
		||||
use Illuminate\Validation\Rules\Unique;
 | 
			
		||||
 | 
			
		||||
class EditEgg extends EditRecord
 | 
			
		||||
@ -50,36 +57,215 @@ class EditEgg extends EditRecord
 | 
			
		||||
                Tabs::make()->tabs([
 | 
			
		||||
                    Tab::make('configuration')
 | 
			
		||||
                        ->label(trans('admin/egg.tabs.configuration'))
 | 
			
		||||
                        ->columns(['default' => 1, 'sm' => 1, 'md' => 2, 'lg' => 4])
 | 
			
		||||
                        ->columns(['default' => 2, 'sm' => 2, 'md' => 4, 'lg' => 6])
 | 
			
		||||
                        ->icon('tabler-egg')
 | 
			
		||||
                        ->schema([
 | 
			
		||||
                            Grid::make(2)
 | 
			
		||||
                                ->columnSpan(1)
 | 
			
		||||
                                ->schema([
 | 
			
		||||
                                    Image::make('', '')
 | 
			
		||||
                                        ->hidden(fn ($record) => !$record->image)
 | 
			
		||||
                                        ->url(fn ($record) => $record->image)
 | 
			
		||||
                                        ->alt('')
 | 
			
		||||
                                        ->alignJustify()
 | 
			
		||||
                                        ->imageSize(150)
 | 
			
		||||
                                        ->columnSpanFull(),
 | 
			
		||||
                                    Flex::make([
 | 
			
		||||
                                        Action::make('uploadImage')
 | 
			
		||||
                                            ->iconButton()
 | 
			
		||||
                                            ->iconSize(IconSize::Large)
 | 
			
		||||
                                            ->icon('tabler-photo-up')
 | 
			
		||||
                                            ->modal()
 | 
			
		||||
                                            ->modalHeading('')
 | 
			
		||||
                                            ->modalSubmitActionLabel(trans('admin/egg.import.import_image'))
 | 
			
		||||
                                            ->schema([
 | 
			
		||||
                                                Tabs::make()
 | 
			
		||||
                                                    ->contained(false)
 | 
			
		||||
                                                    ->tabs([
 | 
			
		||||
                                                        Tab::make(trans('admin/egg.import.url'))
 | 
			
		||||
                                                            ->schema([
 | 
			
		||||
                                                                Hidden::make('base64Image'),
 | 
			
		||||
                                                                TextInput::make('image_url')
 | 
			
		||||
                                                                    ->label(trans('admin/egg.import.image_url'))
 | 
			
		||||
                                                                    ->reactive()
 | 
			
		||||
                                                                    ->autocomplete(false)
 | 
			
		||||
                                                                    ->debounce(500)
 | 
			
		||||
                                                                    ->afterStateUpdated(function ($state, Set $set) {
 | 
			
		||||
                                                                        if (!$state) {
 | 
			
		||||
                                                                            $set('image_url_error', null);
 | 
			
		||||
 | 
			
		||||
                                                                            return;
 | 
			
		||||
                                                                        }
 | 
			
		||||
 | 
			
		||||
                                                                        try {
 | 
			
		||||
                                                                            if (!filter_var($state, FILTER_VALIDATE_URL)) {
 | 
			
		||||
                                                                                throw new \Exception(trans('admin/egg.import.invalid_url'));
 | 
			
		||||
                                                                            }
 | 
			
		||||
 | 
			
		||||
                                                                            $allowedExtensions = [
 | 
			
		||||
                                                                                'png' => 'image/png',
 | 
			
		||||
                                                                                'jpg' => 'image/jpeg',
 | 
			
		||||
                                                                                'jpeg' => 'image/jpeg',
 | 
			
		||||
                                                                                'gif' => 'image/gif',
 | 
			
		||||
                                                                                'webp' => 'image/webp',
 | 
			
		||||
                                                                                'svg' => 'image/svg+xml',
 | 
			
		||||
                                                                            ];
 | 
			
		||||
 | 
			
		||||
                                                                            $extension = strtolower(pathinfo(parse_url($state, PHP_URL_PATH), PATHINFO_EXTENSION));
 | 
			
		||||
 | 
			
		||||
                                                                            if (!array_key_exists($extension, $allowedExtensions)) {
 | 
			
		||||
                                                                                throw new \Exception(trans('admin/egg.import.unsupported_format', ['format' => implode(', ', $allowedExtensions)]));
 | 
			
		||||
                                                                            }
 | 
			
		||||
 | 
			
		||||
                                                                            $host = parse_url($state, PHP_URL_HOST);
 | 
			
		||||
                                                                            $ip = gethostbyname($host);
 | 
			
		||||
 | 
			
		||||
                                                                            if (
 | 
			
		||||
                                                                                filter_var($ip, FILTER_VALIDATE_IP, FILTER_FLAG_NO_PRIV_RANGE | FILTER_FLAG_NO_RES_RANGE) === false
 | 
			
		||||
                                                                            ) {
 | 
			
		||||
                                                                                throw new \Exception(trans('admin/egg.import.no_local_ip'));
 | 
			
		||||
                                                                            }
 | 
			
		||||
 | 
			
		||||
                                                                            $context = stream_context_create([
 | 
			
		||||
                                                                                'http' => ['timeout' => 3],
 | 
			
		||||
                                                                                'https' => [
 | 
			
		||||
                                                                                    'timeout' => 3,
 | 
			
		||||
                                                                                    'verify_peer' => true,
 | 
			
		||||
                                                                                    'verify_peer_name' => true,
 | 
			
		||||
                                                                                ],
 | 
			
		||||
                                                                            ]);
 | 
			
		||||
 | 
			
		||||
                                                                            $imageContent = @file_get_contents($state, false, $context, 0, 1048576); // 1024KB
 | 
			
		||||
 | 
			
		||||
                                                                            if (!$imageContent) {
 | 
			
		||||
                                                                                throw new \Exception(trans('admin/egg.import.image_error'));
 | 
			
		||||
                                                                            }
 | 
			
		||||
 | 
			
		||||
                                                                            if (strlen($imageContent) >= 1048576) {
 | 
			
		||||
                                                                                throw new \Exception(trans('admin/egg.import.image_too_large'));
 | 
			
		||||
                                                                            }
 | 
			
		||||
 | 
			
		||||
                                                                            $mimeType = $allowedExtensions[$extension];
 | 
			
		||||
                                                                            $base64 = 'data:' . $mimeType . ';base64,' . base64_encode($imageContent);
 | 
			
		||||
 | 
			
		||||
                                                                            $set('base64Image', $base64);
 | 
			
		||||
                                                                            $set('image_url_error', null);
 | 
			
		||||
 | 
			
		||||
                                                                        } catch (\Exception $e) {
 | 
			
		||||
                                                                            $set('image_url_error', $e->getMessage());
 | 
			
		||||
                                                                            $set('base64Image', null);
 | 
			
		||||
                                                                        }
 | 
			
		||||
                                                                    }),
 | 
			
		||||
                                                                TextEntry::make('image_url_error')
 | 
			
		||||
                                                                    ->hiddenLabel()
 | 
			
		||||
                                                                    ->visible(fn ($get) => $get('image_url_error') !== null)
 | 
			
		||||
                                                                    ->afterStateHydrated(fn ($set, $get) => $get('image_url_error')),
 | 
			
		||||
                                                                Image::make(fn (Get $get) => $get('image_url'), '')
 | 
			
		||||
                                                                    ->imageSize(150)
 | 
			
		||||
                                                                    ->visible(fn ($get) => $get('image_url') && !$get('image_url_error'))
 | 
			
		||||
                                                                    ->alignCenter(),
 | 
			
		||||
                                                            ]),
 | 
			
		||||
                                                        Tab::make(trans('admin/egg.import.file'))
 | 
			
		||||
                                                            ->schema([
 | 
			
		||||
                                                                FileUpload::make('image')
 | 
			
		||||
                                                                    ->hiddenLabel()
 | 
			
		||||
                                                                    ->previewable()
 | 
			
		||||
                                                                    ->openable(false)
 | 
			
		||||
                                                                    ->downloadable(false)
 | 
			
		||||
                                                                    ->maxSize(1024)
 | 
			
		||||
                                                                    ->maxFiles(1)
 | 
			
		||||
                                                                    ->columnSpanFull()
 | 
			
		||||
                                                                    ->alignCenter()
 | 
			
		||||
                                                                    ->imageEditor()
 | 
			
		||||
                                                                    ->saveUploadedFileUsing(function ($file, Set $set) {
 | 
			
		||||
                                                                        $base64 = "data:{$file->getMimeType()};base64,". base64_encode(file_get_contents($file->getRealPath()));
 | 
			
		||||
                                                                        $set('base64Image', $base64);
 | 
			
		||||
 | 
			
		||||
                                                                        return $base64;
 | 
			
		||||
                                                                    }),
 | 
			
		||||
                                                            ]),
 | 
			
		||||
                                                    ]),
 | 
			
		||||
                                            ])
 | 
			
		||||
                                            ->action(function (array $data, $record): void {
 | 
			
		||||
                                                $base64 = $data['base64Image'] ?? null;
 | 
			
		||||
 | 
			
		||||
                                                if (empty($base64) && !empty($data['image'])) {
 | 
			
		||||
                                                    $base64 = $data['image'];
 | 
			
		||||
                                                }
 | 
			
		||||
 | 
			
		||||
                                                if (!empty($base64)) {
 | 
			
		||||
                                                    $record->update([
 | 
			
		||||
                                                        'image' => $base64,
 | 
			
		||||
                                                    ]);
 | 
			
		||||
 | 
			
		||||
                                                    Notification::make()
 | 
			
		||||
                                                        ->title(trans('admin/egg.import.image_updated'))
 | 
			
		||||
                                                        ->success()
 | 
			
		||||
                                                        ->send();
 | 
			
		||||
 | 
			
		||||
                                                    $record->refresh();
 | 
			
		||||
                                                } else {
 | 
			
		||||
                                                    Notification::make()
 | 
			
		||||
                                                        ->title(trans('admin/egg.import.no_image'))
 | 
			
		||||
                                                        ->warning()
 | 
			
		||||
                                                        ->send();
 | 
			
		||||
                                                }
 | 
			
		||||
                                            }),
 | 
			
		||||
                                        Action::make('deleteImage')
 | 
			
		||||
                                            ->visible(fn ($record) => $record->image)
 | 
			
		||||
                                            ->label('')
 | 
			
		||||
                                            ->icon('tabler-trash')
 | 
			
		||||
                                            ->iconButton()
 | 
			
		||||
                                            ->iconSize(IconSize::Large)
 | 
			
		||||
                                            ->color('danger')
 | 
			
		||||
                                            ->action(function ($record) {
 | 
			
		||||
 | 
			
		||||
                                                $record->update([
 | 
			
		||||
                                                    'image' => null,
 | 
			
		||||
                                                ]);
 | 
			
		||||
 | 
			
		||||
                                                Notification::make()
 | 
			
		||||
                                                    ->title(trans('admin/egg.import.image_deleted'))
 | 
			
		||||
                                                    ->success()
 | 
			
		||||
                                                    ->send();
 | 
			
		||||
 | 
			
		||||
                                                $record->refresh();
 | 
			
		||||
                                            }),
 | 
			
		||||
                                    ]),
 | 
			
		||||
                                ]),
 | 
			
		||||
                            TextInput::make('name')
 | 
			
		||||
                                ->label(trans('admin/egg.name'))
 | 
			
		||||
                                ->required()
 | 
			
		||||
                                ->maxLength(255)
 | 
			
		||||
                                ->columnSpan(['default' => 1, 'sm' => 1, 'md' => 2, 'lg' => 1])
 | 
			
		||||
                                ->columnSpan(['default' => 2, 'sm' => 2, 'md' => 3, 'lg' => 2])
 | 
			
		||||
                                ->helperText(trans('admin/egg.name_help')),
 | 
			
		||||
                            Textarea::make('description')
 | 
			
		||||
                                ->label(trans('admin/egg.description'))
 | 
			
		||||
                                ->rows(3)
 | 
			
		||||
                                ->columnSpan(['default' => 2, 'sm' => 2, 'md' => 4, 'lg' => 3])
 | 
			
		||||
                                ->helperText(trans('admin/egg.description_help')),
 | 
			
		||||
                            TextInput::make('id')
 | 
			
		||||
                                ->label(trans('admin/egg.egg_id'))
 | 
			
		||||
                                ->columnSpan(1)
 | 
			
		||||
                                ->disabled(),
 | 
			
		||||
                            TextInput::make('uuid')
 | 
			
		||||
                                ->label(trans('admin/egg.egg_uuid'))
 | 
			
		||||
                                ->disabled()
 | 
			
		||||
                                ->columnSpan(['default' => 1, 'sm' => 1, 'md' => 1, 'lg' => 2])
 | 
			
		||||
                                ->helperText(trans('admin/egg.uuid_help')),
 | 
			
		||||
                            TextInput::make('id')
 | 
			
		||||
                                ->label(trans('admin/egg.egg_id'))
 | 
			
		||||
                                ->disabled(),
 | 
			
		||||
                            Textarea::make('description')
 | 
			
		||||
                                ->label(trans('admin/egg.description'))
 | 
			
		||||
                                ->rows(3)
 | 
			
		||||
                                ->columnSpan(['default' => 1, 'sm' => 1, 'md' => 2, 'lg' => 2])
 | 
			
		||||
                                ->helperText(trans('admin/egg.description_help')),
 | 
			
		||||
                            TextInput::make('author')
 | 
			
		||||
                                ->label(trans('admin/egg.author'))
 | 
			
		||||
                                ->required()
 | 
			
		||||
                                ->maxLength(255)
 | 
			
		||||
                                ->email()
 | 
			
		||||
                                ->disabled()
 | 
			
		||||
                                ->columnSpan(['default' => 1, 'sm' => 1, 'md' => 2, 'lg' => 2])
 | 
			
		||||
                                ->columnSpan(['default' => 1, 'sm' => 1, 'md' => 1, 'lg' => 2])
 | 
			
		||||
                                ->helperText(trans('admin/egg.author_help_edit')),
 | 
			
		||||
                            Toggle::make('force_outgoing_ip')
 | 
			
		||||
                                ->inline(false)
 | 
			
		||||
                                ->label(trans('admin/egg.force_ip'))
 | 
			
		||||
                                ->columnSpan(1)
 | 
			
		||||
                                ->hintIcon('tabler-question-mark', trans('admin/egg.force_ip_help')),
 | 
			
		||||
                            KeyValue::make('startup_commands')
 | 
			
		||||
                                ->label(trans('admin/egg.startup_commands'))
 | 
			
		||||
                                ->live()
 | 
			
		||||
@ -93,24 +279,20 @@ class EditEgg extends EditRecord
 | 
			
		||||
                                ->label(trans('admin/egg.file_denylist'))
 | 
			
		||||
                                ->placeholder('denied-file.txt')
 | 
			
		||||
                                ->helperText(trans('admin/egg.file_denylist_help'))
 | 
			
		||||
                                ->columnSpan(['default' => 1, 'sm' => 1, 'md' => 2, 'lg' => 2]),
 | 
			
		||||
                            TagsInput::make('features')
 | 
			
		||||
                                ->label(trans('admin/egg.features'))
 | 
			
		||||
                                ->columnSpan(['default' => 1, 'sm' => 1, 'md' => 1, 'lg' => 1]),
 | 
			
		||||
                            Toggle::make('force_outgoing_ip')
 | 
			
		||||
                                ->inline(false)
 | 
			
		||||
                                ->label(trans('admin/egg.force_ip'))
 | 
			
		||||
                                ->hintIcon('tabler-question-mark', trans('admin/egg.force_ip_help')),
 | 
			
		||||
                            Hidden::make('script_is_privileged')
 | 
			
		||||
                                ->helperText('The docker images available to servers using this egg.'),
 | 
			
		||||
                            TagsInput::make('tags')
 | 
			
		||||
                                ->label(trans('admin/egg.tags'))
 | 
			
		||||
                                ->columnSpan(['default' => 1, 'sm' => 1, 'md' => 2, 'lg' => 2]),
 | 
			
		||||
                                ->columnSpan(['default' => 2, 'sm' => 2, 'md' => 2, 'lg' => 3]),
 | 
			
		||||
                            TextInput::make('update_url')
 | 
			
		||||
                                ->label(trans('admin/egg.update_url'))
 | 
			
		||||
                                ->url()
 | 
			
		||||
                                ->hintIcon('tabler-question-mark', trans('admin/egg.update_url_help'))
 | 
			
		||||
                                ->columnSpan(['default' => 1, 'sm' => 1, 'md' => 2, 'lg' => 2]),
 | 
			
		||||
                                ->columnSpan(['default' => 2, 'sm' => 2, 'md' => 2, 'lg' => 3]),
 | 
			
		||||
                            TagsInput::make('features')
 | 
			
		||||
                                ->label(trans('admin/egg.features'))
 | 
			
		||||
                                ->columnSpan(['default' => 2, 'sm' => 2, 'md' => 2, 'lg' => 3]),
 | 
			
		||||
                            Hidden::make('script_is_privileged')
 | 
			
		||||
                                ->helperText('The docker images available to servers using this egg.'),
 | 
			
		||||
                            TagsInput::make('tags')
 | 
			
		||||
                                ->label(trans('admin/egg.tags'))
 | 
			
		||||
                                ->columnSpan(['default' => 2, 'sm' => 2, 'md' => 2, 'lg' => 3]),
 | 
			
		||||
                            KeyValue::make('docker_images')
 | 
			
		||||
                                ->label(trans('admin/egg.docker_images'))
 | 
			
		||||
                                ->live()
 | 
			
		||||
 | 
			
		||||
@ -19,6 +19,7 @@ use Filament\Actions\DeleteBulkAction;
 | 
			
		||||
use Filament\Actions\EditAction;
 | 
			
		||||
use Filament\Actions\ReplicateAction;
 | 
			
		||||
use Filament\Resources\Pages\ListRecords;
 | 
			
		||||
use Filament\Tables\Columns\ImageColumn;
 | 
			
		||||
use Filament\Tables\Columns\TextColumn;
 | 
			
		||||
use Filament\Tables\Table;
 | 
			
		||||
use Illuminate\Support\Str;
 | 
			
		||||
@ -42,6 +43,13 @@ class ListEggs extends ListRecords
 | 
			
		||||
                TextColumn::make('id')
 | 
			
		||||
                    ->label('Id')
 | 
			
		||||
                    ->hidden(),
 | 
			
		||||
                ImageColumn::make('image')
 | 
			
		||||
                    ->label('')
 | 
			
		||||
                    ->alignCenter()
 | 
			
		||||
                    ->circular()
 | 
			
		||||
                    ->getStateUsing(fn ($record) => $record->image
 | 
			
		||||
                        ? $record->image
 | 
			
		||||
                        : 'data:image/svg+xml;base64,' . base64_encode(file_get_contents(public_path('pelican.svg')))),
 | 
			
		||||
                TextColumn::make('name')
 | 
			
		||||
                    ->label(trans('admin/egg.name'))
 | 
			
		||||
                    ->description(fn ($record): ?string => (strlen($record->description) > 120) ? substr($record->description, 0, 120).'...' : $record->description)
 | 
			
		||||
 | 
			
		||||
@ -25,6 +25,18 @@ class ServerEntry extends Component
 | 
			
		||||
            </div>
 | 
			
		||||
 | 
			
		||||
            <div class="flex-1 dark:bg-gray-800 dark:text-white rounded-lg overflow-hidden p-3">
 | 
			
		||||
                @if($server->egg->image)
 | 
			
		||||
                    <div style="
 | 
			
		||||
                        position: absolute;
 | 
			
		||||
                        inset: 0;
 | 
			
		||||
                        background: url('{{ $server->egg->image }}') right no-repeat;
 | 
			
		||||
                        background-size: contain;
 | 
			
		||||
                        opacity: 0.20;
 | 
			
		||||
                        max-width: 680px;
 | 
			
		||||
                        max-height: 140px;
 | 
			
		||||
                    "></div>
 | 
			
		||||
                @endif
 | 
			
		||||
 | 
			
		||||
                <div class="flex items-center mb-5 gap-2">
 | 
			
		||||
                    <x-filament::loading-indicator class="h-6 w-6" />
 | 
			
		||||
                    <h2 class="text-xl font-bold">
 | 
			
		||||
 | 
			
		||||
@ -21,6 +21,7 @@ use Illuminate\Support\Str;
 | 
			
		||||
 * @property string $author
 | 
			
		||||
 * @property string $name
 | 
			
		||||
 * @property string|null $description
 | 
			
		||||
 * @property string|null $image
 | 
			
		||||
 * @property string[]|null $features
 | 
			
		||||
 * @property array<string, string> $docker_images
 | 
			
		||||
 * @property string|null $update_url
 | 
			
		||||
@ -80,6 +81,7 @@ class Egg extends Model implements Validatable
 | 
			
		||||
        'name',
 | 
			
		||||
        'author',
 | 
			
		||||
        'description',
 | 
			
		||||
        'image',
 | 
			
		||||
        'features',
 | 
			
		||||
        'docker_images',
 | 
			
		||||
        'force_outgoing_ip',
 | 
			
		||||
@ -104,6 +106,7 @@ class Egg extends Model implements Validatable
 | 
			
		||||
        'uuid' => ['required', 'string', 'size:36'],
 | 
			
		||||
        'name' => ['required', 'string', 'max:255'],
 | 
			
		||||
        'description' => ['string', 'nullable'],
 | 
			
		||||
        'image' => ['string', 'nullable'],
 | 
			
		||||
        'features' => ['array', 'nullable'],
 | 
			
		||||
        'author' => ['required', 'string', 'email'],
 | 
			
		||||
        'file_denylist' => ['array', 'nullable'],
 | 
			
		||||
 | 
			
		||||
@ -29,6 +29,7 @@ class EggExporterService
 | 
			
		||||
            'author' => $egg->author,
 | 
			
		||||
            'uuid' => $egg->uuid,
 | 
			
		||||
            'description' => $egg->description,
 | 
			
		||||
            'image' => $egg->image,
 | 
			
		||||
            'tags' => $egg->tags,
 | 
			
		||||
            'features' => $egg->features,
 | 
			
		||||
            'docker_images' => $egg->docker_images,
 | 
			
		||||
 | 
			
		||||
@ -198,6 +198,7 @@ class EggImporterService
 | 
			
		||||
        return $model->forceFill([
 | 
			
		||||
            'name' => Arr::get($parsed, 'name'),
 | 
			
		||||
            'description' => Arr::get($parsed, 'description'),
 | 
			
		||||
            'image' => Arr::get($parsed, 'image'),
 | 
			
		||||
            'tags' => Arr::get($parsed, 'tags', []),
 | 
			
		||||
            'features' => Arr::get($parsed, 'features'),
 | 
			
		||||
            'docker_images' => Arr::get($parsed, 'docker_images'),
 | 
			
		||||
 | 
			
		||||
@ -46,6 +46,7 @@ class EggTransformer extends BaseTransformer
 | 
			
		||||
            'name' => $model->name,
 | 
			
		||||
            'author' => $model->author,
 | 
			
		||||
            'description' => $model->description,
 | 
			
		||||
            'image' => $model->image,
 | 
			
		||||
            'features' => $model->features,
 | 
			
		||||
            'tags' => $model->tags,
 | 
			
		||||
            'docker_image' => Arr::first($model->docker_images, default: ''), // docker_images, use startup_commands
 | 
			
		||||
 | 
			
		||||
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							@ -2,11 +2,12 @@ _comment: 'DO NOT EDIT: FILE GENERATED AUTOMATICALLY BY PANEL'
 | 
			
		||||
meta:
 | 
			
		||||
  version: PLCN_v3
 | 
			
		||||
  update_url: 'https://github.com/pelican-dev/panel/raw/main/database/Seeders/eggs/minecraft/egg-sponge.yaml'
 | 
			
		||||
exported_at: '2025-09-12T08:38:42+00:00'
 | 
			
		||||
exported_at: '2025-10-31T12:41:03+00:00'
 | 
			
		||||
name: Sponge
 | 
			
		||||
author: panel@example.com
 | 
			
		||||
uuid: f0d2f88f-1ff3-42a0-b03f-ac44c5571e6d
 | 
			
		||||
description: 'A community-driven open source Minecraft: Java Edition modding platform.'
 | 
			
		||||
image: 'data:image/svg+xml;base64,PHN2ZyB4bWxucz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciIHZpZXdCb3g9IjAgMCAyMDAgMjAwIiBmaWxsPSIjRjdDRjBEIj48cGF0aCBkPSJNMTkwIDBIMTBDNC41IDAgMCA0LjUgMCAxMHYxODBjMCA1LjUgNC41IDEwIDEwIDEwaDE2LjFjLTEuNy00NS43LS4xLTUyLjUgMy4xLTU3IDMuOS01LjYgNS41LTYuMyAxMS40LTExIDUtNCAzLjItMTAuNS0uNC0xNS4yLTIuMi0yLjktNS4zLTYuMy03LjctOS42LTEuNS0yLjIgMi4yLTE1LjEgMy42LTE5LjggMS40LTQuNyAzLjgtMjAgMjQuOC0yNC4xIDcuOS0xLjYgMjkuNi0yLjcgNDQuNS0xLjgtLjEtLjYtLjMtMS4zLS40LTItLjMtMS4yLS41LTIuNS0uOC0zLjktLjMtMS4zLS42LTIuNy0uOS00LjEtLjMtMS40LS43LTIuOC0xLTQuMy0uNC0xLjUtLjctMi45LTEuMi00LjQtLjgtMy0xLjgtNS45LTMtOC43LS42LTEuNC0xLjItMi43LTEuOS0zLjktLjctMS4xLTEuNC0yLjEtMi0yLjUtLjEtLjEtLjItLjItLjMtLjJoLS4xLjJzLjEgMCAwIDBsLS4zLS4xaC0uMmwtLjQtLjFoLS41Yy0xLjMtLjEtMi43LS4xLTQuMiAwLTIuOS4yLTYgLjgtOSAxLjVzLTUuOSAxLjYtOC43IDIuNGMtMS4yLjQtMi4zLjgtMy40IDEuMS4xLjkuMiAxLjcuMiAyLjYgMCAxMy0xMC41IDIzLjUtMjMuNSAyMy41UzIwLjYgNDcuOSAyMC42IDM0LjlzMTAuNS0yMy41IDIzLjUtMjMuNWM4LjcgMCAxNi4zIDQuNyAyMC40IDExLjggMS0uNCAyLjEtLjggMy4yLTEuMiAyLjgtMS4xIDUuOS0yLjIgOS4xLTMuMiAzLjMtMSA2LjctMiAxMC41LTIuNSAxLjktLjMgMy45LS40IDYuMS0uM2guOGMuMyAwIC42LjEuOC4xSDk1LjdsLjMuMWguMWwuMy4xcy4yIDAgLjMuMWwuNC4xYy42LjIuOS4zIDEuMy41cy43LjMgMS4xLjVjLjcuNCAxLjMuOCAxLjkgMS4yIDEuMS45IDIgMS44IDIuNyAyLjcuOC45IDEuNCAxLjggMiAyLjcgMS4yIDEuOCAyLjEgMy41IDIuOSA1LjIgMS42IDMuNCAyLjkgNi44IDMuOSAxMGwxLjUgNC44Yy41IDEuNi44IDMuMSAxLjIgNC42LjIuNy40IDEuNS41IDIuMi4yLjcuMyAxLjQuNSAyLjEuMyAxLjQuNiAyLjguOSA0LjEuNCAyIC43IDMuOSAxIDUuNiAyMi40IDIuMiAzOS41IDUuMSA0Ny4yIDEyLjggMTEuMyAxMSAyMCA2MSAxNC4zIDEyNC41aDEwYzUuNSAwIDEwLTQuNSAxMC0xMFYxMGMwLTUuNS00LjUtMTAtMTAtMTB6Ii8+PHBhdGggZD0iTTkxLjQgMTQwLjhjLTEuMyAzLjYtMi40IDQ1LjcgMTAgNDUuN3MxMi41LTQzLjIgMTIuMS00NS43Yy0uNC0yLjQtMjAuOC0zLjUtMjIuMSAwek03NSAxMDBjLTguNS0xLjItMTMuNiA0MC4yLTEuNyA0Mi42IDExLjIgMi4yIDEwLjEtNDEuNCAxLjctNDIuNnpNMTMwLjggMTAwYy04LjUtMS4yLTEzLjYgNDAuMi0xLjcgNDIuNiAxMS4yIDIuMiAxMC4yLTQxLjQgMS43LTQyLjZ6Ii8+PC9zdmc+'
 | 
			
		||||
tags:
 | 
			
		||||
  - minecraft
 | 
			
		||||
features:
 | 
			
		||||
 | 
			
		||||
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							@ -2,13 +2,14 @@ _comment: 'DO NOT EDIT: FILE GENERATED AUTOMATICALLY BY PANEL'
 | 
			
		||||
meta:
 | 
			
		||||
  version: PLCN_v3
 | 
			
		||||
  update_url: 'https://github.com/pelican-dev/panel/raw/main/database/Seeders/eggs/source-engine/egg-custom-source-engine-game.yaml'
 | 
			
		||||
exported_at: '2025-09-05T08:55:22+00:00'
 | 
			
		||||
exported_at: '2025-10-31T12:43:00+00:00'
 | 
			
		||||
name: 'Custom Source Engine Game'
 | 
			
		||||
author: panel@example.com
 | 
			
		||||
uuid: 2a42d0c2-c0ba-4067-9a0a-9b95d77a3490
 | 
			
		||||
description: |-
 | 
			
		||||
  This option allows modifying the startup arguments and other details to run a custom SRCDS based
 | 
			
		||||
  game on the panel.
 | 
			
		||||
image: 'data:image/webp;base64,UklGRoAMAABXRUJQVlA4THMMAAAv+QATECUwbtvIkdx/2ZNnLnwjYgLYqi8KnDuYekQOBtxU1VNGcZdt8CWWtnXLbdOCTsNlE6Ix5eSRaVdUVr1HA0Dlx+NLkmTVtm3binROhla09D55Lsi/tN5reIlWJ3xHSKBt7dii9/uN8Cv8yLaxjGXbrrFt20YY2whj2zZy/bbVK8FtI0mS6JnZe9d1ZWVW7QskSZIcN1L9/zwbHDaPlYyAqtAy1NVBG0mOnCyW92E8+9FhG0mKVBkf/fjgKXVZkCSbttXXtm3btm3bfrZt27b5dW3btm3uuyBIkuSoyWLwSHjPBwr8a2Y+MUMVHtiDh/iHYtShBa1oQCUqUYv6V2h4HQO0ogXt6EbXK/Rj4BUGMfYOk5h5hwUsYRGLWMb6O2yBgGATs2jDT6RbYlG4eWMvZksTu/EbXVgF+bEwjZsQvbV53+lhjBtoBwXyw2EUAVfG8CEddaBAviBs48h9QRSXMAbyLeHAZc3HiwfHMAHyPWHLcrsqGKIT5KtCh8V1UWZ4XvNl4dBFIfbjOYYVdOE/bmEPwuEBB9jBGQFIxll8QC1mzkPXhP2h2DWzJtas8qjUohQoemVdDC7uoR8vEQrFf+dh2h+rP78/jSUCe5xHJTYOL+D8UDZVX+9qfHHqdc0oUMvrzcVgzwbm8I5xAlvumP153XRQx5mzt0Rcex7bwtelohao2/WqmPWhbl8Mo4D5JwzjBCNROXJjsSIQpQch86kIdaVohdSJ2lLPanOdv9kOiMfmGyhcsXirx25eQw8fNByDeoPhqa491cUKrpMVXp1RuFljT2M8RQ7ejD3tThy9cU4W7MPcIWj3xPpQuFpdlgVqVTEKFLvWR6GF5owics9A4zgnUxQ6aj4PRpwHdQJyotBUCMHCAbjXlSjACpN1hk9bogBDjFVhZM7w9iUKMKv+R9yPQmfBq/YsGlswaE0UsK8EuVFoLnyuQHh3IIK+PPRxcRe9ozYeiQUyAvSLNgYBGQn+R+Rv4gc0EtIb+mTjEJGRFvZFG7XxSEhwvyIv1EJgHi4XeAFH5bbr9E4vPjG+Pu912qoYa4itQlv1+GyQEE+vl85bIGYRs03i2IIDlbFrbTihPKFzaiUW4YKlUsd8wMX3XmD3f4S7Jh0e+G2E0Kgf7juobrkMvWGFmeuc1/rwcf310gkt3H5GCgyDn1lYg9rYG0aRW5RgDpmrUr9HjhYixhmi2MyYP1Yvh0QRLdkl5AZYZ27RXOvQW/ukle7agULC50llit5rfelXD5xi92jBFCp3lW7cGYrpPjKlZjRuSRuCFpaT8D89Aj0CBzGmk8sBT5eYz9VIrdjkTkU4VCiZZ72ObU9O9A475Ylz2FCpF2K8As/xTvLB94hqfecd4/J2KjAKniTBN2vlkYKzGBUbqSWZVfOGX6M8HIUDTztbnxzvdnIBAfrB7BeUhfTgJIzWbUyyKU6NmhmROgJVrObe8J3zYk1aMRTgNAQCA4vH1fDF/RV2wmb4C2ZfIPaB0zj2wHSYXg2LUSssMAm+pOBUct7w3oHzeAHXmPNCNUJhfwFJzoAHtoPz2N5sqtfjYhRiJ+CUgWVLMUk12JE7Y4ZShbvGcnfpQhPNV7KYImYaOjDGj824HaPNDAcawIiBlinXUHQB5XK++GjyJen3YiSMczKhJQE/opCks/OVZ2siytfOy4vF8B/smNzWLJIXPylW+A2GaL0dMzkYScbEC+idUyKIl3A1rtIHviF+Mi3da5xSLpQnP2k2Gg5N7C0wCC4nwC3pQTHUKQhdeJM0dHw+WZ8GW7mtGbJ1zOH0EH3bD463BAKfPPTYJ3wx6Tf5L761XQkGQhfVH7DxwlW//FqzZeNK+HcFDcDyMzSNgzlp+dH2I70CUhlteYu3PUEYTFZwqmYu2OPSX4LRRambdpl5CaLqDPMKdYVTdHo+2NJLePZmFZohuN/LODIADvSf+vxWFNijJlFmVN8jXDM/mAF2GnPhcNHT8PoBeDJ7M2vBTlnsa324rbEpgGGkD398YzOhtA3J5hQ4B78+wRyk6gwKDR5oli2L8CHMdrxZqExoGAbMvh+5yDAR9r3zoEDEO0wkxRRQ0ys4WCDH1V5H0wAOfYI3UcjilgY6Cq07fLRM4UTRmXoq5H7nzv1MLtAJYQrXc/ZAeUyEdwm53AIudFIO1HYuD8D5E9ikfYt4my97P7Sq57vIJ3J6KVo+n1I7G0S5my/v+MD43oWY/WGbWGKgX4Kx8Ij+92YoB8uJtvFvk8/mjBrW9lDx/7bSRSGdKeApTHp1WSIZNbGw829jTySF2wOfu1mYimnvPC8wDm+G/k846ZjjKY45Td3PEh5n4MPEHtKikPeO83C0TCJwRR36eae4W+BkO4RBQNrNjAmNi8jOoTwp4OuRdvbr/6dD9xbGx7mEopCo/GQczwa2F651zhyhEoW70c2MpGowPnRW5Km9gcZBV1Rt4VEUSrG0iw6cxbDcw+mg/VzXYGfCzXxMvd4IfnVm5Zl0AzMGkL8DyjKKQjH+9uLOtoVR/Qe3O21HLAx0Js55NuNugux/Z1qeyTewYADZOyiIQj00+dp8mPxxahFPrndOnX9A2Kkbe0e8mdGVIPjZWTSn4QbmDiBvB7EFbskbZrkqs9ULYvAQ3U8Od24dbTZNJyshR41gfJ1xPYH2rrNzTnaYsitXoHAWJVwHULH7KMviK3Bftr0Mu/2LaHkUsDSsRH9An4gHO34J1bKPEAy+rICusDA4gUNhfQkKrMAw02Bd4Lb8fyy06NjArSjcm01U7H36UF5AQbjkeEZ5HWEEvYDSjp53hEPxOts7A5t0YDElwSUU5m8ybh/shNzH4MLYCpvQKdk0tGvykYR3KjC7ALewijP+xM3OwwIF5IWiMiPp4OuUhGYSD9xJ6ydPegUtN/7XDSzvvoxU4j+u0c8QeIkQyyw2UWd+gQJehNXpn8jhPkQT54ISAosiO2/AgbRzNEpHmXp7pqvczRMxlpj/GYtVDsB2sxnBNXr7wMAlgvYtTREKWJAuuSycj/QCnvilx8A/otPUzOjU8EkUrbDZxed1gQI7iKFP5yVkb4Q+FFg1gJ0r9Mw5OGp0zW5K7Ak7GR/nA0kYSHjNZja6PwfkUBTU0H2DOA7eQX6o5jXrQCZ9ONXzFM7XbZc80H0yT8l9AJ9WOBeFGtM6IqFBOrPJsQ8Xi14JfT1vEUPHfHkV4WNa9Ktof8YZ8BYwnoVuHnLsz2NIwr7A+SAtNPM5nI+89VReaDCs6yGxMvZULuIQhhJcVQt3tUiWMB5ucptlKD2TdUk5L0zsHW522sGgk5eu9saHwp5ZqVmzRS6SgvcoIIU5vVewSRwOJm+1bqUm1l+qOR960Lighj8WJm/lzlCG48tzAPrYbvArCtVsTvvFN9tUSZOqyq44kZDTR/64Q6locw29i3UhDRzkLgVsyf3oKJnzk/RNvZ4UBvPylYbK2GHKPIAT62Z4lKH6DhZyoXWz6oVmZO3lzsJ4P8LowS02TieYjkpcPx+p5tVc/xE94P2iXV8honm8TuYolBOCP+fLsuvdbBwjlRGGyBFtim5RYFHcnKJLtzAFlFNebFqBBJiuqzgchQMJ9XfKS47xiRHVjG/CUzNbB1x7IEruUWCTtA5fFcrOM3T/ZtxELzCBZwvMW9JROBGGg7QTlDYsi9xNsDuTq8C2vCzaiC3MB8z2IBZQNnt7rXOy4Yr9PmN29bQAYquVfvAyz9IHvsAEZDC/wMcoHEqAfQZODNjK7VgNWprrVJkhu8/DuUyv4wDohfTLkMz4CLT5ydHewZXRdrMy2UfmfCxdk7yRczXBDhMeHlXZs9wKzIJr6+axp0MUjvWwlGKZSz7qxycmxtPnvfMWiWcxqvYVwlG9Vs/9xyUiNOyHm9bKYDkptrFirXXHb6NERAa9sk3845PnZbE9tvuj7fHxupNd74A53O6NFW6BK74aJiDG0+uVdo2chtUWQ7EdHvprlIgY118P7VCwaQWGgTxmFqiezxd9FI72J7LJ23axsbYN0ovdHoTNwGK/KRv9Sq93zKZjYrIs8FcFijX7xd42i/hiNTh0TCx0uFd64dW6ioQodBucQC3QaU+ubs0ZHjSvq0hvdxVP1kGLGfZuIQFkhcBuBzZYXuFvtwOtzdt+mIVKszjV0AfS9WFguBt8aXagiLHNoHmcU6BZeLAZTEE7Cs1C9ebn8XCNQrdQvMAGQqPQLpxaBhFR6Ne/c/Gg9gEZUeg4QwRvsQiC9rM/jCoAAA=='
 | 
			
		||||
tags:
 | 
			
		||||
  - source
 | 
			
		||||
  - steamcmd
 | 
			
		||||
 | 
			
		||||
@ -2,13 +2,14 @@ _comment: 'DO NOT EDIT: FILE GENERATED AUTOMATICALLY BY PANEL'
 | 
			
		||||
meta:
 | 
			
		||||
  version: PLCN_v3
 | 
			
		||||
  update_url: 'https://github.com/pelican-dev/panel/raw/main/database/Seeders/eggs/source-engine/egg-garrys-mod.yaml'
 | 
			
		||||
exported_at: '2025-09-05T08:54:21+00:00'
 | 
			
		||||
exported_at: '2025-10-31T12:37:53+00:00'
 | 
			
		||||
name: 'Garrys Mod'
 | 
			
		||||
author: panel@example.com
 | 
			
		||||
uuid: 60ef81d4-30a2-4d98-ab64-f59c69e2f915
 | 
			
		||||
description: |-
 | 
			
		||||
  Garrys Mod, is a sandbox physics game created by Garry Newman, and developed by his company,
 | 
			
		||||
  Facepunch Studios.
 | 
			
		||||
image: 'data:image/svg+xml;base64,PD94bWwgdmVyc2lvbj0iMS4wIiBlbmNvZGluZz0idXRmLTgiPz4NCjwhLS0gR2VuZXJhdG9yOiBBZG9iZSBJbGx1c3RyYXRvciAyNC4zLjAsIFNWRyBFeHBvcnQgUGx1Zy1JbiAuIFNWRyBWZXJzaW9uOiA2LjAwIEJ1aWxkIDApICAtLT4NCjxzdmcgdmVyc2lvbj0iMS4xIiBpZD0iTGF5ZXJfMSIgeG1sbnM9Imh0dHA6Ly93d3cudzMub3JnLzIwMDAvc3ZnIiB4bWxuczp4bGluaz0iaHR0cDovL3d3dy53My5vcmcvMTk5OS94bGluayIgeD0iMHB4IiB5PSIwcHgiDQoJIHZpZXdCb3g9IjAgMCAzODQgMzg0IiBzdHlsZT0iZW5hYmxlLWJhY2tncm91bmQ6bmV3IDAgMCAzODQgMzg0OyIgeG1sOnNwYWNlPSJwcmVzZXJ2ZSI+DQo8c3R5bGUgdHlwZT0idGV4dC9jc3MiPg0KCS5zdDB7ZmlsbDojMDA4MUZGO30NCgkuc3Qxe2ZpbGw6I0ZGRkZGRjt9DQo8L3N0eWxlPg0KPHBhdGggY2xhc3M9InN0MCIgZD0iTTM0My45MiwzODRjLTEwMS42LDAtMjAzLjIsMC0zMDQuOCwwYy0wLjg5LTAuNDQtMS44NC0wLjMxLTIuNzktMC4zOGMtNC41MS0wLjM2LTguNjEtMi4wNy0xMi41Ni00LjEzDQoJYy01LjktMy4wNy0xMC44NC03LjMxLTE0LjkyLTEyLjU0Yy0zLjkzLTUuMDQtNi43My0xMC42NS04LjE3LTE2LjljLTAuMzktMS42OC0wLjMxLTMuNDEtMC40MS01LjEyYy0wLjAyLTAuMzQsMC4wOS0wLjc0LTAuMjctMQ0KCUMwLDI0Mi42NCwwLDE0MS4zNiwwLDQwLjA4YzAuNDMtMC42NywwLjI1LTEuNDQsMC4yNy0yLjE1YzAuMTEtNC42NCwxLjU3LTguOTIsMy42My0xMi45N0M5LjIyLDE0LjQ4LDE3LjI1LDYuODgsMjguMjUsMi40Ng0KCWMzLjM2LTEuMzUsNi44MS0yLjA5LDEwLjQtMi4yN0MzOS4xMywwLjE2LDM5LjY3LDAuNiw0MC4wOCwwQzE0MS45MiwwLDI0My43NiwwLDM0NS42LDBjMC43MSwwLjc3LDEuNjgsMC4zNiwyLjQ5LDAuNDINCgljMS45MiwwLjE1LDMuNzQsMC42NCw1LjUzLDEuMjVjNy4zMSwyLjQ2LDEzLjU4LDYuNTYsMTguODIsMTIuMmM1LjM5LDUuOCw5LjEyLDEyLjUsMTAuOSwyMC4yNWMwLjM4LDEuNjYsMC4yOSwzLjM3LDAuMzksNS4wNg0KCWMwLjAyLDAuMzItMC4wNywwLjY3LDAuMjYsMC45YzAsMTAxLjI4LDAsMjAyLjU2LDAsMzAzLjg0Yy0wLjU4LDAuOC0wLjM1LDEuNzYtMC4zNCwyLjU5YzAuMDMsMi43LTAuNjMsNS4yNC0xLjUzLDcuNzENCgljLTQuMjMsMTEuNTUtMTEuODgsMjAuMTItMjIuOCwyNS43M2MtMy4xMywxLjYxLTYuNDIsMi45MS05Ljk2LDMuNDNjLTEuNTIsMC4yMi0zLjA0LDAuMjYtNC41NiwwLjMzDQoJQzM0NC40OCwzODMuNzMsMzQ0LjE1LDM4My42OCwzNDMuOTIsMzg0eiIvPg0KPHBhdGggY2xhc3M9InN0MSIgZD0iTTIzNS41NywyMjEuOTFjLTIuODQsMy40LTUuNjgsNy41MS05LjIyLDEwLjg4Yy03LjIsNi44NC0xNi4yLDEwLjQxLTI1Ljc3LDEyLjQ5DQoJYy0xMS41MywyLjUxLTIzLjEzLDIuMTgtMzQuNzYsMC42MWMtMTQuNTItMS45NS0yNy4yNy03LjgzLTM4LjM4LTE3LjIyYy05LjM1LTcuOTEtMTYuMDYtMTcuOC0yMC44NC0yOS4wNA0KCWMtNC44OC0xMS40Ni03LjQtMjMuNTEtOC42OC0zNS44MWMtMS44NS0xNy43OC0wLjk5LTM1LjQyLDQuMzMtNTIuNmM0LjM3LTE0LjExLDExLjAxLTI3LDIxLjA4LTM4LjAzDQoJQzEzNSw2MC40LDE0OS4xOCw1Mi4wOCwxNjYuMSw0OC41OWM5LjQ2LTEuOTUsMTkuMDEtMS45NSwyOC41Ny0wLjRjMTcsMi43NywzMC4xMywxMS40OSwzOS43NSwyNS42OGMwLjI1LDAuMzIsMS4xOSwxLjcyLDEuMTksMS43Mg0KCXMwLTQuNzgsMC02LjgzYy0wLjAxLTUuMTYsMC0xNi41MSwwLTE2LjUxczEuMDksMCwxLjM1LDBjMTUuMjgtMC4wNSw0OS45Ny0wLjAxLDQ5Ljk3LTAuMDFzLTAuMDEsMS45Ni0wLjAxLDQuMjgNCgljMCw2NC40LDAuMDksMTI4LjgtMC4wNSwxOTMuMTljLTAuMDQsMjAuMzYtNS43OSwzOS0xOC44NCw1NC45NWMtMTAuMTUsMTIuNC0yMy4yNSwyMC41Ny0zOC4yNSwyNS44NQ0KCWMtMTQuNiw1LjE0LTI5LjcsNi45MS00NS4wOCw2LjI2Yy0xNi44MS0wLjcxLTMyLjk5LTQuMzUtNDcuNzUtMTIuNjRjLTIyLjIyLTEyLjQ5LTM1LjcyLTMxLjIyLTM5LjA3LTU2Ljc4DQoJYy0wLjQ1LTMuMzktMC41Ni02Ljg0LTAuNjktMTAuMjZjLTAuMDEtMC4yMS0wLjA3LTEuMzEtMC4wNy0xLjMxczEuMDQsMCwxLjI2LDBjMTUuMjQtMC4wNywzMC40OC0wLjA4LDQ1LjcyLTAuMDMNCgljMC4xOCwwLDAuOSwwLjAxLDAuOSwwLjAxcy0wLjAxLDAuOTYtMC4wMSwxLjEyYzAuMTYsOC44NiwyLjYyLDE2Ljg2LDguNzcsMjMuNDRjNC41OSw0LjkxLDEwLjI2LDguMTYsMTYuNyw5Ljg5DQoJYzE0Ljk2LDQuMDMsMjkuNTksMy4xOSw0My40OS0zLjk4YzEyLjUtNi40NSwyMC40LTE2LjU0LDIxLjMtMzAuODVjMC42Ny0xMC43OSwwLjI3LTIxLjY2LDAuMzQtMzIuNDkNCglDMjM1LjU3LDIyMi4zNSwyMzUuNTcsMjIxLjc5LDIzNS41NywyMjEuOTF6Ii8+DQo8cGF0aCBjbGFzcz0ic3QwIiBkPSJNMTUwLjc3LDE0OS44MmMtMC4zLTExLjc5LDAuOTMtMjMuMzcsNS41NS0zNC4zNWM1LjQzLTEyLjg5LDE0LjkxLTIwLjc0LDI4LjktMjIuODgNCgljOS42Ny0xLjQ4LDE5LjA4LTAuODUsMjcuOTksMy40N2M4LjksNC4zMiwxNC4zOSwxMS43LDE3Ljk5LDIwLjY0YzMuNTMsOC43NSw1LjMzLDE3Ljk0LDUuNDYsMjcuMzFjMC4xMiw4LjM4LDAuMDksMTYuODktMS4yLDI1LjE0DQoJYy0yLjU5LDE2LjU3LTE0LjA5LDMyLjA4LTMxLjczLDM2LjRjLTE2LjA5LDMuOTQtMzIuNjktMi4yMy00Mi4yOC0xNS44MWMtNi43NC05LjUzLTkuODUtMjAuMjEtMTAuNjYtMzEuNjgNCglDMTUwLjYxLDE1NS4zMywxNTAuNzcsMTUyLjU3LDE1MC43NywxNDkuODJ6Ii8+DQo8L3N2Zz4NCg=='
 | 
			
		||||
tags:
 | 
			
		||||
  - source
 | 
			
		||||
  - steamcmd
 | 
			
		||||
 | 
			
		||||
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							@ -2,13 +2,14 @@ _comment: 'DO NOT EDIT: FILE GENERATED AUTOMATICALLY BY PANEL'
 | 
			
		||||
meta:
 | 
			
		||||
  version: PLCN_v3
 | 
			
		||||
  update_url: 'https://github.com/pelican-dev/panel/raw/main/database/Seeders/eggs/source-engine/egg-team-fortress2.yaml'
 | 
			
		||||
exported_at: '2025-09-05T08:55:44+00:00'
 | 
			
		||||
exported_at: '2025-10-31T12:31:09+00:00'
 | 
			
		||||
name: 'Team Fortress 2'
 | 
			
		||||
author: panel@example.com
 | 
			
		||||
uuid: 7f8eb681-b2c8-4bf8-b9f4-d79ff70b6e5d
 | 
			
		||||
description: |-
 | 
			
		||||
  Team Fortress 2 is a team-based first-person shooter multiplayer video game developed and published
 | 
			
		||||
  by Valve Corporation. It is the sequel to the 1996 mod Team Fortress for Quake and its 1999 remake.
 | 
			
		||||
image: 'data:image/svg+xml;base64,PD94bWwgdmVyc2lvbj0iMS4wIiBlbmNvZGluZz0idXRmLTgiPz4NCjwhLS0gR2VuZXJhdG9yOiBBZG9iZSBJbGx1c3RyYXRvciAxNi4wLjAsIFNWRyBFeHBvcnQgUGx1Zy1JbiAuIFNWRyBWZXJzaW9uOiA2LjAwIEJ1aWxkIDApICAtLT4NCjwhRE9DVFlQRSBzdmcgUFVCTElDICItLy9XM0MvL0RURCBTVkcgMS4xLy9FTiIgImh0dHA6Ly93d3cudzMub3JnL0dyYXBoaWNzL1NWRy8xLjEvRFREL3N2ZzExLmR0ZCI+DQo8c3ZnIHZlcnNpb249IjEuMSIgaWQ9IkxheWVyXzEiIHhtbG5zPSJodHRwOi8vd3d3LnczLm9yZy8yMDAwL3N2ZyIgeG1sbnM6eGxpbms9Imh0dHA6Ly93d3cudzMub3JnLzE5OTkveGxpbmsiIHg9IjBweCIgeT0iMHB4Ig0KCSB3aWR0aD0iNTAwcHgiIGhlaWdodD0iNTAwLjAwOXB4IiB2aWV3Qm94PSItNTAgLTUwLjAwNSA1MDAgNTAwLjAwOSIgZW5hYmxlLWJhY2tncm91bmQ9Im5ldyAtNTAgLTUwLjAwNSA1MDAgNTAwLjAwOSINCgkgeG1sOnNwYWNlPSJwcmVzZXJ2ZSI+DQo8bGluZWFyR3JhZGllbnQgaWQ9IlNWR0lEXzFfIiBncmFkaWVudFVuaXRzPSJ1c2VyU3BhY2VPblVzZSIgeDE9IjI1NS4zOTk0IiB5MT0iLTQzMy44NDM4IiB4Mj0iMTQ0LjU5ODkiIHkyPSI1My44NDUxIiBncmFkaWVudFRyYW5zZm9ybT0ibWF0cml4KDEgMCAwIC0xIDAgMTApIj4NCgk8c3RvcCAgb2Zmc2V0PSIwIiBzdHlsZT0ic3RvcC1jb2xvcjojNUQxRjBFIi8+DQoJPHN0b3AgIG9mZnNldD0iMSIgc3R5bGU9InN0b3AtY29sb3I6I0QwOTczNyIvPg0KPC9saW5lYXJHcmFkaWVudD4NCjxwYXRoIGZpbGw9InVybCgjU1ZHSURfMV8pIiBkPSJNMjYyLjA0NC00Mi4yMzlDMzcwLjA5Ny0xNC42NDQsNDUwLDgzLjM0NCw0NTAsMTk5Ljk5OGMwLDIuNjYxLTAuMDU3LDUuMzExLTAuMTQzLDcuOTUxDQoJbC0xNjguMTUxLTIzLjkxNmMtNC45MjItMjUuMzM4LTIxLjMzMy00Ni41NTYtNDMuNTk0LTU4LjA0NUwyNjIuMDQ0LTQyLjIzOXogTTEyNS45OSwxNjEuODgNCgljMTEuNDg4LTIyLjI2MSwzMi43MDctMzguNjcsNTguMDQzLTQzLjU5M2wyMy45Mi0xNjguMTUzYy0yLjY0My0wLjA4My01LjI5LTAuMTM5LTcuOTUzLTAuMTM5DQoJYy0xMTYuNjUyLDAtMjE0LjYzOSw3OS44OTgtMjQyLjIzNSwxODcuOTUzTDEyNS45OSwxNjEuODh6IE0xNjEuODgzLDI3NC4wMDhjLTIyLjI1OS0xMS40ODktMzguNjctMzIuNzEtNDMuNTkyLTU4LjA0NQ0KCWwtMTY4LjE1Mi0yMy45MmMtMC4wODMsMi42NDMtMC4xMzksNS4yOTMtMC4xMzksNy45NTVjMCwxMTYuNjQ4LDc5Ljg5OCwyMTQuNjM3LDE4Ny45NTIsMjQyLjIzM0wxNjEuODgzLDI3NC4wMDh6IE0yNzQuMDEsMjM4LjExMw0KCWMtMTEuNDksMjIuMjYxLTMyLjcwNywzOC42NjktNTguMDQ2LDQzLjU5M2wtMjMuOTE5LDE2OC4xNThjMi42NDMsMC4wODMsNS4yOTIsMC4xNCw3Ljk1NCwwLjE0DQoJYzExNi42NTMsMCwyMTQuNjQtNzkuOTAxLDI0Mi4yMzItMTg3Ljk1NUwyNzQuMDEsMjM4LjExM3oiLz4NCjxyYWRpYWxHcmFkaWVudCBpZD0iU1ZHSURfMl8iIGN4PSI5OC4xOTE5IiBjeT0iLTE5OC4zNTYiIHI9IjQyNS45ODcxIiBmeD0iOTIuNzU2NSIgZnk9Ii0xOTkuNDgzNCIgZ3JhZGllbnRUcmFuc2Zvcm09Im1hdHJpeCgwLjk3NzQgMC4yMTE0IDAuMTI2NiAtMC41ODUxIC03Ny4wMzAxIC0xNjcuMzcwNykiIGdyYWRpZW50VW5pdHM9InVzZXJTcGFjZU9uVXNlIj4NCgk8c3RvcCAgb2Zmc2V0PSIwIiBzdHlsZT0ic3RvcC1jb2xvcjojRkZGRkZGO3N0b3Atb3BhY2l0eTowIi8+DQoJPHN0b3AgIG9mZnNldD0iMC40NjU3IiBzdHlsZT0ic3RvcC1jb2xvcjojRkZGRkZGO3N0b3Atb3BhY2l0eTowLjA4MDQiLz4NCgk8c3RvcCAgb2Zmc2V0PSIwLjk4NjEiIHN0eWxlPSJzdG9wLWNvbG9yOiNGRkZGRkY7c3RvcC1vcGFjaXR5OjAuMjE0MyIvPg0KCTxzdG9wICBvZmZzZXQ9IjEiIHN0eWxlPSJzdG9wLWNvbG9yOiNGRkZGRkY7c3RvcC1vcGFjaXR5OjAiLz4NCjwvcmFkaWFsR3JhZGllbnQ+DQo8cGF0aCBmaWxsPSJ1cmwoI1NWR0lEXzJfKSIgZD0iTTI2Mi4wNDQtNDIuMjM5QzM3MC4wOTctMTQuNjQ0LDQ1MCw4My4zNDQsNDUwLDE5OS45OThjMCwyLjY2MS0wLjA1Nyw1LjMxMS0wLjE0Myw3Ljk1MQ0KCWwtMTY4LjE1MS0yMy45MTZjLTQuOTIyLTI1LjMzOC0yMS4zMzMtNDYuNTU2LTQzLjU5NC01OC4wNDVMMjYyLjA0NC00Mi4yMzl6IE0xMjUuOTksMTYxLjg4DQoJYzExLjQ4OC0yMi4yNjEsMzIuNzA3LTM4LjY3LDU4LjA0My00My41OTNsMjMuOTItMTY4LjE1M2MtMi42NDMtMC4wODMtNS4yOS0wLjEzOS03Ljk1My0wLjEzOQ0KCWMtMTE2LjY1MiwwLTIxNC42MzksNzkuODk4LTI0Mi4yMzUsMTg3Ljk1M0wxMjUuOTksMTYxLjg4eiBNMTYxLjg4MywyNzQuMDA4Yy0yMi4yNTktMTEuNDg5LTM4LjY3LTMyLjcxLTQzLjU5Mi01OC4wNDUNCglsLTE2OC4xNTItMjMuOTJjLTAuMDgzLDIuNjQzLTAuMTM5LDUuMjkzLTAuMTM5LDcuOTU1YzAsMTE2LjY0OCw3OS44OTgsMjE0LjYzNywxODcuOTUyLDI0Mi4yMzNMMTYxLjg4MywyNzQuMDA4eiBNMjc0LjAxLDIzOC4xMTMNCgljLTExLjQ5LDIyLjI2MS0zMi43MDcsMzguNjY5LTU4LjA0Niw0My41OTNsLTIzLjkxOSwxNjguMTU4YzIuNjQzLDAuMDgzLDUuMjkyLDAuMTQsNy45NTQsMC4xNA0KCWMxMTYuNjUzLDAsMjE0LjY0LTc5LjkwMSwyNDIuMjMyLTE4Ny45NTVMMjc0LjAxLDIzOC4xMTN6Ii8+DQo8L3N2Zz4NCg=='
 | 
			
		||||
tags:
 | 
			
		||||
  - source
 | 
			
		||||
  - steamcmd
 | 
			
		||||
 | 
			
		||||
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							@ -0,0 +1,28 @@
 | 
			
		||||
<?php
 | 
			
		||||
 | 
			
		||||
use Illuminate\Database\Migrations\Migration;
 | 
			
		||||
use Illuminate\Database\Schema\Blueprint;
 | 
			
		||||
use Illuminate\Support\Facades\Schema;
 | 
			
		||||
 | 
			
		||||
return new class extends Migration
 | 
			
		||||
{
 | 
			
		||||
    /**
 | 
			
		||||
     * Run the migrations.
 | 
			
		||||
     */
 | 
			
		||||
    public function up(): void
 | 
			
		||||
    {
 | 
			
		||||
        Schema::table('eggs', function (Blueprint $table) {
 | 
			
		||||
            $table->longText('image')->nullable();
 | 
			
		||||
        });
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /**
 | 
			
		||||
     * Reverse the migrations.
 | 
			
		||||
     */
 | 
			
		||||
    public function down(): void
 | 
			
		||||
    {
 | 
			
		||||
        Schema::table('eggs', function (Blueprint $table) {
 | 
			
		||||
            $table->dropColumn('image');
 | 
			
		||||
        });
 | 
			
		||||
    }
 | 
			
		||||
};
 | 
			
		||||
@ -13,6 +13,9 @@ return [
 | 
			
		||||
    'import' => [
 | 
			
		||||
        'file' => 'File',
 | 
			
		||||
        'url' => 'URL',
 | 
			
		||||
        'image_url' => 'Image URL',
 | 
			
		||||
        'image_error' => 'Could not fetch image',
 | 
			
		||||
        'image_too_large' => 'Image too large. Limit is 1024KB',
 | 
			
		||||
        'egg_help' => 'This should be the raw .json/.yaml file',
 | 
			
		||||
        'url_help' => 'URLs must point directly to the raw .json/.yaml file',
 | 
			
		||||
        'add_url' => 'New URL',
 | 
			
		||||
@ -20,6 +23,13 @@ return [
 | 
			
		||||
        'import_success' => 'Import Success',
 | 
			
		||||
        'github' => 'Add from Github',
 | 
			
		||||
        'refresh' => 'Refresh',
 | 
			
		||||
        'import_image' => 'Import Image',
 | 
			
		||||
        'no_local_ip' => 'Local IP Addresses are not allowed',
 | 
			
		||||
        'unsupported_format' => 'Unsupported Format. Supported Formats: :formats',
 | 
			
		||||
        'invalid_url' => 'The provided URL is invalid',
 | 
			
		||||
        'image_deleted' => 'Image Deleted',
 | 
			
		||||
        'no_image' => 'No Image Provided',
 | 
			
		||||
        'image_updated' => 'Image Updated',
 | 
			
		||||
    ],
 | 
			
		||||
    'export' => [
 | 
			
		||||
        'modal' => 'How would you like to export :egg ?',
 | 
			
		||||
 | 
			
		||||
@ -1,7 +1,6 @@
 | 
			
		||||
@php
 | 
			
		||||
    $actiongroup = \App\Filament\App\Resources\Servers\Pages\ListServers::getPowerActionGroup()->record($server);
 | 
			
		||||
@endphp
 | 
			
		||||
 | 
			
		||||
<div wire:poll.15s
 | 
			
		||||
     class="relative cursor-pointer"
 | 
			
		||||
     x-on:click="window.location.href = '{{ \App\Filament\Server\Pages\Console::getUrl(panel: 'server', tenant: $server) }}'">
 | 
			
		||||
@ -11,6 +10,18 @@
 | 
			
		||||
    </div>
 | 
			
		||||
 | 
			
		||||
    <div class="flex-1 dark:bg-gray-800 dark:text-white rounded-lg overflow-hidden p-3">
 | 
			
		||||
        @if($server->egg->image)
 | 
			
		||||
            <div style="
 | 
			
		||||
                position: absolute;
 | 
			
		||||
                inset: 0;
 | 
			
		||||
                background: url('{{ $server->egg->image }}') right no-repeat;
 | 
			
		||||
                background-size: contain;
 | 
			
		||||
                opacity: 0.20;
 | 
			
		||||
                max-width: 680px;
 | 
			
		||||
                max-height: 140px;
 | 
			
		||||
        "></div>
 | 
			
		||||
        @endif
 | 
			
		||||
 | 
			
		||||
        <div class="flex items-center mb-5 gap-2">
 | 
			
		||||
            <x-filament::icon-button
 | 
			
		||||
                :icon="$server->condition->getIcon()"
 | 
			
		||||
@ -26,7 +37,8 @@
 | 
			
		||||
            </h2>
 | 
			
		||||
            @if ($actiongroup->isVisible())
 | 
			
		||||
                <div class="end-0">
 | 
			
		||||
                    <div class="flex-1 dark:bg-gray-800 dark:text-white rounded-b-lg overflow-hidden p-1" x-on:click.stop>
 | 
			
		||||
                    <div class="flex-1 dark:bg-gray-800 dark:text-white rounded-b-lg overflow-hidden p-1"
 | 
			
		||||
                         x-on:click.stop>
 | 
			
		||||
                        {{ $actiongroup }}
 | 
			
		||||
                    </div>
 | 
			
		||||
                </div>
 | 
			
		||||
@ -59,4 +71,4 @@
 | 
			
		||||
            </div>
 | 
			
		||||
        </div>
 | 
			
		||||
    </div>
 | 
			
		||||
</div>
 | 
			
		||||
</div>
 | 
			
		||||
 | 
			
		||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user