mirror of
https://github.com/pelican-dev/panel.git
synced 2025-05-19 21:04:44 +02:00
160 lines
6.8 KiB
PHP
160 lines
6.8 KiB
PHP
<?php
|
|
|
|
namespace App\Filament\Admin\Resources\NodeResource\RelationManagers;
|
|
|
|
use App\Models\Allocation;
|
|
use App\Models\Node;
|
|
use App\Services\Allocations\AssignmentService;
|
|
use Filament\Forms\Components\TagsInput;
|
|
use Filament\Forms\Components\TextInput;
|
|
use Filament\Forms\Form;
|
|
use Filament\Forms\Set;
|
|
use Filament\Resources\RelationManagers\RelationManager;
|
|
use Filament\Tables;
|
|
use Filament\Tables\Actions\BulkActionGroup;
|
|
use Filament\Tables\Actions\DeleteBulkAction;
|
|
use Filament\Tables\Columns\TextColumn;
|
|
use Filament\Tables\Columns\TextInputColumn;
|
|
use Filament\Tables\Table;
|
|
use Illuminate\Support\HtmlString;
|
|
|
|
/**
|
|
* @method Node getOwnerRecord()
|
|
*/
|
|
class AllocationsRelationManager extends RelationManager
|
|
{
|
|
protected static string $relationship = 'allocations';
|
|
|
|
protected static ?string $icon = 'tabler-plug-connected';
|
|
|
|
public function form(Form $form): Form
|
|
{
|
|
return $form
|
|
->schema([
|
|
TextInput::make('ip')
|
|
->required()
|
|
->maxLength(255),
|
|
]);
|
|
}
|
|
|
|
public function table(Table $table): Table
|
|
{
|
|
return $table
|
|
->recordTitleAttribute('ip')
|
|
|
|
// Non Primary Allocations
|
|
// ->checkIfRecordIsSelectableUsing(fn (Allocation $allocation) => $allocation->id !== $allocation->server?->allocation_id)
|
|
|
|
// All assigned allocations
|
|
->checkIfRecordIsSelectableUsing(fn (Allocation $allocation) => $allocation->server_id === null)
|
|
->paginationPageOptions(['10', '20', '50', '100', '200', '500', '1000'])
|
|
->searchable()
|
|
->selectCurrentPageOnly() //Prevent people from trying to nuke 30,000 ports at once.... -,-
|
|
->columns([
|
|
TextColumn::make('id')
|
|
->toggleable()
|
|
->toggledHiddenByDefault(),
|
|
TextColumn::make('port')
|
|
->searchable()
|
|
->label('Port'),
|
|
TextColumn::make('server.name')
|
|
->label('Server')
|
|
->icon('tabler-brand-docker')
|
|
->visibleFrom('md')
|
|
->searchable()
|
|
->url(fn (Allocation $allocation): string => $allocation->server ? route('filament.admin.resources.servers.edit', ['record' => $allocation->server]) : ''),
|
|
TextInputColumn::make('ip_alias')
|
|
->searchable()
|
|
->label('Alias'),
|
|
TextInputColumn::make('ip')
|
|
->searchable()
|
|
->label('IP'),
|
|
])
|
|
->headerActions([
|
|
Tables\Actions\Action::make('create new allocation')->label('Create Allocations')
|
|
->form(fn () => [
|
|
TextInput::make('allocation_ip')
|
|
->datalist($this->getOwnerRecord()->ipAddresses())
|
|
->label('IP Address')
|
|
->inlineLabel()
|
|
->ipv4()
|
|
->helperText("Usually your machine's public IP unless you are port forwarding.")
|
|
->required(),
|
|
TextInput::make('allocation_alias')
|
|
->label('Alias')
|
|
->inlineLabel()
|
|
->default(null)
|
|
->helperText('Optional display name to help you remember what these are.')
|
|
->required(false),
|
|
TagsInput::make('allocation_ports')
|
|
->placeholder('Examples: 27015, 27017-27019')
|
|
->helperText(new HtmlString('
|
|
These are the ports that users can connect to this Server through.
|
|
<br />
|
|
You would have to port forward these on your home network.
|
|
'))
|
|
->label('Ports')
|
|
->inlineLabel()
|
|
->live()
|
|
->afterStateUpdated(function ($state, 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);
|
|
foreach (range($start, $end) as $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;
|
|
}
|
|
|
|
$ports = $ports->filter(fn ($port) => $port > 1024 && $port < 65535)->values();
|
|
|
|
if ($update) {
|
|
$set('allocation_ports', $ports->all());
|
|
}
|
|
})
|
|
->splitKeys(['Tab', ' ', ','])
|
|
->required(),
|
|
])
|
|
->action(fn (array $data, AssignmentService $service) => $service->handle($this->getOwnerRecord(), $data)),
|
|
])
|
|
->bulkActions([
|
|
BulkActionGroup::make([
|
|
DeleteBulkAction::make()
|
|
->authorize(fn () => auth()->user()->can('delete allocation')),
|
|
]),
|
|
]);
|
|
}
|
|
}
|