mirror of
https://github.com/pelican-dev/panel.git
synced 2025-09-08 23:08:37 +02:00
Merge remote-tracking branch 'upstream/main' into filament-v4
This commit is contained in:
commit
999ac1ebb1
7
.github/ISSUE_TEMPLATE/bug-report.yml
vendored
7
.github/ISSUE_TEMPLATE/bug-report.yml
vendored
@ -64,10 +64,9 @@ body:
|
|||||||
label: Error Logs
|
label: Error Logs
|
||||||
description: |
|
description: |
|
||||||
Run the following command to collect logs on your system.
|
Run the following command to collect logs on your system.
|
||||||
|
Wings: `sudo wings diagnostics --hastebin-url=https://logs.pelican.dev`
|
||||||
Wings: `sudo wings diagnostics`
|
Panel: `tail -n 300 /var/www/pelican/storage/logs/laravel-$(date +%F).log | curl --data-binary @- https://logs.pelican.dev`
|
||||||
Panel: `tail -n 150 /var/www/pelican/storage/logs/laravel-$(date +%F).log | curl -X POST -F 'c=@-' paste.pelistuff.com`
|
placeholder: "https://logs.pelican.dev/c17f750e"
|
||||||
placeholder: "https://pelipaste.com/a1h6z"
|
|
||||||
render: bash
|
render: bash
|
||||||
validations:
|
validations:
|
||||||
required: false
|
required: false
|
||||||
|
@ -85,7 +85,8 @@ RUN chown root:www-data ./ \
|
|||||||
&& ln -s /pelican-data/storage/fonts /var/www/html/storage/app/public/fonts \
|
&& ln -s /pelican-data/storage/fonts /var/www/html/storage/app/public/fonts \
|
||||||
# Allow www-data write permissions where necessary
|
# Allow www-data write permissions where necessary
|
||||||
&& chown -R www-data:www-data /pelican-data ./storage ./bootstrap/cache /var/run/supervisord /var/www/html/public/storage \
|
&& chown -R www-data:www-data /pelican-data ./storage ./bootstrap/cache /var/run/supervisord /var/www/html/public/storage \
|
||||||
&& chmod -R u+rwX,g+rwX,o-rwx /pelican-data ./storage ./bootstrap/cache /var/run/supervisord
|
&& chmod -R u+rwX,g+rwX,o-rwx /pelican-data ./storage ./bootstrap/cache /var/run/supervisord \
|
||||||
|
&& chown -R www-data: /usr/local/etc/php/
|
||||||
|
|
||||||
# Configure Supervisor
|
# Configure Supervisor
|
||||||
COPY docker/supervisord.conf /etc/supervisord.conf
|
COPY docker/supervisord.conf /etc/supervisord.conf
|
||||||
|
@ -89,7 +89,8 @@ RUN chown root:www-data ./ \
|
|||||||
&& ln -s /pelican-data/storage/fonts /var/www/html/storage/app/public/fonts \
|
&& ln -s /pelican-data/storage/fonts /var/www/html/storage/app/public/fonts \
|
||||||
# Allow www-data write permissions where necessary
|
# Allow www-data write permissions where necessary
|
||||||
&& chown -R www-data:www-data /pelican-data ./storage ./bootstrap/cache /var/run/supervisord /var/www/html/public/storage \
|
&& chown -R www-data:www-data /pelican-data ./storage ./bootstrap/cache /var/run/supervisord /var/www/html/public/storage \
|
||||||
&& chmod -R u+rwX,g+rwX,o-rwx /pelican-data ./storage ./bootstrap/cache /var/run/supervisord
|
&& chmod -R u+rwX,g+rwX,o-rwx /pelican-data ./storage ./bootstrap/cache /var/run/supervisord \
|
||||||
|
&& chown -R www-data: /usr/local/etc/php/
|
||||||
|
|
||||||
# Configure Supervisor
|
# Configure Supervisor
|
||||||
COPY docker/supervisord.conf /etc/supervisord.conf
|
COPY docker/supervisord.conf /etc/supervisord.conf
|
||||||
|
10
app/Enums/StartupVariableType.php
Normal file
10
app/Enums/StartupVariableType.php
Normal file
@ -0,0 +1,10 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
namespace App\Enums;
|
||||||
|
|
||||||
|
enum StartupVariableType: string
|
||||||
|
{
|
||||||
|
case Text = 'text';
|
||||||
|
case Select = 'select';
|
||||||
|
case Toggle = 'toggle'; // TODO: add toggle to blade view
|
||||||
|
}
|
@ -5,7 +5,6 @@ namespace App\Filament\Admin\Resources\Nodes\Widgets;
|
|||||||
use App\Models\Node;
|
use App\Models\Node;
|
||||||
use Filament\Support\RawJs;
|
use Filament\Support\RawJs;
|
||||||
use Filament\Widgets\ChartWidget;
|
use Filament\Widgets\ChartWidget;
|
||||||
use Illuminate\Support\Number;
|
|
||||||
|
|
||||||
class NodeCpuChart extends ChartWidget
|
class NodeCpuChart extends ChartWidget
|
||||||
{
|
{
|
||||||
@ -82,8 +81,8 @@ class NodeCpuChart extends ChartWidget
|
|||||||
{
|
{
|
||||||
$data = array_slice(end($this->cpuHistory), -60);
|
$data = array_slice(end($this->cpuHistory), -60);
|
||||||
|
|
||||||
$cpu = Number::format($data['cpu'], maxPrecision: 2, locale: auth()->user()->language);
|
$cpu = format_number($data['cpu'], maxPrecision: 2);
|
||||||
$max = Number::format($this->threads * 100, locale: auth()->user()->language);
|
$max = format_number($this->threads * 100);
|
||||||
|
|
||||||
return trans('admin/node.cpu_chart', ['cpu' => $cpu, 'max' => $max]);
|
return trans('admin/node.cpu_chart', ['cpu' => $cpu, 'max' => $max]);
|
||||||
}
|
}
|
||||||
|
@ -5,7 +5,6 @@ namespace App\Filament\Admin\Resources\Nodes\Widgets;
|
|||||||
use App\Models\Node;
|
use App\Models\Node;
|
||||||
use Filament\Support\RawJs;
|
use Filament\Support\RawJs;
|
||||||
use Filament\Widgets\ChartWidget;
|
use Filament\Widgets\ChartWidget;
|
||||||
use Illuminate\Support\Number;
|
|
||||||
|
|
||||||
class NodeMemoryChart extends ChartWidget
|
class NodeMemoryChart extends ChartWidget
|
||||||
{
|
{
|
||||||
@ -85,12 +84,12 @@ class NodeMemoryChart extends ChartWidget
|
|||||||
$latestMemoryUsed = array_slice(end($this->memoryHistory), -60);
|
$latestMemoryUsed = array_slice(end($this->memoryHistory), -60);
|
||||||
|
|
||||||
$used = config('panel.use_binary_prefix')
|
$used = config('panel.use_binary_prefix')
|
||||||
? Number::format($latestMemoryUsed['memory'], maxPrecision: 2, locale: auth()->user()->language) .' GiB'
|
? format_number($latestMemoryUsed['memory'], maxPrecision: 2) .' GiB'
|
||||||
: Number::format($latestMemoryUsed['memory'], maxPrecision: 2, locale: auth()->user()->language) . ' GB';
|
: format_number($latestMemoryUsed['memory'], maxPrecision: 2) . ' GB';
|
||||||
|
|
||||||
$total = config('panel.use_binary_prefix')
|
$total = config('panel.use_binary_prefix')
|
||||||
? Number::format($this->totalMemory / 1024 / 1024 / 1024, maxPrecision: 2, locale: auth()->user()->language) .' GiB'
|
? format_number($this->totalMemory / 1024 / 1024 / 1024, maxPrecision: 2) .' GiB'
|
||||||
: Number::format($this->totalMemory / 1000 / 1000 / 1000, maxPrecision: 2, locale: auth()->user()->language) . ' GB';
|
: format_number($this->totalMemory / 1000 / 1000 / 1000, maxPrecision: 2) . ' GB';
|
||||||
|
|
||||||
return trans('admin/node.memory_chart', ['used' => $used, 'total' => $total]);
|
return trans('admin/node.memory_chart', ['used' => $used, 'total' => $total]);
|
||||||
}
|
}
|
||||||
|
@ -13,10 +13,8 @@ use App\Services\Servers\ServerCreationService;
|
|||||||
use App\Services\Users\UserCreationService;
|
use App\Services\Users\UserCreationService;
|
||||||
use App\Traits\Filament\CanCustomizeHeaderActions;
|
use App\Traits\Filament\CanCustomizeHeaderActions;
|
||||||
use App\Traits\Filament\CanCustomizeHeaderWidgets;
|
use App\Traits\Filament\CanCustomizeHeaderWidgets;
|
||||||
use Closure;
|
|
||||||
use Exception;
|
use Exception;
|
||||||
use Filament\Actions\Action;
|
use Filament\Actions\Action;
|
||||||
use Filament\Schemas\Components\Component;
|
|
||||||
use Filament\Schemas\Components\Fieldset;
|
use Filament\Schemas\Components\Fieldset;
|
||||||
use Filament\Forms\Components\Hidden;
|
use Filament\Forms\Components\Hidden;
|
||||||
use Filament\Forms\Components\KeyValue;
|
use Filament\Forms\Components\KeyValue;
|
||||||
@ -39,7 +37,6 @@ use Filament\Support\Exceptions\Halt;
|
|||||||
use Illuminate\Database\Eloquent\Builder;
|
use Illuminate\Database\Eloquent\Builder;
|
||||||
use Illuminate\Database\Eloquent\Model;
|
use Illuminate\Database\Eloquent\Model;
|
||||||
use Illuminate\Support\Facades\Blade;
|
use Illuminate\Support\Facades\Blade;
|
||||||
use Illuminate\Support\Facades\Validator;
|
|
||||||
use Illuminate\Support\HtmlString;
|
use Illuminate\Support\HtmlString;
|
||||||
use LogicException;
|
use LogicException;
|
||||||
use Filament\Schemas\Schema;
|
use Filament\Schemas\Schema;
|
||||||
@ -432,7 +429,7 @@ class CreateServer extends CreateRecord
|
|||||||
Egg::query()->find($get('egg_id'))?->variables()?->count()
|
Egg::query()->find($get('egg_id'))?->variables()?->count()
|
||||||
),
|
),
|
||||||
Repeater::make('server_variables')
|
Repeater::make('server_variables')
|
||||||
->label('')
|
->hiddenLabel()
|
||||||
->relationship('serverVariables', fn (Builder $query) => $query->orderByPowerJoins('variable.sort'))
|
->relationship('serverVariables', fn (Builder $query) => $query->orderByPowerJoins('variable.sort'))
|
||||||
->saveRelationshipsBeforeChildrenUsing(null)
|
->saveRelationshipsBeforeChildrenUsing(null)
|
||||||
->saveRelationshipsUsing(null)
|
->saveRelationshipsUsing(null)
|
||||||
@ -442,61 +439,15 @@ class CreateServer extends CreateRecord
|
|||||||
->deletable(false)
|
->deletable(false)
|
||||||
->default([])
|
->default([])
|
||||||
->hidden(fn ($state) => empty($state))
|
->hidden(fn ($state) => empty($state))
|
||||||
->mutateRelationshipDataBeforeCreateUsing(function ($data) {
|
->schema([
|
||||||
$data['variable_value'] = ($data['is_select'] ? $data['variable_value_select'] : $data['variable_value_input']) ?? '';
|
StartupVariable::make('variable_value')
|
||||||
|
->fromForm()
|
||||||
return $data;
|
->afterStateUpdated(function (Set $set, Get $get, $state) {
|
||||||
})
|
$environment = $get($envPath = '../../environment');
|
||||||
->schema(function () {
|
$environment[$get('env_variable')] = $state;
|
||||||
$isSelect = Hidden::make('is_select')
|
$set($envPath, $environment);
|
||||||
->dehydrated(false)
|
}),
|
||||||
->formatStateUsing(fn (Get $get) => (bool) collect($get('rules'))->reduce(
|
])
|
||||||
fn ($result, $value) => $result === true && !str($value)->startsWith('in:'), true
|
|
||||||
));
|
|
||||||
|
|
||||||
$text = TextInput::make('variable_value_input')
|
|
||||||
->hidden($this->shouldHideComponent(...))
|
|
||||||
->dehydratedWhenHidden()
|
|
||||||
->required(fn (Get $get) => in_array('required', $get('rules')))
|
|
||||||
->rules(
|
|
||||||
fn (Get $get): Closure => function (string $attribute, $value, Closure $fail) use ($get) {
|
|
||||||
$validator = Validator::make(['validatorkey' => $value], [
|
|
||||||
'validatorkey' => $get('rules'),
|
|
||||||
]);
|
|
||||||
|
|
||||||
if ($validator->fails()) {
|
|
||||||
$message = str($validator->errors()->first())->replace('validatorkey', $get('name'))->toString();
|
|
||||||
|
|
||||||
$fail($message);
|
|
||||||
}
|
|
||||||
},
|
|
||||||
);
|
|
||||||
|
|
||||||
$select = Select::make('variable_value_select')
|
|
||||||
->hidden($this->shouldHideComponent(...))
|
|
||||||
->dehydratedWhenHidden()
|
|
||||||
->options($this->getSelectOptionsFromRules(...))
|
|
||||||
->selectablePlaceholder(false);
|
|
||||||
|
|
||||||
$components = [$text, $select];
|
|
||||||
|
|
||||||
foreach ($components as &$component) {
|
|
||||||
$component = $component
|
|
||||||
->live(onBlur: true)
|
|
||||||
->hintIcon('tabler-code')
|
|
||||||
->label(fn (Get $get) => $get('name'))
|
|
||||||
->hintIconTooltip(fn (Get $get) => implode('|', $get('rules')))
|
|
||||||
->prefix(fn (Get $get) => '{{' . $get('env_variable') . '}}')
|
|
||||||
->helperText(fn (Get $get) => empty($get('description')) ? '—' : $get('description'))
|
|
||||||
->afterStateUpdated(function (Set $set, Get $get, $state) {
|
|
||||||
$environment = $get($envPath = '../../environment');
|
|
||||||
$environment[$get('env_variable')] = $state;
|
|
||||||
$set($envPath, $environment);
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
return [$isSelect, ...$components];
|
|
||||||
})
|
|
||||||
->columnSpan(2),
|
->columnSpan(2),
|
||||||
]),
|
]),
|
||||||
]),
|
]),
|
||||||
@ -864,40 +815,6 @@ class CreateServer extends CreateRecord
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private function shouldHideComponent(Get $get, Component $component): bool
|
|
||||||
{
|
|
||||||
$containsRuleIn = collect($get('rules'))->reduce(
|
|
||||||
fn ($result, $value) => $result === true && !str($value)->startsWith('in:'), true
|
|
||||||
);
|
|
||||||
|
|
||||||
if ($component instanceof Select) {
|
|
||||||
return $containsRuleIn;
|
|
||||||
}
|
|
||||||
|
|
||||||
if ($component instanceof TextInput) {
|
|
||||||
return !$containsRuleIn;
|
|
||||||
}
|
|
||||||
|
|
||||||
throw new Exception('Component type not supported: ' . $component::class);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @return array<array-key, string>
|
|
||||||
*/
|
|
||||||
private function getSelectOptionsFromRules(Get $get): array
|
|
||||||
{
|
|
||||||
$inRule = collect($get('rules'))->reduce(
|
|
||||||
fn ($result, $value) => str($value)->startsWith('in:') ? $value : $result, ''
|
|
||||||
);
|
|
||||||
|
|
||||||
return str($inRule)
|
|
||||||
->after('in:')
|
|
||||||
->explode(',')
|
|
||||||
->each(fn ($value) => str($value)->trim())
|
|
||||||
->mapWithKeys(fn ($value) => [$value => $value])
|
|
||||||
->all();
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @param string[] $portEntries
|
* @param string[] $portEntries
|
||||||
* @return array<int>
|
* @return array<int>
|
||||||
|
@ -4,8 +4,6 @@ namespace App\Filament\Admin\Resources\Servers\Pages;
|
|||||||
|
|
||||||
use App\Enums\SuspendAction;
|
use App\Enums\SuspendAction;
|
||||||
use App\Filament\Admin\Resources\Servers\ServerResource;
|
use App\Filament\Admin\Resources\Servers\ServerResource;
|
||||||
use App\Filament\Admin\Resources\Servers\RelationManagers\AllocationsRelationManager;
|
|
||||||
use App\Filament\Admin\Resources\Servers\RelationManagers\DatabasesRelationManager;
|
|
||||||
use App\Filament\Components\Actions\PreviewStartupAction;
|
use App\Filament\Components\Actions\PreviewStartupAction;
|
||||||
use App\Filament\Server\Pages\Console;
|
use App\Filament\Server\Pages\Console;
|
||||||
use App\Models\Allocation;
|
use App\Models\Allocation;
|
||||||
@ -24,18 +22,8 @@ use App\Services\Servers\ToggleInstallService;
|
|||||||
use App\Services\Servers\TransferServerService;
|
use App\Services\Servers\TransferServerService;
|
||||||
use App\Traits\Filament\CanCustomizeHeaderActions;
|
use App\Traits\Filament\CanCustomizeHeaderActions;
|
||||||
use App\Traits\Filament\CanCustomizeHeaderWidgets;
|
use App\Traits\Filament\CanCustomizeHeaderWidgets;
|
||||||
use Closure;
|
|
||||||
use Exception;
|
use Exception;
|
||||||
use Filament\Actions\Action;
|
use Filament\Actions;
|
||||||
use Filament\Actions\ActionGroup;
|
|
||||||
use Filament\Forms\Components\CodeEditor;
|
|
||||||
use Filament\Schemas\Components\Actions;
|
|
||||||
use Filament\Schemas\Components\Component;
|
|
||||||
use Filament\Schemas\Components\Fieldset;
|
|
||||||
use Filament\Forms\Components\Hidden;
|
|
||||||
use Filament\Forms\Components\KeyValue;
|
|
||||||
use Filament\Forms\Components\Repeater;
|
|
||||||
use Filament\Forms\Components\Select;
|
|
||||||
use Filament\Schemas\Components\Grid;
|
use Filament\Schemas\Components\Grid;
|
||||||
use Filament\Schemas\Components\Tabs;
|
use Filament\Schemas\Components\Tabs;
|
||||||
use Filament\Schemas\Components\Tabs\Tab;
|
use Filament\Schemas\Components\Tabs\Tab;
|
||||||
@ -53,7 +41,6 @@ use Filament\Support\Enums\Alignment;
|
|||||||
use Illuminate\Database\Eloquent\Builder;
|
use Illuminate\Database\Eloquent\Builder;
|
||||||
use Illuminate\Http\Client\ConnectionException;
|
use Illuminate\Http\Client\ConnectionException;
|
||||||
use Illuminate\Support\Arr;
|
use Illuminate\Support\Arr;
|
||||||
use Illuminate\Support\Facades\Validator;
|
|
||||||
use Illuminate\Support\HtmlString;
|
use Illuminate\Support\HtmlString;
|
||||||
use LogicException;
|
use LogicException;
|
||||||
use Filament\Schemas\Schema;
|
use Filament\Schemas\Schema;
|
||||||
@ -629,7 +616,7 @@ class EditServer extends EditRecord
|
|||||||
}),
|
}),
|
||||||
|
|
||||||
Repeater::make('server_variables')
|
Repeater::make('server_variables')
|
||||||
->label('')
|
->hiddenLabel()
|
||||||
->relationship('serverVariables', function (Builder $query) {
|
->relationship('serverVariables', function (Builder $query) {
|
||||||
/** @var Server $server */
|
/** @var Server $server */
|
||||||
$server = $this->getRecord();
|
$server = $this->getRecord();
|
||||||
@ -646,56 +633,16 @@ class EditServer extends EditRecord
|
|||||||
return $query->orderByPowerJoins('variable.sort');
|
return $query->orderByPowerJoins('variable.sort');
|
||||||
})
|
})
|
||||||
->grid()
|
->grid()
|
||||||
->mutateRelationshipDataBeforeSaveUsing(function (array $data) {
|
->mutateRelationshipDataBeforeSaveUsing(function (array $data): array {
|
||||||
$data['variable_value'] = ($data['is_select'] ? $data['variable_value_select'] : $data['variable_value_input']) ?? '';
|
$data['variable_value'] ??= '';
|
||||||
|
|
||||||
return $data;
|
return $data;
|
||||||
})
|
})
|
||||||
->reorderable(false)->addable(false)->deletable(false)
|
->reorderable(false)->addable(false)->deletable(false)
|
||||||
->schema(function () {
|
->schema([
|
||||||
$isSelect = Hidden::make('is_select')
|
StartupVariable::make('variable_value')
|
||||||
->dehydrated(false)
|
->fromRecord(),
|
||||||
->formatStateUsing(fn (ServerVariable $serverVariable) => (bool) array_first($serverVariable->variable->rules, fn ($value) => str($value)->startsWith('in:'), false));
|
])
|
||||||
|
|
||||||
$text = TextInput::make('variable_value_input')
|
|
||||||
->hidden($this->shouldHideComponent(...))
|
|
||||||
->dehydratedWhenHidden()
|
|
||||||
->required(fn (ServerVariable $serverVariable) => $serverVariable->variable->getRequiredAttribute())
|
|
||||||
->rules([
|
|
||||||
fn (ServerVariable $serverVariable): Closure => function (string $attribute, $value, Closure $fail) use ($serverVariable) {
|
|
||||||
$validator = Validator::make(['validatorkey' => $value], [
|
|
||||||
'validatorkey' => $serverVariable->variable->rules,
|
|
||||||
]);
|
|
||||||
|
|
||||||
if ($validator->fails()) {
|
|
||||||
$message = str($validator->errors()->first())->replace('validatorkey', $serverVariable->variable->name);
|
|
||||||
|
|
||||||
$fail($message);
|
|
||||||
}
|
|
||||||
},
|
|
||||||
]);
|
|
||||||
|
|
||||||
$select = Select::make('variable_value_select')
|
|
||||||
->hidden($this->shouldHideComponent(...))
|
|
||||||
->dehydratedWhenHidden()
|
|
||||||
->options($this->getSelectOptionsFromRules(...))
|
|
||||||
->selectablePlaceholder(false);
|
|
||||||
|
|
||||||
$components = [$text, $select];
|
|
||||||
|
|
||||||
foreach ($components as &$component) {
|
|
||||||
$component = $component
|
|
||||||
->live(onBlur: true)
|
|
||||||
->hintIcon('tabler-code')
|
|
||||||
->label(fn (ServerVariable $serverVariable) => $serverVariable->variable->name)
|
|
||||||
->hintIconTooltip(fn (ServerVariable $serverVariable) => implode('|', $serverVariable->variable->rules))
|
|
||||||
->prefix(fn (ServerVariable $serverVariable) => '{{' . $serverVariable->variable->env_variable . '}}')
|
|
||||||
->helperText(fn (ServerVariable $serverVariable) => empty($serverVariable->variable->description) ? '—' : $serverVariable->variable->description)
|
|
||||||
->formatStateUsing(fn (ServerVariable $serverVariable) => $serverVariable->variable_value);
|
|
||||||
}
|
|
||||||
|
|
||||||
return [$isSelect, ...$components];
|
|
||||||
})
|
|
||||||
->columnSpan(6),
|
->columnSpan(6),
|
||||||
]),
|
]),
|
||||||
Tab::make(trans('admin/server.mounts'))
|
Tab::make(trans('admin/server.mounts'))
|
||||||
@ -1032,34 +979,4 @@ class EditServer extends EditRecord
|
|||||||
DatabasesRelationManager::class,
|
DatabasesRelationManager::class,
|
||||||
];
|
];
|
||||||
}
|
}
|
||||||
|
|
||||||
private function shouldHideComponent(ServerVariable $serverVariable, Component $component): bool
|
|
||||||
{
|
|
||||||
$containsRuleIn = array_first($serverVariable->variable->rules, fn ($value) => str($value)->startsWith('in:'), false);
|
|
||||||
|
|
||||||
if ($component instanceof Select) {
|
|
||||||
return !$containsRuleIn;
|
|
||||||
}
|
|
||||||
|
|
||||||
if ($component instanceof TextInput) {
|
|
||||||
return $containsRuleIn;
|
|
||||||
}
|
|
||||||
|
|
||||||
throw new Exception('Component type not supported: ' . $component::class);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @return array<string, string>
|
|
||||||
*/
|
|
||||||
private function getSelectOptionsFromRules(ServerVariable $serverVariable): array
|
|
||||||
{
|
|
||||||
$inRule = array_first($serverVariable->variable->rules, fn ($value) => str($value)->startsWith('in:'));
|
|
||||||
|
|
||||||
return str($inRule)
|
|
||||||
->after('in:')
|
|
||||||
->explode(',')
|
|
||||||
->each(fn ($value) => str($value)->trim())
|
|
||||||
->mapWithKeys(fn ($value) => [$value => $value])
|
|
||||||
->all();
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
@ -39,26 +39,26 @@ class ImportScheduleAction extends Action
|
|||||||
Tabs::make('Tabs')
|
Tabs::make('Tabs')
|
||||||
->contained(false)
|
->contained(false)
|
||||||
->tabs([
|
->tabs([
|
||||||
Tab::make(trans('admin/schedule.import.file'))
|
Tab::make(trans('server/schedule.import_action.file'))
|
||||||
->icon('tabler-file-upload')
|
->icon('tabler-file-upload')
|
||||||
->schema([
|
->schema([
|
||||||
FileUpload::make('files')
|
FileUpload::make('files')
|
||||||
->label(trans('admin/schedule.model_label'))
|
->hiddenLabel()
|
||||||
->hint(trans('admin/schedule.import.schedule_help'))
|
->hint(trans('server/schedule.import_action.schedule_help'))
|
||||||
->acceptedFileTypes(['application/json'])
|
->acceptedFileTypes(['application/json'])
|
||||||
->preserveFilenames()
|
->preserveFilenames()
|
||||||
->previewable(false)
|
->previewable(false)
|
||||||
->storeFiles(false)
|
->storeFiles(false)
|
||||||
->multiple(true),
|
->multiple(true),
|
||||||
]),
|
]),
|
||||||
Tab::make(trans('admin/schedule.import.url'))
|
Tab::make(trans('server/schedule.import_action.url'))
|
||||||
->icon('tabler-world-upload')
|
->icon('tabler-world-upload')
|
||||||
->schema([
|
->schema([
|
||||||
Repeater::make('urls')
|
Repeater::make('urls')
|
||||||
->label('')
|
->hiddenLabel()
|
||||||
->itemLabel(fn (array $state) => str($state['url'])->afterLast('/schedule-')->before('.json')->headline())
|
->itemLabel(fn (array $state) => str($state['url'])->afterLast('/schedule-')->before('.json')->headline())
|
||||||
->hint(trans('admin/schedule.import.url_help'))
|
->hint(trans('server/schedule.import_action.url_help'))
|
||||||
->addActionLabel(trans('admin/schedule.import.add_url'))
|
->addActionLabel(trans('server/schedule.import_action.add_url'))
|
||||||
->grid(2)
|
->grid(2)
|
||||||
->reorderable(false)
|
->reorderable(false)
|
||||||
->addable(true)
|
->addable(true)
|
||||||
@ -66,10 +66,10 @@ class ImportScheduleAction extends Action
|
|||||||
->schema([
|
->schema([
|
||||||
TextInput::make('url')
|
TextInput::make('url')
|
||||||
->live()
|
->live()
|
||||||
->label(trans('admin/schedule.import.url'))
|
->label(trans('server/schedule.import_action.url'))
|
||||||
->url()
|
->url()
|
||||||
->endsWith('.json')
|
->endsWith('.json')
|
||||||
->validationAttribute(trans('admin/schedule.import.url')),
|
->validationAttribute(trans('server/schedule.import_action.url')),
|
||||||
]),
|
]),
|
||||||
]),
|
]),
|
||||||
]),
|
]),
|
||||||
@ -104,14 +104,14 @@ class ImportScheduleAction extends Action
|
|||||||
|
|
||||||
if ($failed->count() > 0) {
|
if ($failed->count() > 0) {
|
||||||
Notification::make()
|
Notification::make()
|
||||||
->title(trans('admin/schedule.import.import_failed'))
|
->title(trans('server/schedule.import_action.import_failed'))
|
||||||
->body($failed->join(', '))
|
->body($failed->join(', '))
|
||||||
->danger()
|
->danger()
|
||||||
->send();
|
->send();
|
||||||
}
|
}
|
||||||
if ($success->count() > 0) {
|
if ($success->count() > 0) {
|
||||||
Notification::make()
|
Notification::make()
|
||||||
->title(trans('admin/schedule.import.import_success'))
|
->title(trans('server/schedule.import_action.import_success'))
|
||||||
->body($success->join(', '))
|
->body($success->join(', '))
|
||||||
->success()
|
->success()
|
||||||
->send();
|
->send();
|
||||||
|
181
app/Filament/Components/Forms/Fields/StartupVariable.php
Normal file
181
app/Filament/Components/Forms/Fields/StartupVariable.php
Normal file
@ -0,0 +1,181 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
namespace App\Filament\Components\Forms\Fields;
|
||||||
|
|
||||||
|
use App\Enums\StartupVariableType;
|
||||||
|
use App\Models\ServerVariable;
|
||||||
|
use Closure;
|
||||||
|
use Filament\Forms\Components\Concerns\HasAffixes;
|
||||||
|
use Filament\Forms\Components\Concerns\HasExtraInputAttributes;
|
||||||
|
use Filament\Forms\Components\Concerns\HasPlaceholder;
|
||||||
|
use Filament\Forms\Components\Field;
|
||||||
|
use Filament\Forms\Get;
|
||||||
|
use Filament\Support\Concerns\HasExtraAlpineAttributes;
|
||||||
|
use Illuminate\Support\Arr;
|
||||||
|
use Illuminate\Support\Str;
|
||||||
|
|
||||||
|
class StartupVariable extends Field
|
||||||
|
{
|
||||||
|
use HasAffixes;
|
||||||
|
use HasExtraAlpineAttributes;
|
||||||
|
use HasExtraInputAttributes;
|
||||||
|
use HasPlaceholder;
|
||||||
|
|
||||||
|
/** @var view-string */
|
||||||
|
protected string $view = 'filament.components.startup-variable';
|
||||||
|
|
||||||
|
protected string|Closure|null $variableName = null;
|
||||||
|
|
||||||
|
protected string|Closure|null $variableDesc = null;
|
||||||
|
|
||||||
|
protected string|Closure|null $variableEnv = null;
|
||||||
|
|
||||||
|
protected string|Closure|null $variableDefault = null;
|
||||||
|
|
||||||
|
/** @var string[]|Closure|null */
|
||||||
|
protected array|Closure|null $variableRules = [];
|
||||||
|
|
||||||
|
protected function setUp(): void
|
||||||
|
{
|
||||||
|
parent::setUp();
|
||||||
|
|
||||||
|
$this->label(fn (StartupVariable $component) => $component->getVariableName());
|
||||||
|
|
||||||
|
$this->prefix(fn (StartupVariable $component) => '{{' . $component->getVariableEnv() . '}}');
|
||||||
|
|
||||||
|
$this->hintIcon('tabler-code');
|
||||||
|
|
||||||
|
$this->hintIconTooltip(fn (StartupVariable $component) => implode('|', $component->getVariableRules()));
|
||||||
|
|
||||||
|
$this->helperText(fn (StartupVariable $component) => !$component->getVariableDesc() ? '—' : $component->getVariableDesc());
|
||||||
|
|
||||||
|
$this->rules(fn (StartupVariable $component) => $component->getVariableRules());
|
||||||
|
|
||||||
|
$this->placeholder(fn (StartupVariable $component) => $component->getVariableDefault());
|
||||||
|
|
||||||
|
$this->live(onBlur: true);
|
||||||
|
}
|
||||||
|
|
||||||
|
public function fromForm(): static
|
||||||
|
{
|
||||||
|
$this->variableName(fn (Get $get) => $get('name'));
|
||||||
|
$this->variableDesc(fn (Get $get) => $get('description'));
|
||||||
|
$this->variableEnv(fn (Get $get) => $get('env_variable'));
|
||||||
|
$this->variableDefault(fn (Get $get) => $get('default_value'));
|
||||||
|
$this->variableRules(fn (Get $get) => $get('rules'));
|
||||||
|
|
||||||
|
return $this;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function fromRecord(): static
|
||||||
|
{
|
||||||
|
$this->variableName(fn (ServerVariable $record) => $record->variable->name);
|
||||||
|
$this->variableDesc(fn (ServerVariable $record) => $record->variable->description);
|
||||||
|
$this->variableEnv(fn (ServerVariable $record) => $record->variable->env_variable);
|
||||||
|
$this->variableDefault(fn (ServerVariable $record) => $record->variable->default_value);
|
||||||
|
$this->variableRules(fn (ServerVariable $record) => $record->variable->rules);
|
||||||
|
|
||||||
|
return $this;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function variableName(string|Closure|null $name): static
|
||||||
|
{
|
||||||
|
$this->variableName = $name;
|
||||||
|
|
||||||
|
return $this;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function variableDesc(string|Closure|null $desc): static
|
||||||
|
{
|
||||||
|
$this->variableDesc = $desc;
|
||||||
|
|
||||||
|
return $this;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function variableEnv(string|Closure|null $envVariable): static
|
||||||
|
{
|
||||||
|
$this->variableEnv = $envVariable;
|
||||||
|
|
||||||
|
return $this;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function variableDefault(string|Closure|null $default): static
|
||||||
|
{
|
||||||
|
$this->variableDefault = $default;
|
||||||
|
|
||||||
|
return $this;
|
||||||
|
}
|
||||||
|
|
||||||
|
/** @param string[]|Closure|null $rules */
|
||||||
|
public function variableRules(array|Closure|null $rules): static
|
||||||
|
{
|
||||||
|
$this->variableRules = $rules;
|
||||||
|
|
||||||
|
return $this;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getVariableName(): ?string
|
||||||
|
{
|
||||||
|
return $this->evaluate($this->variableName);
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getVariableDesc(): ?string
|
||||||
|
{
|
||||||
|
return $this->evaluate($this->variableDesc);
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getVariableEnv(): ?string
|
||||||
|
{
|
||||||
|
return $this->evaluate($this->variableEnv);
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getVariableDefault(): ?string
|
||||||
|
{
|
||||||
|
return $this->evaluate($this->variableDefault);
|
||||||
|
}
|
||||||
|
|
||||||
|
/** @return string[] */
|
||||||
|
public function getVariableRules(): array
|
||||||
|
{
|
||||||
|
return (array) ($this->evaluate($this->variableRules) ?? []);
|
||||||
|
}
|
||||||
|
|
||||||
|
public function isRequired(): bool
|
||||||
|
{
|
||||||
|
$rules = $this->getVariableRules();
|
||||||
|
|
||||||
|
return in_array('required', $rules);
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getType(): StartupVariableType
|
||||||
|
{
|
||||||
|
$rules = $this->getVariableRules();
|
||||||
|
|
||||||
|
if (Arr::first($rules, fn ($value) => str($value)->startsWith('in:'), false)) {
|
||||||
|
return StartupVariableType::Select;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (in_array('boolean', $rules)) {
|
||||||
|
return StartupVariableType::Toggle;
|
||||||
|
}
|
||||||
|
|
||||||
|
return StartupVariableType::Text;
|
||||||
|
}
|
||||||
|
|
||||||
|
/** @return string[] */
|
||||||
|
public function getSelectOptions(): array
|
||||||
|
{
|
||||||
|
$rules = $this->getVariableRules();
|
||||||
|
|
||||||
|
$inRule = Arr::first($rules, fn ($value) => str($value)->startsWith('in:'));
|
||||||
|
if ($inRule) {
|
||||||
|
return str($inRule)
|
||||||
|
->after('in:')
|
||||||
|
->explode(',')
|
||||||
|
->each(fn ($value) => Str::trim($value))
|
||||||
|
->all();
|
||||||
|
}
|
||||||
|
|
||||||
|
return [];
|
||||||
|
}
|
||||||
|
}
|
@ -16,7 +16,6 @@ use Filament\Forms\Components\TextInput;
|
|||||||
use Filament\Notifications\Notification;
|
use Filament\Notifications\Notification;
|
||||||
use Filament\Schemas\Schema;
|
use Filament\Schemas\Schema;
|
||||||
use Filament\Support\Enums\Alignment;
|
use Filament\Support\Enums\Alignment;
|
||||||
use Illuminate\Support\Number;
|
|
||||||
|
|
||||||
class Settings extends ServerFormPage
|
class Settings extends ServerFormPage
|
||||||
{
|
{
|
||||||
@ -110,7 +109,7 @@ class Settings extends ServerFormPage
|
|||||||
->prefixIcon('tabler-cpu')
|
->prefixIcon('tabler-cpu')
|
||||||
->columnSpan(1)
|
->columnSpan(1)
|
||||||
->disabled()
|
->disabled()
|
||||||
->formatStateUsing(fn ($state, Server $server) => !$state ? trans('server/setting.server_info.limits.unlimited') : Number::format($server->cpu, locale: auth()->user()->language) . '%'),
|
->formatStateUsing(fn ($state, Server $server) => !$state ? trans('server/setting.server_info.limits.unlimited') : format_number($server->cpu) . '%'),
|
||||||
TextInput::make('memory')
|
TextInput::make('memory')
|
||||||
->label('')
|
->label('')
|
||||||
->prefix(trans('server/setting.server_info.limits.memory'))
|
->prefix(trans('server/setting.server_info.limits.memory'))
|
||||||
|
@ -127,7 +127,15 @@ class ScheduleResource extends Resource
|
|||||||
->visibleOn('view'),
|
->visibleOn('view'),
|
||||||
Section::make('Cron')
|
Section::make('Cron')
|
||||||
->label(trans('server/schedule.cron'))
|
->label(trans('server/schedule.cron'))
|
||||||
->description(fn (Get $get) => new HtmlString(trans('server/schedule.cron_body') . '<br>' . trans('server/schedule.cron_timezone', ['timezone' => auth()->user()->timezone, 'next_run' => Utilities::getScheduleNextRunDate($get('cron_minute'), $get('cron_hour'), $get('cron_day_of_month'), $get('cron_month'), $get('cron_day_of_week'))->timezone(auth()->user()->timezone)])))
|
->description(function (Get $get) {
|
||||||
|
try {
|
||||||
|
$nextRun = Utilities::getScheduleNextRunDate($get('cron_minute'), $get('cron_hour'), $get('cron_day_of_month'), $get('cron_month'), $get('cron_day_of_week'))->timezone(auth()->user()->timezone);
|
||||||
|
} catch (Exception) {
|
||||||
|
$nextRun = trans('server/schedule.invalid');
|
||||||
|
}
|
||||||
|
|
||||||
|
return new HtmlString(trans('server/schedule.cron_body') . '<br>' . trans('server/schedule.cron_timezone', ['timezone' => auth()->user()->timezone, 'next_run' => $nextRun]));
|
||||||
|
})
|
||||||
->schema([
|
->schema([
|
||||||
Actions::make([
|
Actions::make([
|
||||||
CronPresetAction::make('hourly')
|
CronPresetAction::make('hourly')
|
||||||
|
@ -7,7 +7,6 @@ use Carbon\Carbon;
|
|||||||
use Filament\Facades\Filament;
|
use Filament\Facades\Filament;
|
||||||
use Filament\Support\RawJs;
|
use Filament\Support\RawJs;
|
||||||
use Filament\Widgets\ChartWidget;
|
use Filament\Widgets\ChartWidget;
|
||||||
use Illuminate\Support\Number;
|
|
||||||
|
|
||||||
class ServerCpuChart extends ChartWidget
|
class ServerCpuChart extends ChartWidget
|
||||||
{
|
{
|
||||||
@ -31,7 +30,7 @@ class ServerCpuChart extends ChartWidget
|
|||||||
$cpu = collect(cache()->get("servers.{$this->server->id}.cpu_absolute"))
|
$cpu = collect(cache()->get("servers.{$this->server->id}.cpu_absolute"))
|
||||||
->slice(-$period)
|
->slice(-$period)
|
||||||
->map(fn ($value, $key) => [
|
->map(fn ($value, $key) => [
|
||||||
'cpu' => Number::format($value, maxPrecision: 2),
|
'cpu' => round($value, 2),
|
||||||
'timestamp' => Carbon::createFromTimestamp($key, auth()->user()->timezone ?? 'UTC')->format('H:i:s'),
|
'timestamp' => Carbon::createFromTimestamp($key, auth()->user()->timezone ?? 'UTC')->format('H:i:s'),
|
||||||
])
|
])
|
||||||
->all();
|
->all();
|
||||||
|
@ -7,7 +7,6 @@ use Carbon\Carbon;
|
|||||||
use Filament\Facades\Filament;
|
use Filament\Facades\Filament;
|
||||||
use Filament\Support\RawJs;
|
use Filament\Support\RawJs;
|
||||||
use Filament\Widgets\ChartWidget;
|
use Filament\Widgets\ChartWidget;
|
||||||
use Illuminate\Support\Number;
|
|
||||||
|
|
||||||
class ServerMemoryChart extends ChartWidget
|
class ServerMemoryChart extends ChartWidget
|
||||||
{
|
{
|
||||||
@ -31,7 +30,7 @@ class ServerMemoryChart extends ChartWidget
|
|||||||
$memUsed = collect(cache()->get("servers.{$this->server->id}.memory_bytes"))
|
$memUsed = collect(cache()->get("servers.{$this->server->id}.memory_bytes"))
|
||||||
->slice(-$period)
|
->slice(-$period)
|
||||||
->map(fn ($value, $key) => [
|
->map(fn ($value, $key) => [
|
||||||
'memory' => Number::format(config('panel.use_binary_prefix') ? $value / 1024 / 1024 / 1024 : $value / 1000 / 1000 / 1000, maxPrecision: 2),
|
'memory' => round(config('panel.use_binary_prefix') ? $value / 1024 / 1024 / 1024 : $value / 1000 / 1000 / 1000, 2),
|
||||||
'timestamp' => Carbon::createFromTimestamp($key, auth()->user()->timezone ?? 'UTC')->format('H:i:s'),
|
'timestamp' => Carbon::createFromTimestamp($key, auth()->user()->timezone ?? 'UTC')->format('H:i:s'),
|
||||||
])
|
])
|
||||||
->all();
|
->all();
|
||||||
|
@ -8,7 +8,6 @@ use App\Models\Server;
|
|||||||
use Carbon\CarbonInterface;
|
use Carbon\CarbonInterface;
|
||||||
use Filament\Notifications\Notification;
|
use Filament\Notifications\Notification;
|
||||||
use Filament\Widgets\StatsOverviewWidget;
|
use Filament\Widgets\StatsOverviewWidget;
|
||||||
use Illuminate\Support\Number;
|
|
||||||
use Livewire\Attributes\On;
|
use Livewire\Attributes\On;
|
||||||
|
|
||||||
class ServerOverview extends StatsOverviewWidget
|
class ServerOverview extends StatsOverviewWidget
|
||||||
@ -54,9 +53,9 @@ class ServerOverview extends StatsOverviewWidget
|
|||||||
}
|
}
|
||||||
|
|
||||||
$data = collect(cache()->get("servers.{$this->server->id}.cpu_absolute"))->last(default: 0);
|
$data = collect(cache()->get("servers.{$this->server->id}.cpu_absolute"))->last(default: 0);
|
||||||
$cpu = Number::format($data, maxPrecision: 2, locale: auth()->user()->language) . ' %';
|
$cpu = format_number($data, maxPrecision: 2) . ' %';
|
||||||
|
|
||||||
return $cpu . ($this->server->cpu > 0 ? ' / ' . Number::format($this->server->cpu, locale: auth()->user()->language) . ' %' : ' / ∞');
|
return $cpu . ($this->server->cpu > 0 ? ' / ' . format_number($this->server->cpu) . ' %' : ' / ∞');
|
||||||
}
|
}
|
||||||
|
|
||||||
public function memoryUsage(): string
|
public function memoryUsage(): string
|
||||||
@ -68,7 +67,7 @@ class ServerOverview extends StatsOverviewWidget
|
|||||||
}
|
}
|
||||||
|
|
||||||
$latestMemoryUsed = collect(cache()->get("servers.{$this->server->id}.memory_bytes"))->last(default: 0);
|
$latestMemoryUsed = collect(cache()->get("servers.{$this->server->id}.memory_bytes"))->last(default: 0);
|
||||||
$totalMemory = $this->server->memory * 2 ** 20;
|
$totalMemory = $this->server->memory * (config('panel.use_binary_prefix') ? 1024 * 1024 : 1000 * 1000);
|
||||||
|
|
||||||
$used = convert_bytes_to_readable($latestMemoryUsed);
|
$used = convert_bytes_to_readable($latestMemoryUsed);
|
||||||
$total = convert_bytes_to_readable($totalMemory);
|
$total = convert_bytes_to_readable($totalMemory);
|
||||||
|
@ -2,7 +2,6 @@
|
|||||||
|
|
||||||
namespace App\Livewire\Installer;
|
namespace App\Livewire\Installer;
|
||||||
|
|
||||||
use App\Filament\Admin\Pages\Dashboard;
|
|
||||||
use App\Livewire\Installer\Steps\CacheStep;
|
use App\Livewire\Installer\Steps\CacheStep;
|
||||||
use App\Livewire\Installer\Steps\DatabaseStep;
|
use App\Livewire\Installer\Steps\DatabaseStep;
|
||||||
use App\Livewire\Installer\Steps\EnvironmentStep;
|
use App\Livewire\Installer\Steps\EnvironmentStep;
|
||||||
@ -15,6 +14,7 @@ use App\Traits\CheckMigrationsTrait;
|
|||||||
use App\Traits\EnvironmentWriterTrait;
|
use App\Traits\EnvironmentWriterTrait;
|
||||||
use Exception;
|
use Exception;
|
||||||
use Filament\Actions\Action;
|
use Filament\Actions\Action;
|
||||||
|
use Filament\Facades\Filament;
|
||||||
use Filament\Forms\Contracts\HasForms;
|
use Filament\Forms\Contracts\HasForms;
|
||||||
use Filament\Schemas\Components\Wizard;
|
use Filament\Schemas\Components\Wizard;
|
||||||
use Filament\Forms\Concerns\InteractsWithForms;
|
use Filament\Forms\Concerns\InteractsWithForms;
|
||||||
@ -110,7 +110,7 @@ class PanelInstaller extends SimplePage implements HasForms
|
|||||||
$this->writeToEnv('env_session');
|
$this->writeToEnv('env_session');
|
||||||
|
|
||||||
// Redirect to admin panel
|
// Redirect to admin panel
|
||||||
$this->redirect(Dashboard::getUrl());
|
$this->redirect(Filament::getPanel('admin')->getUrl());
|
||||||
} catch (Halt) {
|
} catch (Halt) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -38,7 +38,7 @@ class ServerEntry extends Component
|
|||||||
<div class="flex justify-between text-center items-center gap-4">
|
<div class="flex justify-between text-center items-center gap-4">
|
||||||
<div>
|
<div>
|
||||||
<p class="text-sm dark:text-gray-400">{{ trans('server/dashboard.cpu') }}</p>
|
<p class="text-sm dark:text-gray-400">{{ trans('server/dashboard.cpu') }}</p>
|
||||||
<p class="text-md font-semibold">{{ Number::format(0, precision: 2, locale: auth()->user()->language ?? 'en') . '%' }}</p>
|
<p class="text-md font-semibold">{{ format_number(0, precision: 2) . '%' }}</p>
|
||||||
<hr class="p-0.5">
|
<hr class="p-0.5">
|
||||||
<p class="text-xs dark:text-gray-400">{{ $server->formatResource(\App\Enums\ServerResourceType::CPULimit) }}</p>
|
<p class="text-xs dark:text-gray-400">{{ $server->formatResource(\App\Enums\ServerResourceType::CPULimit) }}</p>
|
||||||
</div>
|
</div>
|
||||||
|
@ -46,7 +46,7 @@ class Role extends BaseRole
|
|||||||
'health' => [
|
'health' => [
|
||||||
'view',
|
'view',
|
||||||
],
|
],
|
||||||
'activity' => [
|
'activityLog' => [
|
||||||
'seeIps',
|
'seeIps',
|
||||||
],
|
],
|
||||||
];
|
];
|
||||||
|
@ -23,7 +23,6 @@ use Illuminate\Notifications\Notifiable;
|
|||||||
use Illuminate\Database\Query\JoinClause;
|
use Illuminate\Database\Query\JoinClause;
|
||||||
use Illuminate\Support\Arr;
|
use Illuminate\Support\Arr;
|
||||||
use Illuminate\Support\Facades\Http;
|
use Illuminate\Support\Facades\Http;
|
||||||
use Illuminate\Support\Number;
|
|
||||||
use Psr\Http\Message\ResponseInterface;
|
use Psr\Http\Message\ResponseInterface;
|
||||||
use Illuminate\Database\Eloquent\Relations\HasOne;
|
use Illuminate\Database\Eloquent\Relations\HasOne;
|
||||||
use Illuminate\Database\Eloquent\Relations\HasMany;
|
use Illuminate\Database\Eloquent\Relations\HasMany;
|
||||||
@ -490,7 +489,7 @@ class Server extends Model implements Validatable
|
|||||||
}
|
}
|
||||||
|
|
||||||
if ($resourceType->isPercentage()) {
|
if ($resourceType->isPercentage()) {
|
||||||
return Number::format($resourceAmount, precision: 2, locale: auth()->user()->language ?? 'en') . '%';
|
return format_number($resourceAmount, precision: 2) . '%';
|
||||||
}
|
}
|
||||||
|
|
||||||
return convert_bytes_to_readable($resourceAmount, base: 3);
|
return convert_bytes_to_readable($resourceAmount, base: 3);
|
||||||
|
@ -3,6 +3,7 @@
|
|||||||
namespace App\Providers\Filament;
|
namespace App\Providers\Filament;
|
||||||
|
|
||||||
use Filament\Actions\Action;
|
use Filament\Actions\Action;
|
||||||
|
use Filament\Facades\Filament;
|
||||||
use Filament\Navigation\NavigationGroup;
|
use Filament\Navigation\NavigationGroup;
|
||||||
use Filament\Panel;
|
use Filament\Panel;
|
||||||
|
|
||||||
@ -20,7 +21,7 @@ class AdminPanelProvider extends PanelProvider
|
|||||||
'profile' => fn (Action $action) => $action->label(auth()->user()->username),
|
'profile' => fn (Action $action) => $action->label(auth()->user()->username),
|
||||||
Action::make('exitAdmin')
|
Action::make('exitAdmin')
|
||||||
->label(fn () => trans('profile.exit_admin'))
|
->label(fn () => trans('profile.exit_admin'))
|
||||||
->url('/')
|
->url(fn () => Filament::getPanel('app')->getUrl())
|
||||||
->icon('tabler-arrow-back')
|
->icon('tabler-arrow-back')
|
||||||
->sort(24),
|
->sort(24),
|
||||||
])
|
])
|
||||||
|
@ -19,7 +19,7 @@ class AppPanelProvider extends PanelProvider
|
|||||||
'profile' => fn (Action $action) => $action->label(auth()->user()->username),
|
'profile' => fn (Action $action) => $action->label(auth()->user()->username),
|
||||||
Action::make('toAdmin')
|
Action::make('toAdmin')
|
||||||
->label(trans('profile.admin'))
|
->label(trans('profile.admin'))
|
||||||
->url('/admin')
|
->url(fn () => Filament::getPanel('admin')->getUrl())
|
||||||
->icon('tabler-arrow-forward')
|
->icon('tabler-arrow-forward')
|
||||||
->sort(5)
|
->sort(5)
|
||||||
->visible(fn () => auth()->user()->canAccessPanel(Filament::getPanel('admin'))),
|
->visible(fn () => auth()->user()->canAccessPanel(Filament::getPanel('admin'))),
|
||||||
|
@ -45,7 +45,7 @@ if (!function_exists('convert_bytes_to_readable')) {
|
|||||||
$fromBase = log($bytes) / log($conversionUnit);
|
$fromBase = log($bytes) / log($conversionUnit);
|
||||||
$base ??= floor($fromBase);
|
$base ??= floor($fromBase);
|
||||||
|
|
||||||
return Number::format(pow($conversionUnit, $fromBase - $base), $decimals, locale: auth()->user()->language) . ' ' . $suffix[$base];
|
return format_number(pow($conversionUnit, $fromBase - $base), precision: $decimals) . ' ' . $suffix[$base];
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -98,3 +98,15 @@ if (!function_exists('get_ip_from_hostname')) {
|
|||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (!function_exists('format_number')) {
|
||||||
|
function format_number(int|float $number, ?int $precision = null, ?int $maxPrecision = null): false|string
|
||||||
|
{
|
||||||
|
try {
|
||||||
|
return Number::format($number, $precision, $maxPrecision, auth()->user()->language ?? 'en');
|
||||||
|
} catch (Throwable) {
|
||||||
|
// User language is invalid, so default to english
|
||||||
|
return Number::format($number, $precision, $maxPrecision, 'en');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
@ -0,0 +1,27 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
use Illuminate\Database\Migrations\Migration;
|
||||||
|
use Illuminate\Support\Facades\DB;
|
||||||
|
|
||||||
|
return new class extends Migration
|
||||||
|
{
|
||||||
|
/**
|
||||||
|
* Run the migrations.
|
||||||
|
*/
|
||||||
|
public function up(): void
|
||||||
|
{
|
||||||
|
DB::table('permissions')
|
||||||
|
->where('name', 'seeIps activity')
|
||||||
|
->update(['name' => 'seeIps activityLog']);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Reverse the migrations.
|
||||||
|
*/
|
||||||
|
public function down(): void
|
||||||
|
{
|
||||||
|
DB::table('permissions')
|
||||||
|
->where('name', 'seeIps activityLog')
|
||||||
|
->update(['name' => 'seeIps activity']);
|
||||||
|
}
|
||||||
|
};
|
@ -1,15 +0,0 @@
|
|||||||
<?php
|
|
||||||
|
|
||||||
return [
|
|
||||||
'model_label' => 'Schedule',
|
|
||||||
'model_label_plural' => 'Schedule',
|
|
||||||
'import' => [
|
|
||||||
'file' => 'File',
|
|
||||||
'url' => 'URL',
|
|
||||||
'schedule_help' => 'This should be the raw .json file ( schedule-daily-restart.json )',
|
|
||||||
'url_help' => 'URLs must point directly to the raw .json file',
|
|
||||||
'add_url' => 'New URL',
|
|
||||||
'import_failed' => 'Import Failed',
|
|
||||||
'import_success' => 'Import Success',
|
|
||||||
],
|
|
||||||
];
|
|
@ -30,6 +30,8 @@ return [
|
|||||||
'cron_body' => 'Please keep in mind that the cron inputs below always assume UTC.',
|
'cron_body' => 'Please keep in mind that the cron inputs below always assume UTC.',
|
||||||
'cron_timezone' => 'Next run in your timezone (:timezone): <b> :next_run </b>',
|
'cron_timezone' => 'Next run in your timezone (:timezone): <b> :next_run </b>',
|
||||||
|
|
||||||
|
'invalid' => 'Invalid',
|
||||||
|
|
||||||
'time' => [
|
'time' => [
|
||||||
'minute' => 'Minute',
|
'minute' => 'Minute',
|
||||||
'hour' => 'Hour',
|
'hour' => 'Hour',
|
||||||
@ -104,4 +106,13 @@ return [
|
|||||||
|
|
||||||
'notification_invalid_cron' => 'The cron data provided does not evaluate to a valid expression',
|
'notification_invalid_cron' => 'The cron data provided does not evaluate to a valid expression',
|
||||||
|
|
||||||
|
'import_action' => [
|
||||||
|
'file' => 'File',
|
||||||
|
'url' => 'URL',
|
||||||
|
'schedule_help' => 'This should be the raw .json file ( schedule-daily-restart.json )',
|
||||||
|
'url_help' => 'URLs must point directly to the raw .json file',
|
||||||
|
'add_url' => 'New URL',
|
||||||
|
'import_failed' => 'Import Failed',
|
||||||
|
'import_success' => 'Import Success',
|
||||||
|
],
|
||||||
];
|
];
|
||||||
|
@ -0,0 +1,67 @@
|
|||||||
|
@php
|
||||||
|
$statePath = $getStatePath();
|
||||||
|
$isRequired = $isRequired();
|
||||||
|
$isDisabled = $isDisabled();
|
||||||
|
$type = $getType();
|
||||||
|
@endphp
|
||||||
|
|
||||||
|
<x-dynamic-component
|
||||||
|
:component="$getFieldWrapperView()"
|
||||||
|
:field="$field"
|
||||||
|
>
|
||||||
|
<x-slot name="label">
|
||||||
|
{{ $getLabel() }}
|
||||||
|
</x-slot>
|
||||||
|
|
||||||
|
<x-filament::input.wrapper
|
||||||
|
:disabled="$isDisabled"
|
||||||
|
:prefix="$getPrefixLabel()"
|
||||||
|
:valid="! $errors->has($statePath)"
|
||||||
|
:attributes="\Filament\Support\prepare_inherited_attributes($getExtraAttributeBag())->class([
|
||||||
|
'fi-fo-text-input overflow-hidden' => $type === \App\Enums\StartupVariableType::Text,
|
||||||
|
'fi-fo-select' => $type === \App\Enums\StartupVariableType::Select
|
||||||
|
])"
|
||||||
|
>
|
||||||
|
@if ($type === \App\Enums\StartupVariableType::Select)
|
||||||
|
<x-filament::input.select
|
||||||
|
:id="$getId()"
|
||||||
|
:required="$isRequired"
|
||||||
|
:disabled="$isDisabled"
|
||||||
|
:attributes="
|
||||||
|
$getExtraInputAttributeBag()
|
||||||
|
->merge([
|
||||||
|
$applyStateBindingModifiers('wire:model') => $statePath,
|
||||||
|
], escape: false)
|
||||||
|
"
|
||||||
|
>
|
||||||
|
@if (!$isRequired)
|
||||||
|
<option value="">
|
||||||
|
@if (!$isDisabled)
|
||||||
|
{{ trans('filament-forms::components.select.placeholder') }}
|
||||||
|
@endif
|
||||||
|
</option>
|
||||||
|
@endif
|
||||||
|
|
||||||
|
@foreach ($getSelectOptions() as $value)
|
||||||
|
<option value="{{ $value }}">
|
||||||
|
{{ $value }}
|
||||||
|
</option>
|
||||||
|
@endforeach
|
||||||
|
</x-filament::input.select>
|
||||||
|
@else
|
||||||
|
<x-filament::input
|
||||||
|
:id="$getId()"
|
||||||
|
:required="$isRequired"
|
||||||
|
:disabled="$isDisabled"
|
||||||
|
:placeholder="$getPlaceholder()"
|
||||||
|
:attributes="
|
||||||
|
\Filament\Support\prepare_inherited_attributes($getExtraInputAttributeBag())
|
||||||
|
->merge($getExtraAlpineAttributes(), escape: false)
|
||||||
|
->merge([
|
||||||
|
$applyStateBindingModifiers('wire:model') => $statePath,
|
||||||
|
], escape: false)
|
||||||
|
"
|
||||||
|
/>
|
||||||
|
@endif
|
||||||
|
</x-filament::input.wrapper>
|
||||||
|
</x-dynamic-component>
|
Loading…
x
Reference in New Issue
Block a user