mirror of
https://github.com/pelican-dev/panel.git
synced 2025-05-31 04:54:45 +02: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