Update allocations to support ids; protect endpoints; support notes
This commit is contained in:
parent
9c3b9a0fae
commit
2278927fb6
19 changed files with 217 additions and 62 deletions
|
@ -3,16 +3,19 @@
|
||||||
namespace Pterodactyl\Http\Controllers\Api\Client\Servers;
|
namespace Pterodactyl\Http\Controllers\Api\Client\Servers;
|
||||||
|
|
||||||
use Pterodactyl\Models\Server;
|
use Pterodactyl\Models\Server;
|
||||||
|
use Illuminate\Http\JsonResponse;
|
||||||
|
use Pterodactyl\Models\Allocation;
|
||||||
use Pterodactyl\Exceptions\DisplayException;
|
use Pterodactyl\Exceptions\DisplayException;
|
||||||
use Pterodactyl\Repositories\Eloquent\ServerRepository;
|
use Pterodactyl\Repositories\Eloquent\ServerRepository;
|
||||||
use Illuminate\Database\Eloquent\ModelNotFoundException;
|
|
||||||
use Pterodactyl\Repositories\Eloquent\AllocationRepository;
|
use Pterodactyl\Repositories\Eloquent\AllocationRepository;
|
||||||
use Pterodactyl\Transformers\Api\Client\AllocationTransformer;
|
use Pterodactyl\Transformers\Api\Client\AllocationTransformer;
|
||||||
use Pterodactyl\Http\Controllers\Api\Client\ClientApiController;
|
use Pterodactyl\Http\Controllers\Api\Client\ClientApiController;
|
||||||
use Pterodactyl\Http\Requests\Api\Client\Servers\Network\GetNetworkRequest;
|
use Pterodactyl\Http\Requests\Api\Client\Servers\Network\GetNetworkRequest;
|
||||||
|
use Pterodactyl\Http\Requests\Api\Client\Servers\Network\DeleteAllocationRequest;
|
||||||
|
use Pterodactyl\Http\Requests\Api\Client\Servers\Network\UpdateAllocationRequest;
|
||||||
use Pterodactyl\Http\Requests\Api\Client\Servers\Network\SetPrimaryAllocationRequest;
|
use Pterodactyl\Http\Requests\Api\Client\Servers\Network\SetPrimaryAllocationRequest;
|
||||||
|
|
||||||
class NetworkController extends ClientApiController
|
class NetworkAllocationController extends ClientApiController
|
||||||
{
|
{
|
||||||
/**
|
/**
|
||||||
* @var \Pterodactyl\Repositories\Eloquent\AllocationRepository
|
* @var \Pterodactyl\Repositories\Eloquent\AllocationRepository
|
||||||
|
@ -58,33 +61,67 @@ class NetworkController extends ClientApiController
|
||||||
/**
|
/**
|
||||||
* Set the primary allocation for a server.
|
* Set the primary allocation for a server.
|
||||||
*
|
*
|
||||||
* @param \Pterodactyl\Http\Requests\Api\Client\Servers\Network\SetPrimaryAllocationRequest $request
|
* @param \Pterodactyl\Http\Requests\Api\Client\Servers\Network\UpdateAllocationRequest $request
|
||||||
* @param \Pterodactyl\Models\Server $server
|
* @param \Pterodactyl\Models\Server $server
|
||||||
|
* @param \Pterodactyl\Models\Allocation $allocation
|
||||||
* @return array
|
* @return array
|
||||||
*
|
*
|
||||||
* @throws \Pterodactyl\Exceptions\DisplayException
|
|
||||||
* @throws \Pterodactyl\Exceptions\Repository\RecordNotFoundException
|
|
||||||
* @throws \Pterodactyl\Exceptions\Model\DataValidationException
|
* @throws \Pterodactyl\Exceptions\Model\DataValidationException
|
||||||
|
* @throws \Pterodactyl\Exceptions\Repository\RecordNotFoundException
|
||||||
*/
|
*/
|
||||||
public function storePrimary(SetPrimaryAllocationRequest $request, Server $server): array
|
public function update(UpdateAllocationRequest $request, Server $server, Allocation $allocation): array
|
||||||
{
|
{
|
||||||
try {
|
$allocation = $this->repository->update($allocation->id, [
|
||||||
/** @var \Pterodactyl\Models\Allocation $allocation */
|
'notes' => $request->input('notes'),
|
||||||
$allocation = $this->repository->findFirstWhere([
|
|
||||||
'server_id' => $server->id,
|
|
||||||
'ip' => $request->input('ip'),
|
|
||||||
'port' => $request->input('port'),
|
|
||||||
]);
|
]);
|
||||||
} catch (ModelNotFoundException $exception) {
|
|
||||||
throw new DisplayException(
|
return $this->fractal->item($allocation)
|
||||||
'The IP and port you selected are not available for this server.'
|
->transformWith($this->getTransformer(AllocationTransformer::class))
|
||||||
);
|
->toArray();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Set the primary allocation for a server.
|
||||||
|
*
|
||||||
|
* @param \Pterodactyl\Http\Requests\Api\Client\Servers\Network\SetPrimaryAllocationRequest $request
|
||||||
|
* @param \Pterodactyl\Models\Server $server
|
||||||
|
* @param \Pterodactyl\Models\Allocation $allocation
|
||||||
|
* @return array
|
||||||
|
*
|
||||||
|
* @throws \Pterodactyl\Exceptions\Model\DataValidationException
|
||||||
|
* @throws \Pterodactyl\Exceptions\Repository\RecordNotFoundException
|
||||||
|
*/
|
||||||
|
public function setPrimary(SetPrimaryAllocationRequest $request, Server $server, Allocation $allocation): array
|
||||||
|
{
|
||||||
$this->serverRepository->update($server->id, ['allocation_id' => $allocation->id]);
|
$this->serverRepository->update($server->id, ['allocation_id' => $allocation->id]);
|
||||||
|
|
||||||
return $this->fractal->item($allocation)
|
return $this->fractal->item($allocation)
|
||||||
->transformWith($this->getTransformer(AllocationTransformer::class))
|
->transformWith($this->getTransformer(AllocationTransformer::class))
|
||||||
->toArray();
|
->toArray();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Delete an allocation from a server.
|
||||||
|
*
|
||||||
|
* @param \Pterodactyl\Http\Requests\Api\Client\Servers\Network\DeleteAllocationRequest $request
|
||||||
|
* @param \Pterodactyl\Models\Server $server
|
||||||
|
* @param \Pterodactyl\Models\Allocation $allocation
|
||||||
|
* @return \Illuminate\Http\JsonResponse
|
||||||
|
*
|
||||||
|
* @throws \Pterodactyl\Exceptions\DisplayException
|
||||||
|
* @throws \Pterodactyl\Exceptions\Model\DataValidationException
|
||||||
|
* @throws \Pterodactyl\Exceptions\Repository\RecordNotFoundException
|
||||||
|
*/
|
||||||
|
public function delete(DeleteAllocationRequest $request, Server $server, Allocation $allocation)
|
||||||
|
{
|
||||||
|
if ($allocation->id === $server->allocation_id) {
|
||||||
|
throw new DisplayException(
|
||||||
|
'Cannot delete the primary allocation for a server.'
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
$this->repository->update($allocation->id, ['server_id' => null, 'notes' => null]);
|
||||||
|
|
||||||
|
return new JsonResponse([], JsonResponse::HTTP_NO_CONTENT);
|
||||||
|
}
|
||||||
}
|
}
|
|
@ -0,0 +1,33 @@
|
||||||
|
<?php
|
||||||
|
|
||||||
|
namespace Pterodactyl\Http\Middleware\Api\Client\Server;
|
||||||
|
|
||||||
|
use Closure;
|
||||||
|
use Illuminate\Http\Request;
|
||||||
|
use Symfony\Component\HttpKernel\Exception\NotFoundHttpException;
|
||||||
|
|
||||||
|
class AllocationBelongsToServer
|
||||||
|
{
|
||||||
|
/**
|
||||||
|
* Ensure that the allocation found in the URL belongs to the server being queried.
|
||||||
|
*
|
||||||
|
* @param \Illuminate\Http\Request $request
|
||||||
|
* @param \Closure $next
|
||||||
|
* @return mixed
|
||||||
|
*
|
||||||
|
* @throws \Symfony\Component\HttpKernel\Exception\NotFoundHttpException
|
||||||
|
*/
|
||||||
|
public function handle(Request $request, Closure $next)
|
||||||
|
{
|
||||||
|
/** @var \Pterodactyl\Models\Server $server */
|
||||||
|
$server = $request->route()->parameter('server');
|
||||||
|
/** @var \Pterodactyl\Models\Allocation|null $allocation */
|
||||||
|
$allocation = $request->route()->parameter('allocation');
|
||||||
|
|
||||||
|
if ($allocation && $allocation->server_id !== $server->id) {
|
||||||
|
throw new NotFoundHttpException;
|
||||||
|
}
|
||||||
|
|
||||||
|
return $next($request);
|
||||||
|
}
|
||||||
|
}
|
|
@ -4,12 +4,12 @@ namespace Pterodactyl\Http\Middleware\Api\Client;
|
||||||
|
|
||||||
use Closure;
|
use Closure;
|
||||||
use Pterodactyl\Models\Backup;
|
use Pterodactyl\Models\Backup;
|
||||||
|
use Pterodactyl\Models\Database;
|
||||||
use Illuminate\Container\Container;
|
use Illuminate\Container\Container;
|
||||||
use Pterodactyl\Contracts\Extensions\HashidsInterface;
|
use Pterodactyl\Contracts\Extensions\HashidsInterface;
|
||||||
use Pterodactyl\Http\Middleware\Api\ApiSubstituteBindings;
|
use Pterodactyl\Http\Middleware\Api\ApiSubstituteBindings;
|
||||||
use Pterodactyl\Exceptions\Repository\RecordNotFoundException;
|
use Pterodactyl\Exceptions\Repository\RecordNotFoundException;
|
||||||
use Pterodactyl\Contracts\Repository\ServerRepositoryInterface;
|
use Pterodactyl\Contracts\Repository\ServerRepositoryInterface;
|
||||||
use Pterodactyl\Contracts\Repository\DatabaseRepositoryInterface;
|
|
||||||
|
|
||||||
class SubstituteClientApiBindings extends ApiSubstituteBindings
|
class SubstituteClientApiBindings extends ApiSubstituteBindings
|
||||||
{
|
{
|
||||||
|
@ -43,17 +43,9 @@ class SubstituteClientApiBindings extends ApiSubstituteBindings
|
||||||
});
|
});
|
||||||
|
|
||||||
$this->router->bind('database', function ($value) use ($request) {
|
$this->router->bind('database', function ($value) use ($request) {
|
||||||
try {
|
|
||||||
$id = Container::getInstance()->make(HashidsInterface::class)->decodeFirst($value);
|
$id = Container::getInstance()->make(HashidsInterface::class)->decodeFirst($value);
|
||||||
|
|
||||||
return Container::getInstance()->make(DatabaseRepositoryInterface::class)->findFirstWhere([
|
return Database::query()->where('id', $id)->firstOrFail();
|
||||||
['id', '=', $id],
|
|
||||||
]);
|
|
||||||
} catch (RecordNotFoundException $exception) {
|
|
||||||
$request->attributes->set('is_missing_model', true);
|
|
||||||
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
});
|
});
|
||||||
|
|
||||||
$this->router->model('backup', Backup::class, function ($value) {
|
$this->router->model('backup', Backup::class, function ($value) {
|
||||||
|
|
|
@ -0,0 +1,17 @@
|
||||||
|
<?php
|
||||||
|
|
||||||
|
namespace Pterodactyl\Http\Requests\Api\Client\Servers\Network;
|
||||||
|
|
||||||
|
use Pterodactyl\Models\Permission;
|
||||||
|
use Pterodactyl\Http\Requests\Api\Client\ClientApiRequest;
|
||||||
|
|
||||||
|
class DeleteAllocationRequest extends ClientApiRequest
|
||||||
|
{
|
||||||
|
/**
|
||||||
|
* @return string
|
||||||
|
*/
|
||||||
|
public function permission(): string
|
||||||
|
{
|
||||||
|
return Permission::ACTION_ALLOCATION_DELETE;
|
||||||
|
}
|
||||||
|
}
|
|
@ -2,27 +2,13 @@
|
||||||
|
|
||||||
namespace Pterodactyl\Http\Requests\Api\Client\Servers\Network;
|
namespace Pterodactyl\Http\Requests\Api\Client\Servers\Network;
|
||||||
|
|
||||||
use Pterodactyl\Models\Permission;
|
class SetPrimaryAllocationRequest extends UpdateAllocationRequest
|
||||||
use Pterodactyl\Http\Requests\Api\Client\ClientApiRequest;
|
|
||||||
|
|
||||||
class SetPrimaryAllocationRequest extends ClientApiRequest
|
|
||||||
{
|
{
|
||||||
/**
|
|
||||||
* @return string
|
|
||||||
*/
|
|
||||||
public function permission(): string
|
|
||||||
{
|
|
||||||
return Permission::ACTION_ALLOCIATION_UPDATE;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @return array
|
* @return array
|
||||||
*/
|
*/
|
||||||
public function rules(): array
|
public function rules(): array
|
||||||
{
|
{
|
||||||
return [
|
return [];
|
||||||
'ip' => 'required|string',
|
|
||||||
'port' => 'required|numeric|min:1024|max:65535',
|
|
||||||
];
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -0,0 +1,28 @@
|
||||||
|
<?php
|
||||||
|
|
||||||
|
namespace Pterodactyl\Http\Requests\Api\Client\Servers\Network;
|
||||||
|
|
||||||
|
use Pterodactyl\Models\Allocation;
|
||||||
|
use Pterodactyl\Models\Permission;
|
||||||
|
use Pterodactyl\Http\Requests\Api\Client\ClientApiRequest;
|
||||||
|
|
||||||
|
class UpdateAllocationRequest extends ClientApiRequest
|
||||||
|
{
|
||||||
|
/**
|
||||||
|
* @return string
|
||||||
|
*/
|
||||||
|
public function permission(): string
|
||||||
|
{
|
||||||
|
return Permission::ACTION_ALLOCATION_UPDATE;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return array
|
||||||
|
*/
|
||||||
|
public function rules(): array
|
||||||
|
{
|
||||||
|
return [
|
||||||
|
'notes' => Allocation::$validationRules['notes'],
|
||||||
|
];
|
||||||
|
}
|
||||||
|
}
|
|
@ -9,6 +9,7 @@ namespace Pterodactyl\Models;
|
||||||
* @property string|null $ip_alias
|
* @property string|null $ip_alias
|
||||||
* @property int $port
|
* @property int $port
|
||||||
* @property int|null $server_id
|
* @property int|null $server_id
|
||||||
|
* @property string|null $notes
|
||||||
* @property \Carbon\Carbon|null $created_at
|
* @property \Carbon\Carbon|null $created_at
|
||||||
* @property \Carbon\Carbon|null $updated_at
|
* @property \Carbon\Carbon|null $updated_at
|
||||||
*
|
*
|
||||||
|
@ -60,6 +61,7 @@ class Allocation extends Model
|
||||||
'port' => 'required|numeric|between:1024,65553',
|
'port' => 'required|numeric|between:1024,65553',
|
||||||
'ip_alias' => 'nullable|string',
|
'ip_alias' => 'nullable|string',
|
||||||
'server_id' => 'nullable|exists:servers,id',
|
'server_id' => 'nullable|exists:servers,id',
|
||||||
|
'notes' => 'nullable|string|max:256',
|
||||||
];
|
];
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|
|
@ -44,7 +44,9 @@ class Permission extends Model
|
||||||
const ACTION_BACKUP_DOWNLOAD = 'backup.download';
|
const ACTION_BACKUP_DOWNLOAD = 'backup.download';
|
||||||
|
|
||||||
const ACTION_ALLOCATION_READ = 'allocation.read';
|
const ACTION_ALLOCATION_READ = 'allocation.read';
|
||||||
const ACTION_ALLOCIATION_UPDATE = 'allocation.update';
|
const ACTION_ALLOCATION_CREATE = 'allocation.create';
|
||||||
|
const ACTION_ALLOCATION_UPDATE = 'allocation.update';
|
||||||
|
const ACTION_ALLOCATION_DELETE = 'allocation.delete';
|
||||||
|
|
||||||
const ACTION_FILE_READ = 'file.read';
|
const ACTION_FILE_READ = 'file.read';
|
||||||
const ACTION_FILE_CREATE = 'file.create';
|
const ACTION_FILE_CREATE = 'file.create';
|
||||||
|
@ -157,7 +159,9 @@ class Permission extends Model
|
||||||
'description' => 'Permissions that control a user\'s ability to modify the port allocations for this server.',
|
'description' => 'Permissions that control a user\'s ability to modify the port allocations for this server.',
|
||||||
'keys' => [
|
'keys' => [
|
||||||
'read' => 'Allows a user to view the allocations assigned to this server.',
|
'read' => 'Allows a user to view the allocations assigned to this server.',
|
||||||
'update' => 'Allows a user to modify the allocations assigned to this server.',
|
'create' => 'Allows a user to assign additional allocations to the server.',
|
||||||
|
'update' => 'Allows a user to change the primary server allocation and attach notes to each allocation.',
|
||||||
|
'delete' => 'Allows a user to delete an allocation from the server.',
|
||||||
],
|
],
|
||||||
],
|
],
|
||||||
|
|
||||||
|
|
|
@ -39,6 +39,7 @@ class AllocationTransformer extends BaseTransformer
|
||||||
'ip' => $allocation->ip,
|
'ip' => $allocation->ip,
|
||||||
'alias' => $allocation->ip_alias,
|
'alias' => $allocation->ip_alias,
|
||||||
'port' => $allocation->port,
|
'port' => $allocation->port,
|
||||||
|
'notes' => $allocation->notes,
|
||||||
'assigned' => ! is_null($allocation->server_id),
|
'assigned' => ! is_null($allocation->server_id),
|
||||||
];
|
];
|
||||||
}
|
}
|
||||||
|
|
|
@ -25,9 +25,11 @@ class AllocationTransformer extends BaseClientTransformer
|
||||||
public function transform(Allocation $model)
|
public function transform(Allocation $model)
|
||||||
{
|
{
|
||||||
return [
|
return [
|
||||||
|
'id' => $model->id,
|
||||||
'ip' => $model->ip,
|
'ip' => $model->ip,
|
||||||
'ip_alias' => $model->ip_alias,
|
'ip_alias' => $model->ip_alias,
|
||||||
'port' => $model->port,
|
'port' => $model->port,
|
||||||
|
'notes' => $model->notes,
|
||||||
'is_default' => $model->server->allocation_id === $model->id,
|
'is_default' => $model->server->allocation_id === $model->id,
|
||||||
];
|
];
|
||||||
}
|
}
|
||||||
|
|
|
@ -0,0 +1,32 @@
|
||||||
|
<?php
|
||||||
|
|
||||||
|
use Illuminate\Database\Migrations\Migration;
|
||||||
|
use Illuminate\Database\Schema\Blueprint;
|
||||||
|
use Illuminate\Support\Facades\Schema;
|
||||||
|
|
||||||
|
class AddNotesColumnForAllocations extends Migration
|
||||||
|
{
|
||||||
|
/**
|
||||||
|
* Run the migrations.
|
||||||
|
*
|
||||||
|
* @return void
|
||||||
|
*/
|
||||||
|
public function up()
|
||||||
|
{
|
||||||
|
Schema::table('allocations', function (Blueprint $table) {
|
||||||
|
$table->string('notes')->nullable()->after('server_id');
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Reverse the migrations.
|
||||||
|
*
|
||||||
|
* @return void
|
||||||
|
*/
|
||||||
|
public function down()
|
||||||
|
{
|
||||||
|
Schema::table('allocations', function (Blueprint $table) {
|
||||||
|
$table->dropColumn('notes');
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
|
@ -2,9 +2,11 @@ import http, { FractalResponseData, FractalResponseList } from '@/api/http';
|
||||||
import { rawDataToServerAllocation } from '@/api/transformers';
|
import { rawDataToServerAllocation } from '@/api/transformers';
|
||||||
|
|
||||||
export interface Allocation {
|
export interface Allocation {
|
||||||
|
id: number;
|
||||||
ip: string;
|
ip: string;
|
||||||
alias: string | null;
|
alias: string | null;
|
||||||
port: number;
|
port: number;
|
||||||
|
notes: string | null;
|
||||||
isDefault: boolean;
|
isDefault: boolean;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -0,0 +1,4 @@
|
||||||
|
import { Allocation } from '@/api/server/getServer';
|
||||||
|
import http from '@/api/http';
|
||||||
|
|
||||||
|
export default async (uuid: string, id: number): Promise<Allocation> => await http.delete(`/api/client/servers/${uuid}/network/allocations/${id}`);
|
|
@ -3,7 +3,7 @@ import { rawDataToServerAllocation } from '@/api/transformers';
|
||||||
import { Allocation } from '@/api/server/getServer';
|
import { Allocation } from '@/api/server/getServer';
|
||||||
|
|
||||||
export default async (uuid: string): Promise<Allocation[]> => {
|
export default async (uuid: string): Promise<Allocation[]> => {
|
||||||
const { data } = await http.get(`/api/client/servers/${uuid}/network`);
|
const { data } = await http.get(`/api/client/servers/${uuid}/network/allocations`);
|
||||||
|
|
||||||
return (data.data || []).map(rawDataToServerAllocation);
|
return (data.data || []).map(rawDataToServerAllocation);
|
||||||
};
|
};
|
||||||
|
|
|
@ -2,8 +2,8 @@ import { Allocation } from '@/api/server/getServer';
|
||||||
import http from '@/api/http';
|
import http from '@/api/http';
|
||||||
import { rawDataToServerAllocation } from '@/api/transformers';
|
import { rawDataToServerAllocation } from '@/api/transformers';
|
||||||
|
|
||||||
export default async (uuid: string, ip: string, port: number): Promise<Allocation> => {
|
export default async (uuid: string, id: number): Promise<Allocation> => {
|
||||||
const { data } = await http.put(`/api/client/servers/${uuid}/network/primary`, { ip, port });
|
const { data } = await http.post(`/api/client/servers/${uuid}/network/allocations/${id}/primary`);
|
||||||
|
|
||||||
return rawDataToServerAllocation(data);
|
return rawDataToServerAllocation(data);
|
||||||
};
|
};
|
||||||
|
|
|
@ -0,0 +1,9 @@
|
||||||
|
import { Allocation } from '@/api/server/getServer';
|
||||||
|
import http from '@/api/http';
|
||||||
|
import { rawDataToServerAllocation } from '@/api/transformers';
|
||||||
|
|
||||||
|
export default async (uuid: string, id: number, notes: string | null): Promise<Allocation> => {
|
||||||
|
const { data } = await http.post(`/api/client/servers/${uuid}/network/allocations/${id}`, { notes });
|
||||||
|
|
||||||
|
return rawDataToServerAllocation(data);
|
||||||
|
};
|
|
@ -2,8 +2,10 @@ import { Allocation } from '@/api/server/getServer';
|
||||||
import { FractalResponseData } from '@/api/http';
|
import { FractalResponseData } from '@/api/http';
|
||||||
|
|
||||||
export const rawDataToServerAllocation = (data: FractalResponseData): Allocation => ({
|
export const rawDataToServerAllocation = (data: FractalResponseData): Allocation => ({
|
||||||
|
id: data.attributes.id,
|
||||||
ip: data.attributes.ip,
|
ip: data.attributes.ip,
|
||||||
alias: data.attributes.ip_alias,
|
alias: data.attributes.ip_alias,
|
||||||
port: data.attributes.port,
|
port: data.attributes.port,
|
||||||
|
notes: data.attributes.notes,
|
||||||
isDefault: data.attributes.is_default,
|
isDefault: data.attributes.is_default,
|
||||||
});
|
});
|
||||||
|
|
|
@ -23,16 +23,17 @@ const NetworkContainer = () => {
|
||||||
const { clearFlashes, clearAndAddHttpError } = useFlash();
|
const { clearFlashes, clearAndAddHttpError } = useFlash();
|
||||||
const { data, error, mutate } = useSWR<Allocation[]>(server.uuid, key => getServerAllocations(key), { initialData: server.allocations });
|
const { data, error, mutate } = useSWR<Allocation[]>(server.uuid, key => getServerAllocations(key), { initialData: server.allocations });
|
||||||
|
|
||||||
const setPrimaryAllocation = (ip: string, port: number) => {
|
const setPrimaryAllocation = (id: number) => {
|
||||||
clearFlashes('server:network');
|
clearFlashes('server:network');
|
||||||
|
|
||||||
mutate(data?.map(a => (a.ip === ip && a.port === port) ? { ...a, isDefault: true } : {
|
const initial = data;
|
||||||
...a,
|
mutate(data?.map(a => a.id === id ? { ...a, isDefault: true } : { ...a, isDefault: false }), false);
|
||||||
isDefault: false,
|
|
||||||
}), false);
|
|
||||||
|
|
||||||
setPrimaryServerAllocation(server.uuid, ip, port)
|
setPrimaryServerAllocation(server.uuid, id)
|
||||||
.catch(error => clearAndAddHttpError({ key: 'server:network', error }));
|
.catch(error => {
|
||||||
|
clearAndAddHttpError({ key: 'server:network', error });
|
||||||
|
mutate(initial, false);
|
||||||
|
});
|
||||||
};
|
};
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
|
@ -46,7 +47,7 @@ const NetworkContainer = () => {
|
||||||
{!data ?
|
{!data ?
|
||||||
<Spinner size={'large'} centered/>
|
<Spinner size={'large'} centered/>
|
||||||
:
|
:
|
||||||
data.map(({ ip, port, alias, isDefault }, index) => (
|
data.map(({ id, ip, port, alias, isDefault }, index) => (
|
||||||
<GreyRowBox key={`${ip}:${port}`} css={index > 0 ? tw`mt-2` : undefined}>
|
<GreyRowBox key={`${ip}:${port}`} css={index > 0 ? tw`mt-2` : undefined}>
|
||||||
<div css={tw`pl-4 pr-6 text-neutral-400`}>
|
<div css={tw`pl-4 pr-6 text-neutral-400`}>
|
||||||
<FontAwesomeIcon icon={faNetworkWired}/>
|
<FontAwesomeIcon icon={faNetworkWired}/>
|
||||||
|
@ -70,7 +71,7 @@ const NetworkContainer = () => {
|
||||||
isSecondary
|
isSecondary
|
||||||
size={'xsmall'}
|
size={'xsmall'}
|
||||||
color={'primary'}
|
color={'primary'}
|
||||||
onClick={() => setPrimaryAllocation(ip, port)}
|
onClick={() => setPrimaryAllocation(id)}
|
||||||
>
|
>
|
||||||
Make Primary
|
Make Primary
|
||||||
</Button>
|
</Button>
|
||||||
|
|
|
@ -2,6 +2,7 @@
|
||||||
|
|
||||||
use Illuminate\Support\Facades\Route;
|
use Illuminate\Support\Facades\Route;
|
||||||
use Pterodactyl\Http\Middleware\Api\Client\Server\AuthenticateServerAccess;
|
use Pterodactyl\Http\Middleware\Api\Client\Server\AuthenticateServerAccess;
|
||||||
|
use Pterodactyl\Http\Middleware\Api\Client\Server\AllocationBelongsToServer;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|--------------------------------------------------------------------------
|
|--------------------------------------------------------------------------
|
||||||
|
@ -74,9 +75,11 @@ Route::group(['prefix' => '/servers/{server}', 'middleware' => [AuthenticateServ
|
||||||
Route::delete('/{schedule}/tasks/{task}', 'Servers\ScheduleTaskController@delete');
|
Route::delete('/{schedule}/tasks/{task}', 'Servers\ScheduleTaskController@delete');
|
||||||
});
|
});
|
||||||
|
|
||||||
Route::group(['prefix' => '/network'], function () {
|
Route::group(['prefix' => '/network', 'middleware' => [AllocationBelongsToServer::class]], function () {
|
||||||
Route::get('/', 'Servers\NetworkController@index');
|
Route::get('/allocations', 'Servers\NetworkAllocationController@index');
|
||||||
Route::put('/primary', 'Servers\NetworkController@storePrimary');
|
Route::post('/allocations/{allocation}', 'Servers\NetworkAllocationController@update');
|
||||||
|
Route::post('/allocations/{allocation}/primary', 'Servers\NetworkAllocationController@setPrimary');
|
||||||
|
Route::delete('/allocations/{allocation}', 'Servers\NetworkAllocationController@delete');
|
||||||
});
|
});
|
||||||
|
|
||||||
Route::group(['prefix' => '/users'], function () {
|
Route::group(['prefix' => '/users'], function () {
|
||||||
|
|
Loading…
Reference in a new issue