Skip to content

Commit 5438848

Browse files
committed
Add basic subuser listing for servers
1 parent de464d3 commit 5438848

File tree

14 files changed

+310
-4
lines changed

14 files changed

+310
-4
lines changed
Lines changed: 45 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,45 @@
1+
<?php
2+
3+
namespace Pterodactyl\Http\Controllers\Api\Client\Servers;
4+
5+
use Pterodactyl\Models\Server;
6+
use Pterodactyl\Repositories\Eloquent\SubuserRepository;
7+
use Pterodactyl\Transformers\Api\Client\SubuserTransformer;
8+
use Pterodactyl\Http\Controllers\Api\Client\ClientApiController;
9+
use Pterodactyl\Http\Requests\Api\Client\Servers\Subusers\GetSubuserRequest;
10+
11+
class SubuserController extends ClientApiController
12+
{
13+
/**
14+
* @var \Pterodactyl\Repositories\Eloquent\SubuserRepository
15+
*/
16+
private $repository;
17+
18+
/**
19+
* SubuserController constructor.
20+
*
21+
* @param \Pterodactyl\Repositories\Eloquent\SubuserRepository $repository
22+
*/
23+
public function __construct(SubuserRepository $repository)
24+
{
25+
parent::__construct();
26+
27+
$this->repository = $repository;
28+
}
29+
30+
/**
31+
* Return the users associated with this server instance.
32+
*
33+
* @param \Pterodactyl\Http\Requests\Api\Client\Servers\Subusers\GetSubuserRequest $request
34+
* @param \Pterodactyl\Models\Server $server
35+
* @return array
36+
*/
37+
public function index(GetSubuserRequest $request, Server $server)
38+
{
39+
$users = $this->repository->getSubusersForServer($server->id);
40+
41+
return $this->fractal->collection($users)
42+
->transformWith($this->getTransformer(SubuserTransformer::class))
43+
->toArray();
44+
}
45+
}
Lines changed: 18 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,18 @@
1+
<?php
2+
3+
namespace Pterodactyl\Http\Requests\Api\Client\Servers\Subusers;
4+
5+
use Pterodactyl\Http\Requests\Api\Client\ClientApiRequest;
6+
7+
class GetSubuserRequest extends ClientApiRequest
8+
{
9+
/**
10+
* Confirm that a user is able to view subusers for the specified server.
11+
*
12+
* @return bool
13+
*/
14+
public function authorize(): bool
15+
{
16+
return $this->user()->can('view-subusers', $this->route()->parameter('server'));
17+
}
18+
}

app/Models/Subuser.php

Lines changed: 11 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -4,6 +4,17 @@
44

55
use Illuminate\Notifications\Notifiable;
66

7+
/**
8+
* @property int $id
9+
* @property int $user_id
10+
* @property int $server_id
11+
* @property \Carbon\Carbon $created_at
12+
* @property \Carbon\Carbon $updated_at
13+
*
14+
* @property \Pterodactyl\Models\User $user
15+
* @property \Pterodactyl\Models\Server $server
16+
* @property \Pterodactyl\Models\Permission[]|\Illuminate\Support\Collection $permissions
17+
*/
718
class Subuser extends Validable
819
{
920
use Notifiable;

app/Models/User.php

Lines changed: 1 addition & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -216,7 +216,7 @@ public function setUsernameAttribute(string $value)
216216
*/
217217
public function getNameAttribute()
218218
{
219-
return $this->name_first . ' ' . $this->name_last;
219+
return trim($this->name_first . ' ' . $this->name_last);
220220
}
221221

222222
/**

app/Repositories/Eloquent/SubuserRepository.php

Lines changed: 17 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -3,6 +3,7 @@
33
namespace Pterodactyl\Repositories\Eloquent;
44

55
use Pterodactyl\Models\Subuser;
6+
use Illuminate\Support\Collection;
67
use Pterodactyl\Exceptions\Repository\RecordNotFoundException;
78
use Pterodactyl\Contracts\Repository\SubuserRepositoryInterface;
89

@@ -18,6 +19,22 @@ public function model()
1819
return Subuser::class;
1920
}
2021

22+
/**
23+
* Returns the subusers for the given server instance with the associated user
24+
* and permission relationships pre-loaded.
25+
*
26+
* @param int $server
27+
* @return \Illuminate\Support\Collection
28+
*/
29+
public function getSubusersForServer(int $server): Collection
30+
{
31+
return $this->getBuilder()
32+
->with('user', 'permissions')
33+
->where('server_id', $server)
34+
->get()
35+
->toBase();
36+
}
37+
2138
/**
2239
* Return a subuser with the associated server relationship.
2340
*
Lines changed: 55 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,55 @@
1+
<?php
2+
3+
namespace Pterodactyl\Transformers\Api\Client;
4+
5+
use Illuminate\Support\Str;
6+
use Pterodactyl\Models\Subuser;
7+
8+
class SubuserTransformer extends BaseClientTransformer
9+
{
10+
protected $availableIncludes = ['permissions'];
11+
12+
/**
13+
* Return the resource name for the JSONAPI output.
14+
*
15+
* @return string
16+
*/
17+
public function getResourceName(): string
18+
{
19+
return Subuser::RESOURCE_NAME;
20+
}
21+
22+
/**
23+
* Transforms a User model into a representation that can be shown to regular
24+
* users of the API.
25+
*
26+
* @param \Pterodactyl\Models\Subuser $model
27+
* @return array
28+
*/
29+
public function transform(Subuser $model)
30+
{
31+
$user = $model->user;
32+
33+
return [
34+
'uuid' => $user->uuid,
35+
'username' => $user->username,
36+
'email' => $user->email,
37+
'image' => 'https://gravatar.com/avatar/' . md5(Str::lower($user->email)),
38+
'2fa_enabled' => $user->use_totp,
39+
'created_at' => $model->created_at->toIso8601String(),
40+
];
41+
}
42+
43+
/**
44+
* Include the permissions associated with this subuser.
45+
*
46+
* @param \Pterodactyl\Models\Subuser $model
47+
* @return \League\Fractal\Resource\Item
48+
*/
49+
public function includePermissions(Subuser $model)
50+
{
51+
return $this->item($model, function (Subuser $model) {
52+
return ['permissions' => $model->permissions->pluck('permission')];
53+
});
54+
}
55+
}

resources/scripts/api/http.ts

Lines changed: 10 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -38,6 +38,16 @@ export function httpErrorToHuman (error: any): string {
3838
return error.message;
3939
}
4040

41+
export interface FractalResponseData {
42+
object: string;
43+
attributes: {
44+
[k: string]: any;
45+
relationships?: {
46+
[k: string]: FractalResponseData;
47+
};
48+
};
49+
}
50+
4151
export interface PaginatedResult<T> {
4252
items: T[];
4353
pagination: PaginationDataSet;
Lines changed: 21 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,21 @@
1+
import http, { FractalResponseData } from '@/api/http';
2+
import { Subuser } from '@/state/server/subusers';
3+
4+
export const rawDataToServerSubuser = (data: FractalResponseData): Subuser => ({
5+
uuid: data.attributes.uuid,
6+
username: data.attributes.username,
7+
email: data.attributes.email,
8+
image: data.attributes.image,
9+
twoFactorEnabled: data.attributes['2fa_enabled'],
10+
createdAt: new Date(data.attributes.created_at),
11+
permissions: data.attributes.relationships!.permissions.attributes.permissions,
12+
can: permission => data.attributes.relationships!.permissions.attributes.permissions.indexOf(permission) >= 0,
13+
});
14+
15+
export default (uuid: string): Promise<Subuser[]> => {
16+
return new Promise((resolve, reject) => {
17+
http.get(`/api/client/servers/${uuid}/users`, { params: { include: [ 'permissions' ] } })
18+
.then(({ data }) => resolve((data.data || []).map(rawDataToServerSubuser)))
19+
.catch(reject);
20+
});
21+
};
Lines changed: 69 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,69 @@
1+
import React, { useEffect, useState } from 'react';
2+
import { FontAwesomeIcon } from '@fortawesome/react-fontawesome';
3+
import { faUserPlus } from '@fortawesome/free-solid-svg-icons/faUserPlus';
4+
import { ServerContext } from '@/state/server';
5+
import Spinner from '@/components/elements/Spinner';
6+
7+
export default () => {
8+
const [ loading, setLoading ] = useState(true);
9+
const uuid = ServerContext.useStoreState(state => state.server.data!.uuid);
10+
const subusers = ServerContext.useStoreState(state => state.subusers.data);
11+
const getSubusers = ServerContext.useStoreActions(actions => actions.subusers.getSubusers);
12+
13+
useEffect(() => {
14+
getSubusers(uuid)
15+
.then(() => setLoading(false))
16+
.catch(error => {
17+
console.error(error);
18+
});
19+
}, [ uuid, getSubusers ]);
20+
21+
useEffect(() => {
22+
if (subusers.length > 0) {
23+
setLoading(false);
24+
}
25+
}, [subusers]);
26+
27+
return (
28+
<div className={'flex my-10'}>
29+
<div className={'w-1/2'}>
30+
<h2 className={'text-neutral-300 mb-4'}>Subusers</h2>
31+
<div className={'border-t-4 border-primary-400 grey-box mt-0'}>
32+
{loading ?
33+
<div className={'w-full'}>
34+
<Spinner centered={true}/>
35+
</div>
36+
:
37+
!subusers.length ?
38+
<p className={'text-sm'}>It looks like you don't have any subusers.</p>
39+
:
40+
subusers.map(subuser => (
41+
<div key={subuser.uuid} className={'flex items-center w-full'}>
42+
<img
43+
className={'w-10 h-10 rounded-full bg-white border-2 border-inset border-neutral-800'}
44+
src={`${subuser.image}?s=400`}
45+
/>
46+
<div className={'ml-4 flex-1'}>
47+
<p className={'text-sm'}>{subuser.email}</p>
48+
</div>
49+
<div className={'ml-4'}>
50+
<button className={'btn btn-xs btn-primary'}>
51+
Edit
52+
</button>
53+
<button className={'ml-2 btn btn-xs btn-red btn-secondary'}>
54+
Remove
55+
</button>
56+
</div>
57+
</div>
58+
))
59+
}
60+
</div>
61+
<div className={'flex justify-end mt-4'}>
62+
<button className={'btn btn-primary btn-sm'}>
63+
<FontAwesomeIcon icon={faUserPlus} className={'mr-1'}/> New User
64+
</button>
65+
</div>
66+
</div>
67+
</div>
68+
);
69+
};

resources/scripts/routers/ServerRouter.tsx

Lines changed: 3 additions & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -12,6 +12,7 @@ import FileManagerContainer from '@/components/server/files/FileManagerContainer
1212
import { CSSTransition } from 'react-transition-group';
1313
import SuspenseSpinner from '@/components/elements/SuspenseSpinner';
1414
import FileEditContainer from '@/components/server/files/FileEditContainer';
15+
import UsersContainer from '@/components/server/users/UsersContainer';
1516

1617
const ServerRouter = ({ match, location }: RouteComponentProps<{ id: string }>) => {
1718
const server = ServerContext.useStoreState(state => state.server.data);
@@ -61,7 +62,8 @@ const ServerRouter = ({ match, location }: RouteComponentProps<{ id: string }>)
6162
)}
6263
exact
6364
/>
64-
<Route path={`${match.path}/databases`} component={DatabasesContainer}/>
65+
<Route path={`${match.path}/databases`} component={DatabasesContainer} exact/>
66+
<Route path={`${match.path}/users`} component={UsersContainer} exact/>
6567
</Switch>
6668
</React.Fragment>
6769
}

0 commit comments

Comments
 (0)