mirror of
				https://github.com/pelican-dev/panel.git
				synced 2025-10-26 17:26:51 +01:00 
			
		
		
		
	Add basic relationship manager for allocations
This commit is contained in:
		
							parent
							
								
									076125485d
								
							
						
					
					
						commit
						be35692125
					
				| @ -131,175 +131,6 @@ class EditServer extends EditRecord | |||||||
|                     ->preload() |                     ->preload() | ||||||
|                     ->required(), |                     ->required(), | ||||||
| 
 | 
 | ||||||
|                 Forms\Components\Select::make('node_id') |  | ||||||
|                     ->disabledOn('edit') |  | ||||||
|                     ->prefixIcon('tabler-server-2') |  | ||||||
|                     ->default(fn () => Node::query()->latest()->first()?->id) |  | ||||||
|                     ->columnSpan(2) |  | ||||||
|                     ->live() |  | ||||||
|                     ->relationship('node', 'name') |  | ||||||
|                     ->searchable() |  | ||||||
|                     ->preload() |  | ||||||
|                     ->afterStateUpdated(fn (Forms\Set $set) => $set('allocation_id', null)) |  | ||||||
|                     ->required(), |  | ||||||
| 
 |  | ||||||
|                 Forms\Components\Select::make('allocation_id') |  | ||||||
|                     ->preload() |  | ||||||
|                     ->live() |  | ||||||
|                     ->prefixIcon('tabler-network') |  | ||||||
|                     ->label('Primary Allocation') |  | ||||||
|                     ->columnSpan(2) |  | ||||||
|                     ->disabled(fn (Forms\Get $get) => $get('node_id') === null) |  | ||||||
|                     ->searchable(['ip', 'port', 'ip_alias']) |  | ||||||
|                     ->afterStateUpdated(function (Forms\Set $set) { |  | ||||||
|                         $set('allocation_additional', null); |  | ||||||
|                         $set('allocation_additional.needstobeastringhere.extra_allocations', null); |  | ||||||
|                     }) |  | ||||||
|                     ->getOptionLabelFromRecordUsing( |  | ||||||
|                         fn (Allocation $allocation) => "$allocation->ip:$allocation->port" . |  | ||||||
|                             ($allocation->ip_alias ? " ($allocation->ip_alias)" : '') |  | ||||||
|                     ) |  | ||||||
|                     ->placeholder(function (Forms\Get $get) { |  | ||||||
|                         $node = Node::find($get('node_id')); |  | ||||||
| 
 |  | ||||||
|                         if ($node?->allocations) { |  | ||||||
|                             return 'Select an Allocation'; |  | ||||||
|                         } |  | ||||||
| 
 |  | ||||||
|                         return 'Create a New Allocation'; |  | ||||||
|                     }) |  | ||||||
|                     ->relationship( |  | ||||||
|                         'allocation', |  | ||||||
|                         'ip', |  | ||||||
|                         fn (Builder $query, Forms\Get $get) => $query |  | ||||||
|                             ->where('node_id', $get('node_id')) |  | ||||||
|                             ->whereNull('server_id'), |  | ||||||
|                     ) |  | ||||||
|                     ->createOptionForm(fn (Forms\Get $get) => [ |  | ||||||
|                         Forms\Components\TextInput::make('allocation_ip') |  | ||||||
|                             ->datalist(Node::find($get('node_id'))?->ipAddresses() ?? []) |  | ||||||
|                             ->label('IP Address') |  | ||||||
|                             ->ipv4() |  | ||||||
|                             ->helperText("Usually your machine's public IP unless you are port forwarding.") |  | ||||||
|                             // ->selectablePlaceholder(false)
 |  | ||||||
|                             ->required(), |  | ||||||
|                         Forms\Components\TextInput::make('allocation_alias') |  | ||||||
|                             ->label('Alias') |  | ||||||
|                             ->default(null) |  | ||||||
|                             ->datalist([ |  | ||||||
|                                 $get('name'), |  | ||||||
|                                 Egg::find($get('egg_id'))?->name, |  | ||||||
|                             ]) |  | ||||||
|                             ->helperText('This is just a display only name to help you recognize what this Allocation is used for.') |  | ||||||
|                             ->required(false), |  | ||||||
|                         Forms\Components\TagsInput::make('allocation_ports') |  | ||||||
|                             ->placeholder('Examples: 27015, 27017-27019') |  | ||||||
|                             ->helperText(' |  | ||||||
|                                 These are the ports that users can connect to this Server through. |  | ||||||
|                                 They usually consist of the port forwarded ones. |  | ||||||
|                             ') |  | ||||||
|                             ->label('Ports') |  | ||||||
|                             ->live() |  | ||||||
|                             ->afterStateUpdated(function ($state, Forms\Set $set) { |  | ||||||
|                                 $ports = collect(); |  | ||||||
|                                 $update = false; |  | ||||||
|                                 foreach ($state as $portEntry) { |  | ||||||
|                                     if (!str_contains($portEntry, '-')) { |  | ||||||
|                                         if (is_numeric($portEntry)) { |  | ||||||
|                                             $ports->push((int) $portEntry); |  | ||||||
| 
 |  | ||||||
|                                             continue; |  | ||||||
|                                         } |  | ||||||
| 
 |  | ||||||
|                                         // Do not add non numerical ports
 |  | ||||||
|                                         $update = true; |  | ||||||
| 
 |  | ||||||
|                                         continue; |  | ||||||
|                                     } |  | ||||||
| 
 |  | ||||||
|                                     $update = true; |  | ||||||
|                                     [$start, $end] = explode('-', $portEntry); |  | ||||||
|                                     if (!is_numeric($start) || !is_numeric($end)) { |  | ||||||
|                                         continue; |  | ||||||
|                                     } |  | ||||||
| 
 |  | ||||||
|                                     $start = max((int) $start, 0); |  | ||||||
|                                     $end = min((int) $end, 2 ** 16 - 1); |  | ||||||
|                                     for ($i = $start; $i <= $end; $i++) { |  | ||||||
|                                         $ports->push($i); |  | ||||||
|                                     } |  | ||||||
|                                 } |  | ||||||
| 
 |  | ||||||
|                                 $uniquePorts = $ports->unique()->values(); |  | ||||||
|                                 if ($ports->count() > $uniquePorts->count()) { |  | ||||||
|                                     $update = true; |  | ||||||
|                                     $ports = $uniquePorts; |  | ||||||
|                                 } |  | ||||||
| 
 |  | ||||||
|                                 $sortedPorts = $ports->sort()->values(); |  | ||||||
|                                 if ($sortedPorts->all() !== $ports->all()) { |  | ||||||
|                                     $update = true; |  | ||||||
|                                     $ports = $sortedPorts; |  | ||||||
|                                 } |  | ||||||
| 
 |  | ||||||
|                                 if ($update) { |  | ||||||
|                                     $set('allocation_ports', $ports->all()); |  | ||||||
|                                 } |  | ||||||
|                             }) |  | ||||||
|                             ->splitKeys(['Tab', ' ', ',']) |  | ||||||
|                             ->required(), |  | ||||||
|                     ]) |  | ||||||
|                     ->createOptionUsing(function (array $data, Forms\Get $get): int { |  | ||||||
|                         return collect( |  | ||||||
|                             resolve(AssignmentService::class)->handle(Node::find($get('node_id')), $data) |  | ||||||
|                         )->first(); |  | ||||||
|                     }) |  | ||||||
|                     ->required(), |  | ||||||
| 
 |  | ||||||
|                 Forms\Components\Repeater::make('allocation_additional') |  | ||||||
|                     ->label('Additional Allocations') |  | ||||||
|                     ->columnSpan(2) |  | ||||||
|                     ->addActionLabel('Add Allocation') |  | ||||||
|                     ->disabled(fn (Forms\Get $get) => $get('allocation_id') === null) |  | ||||||
|                     // ->addable() TODO disable when all allocations are taken
 |  | ||||||
|                     // ->addable() TODO disable until first additional allocation is selected
 |  | ||||||
|                     ->simple( |  | ||||||
|                         Forms\Components\Select::make('extra_allocations') |  | ||||||
|                             ->live() |  | ||||||
|                             ->preload() |  | ||||||
|                             ->disableOptionsWhenSelectedInSiblingRepeaterItems() |  | ||||||
|                             ->prefixIcon('tabler-network') |  | ||||||
|                             ->label('Additional Allocations') |  | ||||||
|                             ->columnSpan(2) |  | ||||||
|                             ->disabled(fn (Forms\Get $get) => $get('../../node_id') === null) |  | ||||||
|                             ->searchable(['ip', 'port', 'ip_alias']) |  | ||||||
|                             ->getOptionLabelFromRecordUsing( |  | ||||||
|                                 fn (Allocation $allocation) => "$allocation->ip:$allocation->port" . |  | ||||||
|                                     ($allocation->ip_alias ? " ($allocation->ip_alias)" : '') |  | ||||||
|                             ) |  | ||||||
|                             ->placeholder('Select additional Allocations') |  | ||||||
|                             ->relationship( |  | ||||||
|                                 'allocations', |  | ||||||
|                                 'ip', |  | ||||||
|                                 fn (Builder $query, Forms\Get $get, Forms\Components\Select $component, $state) => $query |  | ||||||
|                                     ->where('node_id', $get('../../node_id')) |  | ||||||
|                                     ->whereNotIn( |  | ||||||
|                                         'id', |  | ||||||
|                                         collect(($repeater = $component->getParentRepeater())->getState()) |  | ||||||
|                                             ->pluck( |  | ||||||
|                                                 (string) str($component->getStatePath()) |  | ||||||
|                                                     ->after("{$repeater->getStatePath()}.") |  | ||||||
|                                                     ->after('.'), |  | ||||||
|                                             ) |  | ||||||
|                                             ->flatten() |  | ||||||
|                                             ->diff(Arr::wrap($state)) |  | ||||||
|                                             ->filter(fn (mixed $siblingItemState): bool => filled($siblingItemState)) |  | ||||||
|                                             ->add($get('../../allocation_id')) |  | ||||||
|                                     ) |  | ||||||
|                                     ->whereNull('server_id'), |  | ||||||
|                             ), |  | ||||||
|                     ), |  | ||||||
| 
 |  | ||||||
|                 Forms\Components\Textarea::make('description') |                 Forms\Components\Textarea::make('description') | ||||||
|                     ->hidden() |                     ->hidden() | ||||||
|                     ->default('') |                     ->default('') | ||||||
| @ -549,23 +380,15 @@ class EditServer extends EditRecord | |||||||
|                             ->required() |                             ->required() | ||||||
|                             ->numeric(), |                             ->numeric(), | ||||||
| 
 | 
 | ||||||
|                         Forms\Components\TextInput::make('threads') |                         Forms\Components\Hidden::make('io') | ||||||
|                             ->hint('Advanced') |  | ||||||
|                             ->hintColor('danger') |  | ||||||
|                             ->helperText('Examples: 0, 0-1,3, or 0,1,3,4') |  | ||||||
|                             ->label('CPU Pinning') |  | ||||||
|                             ->suffixIcon('tabler-cpu') |  | ||||||
|                             ->maxLength(191), |  | ||||||
| 
 |  | ||||||
|                         Forms\Components\TextInput::make('io') |  | ||||||
|                             ->helperText('The IO performance relative to other running containers') |                             ->helperText('The IO performance relative to other running containers') | ||||||
|                             ->label('Block IO Proportion') |                             ->label('Block IO Proportion') | ||||||
|                             ->required() |                             ->required() | ||||||
|                             ->minValue(0) | //                            ->numeric()
 | ||||||
|                             ->maxValue(1000) | //                            ->minValue(0)
 | ||||||
|                             ->step(10) | //                            ->maxValue(1000)
 | ||||||
|                             ->default(0) | //                            ->step(10)
 | ||||||
|                             ->numeric(), |                             ->default(0), | ||||||
| 
 | 
 | ||||||
|                         Forms\Components\ToggleButtons::make('oom_disabled') |                         Forms\Components\ToggleButtons::make('oom_disabled') | ||||||
|                             ->label('OOM Killer') |                             ->label('OOM Killer') | ||||||
| @ -610,4 +433,11 @@ class EditServer extends EditRecord | |||||||
| 
 | 
 | ||||||
|         return $data; |         return $data; | ||||||
|     } |     } | ||||||
|  | 
 | ||||||
|  |     public function getRelationManagers(): array | ||||||
|  |     { | ||||||
|  |         return [ | ||||||
|  |             ServerResource\RelationManagers\AllocationsRelationManager::class, | ||||||
|  |         ]; | ||||||
|  |     } | ||||||
| } | } | ||||||
|  | |||||||
| @ -0,0 +1,71 @@ | |||||||
|  | <?php | ||||||
|  | 
 | ||||||
|  | namespace App\Filament\Resources\ServerResource\RelationManagers; | ||||||
|  | 
 | ||||||
|  | use App\Models\Allocation; | ||||||
|  | use Filament\Forms; | ||||||
|  | use Filament\Forms\Form; | ||||||
|  | use Filament\Resources\RelationManagers\RelationManager; | ||||||
|  | use Filament\Tables; | ||||||
|  | use Filament\Tables\Table; | ||||||
|  | use Illuminate\Database\Eloquent\Builder; | ||||||
|  | use Illuminate\Database\Eloquent\SoftDeletingScope; | ||||||
|  | 
 | ||||||
|  | class AllocationsRelationManager extends RelationManager | ||||||
|  | { | ||||||
|  |     protected static string $relationship = 'allocations'; | ||||||
|  | 
 | ||||||
|  |     public function form(Form $form): Form | ||||||
|  |     { | ||||||
|  |         return $form | ||||||
|  |             ->schema([ | ||||||
|  |                 Forms\Components\TextInput::make('ip') | ||||||
|  |                     ->required() | ||||||
|  |                     ->maxLength(255), | ||||||
|  |             ]); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     public function table(Table $table): Table | ||||||
|  |     { | ||||||
|  |         return $table | ||||||
|  |             ->recordTitleAttribute('ip') | ||||||
|  |             ->checkIfRecordIsSelectableUsing(fn (Allocation $record) => $record->id !== $this->getOwnerRecord()->allocation_id) | ||||||
|  |             // ->actions
 | ||||||
|  |             // ->groups
 | ||||||
|  |             ->columns([ | ||||||
|  |                 Tables\Columns\TextColumn::make('ip_alias')->label('Alias'), | ||||||
|  |                 Tables\Columns\TextColumn::make('ip')->label('IP'), | ||||||
|  |                 Tables\Columns\TextColumn::make('port')->label('Port'), | ||||||
|  |                 Tables\Columns\IconColumn::make('primary') | ||||||
|  |                     ->icon(fn ($state) => match ($state) { | ||||||
|  |                         false => 'tabler-star', | ||||||
|  |                         true => 'tabler-star-filled', | ||||||
|  |                     }) | ||||||
|  |                     ->color(fn ($state) => match ($state) { | ||||||
|  |                         false => 'gray', | ||||||
|  |                         true => 'warning', | ||||||
|  |                     }) | ||||||
|  |                     ->action(fn (Allocation $allocation) => $this->getOwnerRecord()->update(['allocation_id' => $allocation->id])) | ||||||
|  |                     ->default(fn (Allocation $allocation) => $allocation->id === $this->getOwnerRecord()->allocation_id) | ||||||
|  |                     ->label('Primary'), | ||||||
|  |             ]) | ||||||
|  |             ->filters([ | ||||||
|  |                 //
 | ||||||
|  |             ]) | ||||||
|  |             ->actions([ | ||||||
|  |                 Tables\Actions\Action::make('make-primary') | ||||||
|  |                     ->action(fn (Allocation $allocation) => $this->getOwnerRecord()->update(['allocation_id' => $allocation->id])) | ||||||
|  |                     ->label(fn (Allocation $allocation) => $allocation->id === $this->getOwnerRecord()->allocation_id ? '' : 'Make Primary'), | ||||||
|  |             ]) | ||||||
|  |             ->headerActions([ | ||||||
|  |                 Tables\Actions\CreateAction::make()->label('Create Allocation'), | ||||||
|  |                 Tables\Actions\AssociateAction::make()->label('Add Allocation'), | ||||||
|  |             ]) | ||||||
|  |             ->bulkActions([ | ||||||
|  |                 Tables\Actions\BulkActionGroup::make([ | ||||||
|  |                     Tables\Actions\DissociateBulkAction::make(), | ||||||
|  |                     Tables\Actions\DeleteBulkAction::make(), | ||||||
|  |                 ]), | ||||||
|  |             ]); | ||||||
|  |     } | ||||||
|  | } | ||||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user
	 Lance Pioch
						Lance Pioch