Add import & export for schedules (#1530)

This commit is contained in:
Boy132 2025-07-19 16:48:21 +02:00 committed by GitHub
parent 61098b11f2
commit 340d1b543c
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194
7 changed files with 291 additions and 0 deletions

View File

@ -0,0 +1,34 @@
<?php
namespace App\Filament\Components\Actions;
use App\Models\Permission;
use App\Models\Schedule;
use App\Models\Server;
use App\Services\Schedules\Sharing\ScheduleExporterService;
use Filament\Actions\Action;
use Filament\Facades\Filament;
class ExportScheduleAction extends Action
{
public static function getDefaultName(): ?string
{
return 'export';
}
protected function setUp(): void
{
parent::setUp();
/** @var Server $server */
$server = Filament::getTenant();
$this->label(trans('filament-actions::export.modal.actions.export.label'));
$this->authorize(fn () => auth()->user()->can(Permission::ACTION_SCHEDULE_READ, $server));
$this->action(fn (ScheduleExporterService $service, Schedule $schedule) => response()->streamDownload(function () use ($service, $schedule) {
echo $service->handle($schedule);
}, 'schedule-' . str($schedule->name)->kebab()->lower()->trim() . '.json'));
}
}

View File

@ -0,0 +1,121 @@
<?php
namespace App\Filament\Components\Actions;
use App\Models\Permission;
use App\Models\Server;
use App\Services\Schedules\Sharing\ScheduleImporterService;
use Exception;
use Filament\Actions\Action;
use Filament\Facades\Filament;
use Filament\Forms\Components\FileUpload;
use Filament\Forms\Components\Repeater;
use Filament\Forms\Components\Tabs;
use Filament\Forms\Components\Tabs\Tab;
use Filament\Forms\Components\TextInput;
use Filament\Notifications\Notification;
use Illuminate\Support\Arr;
use Livewire\Features\SupportFileUploads\TemporaryUploadedFile;
class ImportScheduleAction extends Action
{
public static function getDefaultName(): ?string
{
return 'import';
}
protected function setUp(): void
{
parent::setUp();
/** @var Server $server */
$server = Filament::getTenant();
$this->label(trans('filament-actions::import.modal.actions.import.label'));
$this->authorize(fn () => auth()->user()->can(Permission::ACTION_SCHEDULE_CREATE, $server));
$this->form([
Tabs::make('Tabs')
->contained(false)
->tabs([
Tab::make(trans('admin/schedule.import.file'))
->icon('tabler-file-upload')
->schema([
FileUpload::make('files')
->label(trans('admin/schedule.model_label'))
->hint(trans('admin/schedule.import.schedule_help'))
->acceptedFileTypes(['application/json'])
->preserveFilenames()
->previewable(false)
->storeFiles(false)
->multiple(true),
]),
Tab::make(trans('admin/schedule.import.url'))
->icon('tabler-world-upload')
->schema([
Repeater::make('urls')
->label('')
->itemLabel(fn (array $state) => str($state['url'])->afterLast('/schedule-')->before('.json')->headline())
->hint(trans('admin/schedule.import.url_help'))
->addActionLabel(trans('admin/schedule.import.add_url'))
->grid(2)
->reorderable(false)
->addable(true)
->deletable(fn (array $state) => count($state) > 1)
->schema([
TextInput::make('url')
->live()
->label(trans('admin/schedule.import.url'))
->url()
->endsWith('.json')
->validationAttribute(trans('admin/schedule.import.url')),
]),
]),
]),
]);
$this->action(function (array $data, ScheduleImporterService $service) use ($server) {
$schedules = array_merge(collect($data['urls'])->flatten()->whereNotNull()->unique()->all(), Arr::wrap($data['files']));
if (empty($schedules)) {
return;
}
[$success, $failed] = [collect(), collect()];
foreach ($schedules as $schedule) {
if ($schedule instanceof TemporaryUploadedFile) {
$name = str($schedule->getClientOriginalName())->afterLast('schedule-')->before('.json')->headline();
$method = 'fromFile';
} else {
$schedule = str($schedule);
$schedule = $schedule->contains('github.com') ? $schedule->replaceFirst('blob', 'raw') : $schedule;
$name = $schedule->afterLast('/schedule-')->before('.json')->headline();
$method = 'fromUrl';
}
try {
$service->$method($schedule, $server);
$success->push($name);
} catch (Exception $exception) {
$failed->push($name);
report($exception);
}
}
if ($failed->count() > 0) {
Notification::make()
->title(trans('admin/schedule.import.import_failed'))
->body($failed->join(', '))
->danger()
->send();
}
if ($success->count() > 0) {
Notification::make()
->title(trans('admin/schedule.import.import_success'))
->body($success->join(', '))
->success()
->send();
}
});
}
}

View File

@ -3,6 +3,7 @@
namespace App\Filament\Server\Resources\ScheduleResource\Pages;
use App\Facades\Activity;
use App\Filament\Components\Actions\ExportScheduleAction;
use App\Filament\Server\Resources\ScheduleResource;
use App\Models\Schedule;
use App\Traits\Filament\CanCustomizeHeaderActions;
@ -50,6 +51,7 @@ class EditSchedule extends EditRecord
->property('name', $record->name)
->log();
}),
ExportScheduleAction::make(),
$this->getSaveFormAction()->formId('form')->label('Save'),
$this->getCancelFormAction()->formId('form'),
];

View File

@ -2,6 +2,7 @@
namespace App\Filament\Server\Resources\ScheduleResource\Pages;
use App\Filament\Components\Actions\ImportScheduleAction;
use App\Filament\Server\Resources\ScheduleResource;
use App\Traits\Filament\CanCustomizeHeaderActions;
use App\Traits\Filament\CanCustomizeHeaderWidgets;
@ -23,6 +24,7 @@ class ListSchedules extends ListRecords
return [
CreateAction::make()
->label('New Schedule'),
ImportScheduleAction::make(),
];
}

View File

@ -0,0 +1,39 @@
<?php
namespace App\Services\Schedules\Sharing;
use App\Models\Schedule;
use App\Models\Task;
class ScheduleExporterService
{
public function handle(Schedule|int $schedule): string
{
if (!$schedule instanceof Schedule) {
$schedule = Schedule::findOrFail($schedule);
}
$data = [
'name' => $schedule->name,
'is_active' => $schedule->is_active,
'only_when_online' => $schedule->only_when_online,
'cron_minute' => $schedule->cron_minute,
'cron_hour' => $schedule->cron_hour,
'cron_day_of_month' => $schedule->cron_day_of_month,
'cron_month' => $schedule->cron_month,
'cron_day_of_week' => $schedule->cron_day_of_week,
'tasks' => $schedule->tasks->map(function (Task $task) {
return [
'sequence_id' => $task->sequence_id,
'action' => $task->action,
'payload' => $task->payload,
'time_offset' => $task->time_offset,
'continue_on_failure' => $task->continue_on_failure,
];
}),
];
return json_encode($data, JSON_PRETTY_PRINT);
}
}

View File

@ -0,0 +1,78 @@
<?php
namespace App\Services\Schedules\Sharing;
use App\Exceptions\Service\InvalidFileUploadException;
use App\Helpers\Utilities;
use Illuminate\Support\Arr;
use Illuminate\Http\UploadedFile;
use App\Models\Schedule;
use App\Models\Server;
use App\Models\Task;
use Illuminate\Database\ConnectionInterface;
use Spatie\TemporaryDirectory\TemporaryDirectory;
class ScheduleImporterService
{
public function __construct(protected ConnectionInterface $connection) {}
public function fromFile(UploadedFile $file, Server $server): Schedule
{
if ($file->getError() !== UPLOAD_ERR_OK) {
throw new InvalidFileUploadException('The selected file was not uploaded successfully');
}
try {
$parsed = json_decode($file->getContent(), true, 512, JSON_THROW_ON_ERROR);
} catch (\JsonException $exception) {
throw new InvalidFileUploadException('Could not read JSON file: ' . $exception->getMessage());
}
return $this->connection->transaction(function () use ($server, $parsed) {
$minute = Arr::get($parsed, 'cron_minute', '0');
$hour = Arr::get($parsed, 'cron_hour', '0');
$dayOfMonth = Arr::get($parsed, 'cron_day_of_month', '*');
$month = Arr::get($parsed, 'cron_month', '*');
$dayOfWeek = Arr::get($parsed, 'cron_day_of_week', '*');
$schedule = Schedule::create([
'server_id' => $server->id,
'name' => Arr::get($parsed, 'name'),
'is_active' => Arr::get($parsed, 'is_active'),
'only_when_online' => Arr::get($parsed, 'only_when_online'),
'cron_minute' => $minute,
'cron_hour' => $hour,
'cron_day_of_month' => $dayOfMonth,
'cron_month' => $month,
'cron_day_of_week' => $dayOfWeek,
'next_run_at' => Utilities::getScheduleNextRunDate($minute, $hour, $dayOfMonth, $month, $dayOfWeek),
]);
foreach (Arr::get($parsed, 'tasks', []) as $task) {
Task::create([
'schedule_id' => $schedule->id,
'sequence_id' => Arr::get($task, 'sequence_id'),
'action' => Arr::get($task, 'action'),
'payload' => Arr::get($task, 'payload'),
'time_offset' => Arr::get($task, 'time_offset'),
'continue_on_failure' => Arr::get($task, 'continue_on_failure'),
]);
}
return $schedule;
});
}
public function fromUrl(string $url, Server $server): Schedule
{
$info = pathinfo($url);
$tmpDir = TemporaryDirectory::make()->deleteWhenDestroyed();
$tmpPath = $tmpDir->path($info['basename']);
if (!file_put_contents($tmpPath, file_get_contents($url))) {
throw new InvalidFileUploadException('Could not write temporary file.');
}
return $this->fromFile(new UploadedFile($tmpPath, $info['basename'], 'application/json'), $server);
}
}

View File

@ -0,0 +1,15 @@
<?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',
],
];