Finish building out schedule management functionality
This commit is contained in:
parent
4ac6507b69
commit
1e0d630e1f
16 changed files with 510 additions and 79 deletions
|
@ -3,6 +3,8 @@
|
||||||
namespace Pterodactyl\Helpers;
|
namespace Pterodactyl\Helpers;
|
||||||
|
|
||||||
use Exception;
|
use Exception;
|
||||||
|
use Carbon\Carbon;
|
||||||
|
use Cron\CronExpression;
|
||||||
use Illuminate\Support\Facades\Log;
|
use Illuminate\Support\Facades\Log;
|
||||||
|
|
||||||
class Utilities
|
class Utilities
|
||||||
|
@ -32,4 +34,20 @@ class Utilities
|
||||||
|
|
||||||
return $string;
|
return $string;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Converts schedule cron data into a carbon object.
|
||||||
|
*
|
||||||
|
* @param string $minute
|
||||||
|
* @param string $hour
|
||||||
|
* @param string $dayOfMonth
|
||||||
|
* @param string $dayOfWeek
|
||||||
|
* @return \Carbon\Carbon
|
||||||
|
*/
|
||||||
|
public static function getScheduleNextRunDate(string $minute, string $hour, string $dayOfMonth, string $dayOfWeek)
|
||||||
|
{
|
||||||
|
return Carbon::instance(CronExpression::factory(
|
||||||
|
sprintf('%s %s %s * %s', $minute, $hour, $dayOfMonth, $dayOfWeek)
|
||||||
|
)->getNextRunDate());
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -2,23 +2,51 @@
|
||||||
|
|
||||||
namespace Pterodactyl\Http\Controllers\Api\Client\Servers;
|
namespace Pterodactyl\Http\Controllers\Api\Client\Servers;
|
||||||
|
|
||||||
|
use Exception;
|
||||||
|
use Carbon\Carbon;
|
||||||
use Illuminate\Http\Request;
|
use Illuminate\Http\Request;
|
||||||
|
use Illuminate\Http\Response;
|
||||||
use Pterodactyl\Models\Server;
|
use Pterodactyl\Models\Server;
|
||||||
use Pterodactyl\Models\Schedule;
|
use Pterodactyl\Models\Schedule;
|
||||||
|
use Illuminate\Http\JsonResponse;
|
||||||
|
use Pterodactyl\Helpers\Utilities;
|
||||||
|
use Pterodactyl\Exceptions\DisplayException;
|
||||||
|
use Pterodactyl\Repositories\Eloquent\ScheduleRepository;
|
||||||
use Pterodactyl\Transformers\Api\Client\ScheduleTransformer;
|
use Pterodactyl\Transformers\Api\Client\ScheduleTransformer;
|
||||||
use Pterodactyl\Http\Controllers\Api\Client\ClientApiController;
|
use Pterodactyl\Http\Controllers\Api\Client\ClientApiController;
|
||||||
use Symfony\Component\HttpKernel\Exception\NotFoundHttpException;
|
use Symfony\Component\HttpKernel\Exception\NotFoundHttpException;
|
||||||
|
use Pterodactyl\Http\Requests\Api\Client\Servers\Schedules\ViewScheduleRequest;
|
||||||
|
use Pterodactyl\Http\Requests\Api\Client\Servers\Schedules\StoreScheduleRequest;
|
||||||
|
use Pterodactyl\Http\Requests\Api\Client\Servers\Schedules\DeleteScheduleRequest;
|
||||||
|
use Pterodactyl\Http\Requests\Api\Client\Servers\Schedules\UpdateScheduleRequest;
|
||||||
|
|
||||||
class ScheduleController extends ClientApiController
|
class ScheduleController extends ClientApiController
|
||||||
{
|
{
|
||||||
|
/**
|
||||||
|
* @var \Pterodactyl\Repositories\Eloquent\ScheduleRepository
|
||||||
|
*/
|
||||||
|
private $repository;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* ScheduleController constructor.
|
||||||
|
*
|
||||||
|
* @param \Pterodactyl\Repositories\Eloquent\ScheduleRepository $repository
|
||||||
|
*/
|
||||||
|
public function __construct(ScheduleRepository $repository)
|
||||||
|
{
|
||||||
|
parent::__construct();
|
||||||
|
|
||||||
|
$this->repository = $repository;
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Returns all of the schedules belonging to a given server.
|
* Returns all of the schedules belonging to a given server.
|
||||||
*
|
*
|
||||||
* @param \Illuminate\Http\Request $request
|
* @param \Pterodactyl\Http\Requests\Api\Client\Servers\Schedules\ViewScheduleRequest $request
|
||||||
* @param \Pterodactyl\Models\Server $server
|
* @param \Pterodactyl\Models\Server $server
|
||||||
* @return array
|
* @return array
|
||||||
*/
|
*/
|
||||||
public function index(Request $request, Server $server)
|
public function index(ViewScheduleRequest $request, Server $server)
|
||||||
{
|
{
|
||||||
$schedules = $server->schedule;
|
$schedules = $server->schedule;
|
||||||
$schedules->loadMissing('tasks');
|
$schedules->loadMissing('tasks');
|
||||||
|
@ -28,15 +56,44 @@ class ScheduleController extends ClientApiController
|
||||||
->toArray();
|
->toArray();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Store a new schedule for a server.
|
||||||
|
*
|
||||||
|
* @param \Pterodactyl\Http\Requests\Api\Client\Servers\Schedules\StoreScheduleRequest $request
|
||||||
|
* @param \Pterodactyl\Models\Server $server
|
||||||
|
* @return array
|
||||||
|
*
|
||||||
|
* @throws \Pterodactyl\Exceptions\DisplayException
|
||||||
|
* @throws \Pterodactyl\Exceptions\Model\DataValidationException
|
||||||
|
*/
|
||||||
|
public function store(StoreScheduleRequest $request, Server $server)
|
||||||
|
{
|
||||||
|
/** @var \Pterodactyl\Models\Schedule $model */
|
||||||
|
$model = $this->repository->create([
|
||||||
|
'server_id' => $server->id,
|
||||||
|
'name' => $request->input('name'),
|
||||||
|
'cron_day_of_week' => $request->input('day_of_week'),
|
||||||
|
'cron_day_of_month' => $request->input('day_of_month'),
|
||||||
|
'cron_hour' => $request->input('hour'),
|
||||||
|
'cron_minute' => $request->input('minute'),
|
||||||
|
'is_active' => (bool) $request->input('is_active'),
|
||||||
|
'next_run_at' => $this->getNextRunAt($request),
|
||||||
|
]);
|
||||||
|
|
||||||
|
return $this->fractal->item($model)
|
||||||
|
->transformWith($this->getTransformer(ScheduleTransformer::class))
|
||||||
|
->toArray();
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Returns a specific schedule for the server.
|
* Returns a specific schedule for the server.
|
||||||
*
|
*
|
||||||
* @param \Illuminate\Http\Request $request
|
* @param \Pterodactyl\Http\Requests\Api\Client\Servers\Schedules\ViewScheduleRequest $request
|
||||||
* @param \Pterodactyl\Models\Server $server
|
* @param \Pterodactyl\Models\Server $server
|
||||||
* @param \Pterodactyl\Models\Schedule $schedule
|
* @param \Pterodactyl\Models\Schedule $schedule
|
||||||
* @return array
|
* @return array
|
||||||
*/
|
*/
|
||||||
public function view(Request $request, Server $server, Schedule $schedule)
|
public function view(ViewScheduleRequest $request, Server $server, Schedule $schedule)
|
||||||
{
|
{
|
||||||
if ($schedule->server_id !== $server->id) {
|
if ($schedule->server_id !== $server->id) {
|
||||||
throw new NotFoundHttpException;
|
throw new NotFoundHttpException;
|
||||||
|
@ -48,4 +105,71 @@ class ScheduleController extends ClientApiController
|
||||||
->transformWith($this->getTransformer(ScheduleTransformer::class))
|
->transformWith($this->getTransformer(ScheduleTransformer::class))
|
||||||
->toArray();
|
->toArray();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Updates a given schedule with the new data provided.
|
||||||
|
*
|
||||||
|
* @param \Pterodactyl\Http\Requests\Api\Client\Servers\Schedules\UpdateScheduleRequest $request
|
||||||
|
* @param \Pterodactyl\Models\Server $server
|
||||||
|
* @param \Pterodactyl\Models\Schedule $schedule
|
||||||
|
* @return array
|
||||||
|
*
|
||||||
|
* @throws \Pterodactyl\Exceptions\DisplayException
|
||||||
|
* @throws \Pterodactyl\Exceptions\Model\DataValidationException
|
||||||
|
* @throws \Pterodactyl\Exceptions\Repository\RecordNotFoundException
|
||||||
|
*/
|
||||||
|
public function update(UpdateScheduleRequest $request, Server $server, Schedule $schedule)
|
||||||
|
{
|
||||||
|
$this->repository->update($schedule->id, [
|
||||||
|
'name' => $request->input('name'),
|
||||||
|
'cron_day_of_week' => $request->input('day_of_week'),
|
||||||
|
'cron_day_of_month' => $request->input('day_of_month'),
|
||||||
|
'cron_hour' => $request->input('hour'),
|
||||||
|
'cron_minute' => $request->input('minute'),
|
||||||
|
'is_active' => (bool) $request->input('is_active'),
|
||||||
|
'next_run_at' => $this->getNextRunAt($request),
|
||||||
|
]);
|
||||||
|
|
||||||
|
return $this->fractal->item($schedule->refresh())
|
||||||
|
->transformWith($this->getTransformer(ScheduleTransformer::class))
|
||||||
|
->toArray();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Deletes a schedule and it's associated tasks.
|
||||||
|
*
|
||||||
|
* @param \Pterodactyl\Http\Requests\Api\Client\Servers\Schedules\DeleteScheduleRequest $request
|
||||||
|
* @param \Pterodactyl\Models\Server $server
|
||||||
|
* @param \Pterodactyl\Models\Schedule $schedule
|
||||||
|
* @return \Illuminate\Http\JsonResponse
|
||||||
|
*/
|
||||||
|
public function delete(DeleteScheduleRequest $request, Server $server, Schedule $schedule)
|
||||||
|
{
|
||||||
|
$this->repository->delete($schedule->id);
|
||||||
|
|
||||||
|
return JsonResponse::create([], Response::HTTP_NO_CONTENT);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get the next run timestamp based on the cron data provided.
|
||||||
|
*
|
||||||
|
* @param \Illuminate\Http\Request $request
|
||||||
|
* @return \Carbon\Carbon
|
||||||
|
* @throws \Pterodactyl\Exceptions\DisplayException
|
||||||
|
*/
|
||||||
|
protected function getNextRunAt(Request $request): Carbon
|
||||||
|
{
|
||||||
|
try {
|
||||||
|
return Utilities::getScheduleNextRunDate(
|
||||||
|
$request->input('minute'),
|
||||||
|
$request->input('hour'),
|
||||||
|
$request->input('day_of_month'),
|
||||||
|
$request->input('day_of_week')
|
||||||
|
);
|
||||||
|
} catch (Exception $exception) {
|
||||||
|
throw new DisplayException(
|
||||||
|
'The cron data provided does not evaluate to a valid expression.'
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -2,7 +2,6 @@
|
||||||
|
|
||||||
namespace Pterodactyl\Http\Requests\Api\Client;
|
namespace Pterodactyl\Http\Requests\Api\Client;
|
||||||
|
|
||||||
use Pterodactyl\Models\User;
|
|
||||||
use Pterodactyl\Models\Server;
|
use Pterodactyl\Models\Server;
|
||||||
use Pterodactyl\Contracts\Http\ClientPermissionsRequest;
|
use Pterodactyl\Contracts\Http\ClientPermissionsRequest;
|
||||||
use Pterodactyl\Http\Requests\Api\Application\ApplicationApiRequest;
|
use Pterodactyl\Http\Requests\Api\Application\ApplicationApiRequest;
|
||||||
|
@ -20,7 +19,15 @@ class ClientApiRequest extends ApplicationApiRequest
|
||||||
public function authorize(): bool
|
public function authorize(): bool
|
||||||
{
|
{
|
||||||
if ($this instanceof ClientPermissionsRequest || method_exists($this, 'permission')) {
|
if ($this instanceof ClientPermissionsRequest || method_exists($this, 'permission')) {
|
||||||
return $this->user()->can($this->permission(), $this->getModel(Server::class));
|
$server = $this->route()->parameter('server');
|
||||||
|
|
||||||
|
if ($server instanceof Server) {
|
||||||
|
return $this->user()->can($this->permission(), $server);
|
||||||
|
}
|
||||||
|
|
||||||
|
// If there is no server available on the reqest, trigger a failure since
|
||||||
|
// we expect there to be one at this point.
|
||||||
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
|
|
|
@ -0,0 +1,16 @@
|
||||||
|
<?php
|
||||||
|
|
||||||
|
namespace Pterodactyl\Http\Requests\Api\Client\Servers\Schedules;
|
||||||
|
|
||||||
|
use Pterodactyl\Models\Permission;
|
||||||
|
|
||||||
|
class DeleteScheduleRequest extends ViewScheduleRequest
|
||||||
|
{
|
||||||
|
/**
|
||||||
|
* @return string
|
||||||
|
*/
|
||||||
|
public function permission(): string
|
||||||
|
{
|
||||||
|
return Permission::ACTION_SCHEDULE_DELETE;
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,31 @@
|
||||||
|
<?php
|
||||||
|
|
||||||
|
namespace Pterodactyl\Http\Requests\Api\Client\Servers\Schedules;
|
||||||
|
|
||||||
|
use Pterodactyl\Models\Permission;
|
||||||
|
|
||||||
|
class StoreScheduleRequest extends ViewScheduleRequest
|
||||||
|
{
|
||||||
|
/**
|
||||||
|
* @return string
|
||||||
|
*/
|
||||||
|
public function permission(): string
|
||||||
|
{
|
||||||
|
return Permission::ACTION_SCHEDULE_CREATE;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return array
|
||||||
|
*/
|
||||||
|
public function rules(): array
|
||||||
|
{
|
||||||
|
return [
|
||||||
|
'name' => 'required|string|min:1',
|
||||||
|
'is_active' => 'boolean',
|
||||||
|
'minute' => 'required|string',
|
||||||
|
'hour' => 'required|string',
|
||||||
|
'day_of_month' => 'required|string',
|
||||||
|
'day_of_week' => 'required|string',
|
||||||
|
];
|
||||||
|
}
|
||||||
|
}
|
|
@ -3,9 +3,8 @@
|
||||||
namespace Pterodactyl\Http\Requests\Api\Client\Servers\Schedules;
|
namespace Pterodactyl\Http\Requests\Api\Client\Servers\Schedules;
|
||||||
|
|
||||||
use Pterodactyl\Models\Permission;
|
use Pterodactyl\Models\Permission;
|
||||||
use Pterodactyl\Http\Requests\Api\Client\ClientApiRequest;
|
|
||||||
|
|
||||||
class StoreTaskRequest extends ClientApiRequest
|
class StoreTaskRequest extends ViewScheduleRequest
|
||||||
{
|
{
|
||||||
/**
|
/**
|
||||||
* Determine if the user is allowed to create a new task for this schedule. We simply
|
* Determine if the user is allowed to create a new task for this schedule. We simply
|
||||||
|
@ -14,7 +13,7 @@ class StoreTaskRequest extends ClientApiRequest
|
||||||
*
|
*
|
||||||
* @return string
|
* @return string
|
||||||
*/
|
*/
|
||||||
public function permission()
|
public function permission(): string
|
||||||
{
|
{
|
||||||
return Permission::ACTION_SCHEDULE_UPDATE;
|
return Permission::ACTION_SCHEDULE_UPDATE;
|
||||||
}
|
}
|
||||||
|
|
|
@ -0,0 +1,16 @@
|
||||||
|
<?php
|
||||||
|
|
||||||
|
namespace Pterodactyl\Http\Requests\Api\Client\Servers\Schedules;
|
||||||
|
|
||||||
|
use Pterodactyl\Models\Permission;
|
||||||
|
|
||||||
|
class UpdateScheduleRequest extends StoreScheduleRequest
|
||||||
|
{
|
||||||
|
/**
|
||||||
|
* @return string
|
||||||
|
*/
|
||||||
|
public function permission(): string
|
||||||
|
{
|
||||||
|
return Permission::ACTION_SCHEDULE_UPDATE;
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,50 @@
|
||||||
|
<?php
|
||||||
|
|
||||||
|
namespace Pterodactyl\Http\Requests\Api\Client\Servers\Schedules;
|
||||||
|
|
||||||
|
use Pterodactyl\Models\Task;
|
||||||
|
use Pterodactyl\Models\Server;
|
||||||
|
use Pterodactyl\Models\Schedule;
|
||||||
|
use Pterodactyl\Models\Permission;
|
||||||
|
use Pterodactyl\Http\Requests\Api\Client\ClientApiRequest;
|
||||||
|
use Symfony\Component\HttpKernel\Exception\NotFoundHttpException;
|
||||||
|
|
||||||
|
class ViewScheduleRequest extends ClientApiRequest
|
||||||
|
{
|
||||||
|
/**
|
||||||
|
* Determine if this resource can be viewed.
|
||||||
|
*
|
||||||
|
* @return bool
|
||||||
|
*/
|
||||||
|
public function authorize(): bool
|
||||||
|
{
|
||||||
|
if (! parent::authorize()) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
$server = $this->route()->parameter('server');
|
||||||
|
$schedule = $this->route()->parameter('schedule');
|
||||||
|
|
||||||
|
// If the schedule does not belong to this server throw a 404 error. Also throw an
|
||||||
|
// error if the task being requested does not belong to the associated schedule.
|
||||||
|
if ($server instanceof Server && $schedule instanceof Schedule) {
|
||||||
|
$task = $this->route()->parameter('task');
|
||||||
|
|
||||||
|
if ($schedule->server_id !== $server->id || ($task instanceof Task && $task->schedule_id !== $schedule->id)) {
|
||||||
|
throw new NotFoundHttpException(
|
||||||
|
'The requested resource does not exist on the system.'
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return string
|
||||||
|
*/
|
||||||
|
public function permission(): string
|
||||||
|
{
|
||||||
|
return Permission::ACTION_SCHEDULE_READ;
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,19 @@
|
||||||
|
import { rawDataToServerSchedule, Schedule } from '@/api/server/schedules/getServerSchedules';
|
||||||
|
import http from '@/api/http';
|
||||||
|
|
||||||
|
type Data = Pick<Schedule, 'cron' | 'name' | 'isActive'> & { id?: number }
|
||||||
|
|
||||||
|
export default (uuid: string, schedule: Data): Promise<Schedule> => {
|
||||||
|
return new Promise((resolve, reject) => {
|
||||||
|
http.post(`/api/client/servers/${uuid}/schedules${schedule.id ? `/${schedule.id}` : ''}`, {
|
||||||
|
is_active: schedule.isActive,
|
||||||
|
name: schedule.name,
|
||||||
|
minute: schedule.cron.minute,
|
||||||
|
hour: schedule.cron.hour,
|
||||||
|
day_of_month: schedule.cron.dayOfMonth,
|
||||||
|
day_of_week: schedule.cron.dayOfWeek,
|
||||||
|
})
|
||||||
|
.then(({ data }) => resolve(rawDataToServerSchedule(data.attributes)))
|
||||||
|
.catch(reject);
|
||||||
|
});
|
||||||
|
};
|
9
resources/scripts/api/server/schedules/deleteSchedule.ts
Normal file
9
resources/scripts/api/server/schedules/deleteSchedule.ts
Normal file
|
@ -0,0 +1,9 @@
|
||||||
|
import http from '@/api/http';
|
||||||
|
|
||||||
|
export default (uuid: string, schedule: number): Promise<void> => {
|
||||||
|
return new Promise((resolve, reject) => {
|
||||||
|
http.delete(`/api/client/servers/${uuid}/schedules/${schedule}`)
|
||||||
|
.then(() => resolve())
|
||||||
|
.catch(reject);
|
||||||
|
});
|
||||||
|
};
|
|
@ -0,0 +1,69 @@
|
||||||
|
import React, { useState } from 'react';
|
||||||
|
import Modal from '@/components/elements/Modal';
|
||||||
|
import deleteSchedule from '@/api/server/schedules/deleteSchedule';
|
||||||
|
import { ServerContext } from '@/state/server';
|
||||||
|
import { Actions, useStoreActions } from 'easy-peasy';
|
||||||
|
import { ApplicationStore } from '@/state';
|
||||||
|
import { httpErrorToHuman } from '@/api/http';
|
||||||
|
|
||||||
|
interface Props {
|
||||||
|
scheduleId: number;
|
||||||
|
onDeleted: () => void;
|
||||||
|
}
|
||||||
|
|
||||||
|
export default ({ scheduleId, onDeleted }: Props) => {
|
||||||
|
const [ visible, setVisible ] = useState(false);
|
||||||
|
const [ isLoading, setIsLoading ] = useState(false);
|
||||||
|
const uuid = ServerContext.useStoreState(state => state.server.data!.uuid);
|
||||||
|
const { addError, clearFlashes } = useStoreActions((actions: Actions<ApplicationStore>) => actions.flashes);
|
||||||
|
|
||||||
|
const onDelete = () => {
|
||||||
|
setIsLoading(true);
|
||||||
|
clearFlashes('schedules');
|
||||||
|
deleteSchedule(uuid, scheduleId)
|
||||||
|
.then(() => {
|
||||||
|
setIsLoading(false);
|
||||||
|
onDeleted();
|
||||||
|
})
|
||||||
|
.catch(error => {
|
||||||
|
console.error(error);
|
||||||
|
|
||||||
|
addError({ key: 'schedules', message: httpErrorToHuman(error) });
|
||||||
|
setIsLoading(false);
|
||||||
|
setVisible(false);
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
return (
|
||||||
|
<>
|
||||||
|
<Modal
|
||||||
|
visible={visible}
|
||||||
|
onDismissed={() => setVisible(false)}
|
||||||
|
showSpinnerOverlay={isLoading}
|
||||||
|
>
|
||||||
|
<h3 className={'mb-6'}>Delete schedule</h3>
|
||||||
|
<p className={'text-sm'}>
|
||||||
|
Are you sure you want to delete this schedule? All tasks will be removed and any running processes
|
||||||
|
will be terminated.
|
||||||
|
</p>
|
||||||
|
<div className={'mt-6 flex justify-end'}>
|
||||||
|
<button
|
||||||
|
className={'btn btn-secondary btn-sm mr-4'}
|
||||||
|
onClick={() => setVisible(false)}
|
||||||
|
>
|
||||||
|
Cancel
|
||||||
|
</button>
|
||||||
|
<button
|
||||||
|
className={'btn btn-red btn-sm'}
|
||||||
|
onClick={() => onDelete()}
|
||||||
|
>
|
||||||
|
Yes, delete schedule
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
</Modal>
|
||||||
|
<button className={'btn btn-red btn-secondary btn-sm mr-4'} onClick={() => setVisible(true)}>
|
||||||
|
Delete
|
||||||
|
</button>
|
||||||
|
</>
|
||||||
|
);
|
||||||
|
};
|
|
@ -1,21 +1,20 @@
|
||||||
import React from 'react';
|
import React, { useEffect, useState } from 'react';
|
||||||
import { Schedule } from '@/api/server/schedules/getServerSchedules';
|
import { Schedule } from '@/api/server/schedules/getServerSchedules';
|
||||||
import Modal, { RequiredModalProps } from '@/components/elements/Modal';
|
import Modal, { RequiredModalProps } from '@/components/elements/Modal';
|
||||||
import Field from '@/components/elements/Field';
|
import Field from '@/components/elements/Field';
|
||||||
import { connect } from 'react-redux';
|
import { Form, Formik, FormikHelpers, useFormikContext } from 'formik';
|
||||||
import { Form, FormikProps, withFormik } from 'formik';
|
|
||||||
import { Actions } from 'easy-peasy';
|
|
||||||
import { ApplicationStore } from '@/state';
|
|
||||||
import Switch from '@/components/elements/Switch';
|
import Switch from '@/components/elements/Switch';
|
||||||
import { boolean, object, string } from 'yup';
|
import createOrUpdateSchedule from '@/api/server/schedules/createOrUpdateSchedule';
|
||||||
|
import { ServerContext } from '@/state/server';
|
||||||
|
import { Actions, useStoreActions } from 'easy-peasy';
|
||||||
|
import { ApplicationStore } from '@/state';
|
||||||
|
import { httpErrorToHuman } from '@/api/http';
|
||||||
|
import FlashMessageRender from '@/components/FlashMessageRender';
|
||||||
|
|
||||||
type OwnProps = { schedule: Schedule } & RequiredModalProps;
|
type Props = {
|
||||||
|
schedule?: Schedule;
|
||||||
interface ReduxProps {
|
onScheduleUpdated: (schedule: Schedule) => void;
|
||||||
addError: ApplicationStore['flashes']['addError'];
|
} & RequiredModalProps;
|
||||||
}
|
|
||||||
|
|
||||||
type ComponentProps = OwnProps & ReduxProps;
|
|
||||||
|
|
||||||
interface Values {
|
interface Values {
|
||||||
name: string;
|
name: string;
|
||||||
|
@ -26,9 +25,13 @@ interface Values {
|
||||||
enabled: boolean;
|
enabled: boolean;
|
||||||
}
|
}
|
||||||
|
|
||||||
const EditScheduleModal = ({ values, schedule, ...props }: ComponentProps & FormikProps<Values>) => {
|
const EditScheduleModal = ({ schedule, ...props }: Omit<Props, 'onScheduleUpdated'>) => {
|
||||||
|
const { isSubmitting } = useFormikContext();
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Modal {...props}>
|
<Modal {...props} showSpinnerOverlay={isSubmitting}>
|
||||||
|
<h3 className={'mb-6'}>{schedule ? 'Edit schedule' : 'Create new schedule'}</h3>
|
||||||
|
<FlashMessageRender byKey={'schedule:edit'} className={'mb-6'}/>
|
||||||
<Form>
|
<Form>
|
||||||
<Field
|
<Field
|
||||||
name={'name'}
|
name={'name'}
|
||||||
|
@ -61,8 +64,8 @@ const EditScheduleModal = ({ values, schedule, ...props }: ComponentProps & Form
|
||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
<div className={'mt-6 text-right'}>
|
<div className={'mt-6 text-right'}>
|
||||||
<button className={'btn btn-lg btn-primary'} type={'button'}>
|
<button className={'btn btn-sm btn-primary'} type={'submit'}>
|
||||||
Save
|
{schedule ? 'Save changes' : 'Create schedule'}
|
||||||
</button>
|
</button>
|
||||||
</div>
|
</div>
|
||||||
</Form>
|
</Form>
|
||||||
|
@ -70,29 +73,60 @@ const EditScheduleModal = ({ values, schedule, ...props }: ComponentProps & Form
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
export default connect(
|
export default ({ schedule, onScheduleUpdated, visible, ...props }: Props) => {
|
||||||
null,
|
const uuid = ServerContext.useStoreState(state => state.server.data!.uuid);
|
||||||
// @ts-ignore
|
const { addError, clearFlashes } = useStoreActions((actions: Actions<ApplicationStore>) => actions.flashes);
|
||||||
(dispatch: Actions<ApplicationStore>) => ({
|
const [ modalVisible, setModalVisible ] = useState(visible);
|
||||||
addError: dispatch.flashes.addError,
|
|
||||||
}),
|
useEffect(() => {
|
||||||
)(
|
setModalVisible(visible);
|
||||||
withFormik<ComponentProps, Values>({
|
clearFlashes('schedule:edit');
|
||||||
handleSubmit: (values, { props }) => {
|
}, [visible]);
|
||||||
|
|
||||||
|
const submit = (values: Values, { setSubmitting }: FormikHelpers<Values>) => {
|
||||||
|
clearFlashes('schedule:edit');
|
||||||
|
createOrUpdateSchedule(uuid, {
|
||||||
|
id: schedule?.id,
|
||||||
|
name: values.name,
|
||||||
|
cron: {
|
||||||
|
minute: values.minute,
|
||||||
|
hour: values.hour,
|
||||||
|
dayOfWeek: values.dayOfWeek,
|
||||||
|
dayOfMonth: values.dayOfMonth,
|
||||||
},
|
},
|
||||||
|
isActive: values.enabled,
|
||||||
|
})
|
||||||
|
.then(schedule => {
|
||||||
|
setSubmitting(false);
|
||||||
|
onScheduleUpdated(schedule);
|
||||||
|
setModalVisible(false);
|
||||||
|
})
|
||||||
|
.catch(error => {
|
||||||
|
console.error(error);
|
||||||
|
|
||||||
mapPropsToValues: ({ schedule }) => ({
|
setSubmitting(false);
|
||||||
name: schedule.name,
|
addError({ key: 'schedule:edit', message: httpErrorToHuman(error) });
|
||||||
dayOfWeek: schedule.cron.dayOfWeek,
|
});
|
||||||
dayOfMonth: schedule.cron.dayOfMonth,
|
};
|
||||||
hour: schedule.cron.hour,
|
|
||||||
minute: schedule.cron.minute,
|
|
||||||
enabled: schedule.isActive,
|
|
||||||
}),
|
|
||||||
|
|
||||||
validationSchema: object().shape({
|
return (
|
||||||
name: string().required(),
|
<Formik
|
||||||
enabled: boolean().required(),
|
onSubmit={submit}
|
||||||
}),
|
initialValues={{
|
||||||
})(EditScheduleModal),
|
name: schedule?.name || '',
|
||||||
|
dayOfWeek: schedule?.cron.dayOfWeek || '*',
|
||||||
|
dayOfMonth: schedule?.cron.dayOfMonth || '*',
|
||||||
|
hour: schedule?.cron.hour || '*',
|
||||||
|
minute: schedule?.cron.minute || '*/5',
|
||||||
|
enabled: schedule ? schedule.isActive : true,
|
||||||
|
} as Values}
|
||||||
|
validationSchema={null}
|
||||||
|
>
|
||||||
|
<EditScheduleModal
|
||||||
|
visible={modalVisible}
|
||||||
|
schedule={schedule}
|
||||||
|
{...props}
|
||||||
|
/>
|
||||||
|
</Formik>
|
||||||
);
|
);
|
||||||
|
};
|
||||||
|
|
|
@ -21,7 +21,7 @@ export default ({ scheduleId, onTaskAdded }: Props) => {
|
||||||
}}
|
}}
|
||||||
/>
|
/>
|
||||||
}
|
}
|
||||||
<button className={'btn btn-primary btn-sm ml-4'} onClick={() => setVisible(true)}>
|
<button className={'btn btn-primary btn-sm'} onClick={() => setVisible(true)}>
|
||||||
New Task
|
New Task
|
||||||
</button>
|
</button>
|
||||||
</>
|
</>
|
||||||
|
|
|
@ -2,16 +2,18 @@ import React, { useMemo, useState } from 'react';
|
||||||
import getServerSchedules, { Schedule } from '@/api/server/schedules/getServerSchedules';
|
import getServerSchedules, { Schedule } from '@/api/server/schedules/getServerSchedules';
|
||||||
import { ServerContext } from '@/state/server';
|
import { ServerContext } from '@/state/server';
|
||||||
import Spinner from '@/components/elements/Spinner';
|
import Spinner from '@/components/elements/Spinner';
|
||||||
import { RouteComponentProps, Link } from 'react-router-dom';
|
import { RouteComponentProps } from 'react-router-dom';
|
||||||
import FlashMessageRender from '@/components/FlashMessageRender';
|
import FlashMessageRender from '@/components/FlashMessageRender';
|
||||||
import ScheduleRow from '@/components/server/schedules/ScheduleRow';
|
import ScheduleRow from '@/components/server/schedules/ScheduleRow';
|
||||||
import { httpErrorToHuman } from '@/api/http';
|
import { httpErrorToHuman } from '@/api/http';
|
||||||
import { Actions, useStoreActions } from 'easy-peasy';
|
import { Actions, useStoreActions } from 'easy-peasy';
|
||||||
import { ApplicationStore } from '@/state';
|
import { ApplicationStore } from '@/state';
|
||||||
|
import EditScheduleModal from '@/components/server/schedules/EditScheduleModal';
|
||||||
|
|
||||||
export default ({ match, history }: RouteComponentProps) => {
|
export default ({ match, history }: RouteComponentProps) => {
|
||||||
const { uuid } = ServerContext.useStoreState(state => state.server.data!);
|
const { uuid } = ServerContext.useStoreState(state => state.server.data!);
|
||||||
const [ schedules, setSchedules ] = useState<Schedule[] | null>(null);
|
const [ schedules, setSchedules ] = useState<Schedule[] | null>(null);
|
||||||
|
const [ visible, setVisible ] = useState(false);
|
||||||
const { clearFlashes, addError } = useStoreActions((actions: Actions<ApplicationStore>) => actions.flashes);
|
const { clearFlashes, addError } = useStoreActions((actions: Actions<ApplicationStore>) => actions.flashes);
|
||||||
|
|
||||||
useMemo(() => {
|
useMemo(() => {
|
||||||
|
@ -30,11 +32,19 @@ export default ({ match, history }: RouteComponentProps) => {
|
||||||
{!schedules ?
|
{!schedules ?
|
||||||
<Spinner size={'large'} centered={true}/>
|
<Spinner size={'large'} centered={true}/>
|
||||||
:
|
:
|
||||||
|
<>
|
||||||
|
{
|
||||||
|
schedules.length === 0 ?
|
||||||
|
<p className={'text-sm text-neutral-400'}>
|
||||||
|
There are no schedules configured for this server. Click the button below to get
|
||||||
|
started.
|
||||||
|
</p>
|
||||||
|
:
|
||||||
schedules.map(schedule => (
|
schedules.map(schedule => (
|
||||||
<a
|
<a
|
||||||
key={schedule.id}
|
key={schedule.id}
|
||||||
href={`${match.url}/${schedule.id}`}
|
href={`${match.url}/${schedule.id}`}
|
||||||
className={'grey-row-box cursor-pointer'}
|
className={'grey-row-box cursor-pointer mb-2'}
|
||||||
onClick={e => {
|
onClick={e => {
|
||||||
e.preventDefault();
|
e.preventDefault();
|
||||||
history.push(`${match.url}/${schedule.id}`, { schedule });
|
history.push(`${match.url}/${schedule.id}`, { schedule });
|
||||||
|
@ -44,6 +54,23 @@ export default ({ match, history }: RouteComponentProps) => {
|
||||||
</a>
|
</a>
|
||||||
))
|
))
|
||||||
}
|
}
|
||||||
|
<div className={'mt-8 flex justify-end'}>
|
||||||
|
{visible && <EditScheduleModal
|
||||||
|
appear={true}
|
||||||
|
visible={true}
|
||||||
|
onScheduleUpdated={schedule => setSchedules(s => [...(s || []), schedule])}
|
||||||
|
onDismissed={() => setVisible(false)}
|
||||||
|
/>}
|
||||||
|
<button
|
||||||
|
type={'button'}
|
||||||
|
className={'btn btn-lg btn-primary'}
|
||||||
|
onClick={() => setVisible(true)}
|
||||||
|
>
|
||||||
|
Create schedule
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
</>
|
||||||
|
}
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
|
@ -12,6 +12,7 @@ import ScheduleRow from '@/components/server/schedules/ScheduleRow';
|
||||||
import ScheduleTaskRow from '@/components/server/schedules/ScheduleTaskRow';
|
import ScheduleTaskRow from '@/components/server/schedules/ScheduleTaskRow';
|
||||||
import EditScheduleModal from '@/components/server/schedules/EditScheduleModal';
|
import EditScheduleModal from '@/components/server/schedules/EditScheduleModal';
|
||||||
import NewTaskButton from '@/components/server/schedules/NewTaskButton';
|
import NewTaskButton from '@/components/server/schedules/NewTaskButton';
|
||||||
|
import DeleteScheduleButton from '@/components/server/schedules/DeleteScheduleButton';
|
||||||
|
|
||||||
interface Params {
|
interface Params {
|
||||||
id: string;
|
id: string;
|
||||||
|
@ -21,8 +22,8 @@ interface State {
|
||||||
schedule?: Schedule;
|
schedule?: Schedule;
|
||||||
}
|
}
|
||||||
|
|
||||||
export default ({ match, location: { state } }: RouteComponentProps<Params, {}, State>) => {
|
export default ({ match, history, location: { state } }: RouteComponentProps<Params, {}, State>) => {
|
||||||
const uuid = ServerContext.useStoreState(state => state.server.data!.uuid);
|
const { id, uuid } = ServerContext.useStoreState(state => state.server.data!);
|
||||||
const [ isLoading, setIsLoading ] = useState(true);
|
const [ isLoading, setIsLoading ] = useState(true);
|
||||||
const [ showEditModal, setShowEditModal ] = useState(false);
|
const [ showEditModal, setShowEditModal ] = useState(false);
|
||||||
const [ schedule, setSchedule ] = useState<Schedule | undefined>(state?.schedule);
|
const [ schedule, setSchedule ] = useState<Schedule | undefined>(state?.schedule);
|
||||||
|
@ -57,21 +58,13 @@ export default ({ match, location: { state } }: RouteComponentProps<Params, {},
|
||||||
<EditScheduleModal
|
<EditScheduleModal
|
||||||
visible={showEditModal}
|
visible={showEditModal}
|
||||||
schedule={schedule}
|
schedule={schedule}
|
||||||
|
onScheduleUpdated={schedule => setSchedule(schedule)}
|
||||||
onDismissed={() => setShowEditModal(false)}
|
onDismissed={() => setShowEditModal(false)}
|
||||||
/>
|
/>
|
||||||
<div className={'flex items-center my-4'}>
|
<div className={'flex items-center mt-8 mb-4'}>
|
||||||
<div className={'flex-1'}>
|
<div className={'flex-1'}>
|
||||||
<h2>Schedule Tasks</h2>
|
<h2>Configured Tasks</h2>
|
||||||
</div>
|
</div>
|
||||||
<button className={'btn btn-secondary btn-sm'} onClick={() => setShowEditModal(true)}>
|
|
||||||
Edit
|
|
||||||
</button>
|
|
||||||
<NewTaskButton
|
|
||||||
scheduleId={schedule.id}
|
|
||||||
onTaskAdded={task => setSchedule(s => ({
|
|
||||||
...s!, tasks: [ ...s!.tasks, task ],
|
|
||||||
}))}
|
|
||||||
/>
|
|
||||||
</div>
|
</div>
|
||||||
{schedule?.tasks.length > 0 ?
|
{schedule?.tasks.length > 0 ?
|
||||||
<>
|
<>
|
||||||
|
@ -101,9 +94,24 @@ export default ({ match, location: { state } }: RouteComponentProps<Params, {},
|
||||||
:
|
:
|
||||||
<p className={'text-sm text-neutral-400'}>
|
<p className={'text-sm text-neutral-400'}>
|
||||||
There are no tasks configured for this schedule. Consider adding a new one using the
|
There are no tasks configured for this schedule. Consider adding a new one using the
|
||||||
button above.
|
button below.
|
||||||
</p>
|
</p>
|
||||||
}
|
}
|
||||||
|
<div className={'mt-8 flex justify-end'}>
|
||||||
|
<DeleteScheduleButton
|
||||||
|
scheduleId={schedule.id}
|
||||||
|
onDeleted={() => history.push(`/server/${id}/schedules`)}
|
||||||
|
/>
|
||||||
|
<button className={'btn btn-primary btn-sm mr-4'} onClick={() => setShowEditModal(true)}>
|
||||||
|
Edit
|
||||||
|
</button>
|
||||||
|
<NewTaskButton
|
||||||
|
scheduleId={schedule.id}
|
||||||
|
onTaskAdded={task => setSchedule(s => ({
|
||||||
|
...s!, tasks: [ ...s!.tasks, task ],
|
||||||
|
}))}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
</>
|
</>
|
||||||
}
|
}
|
||||||
</div>
|
</div>
|
||||||
|
|
|
@ -61,7 +61,11 @@ Route::group(['prefix' => '/servers/{server}', 'middleware' => [AuthenticateServ
|
||||||
|
|
||||||
Route::group(['prefix' => '/schedules'], function () {
|
Route::group(['prefix' => '/schedules'], function () {
|
||||||
Route::get('/', 'Servers\ScheduleController@index');
|
Route::get('/', 'Servers\ScheduleController@index');
|
||||||
|
Route::post('/', 'Servers\ScheduleController@store');
|
||||||
Route::get('/{schedule}', 'Servers\ScheduleController@view');
|
Route::get('/{schedule}', 'Servers\ScheduleController@view');
|
||||||
|
Route::post('/{schedule}', 'Servers\ScheduleController@update');
|
||||||
|
Route::delete('/{schedule}', 'Servers\ScheduleController@delete');
|
||||||
|
|
||||||
Route::post('/{schedule}/tasks', 'Servers\ScheduleTaskController@store');
|
Route::post('/{schedule}/tasks', 'Servers\ScheduleTaskController@store');
|
||||||
Route::post('/{schedule}/tasks/{task}', 'Servers\ScheduleTaskController@update');
|
Route::post('/{schedule}/tasks/{task}', 'Servers\ScheduleTaskController@update');
|
||||||
Route::delete('/{schedule}/tasks/{task}', 'Servers\ScheduleTaskController@delete');
|
Route::delete('/{schedule}/tasks/{task}', 'Servers\ScheduleTaskController@delete');
|
||||||
|
|
Loading…
Add table
Reference in a new issue