Merge branch 'develop' into cputhreads

This commit is contained in:
Dane Everitt 2020-04-03 13:48:06 -07:00 committed by GitHub
commit 78d6e59fc5
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
31 changed files with 1883 additions and 1323 deletions

View File

@ -33,6 +33,7 @@ return PhpCsFixer\Config::create()
'new_with_braces' => false, 'new_with_braces' => false,
'no_alias_functions' => true, 'no_alias_functions' => true,
'no_multiline_whitespace_before_semicolons' => true, 'no_multiline_whitespace_before_semicolons' => true,
'no_superfluous_phpdoc_tags' => false,
'no_unreachable_default_argument_value' => true, 'no_unreachable_default_argument_value' => true,
'no_useless_return' => true, 'no_useless_return' => true,
'not_operator_with_successor_space' => true, 'not_operator_with_successor_space' => true,

View File

@ -18,11 +18,10 @@ class SendPowerRequest extends ClientApiRequest
case 'start': case 'start':
return Permission::ACTION_CONTROL_START; return Permission::ACTION_CONTROL_START;
case 'stop': case 'stop':
case 'kill':
return Permission::ACTION_CONTROL_STOP; return Permission::ACTION_CONTROL_STOP;
case 'restart': case 'restart':
return Permission::ACTION_CONTROL_RESTART; return Permission::ACTION_CONTROL_RESTART;
case 'kill':
return Permission::ACTION_CONTROL_KILL;
} }
return '__invalid'; return '__invalid';

View File

@ -20,7 +20,6 @@ class Permission extends Validable
const ACTION_CONTROL_START = 'control.start'; const ACTION_CONTROL_START = 'control.start';
const ACTION_CONTROL_STOP = 'control.stop'; const ACTION_CONTROL_STOP = 'control.stop';
const ACTION_CONTROL_RESTART = 'control.restart'; const ACTION_CONTROL_RESTART = 'control.restart';
const ACTION_CONTROL_KILL = 'control.kill';
const ACTION_DATABASE_READ = 'database.read'; const ACTION_DATABASE_READ = 'database.read';
const ACTION_DATABASE_CREATE = 'database.create'; const ACTION_DATABASE_CREATE = 'database.create';
@ -111,7 +110,6 @@ class Permission extends Validable
'start' => 'Allows a user to start the server if it is stopped.', 'start' => 'Allows a user to start the server if it is stopped.',
'stop' => 'Allows a user to stop a server if it is running.', 'stop' => 'Allows a user to stop a server if it is running.',
'restart' => 'Allows a user to perform a server restart. This allows them to start the server if it is offline, but not put the server in a completely stopped state.', 'restart' => 'Allows a user to perform a server restart. This allows them to start the server if it is offline, but not put the server in a completely stopped state.',
'kill' => 'Allows a user to terminate a server process.',
], ],
], ],

View File

@ -61,6 +61,10 @@ class Server extends Validable
*/ */
const RESOURCE_NAME = 'server'; const RESOURCE_NAME = 'server';
const STATUS_INSTALLING = 0;
const STATUS_INSTALLED = 1;
const STATUS_INSTALL_FAILED = 2;
/** /**
* The table associated with the model. * The table associated with the model.
* *

View File

@ -2,7 +2,6 @@
namespace Pterodactyl\Repositories\Wings; namespace Pterodactyl\Repositories\Wings;
use BadMethodCallException;
use Webmozart\Assert\Assert; use Webmozart\Assert\Assert;
use Pterodactyl\Models\Server; use Pterodactyl\Models\Server;
use GuzzleHttp\Exception\TransferException; use GuzzleHttp\Exception\TransferException;
@ -13,7 +12,6 @@ class DaemonServerRepository extends DaemonRepository
/** /**
* Returns details about a server from the Daemon instance. * Returns details about a server from the Daemon instance.
* *
* @return array
* @throws \Pterodactyl\Exceptions\Http\Connection\DaemonConnectionException * @throws \Pterodactyl\Exceptions\Http\Connection\DaemonConnectionException
*/ */
public function getDetails(): array public function getDetails(): array
@ -89,10 +87,20 @@ class DaemonServerRepository extends DaemonRepository
/** /**
* Reinstall a server on the daemon. * Reinstall a server on the daemon.
*
* @throws \Pterodactyl\Exceptions\Http\Connection\DaemonConnectionException
*/ */
public function reinstall(): void public function reinstall(): void
{ {
throw new BadMethodCallException('Method is not implemented.'); Assert::isInstanceOf($this->server, Server::class);
try {
$this->getHttpClient()->post(sprintf(
'/api/servers/%s/reinstall', $this->server->uuid
));
} catch (TransferException $exception) {
throw new DaemonConnectionException($exception);
}
} }
/** /**

View File

@ -3,11 +3,9 @@
namespace Pterodactyl\Services\Servers; namespace Pterodactyl\Services\Servers;
use Pterodactyl\Models\Server; use Pterodactyl\Models\Server;
use GuzzleHttp\Exception\RequestException;
use Illuminate\Database\ConnectionInterface; use Illuminate\Database\ConnectionInterface;
use Pterodactyl\Repositories\Wings\DaemonServerRepository; use Pterodactyl\Repositories\Wings\DaemonServerRepository;
use Pterodactyl\Contracts\Repository\ServerRepositoryInterface; use Pterodactyl\Contracts\Repository\ServerRepositoryInterface;
use Pterodactyl\Exceptions\Http\Connection\DaemonConnectionException;
class ReinstallServerService class ReinstallServerService
{ {
@ -44,28 +42,23 @@ class ReinstallServerService
} }
/** /**
* @param int|\Pterodactyl\Models\Server $server * Reinstall a server on the remote daemon.
* *
* @throws \Pterodactyl\Exceptions\DisplayException * @param \Pterodactyl\Models\Server $server
* @throws \Pterodactyl\Exceptions\Model\DataValidationException * @return \Pterodactyl\Models\Server
* @throws \Pterodactyl\Exceptions\Repository\RecordNotFoundException *
* @throws \Throwable
*/ */
public function reinstall($server) public function reinstall(Server $server)
{ {
if (! $server instanceof Server) { $this->database->transaction(function () use ($server) {
$server = $this->repository->find($server);
}
$this->database->beginTransaction();
$this->repository->withoutFreshModel()->update($server->id, [ $this->repository->withoutFreshModel()->update($server->id, [
'installed' => 0, 'installed' => Server::STATUS_INSTALLING,
], true, true); ]);
try {
$this->daemonServerRepository->setServer($server)->reinstall(); $this->daemonServerRepository->setServer($server)->reinstall();
$this->database->commit(); });
} catch (RequestException $exception) {
throw new DaemonConnectionException($exception); return $server->refresh();
}
} }
} }

View File

@ -15,16 +15,16 @@
"ext-mbstring": "*", "ext-mbstring": "*",
"ext-pdo_mysql": "*", "ext-pdo_mysql": "*",
"ext-zip": "*", "ext-zip": "*",
"appstract/laravel-blade-directives": "^1.6", "appstract/laravel-blade-directives": "^1.8",
"aws/aws-sdk-php": "^3.110", "aws/aws-sdk-php": "^3.134",
"cakephp/chronos": "^1.2", "cakephp/chronos": "^1.3",
"doctrine/dbal": "^2.9", "doctrine/dbal": "^2.10",
"fideloper/proxy": "^4.2", "fideloper/proxy": "^4.2",
"guzzlehttp/guzzle": "^6.3", "guzzlehttp/guzzle": "^6.5",
"hashids/hashids": "^4.0", "hashids/hashids": "^4.0",
"laracasts/utilities": "^3.0", "laracasts/utilities": "^3.1",
"laravel/framework": "^6.0.0", "laravel/framework": "^6.18",
"laravel/helpers": "^1.1", "laravel/helpers": "^1.2",
"laravel/tinker": "^1.0", "laravel/tinker": "^1.0",
"lcobucci/jwt": "^3.3", "lcobucci/jwt": "^3.3",
"matriphe/iso-639": "^1.2", "matriphe/iso-639": "^1.2",
@ -32,18 +32,18 @@
"predis/predis": "^1.1", "predis/predis": "^1.1",
"prologue/alerts": "^0.4", "prologue/alerts": "^0.4",
"s1lentium/iptools": "^1.1", "s1lentium/iptools": "^1.1",
"spatie/laravel-fractal": "^5.6", "spatie/laravel-fractal": "^5.7",
"staudenmeir/belongs-to-through": "^2.6", "staudenmeir/belongs-to-through": "^2.9",
"symfony/yaml": "^4.0", "symfony/yaml": "^4.4",
"webmozart/assert": "^1.5" "webmozart/assert": "^1.7"
}, },
"require-dev": { "require-dev": {
"barryvdh/laravel-debugbar": "^3.2", "barryvdh/laravel-debugbar": "^3.2",
"barryvdh/laravel-ide-helper": "^2.6", "barryvdh/laravel-ide-helper": "^2.6",
"codedungeon/phpunit-result-printer": "0.25.1", "codedungeon/phpunit-result-printer": "0.25.1",
"friendsofphp/php-cs-fixer": "^2.15.1", "friendsofphp/php-cs-fixer": "^2.16.1",
"laravel/dusk": "^5.5", "laravel/dusk": "^5.11",
"php-mock/php-mock-phpunit": "^2.4", "php-mock/php-mock-phpunit": "^2.6",
"phpunit/phpunit": "^7" "phpunit/phpunit": "^7"
}, },
"autoload": { "autoload": {

2317
composer.lock generated

File diff suppressed because it is too large Load Diff

View File

@ -1,41 +1,27 @@
import React, { useMemo } from 'react'; import React from 'react';
import { ServerContext } from '@/state/server'; import { usePermissions } from '@/plugins/usePermissions';
interface Props { interface Props {
action: string | string[]; action: string | string[];
matchAny?: boolean;
renderOnError?: React.ReactNode | null; renderOnError?: React.ReactNode | null;
children: React.ReactNode; children: React.ReactNode;
} }
const Can = ({ action, renderOnError, children }: Props) => { const Can = ({ action, matchAny = false, renderOnError, children }: Props) => {
const userPermissions = ServerContext.useStoreState(state => state.server.permissions); const can = usePermissions(action);
const actions = Array.isArray(action) ? action : [ action ];
const missingPermissionCount = useMemo(() => { if (matchAny) {
if (userPermissions[0] === '*') { console.log('Can.tsx', can);
return 0;
} }
return actions.filter(permission => {
return !(
// Allows checking for any permission matching a name, for example files.*
// will return if the user has any permission under the file.XYZ namespace.
(
permission.endsWith('.*')
&& permission !== 'websocket.*'
&& userPermissions.filter(p => p.startsWith(permission.split('.')[0])).length > 0
)
// Otherwise just check if the entire permission exists in the array or not.
|| userPermissions.indexOf(permission) >= 0);
}).length;
}, [ action, userPermissions ]);
return ( return (
<> <>
{missingPermissionCount > 0 ? {
renderOnError ((matchAny && can.filter(p => p).length > 0) || (!matchAny && can.every(p => p))) ?
:
children children
:
renderOnError
} }
</> </>
); );

View File

@ -4,6 +4,9 @@ import * as TerminalFit from 'xterm/lib/addons/fit/fit';
import SpinnerOverlay from '@/components/elements/SpinnerOverlay'; import SpinnerOverlay from '@/components/elements/SpinnerOverlay';
import { ServerContext } from '@/state/server'; import { ServerContext } from '@/state/server';
import styled from 'styled-components'; import styled from 'styled-components';
import Can from '@/components/elements/Can';
import { usePermissions } from '@/plugins/usePermissions';
import classNames from 'classnames';
const theme = { const theme = {
background: 'transparent', background: 'transparent',
@ -52,6 +55,7 @@ export default () => {
const useRef = useCallback(node => setTerminalElement(node), []); const useRef = useCallback(node => setTerminalElement(node), []);
const terminal = useMemo(() => new Terminal({ ...terminalProps }), []); const terminal = useMemo(() => new Terminal({ ...terminalProps }), []);
const { connected, instance } = ServerContext.useStoreState(state => state.socket); const { connected, instance } = ServerContext.useStoreState(state => state.socket);
const [ canSendCommands ] = usePermissions([ 'control.console']);
const handleConsoleOutput = (line: string, prelude = false) => terminal.writeln( const handleConsoleOutput = (line: string, prelude = false) => terminal.writeln(
(prelude ? TERMINAL_PRELUDE : '') + line.replace(/(?:\r\n|\r|\n)$/im, '') + '\u001b[0m', (prelude ? TERMINAL_PRELUDE : '') + line.replace(/(?:\r\n|\r|\n)$/im, '') + '\u001b[0m',
@ -121,7 +125,9 @@ export default () => {
<div className={'text-xs font-mono relative'}> <div className={'text-xs font-mono relative'}>
<SpinnerOverlay visible={!connected} size={'large'}/> <SpinnerOverlay visible={!connected} size={'large'}/>
<div <div
className={'rounded-t p-2 bg-black w-full'} className={classNames('rounded-t p-2 bg-black w-full', {
'rounded-b': !canSendCommands,
})}
style={{ style={{
minHeight: '16rem', minHeight: '16rem',
maxHeight: '32rem', maxHeight: '32rem',
@ -129,6 +135,7 @@ export default () => {
> >
<TerminalDiv id={'terminal'} ref={useRef}/> <TerminalDiv id={'terminal'} ref={useRef}/>
</div> </div>
{canSendCommands &&
<div className={'rounded-b bg-neutral-900 text-neutral-100 flex'}> <div className={'rounded-b bg-neutral-900 text-neutral-100 flex'}>
<div className={'flex-no-shrink p-2 font-bold'}>$</div> <div className={'flex-no-shrink p-2 font-bold'}>$</div>
<div className={'w-full'}> <div className={'w-full'}>
@ -140,6 +147,7 @@ export default () => {
/> />
</div> </div>
</div> </div>
}
</div> </div>
); );
}; };

View File

@ -9,6 +9,7 @@ import { faMicrochip } from '@fortawesome/free-solid-svg-icons/faMicrochip';
import { bytesToHuman } from '@/helpers'; import { bytesToHuman } from '@/helpers';
import SuspenseSpinner from '@/components/elements/SuspenseSpinner'; import SuspenseSpinner from '@/components/elements/SuspenseSpinner';
import TitledGreyBox from '@/components/elements/TitledGreyBox'; import TitledGreyBox from '@/components/elements/TitledGreyBox';
import Can from '@/components/elements/Can';
type PowerAction = 'start' | 'stop' | 'restart' | 'kill'; type PowerAction = 'start' | 'stop' | 'restart' | 'kill';
@ -109,7 +110,9 @@ export default () => {
&nbsp;{cpu.toFixed(2)} % &nbsp;{cpu.toFixed(2)} %
</p> </p>
</TitledGreyBox> </TitledGreyBox>
<Can action={[ 'control.start', 'control.stop', 'control.restart' ]} matchAny={true}>
<div className={'grey-box justify-center'}> <div className={'grey-box justify-center'}>
<Can action={'control.start'}>
<button <button
className={'btn btn-secondary btn-xs mr-2'} className={'btn btn-secondary btn-xs mr-2'}
disabled={status !== 'offline'} disabled={status !== 'offline'}
@ -120,6 +123,8 @@ export default () => {
> >
Start Start
</button> </button>
</Can>
<Can action={'control.restart'}>
<button <button
className={'btn btn-secondary btn-xs mr-2'} className={'btn btn-secondary btn-xs mr-2'}
onClick={e => { onClick={e => {
@ -129,8 +134,12 @@ export default () => {
> >
Restart Restart
</button> </button>
</Can>
<Can action={'control.stop'}>
<StopOrKillButton onPress={action => sendPowerCommand(action)}/> <StopOrKillButton onPress={action => sendPowerCommand(action)}/>
</Can>
</div> </div>
</Can>
</div> </div>
<div className={'flex-1 mx-4 mr-4'}> <div className={'flex-1 mx-4 mr-4'}>
<SuspenseSpinner> <SuspenseSpinner>

View File

@ -1,5 +1,4 @@
import React, { useState } from 'react'; import React, { useState } from 'react';
import { ServerDatabase } from '@/api/server/getServerDatabases';
import { FontAwesomeIcon } from '@fortawesome/react-fontawesome'; import { FontAwesomeIcon } from '@fortawesome/react-fontawesome';
import { faDatabase } from '@fortawesome/free-solid-svg-icons/faDatabase'; import { faDatabase } from '@fortawesome/free-solid-svg-icons/faDatabase';
import { faTrashAlt } from '@fortawesome/free-solid-svg-icons/faTrashAlt'; import { faTrashAlt } from '@fortawesome/free-solid-svg-icons/faTrashAlt';
@ -16,6 +15,7 @@ import { ServerContext } from '@/state/server';
import deleteServerDatabase from '@/api/server/deleteServerDatabase'; import deleteServerDatabase from '@/api/server/deleteServerDatabase';
import { httpErrorToHuman } from '@/api/http'; import { httpErrorToHuman } from '@/api/http';
import RotatePasswordButton from '@/components/server/databases/RotatePasswordButton'; import RotatePasswordButton from '@/components/server/databases/RotatePasswordButton';
import Can from '@/components/elements/Can';
interface Props { interface Props {
databaseId: string | number; databaseId: string | number;
@ -73,7 +73,10 @@ export default ({ databaseId, className, onDelete }: Props) => {
visible={visible} visible={visible}
dismissable={!isSubmitting} dismissable={!isSubmitting}
showSpinnerOverlay={isSubmitting} showSpinnerOverlay={isSubmitting}
onDismissed={() => { setVisible(false); resetForm(); }} onDismissed={() => {
setVisible(false);
resetForm();
}}
> >
<FlashMessageRender byKey={'delete-database-modal'} className={'mb-6'}/> <FlashMessageRender byKey={'delete-database-modal'} className={'mb-6'}/>
<h3 className={'mb-6'}>Confirm database deletion</h3> <h3 className={'mb-6'}>Confirm database deletion</h3>
@ -113,10 +116,12 @@ export default ({ databaseId, className, onDelete }: Props) => {
<Modal visible={connectionVisible} onDismissed={() => setConnectionVisible(false)}> <Modal visible={connectionVisible} onDismissed={() => setConnectionVisible(false)}>
<FlashMessageRender byKey={'database-connection-modal'} className={'mb-6'}/> <FlashMessageRender byKey={'database-connection-modal'} className={'mb-6'}/>
<h3 className={'mb-6'}>Database connection details</h3> <h3 className={'mb-6'}>Database connection details</h3>
<Can action={'database.view_password'}>
<div> <div>
<label className={'input-dark-label'}>Password</label> <label className={'input-dark-label'}>Password</label>
<input type={'text'} className={'input-dark'} readOnly={true} value={database.password}/> <input type={'text'} className={'input-dark'} readOnly={true} value={database.password}/>
</div> </div>
</Can>
<div className={'mt-6'}> <div className={'mt-6'}>
<label className={'input-dark-label'}>JBDC Connection String</label> <label className={'input-dark-label'}>JBDC Connection String</label>
<input <input
@ -127,7 +132,9 @@ export default ({ databaseId, className, onDelete }: Props) => {
/> />
</div> </div>
<div className={'mt-6 text-right'}> <div className={'mt-6 text-right'}>
<Can action={'database.update'}>
<RotatePasswordButton databaseId={database.id} onUpdate={appendDatabase}/> <RotatePasswordButton databaseId={database.id} onUpdate={appendDatabase}/>
</Can>
<button className={'btn btn-sm btn-secondary'} onClick={() => setConnectionVisible(false)}> <button className={'btn btn-sm btn-secondary'} onClick={() => setConnectionVisible(false)}>
Close Close
</button> </button>
@ -156,9 +163,11 @@ export default ({ databaseId, className, onDelete }: Props) => {
<button className={'btn btn-sm btn-secondary mr-2'} onClick={() => setConnectionVisible(true)}> <button className={'btn btn-sm btn-secondary mr-2'} onClick={() => setConnectionVisible(true)}>
<FontAwesomeIcon icon={faEye} fixedWidth={true}/> <FontAwesomeIcon icon={faEye} fixedWidth={true}/>
</button> </button>
<Can action={'database.delete'}>
<button className={'btn btn-sm btn-secondary btn-red'} onClick={() => setVisible(true)}> <button className={'btn btn-sm btn-secondary btn-red'} onClick={() => setVisible(true)}>
<FontAwesomeIcon icon={faTrashAlt} fixedWidth={true}/> <FontAwesomeIcon icon={faTrashAlt} fixedWidth={true}/>
</button> </button>
</Can>
</div> </div>
</div> </div>
</React.Fragment> </React.Fragment>

View File

@ -1,5 +1,5 @@
import React, { useEffect, useState } from 'react'; import React, { useEffect, useState } from 'react';
import getServerDatabases, { ServerDatabase } from '@/api/server/getServerDatabases'; import getServerDatabases from '@/api/server/getServerDatabases';
import { ServerContext } from '@/state/server'; import { ServerContext } from '@/state/server';
import { Actions, useStoreActions } from 'easy-peasy'; import { Actions, useStoreActions } from 'easy-peasy';
import { ApplicationStore } from '@/state'; import { ApplicationStore } from '@/state';
@ -9,6 +9,7 @@ import DatabaseRow from '@/components/server/databases/DatabaseRow';
import Spinner from '@/components/elements/Spinner'; import Spinner from '@/components/elements/Spinner';
import { CSSTransition } from 'react-transition-group'; import { CSSTransition } from 'react-transition-group';
import CreateDatabaseButton from '@/components/server/databases/CreateDatabaseButton'; import CreateDatabaseButton from '@/components/server/databases/CreateDatabaseButton';
import Can from '@/components/elements/Can';
export default () => { export default () => {
const [ loading, setLoading ] = useState(true); const [ loading, setLoading ] = useState(true);
@ -41,7 +42,7 @@ export default () => {
<Spinner size={'large'} centered={true}/> <Spinner size={'large'} centered={true}/>
: :
<CSSTransition classNames={'fade'} timeout={250}> <CSSTransition classNames={'fade'} timeout={250}>
<React.Fragment> <>
{databases.length > 0 ? {databases.length > 0 ?
databases.map((database, index) => ( databases.map((database, index) => (
<DatabaseRow <DatabaseRow
@ -54,18 +55,20 @@ export default () => {
: :
<p className={'text-center text-sm text-neutral-400'}> <p className={'text-center text-sm text-neutral-400'}>
{server.featureLimits.databases > 0 ? {server.featureLimits.databases > 0 ?
`It looks like you have no databases. Click the button below to create one now.` `It looks like you have no databases.`
: :
`Databases cannot be created for this server.` `Databases cannot be created for this server.`
} }
</p> </p>
} }
<Can action={'database.create'}>
{server.featureLimits.databases > 0 && {server.featureLimits.databases > 0 &&
<div className={'mt-6 flex justify-end'}> <div className={'mt-6 flex justify-end'}>
<CreateDatabaseButton onCreated={appendDatabase}/> <CreateDatabaseButton onCreated={appendDatabase}/>
</div> </div>
} }
</React.Fragment> </Can>
</>
</CSSTransition> </CSSTransition>
} }
</div> </div>

View File

@ -13,7 +13,8 @@ import { join } from 'path';
import deleteFile from '@/api/server/files/deleteFile'; import deleteFile from '@/api/server/files/deleteFile';
import SpinnerOverlay from '@/components/elements/SpinnerOverlay'; import SpinnerOverlay from '@/components/elements/SpinnerOverlay';
import copyFile from '@/api/server/files/copyFile'; import copyFile from '@/api/server/files/copyFile';
import http, { httpErrorToHuman } from '@/api/http'; import { httpErrorToHuman } from '@/api/http';
import Can from '@/components/elements/Can';
type ModalType = 'rename' | 'move'; type ModalType = 'rename' | 'move';
@ -118,9 +119,13 @@ export default ({ uuid }: { uuid: string }) => {
<CSSTransition timeout={250} in={menuVisible} unmountOnExit={true} classNames={'fade'}> <CSSTransition timeout={250} in={menuVisible} unmountOnExit={true} classNames={'fade'}>
<div <div
ref={menu} ref={menu}
onClick={e => { e.stopPropagation(); setMenuVisible(false); }} onClick={e => {
e.stopPropagation();
setMenuVisible(false);
}}
className={'absolute bg-white p-2 rounded border border-neutral-700 shadow-lg text-neutral-500 min-w-48'} className={'absolute bg-white p-2 rounded border border-neutral-700 shadow-lg text-neutral-500 min-w-48'}
> >
<Can action={'file.update'}>
<div <div
onClick={() => setModal('rename')} onClick={() => setModal('rename')}
className={'hover:text-neutral-700 p-2 flex items-center hover:bg-neutral-100 rounded'} className={'hover:text-neutral-700 p-2 flex items-center hover:bg-neutral-100 rounded'}
@ -135,6 +140,8 @@ export default ({ uuid }: { uuid: string }) => {
<FontAwesomeIcon icon={faLevelUpAlt} className={'text-xs'}/> <FontAwesomeIcon icon={faLevelUpAlt} className={'text-xs'}/>
<span className={'ml-2'}>Move</span> <span className={'ml-2'}>Move</span>
</div> </div>
</Can>
<Can action={'file.create'}>
<div <div
onClick={() => doCopy()} onClick={() => doCopy()}
className={'hover:text-neutral-700 p-2 flex items-center hover:bg-neutral-100 rounded'} className={'hover:text-neutral-700 p-2 flex items-center hover:bg-neutral-100 rounded'}
@ -142,6 +149,7 @@ export default ({ uuid }: { uuid: string }) => {
<FontAwesomeIcon icon={faCopy} className={'text-xs'}/> <FontAwesomeIcon icon={faCopy} className={'text-xs'}/>
<span className={'ml-2'}>Copy</span> <span className={'ml-2'}>Copy</span>
</div> </div>
</Can>
<div <div
className={'hover:text-neutral-700 p-2 flex items-center hover:bg-neutral-100 rounded'} className={'hover:text-neutral-700 p-2 flex items-center hover:bg-neutral-100 rounded'}
onClick={() => doDownload()} onClick={() => doDownload()}
@ -149,6 +157,7 @@ export default ({ uuid }: { uuid: string }) => {
<FontAwesomeIcon icon={faFileDownload} className={'text-xs'}/> <FontAwesomeIcon icon={faFileDownload} className={'text-xs'}/>
<span className={'ml-2'}>Download</span> <span className={'ml-2'}>Download</span>
</div> </div>
<Can action={'file.delete'}>
<div <div
onClick={() => doDeletion()} onClick={() => doDeletion()}
className={'hover:text-red-700 p-2 flex items-center hover:bg-red-100 rounded'} className={'hover:text-red-700 p-2 flex items-center hover:bg-red-100 rounded'}
@ -156,6 +165,7 @@ export default ({ uuid }: { uuid: string }) => {
<FontAwesomeIcon icon={faTrashAlt} className={'text-xs'}/> <FontAwesomeIcon icon={faTrashAlt} className={'text-xs'}/>
<span className={'ml-2'}>Delete</span> <span className={'ml-2'}>Delete</span>
</div> </div>
</Can>
</div> </div>
</CSSTransition> </CSSTransition>
</div> </div>

View File

@ -2,7 +2,7 @@ import React, { lazy, useEffect, useState } from 'react';
import { ServerContext } from '@/state/server'; import { ServerContext } from '@/state/server';
import getFileContents from '@/api/server/files/getFileContents'; import getFileContents from '@/api/server/files/getFileContents';
import useRouter from 'use-react-router'; import useRouter from 'use-react-router';
import { Actions, useStoreState } from 'easy-peasy'; import { Actions, useStoreActions, useStoreState } from 'easy-peasy';
import { ApplicationStore } from '@/state'; import { ApplicationStore } from '@/state';
import { httpErrorToHuman } from '@/api/http'; import { httpErrorToHuman } from '@/api/http';
import SpinnerOverlay from '@/components/elements/SpinnerOverlay'; import SpinnerOverlay from '@/components/elements/SpinnerOverlay';
@ -10,6 +10,8 @@ import saveFileContents from '@/api/server/files/saveFileContents';
import FileManagerBreadcrumbs from '@/components/server/files/FileManagerBreadcrumbs'; import FileManagerBreadcrumbs from '@/components/server/files/FileManagerBreadcrumbs';
import { useParams } from 'react-router'; import { useParams } from 'react-router';
import FileNameModal from '@/components/server/files/FileNameModal'; import FileNameModal from '@/components/server/files/FileNameModal';
import Can from '@/components/elements/Can';
import FlashMessageRender from '@/components/FlashMessageRender';
const LazyAceEditor = lazy(() => import(/* webpackChunkName: "editor" */'@/components/elements/AceEditor')); const LazyAceEditor = lazy(() => import(/* webpackChunkName: "editor" */'@/components/elements/AceEditor'));
@ -21,15 +23,20 @@ export default () => {
const [ modalVisible, setModalVisible ] = useState(false); const [ modalVisible, setModalVisible ] = useState(false);
const { id, uuid } = ServerContext.useStoreState(state => state.server.data!); const { id, uuid } = ServerContext.useStoreState(state => state.server.data!);
const addError = useStoreState((state: Actions<ApplicationStore>) => state.flashes.addError); const { addError, clearFlashes } = useStoreActions((actions: Actions<ApplicationStore>) => actions.flashes);
let fetchFileContent: null | (() => Promise<string>) = null; let fetchFileContent: null | (() => Promise<string>) = null;
if (action !== 'new') { if (action !== 'new') {
useEffect(() => { useEffect(() => {
setLoading(true);
clearFlashes('files:view');
getFileContents(uuid, hash.replace(/^#/, '')) getFileContents(uuid, hash.replace(/^#/, ''))
.then(setContent) .then(setContent)
.catch(error => console.error(error)) .catch(error => {
console.error(error);
addError({ key: 'files:view', message: httpErrorToHuman(error) });
})
.then(() => setLoading(false)); .then(() => setLoading(false));
}, [ uuid, hash ]); }, [ uuid, hash ]);
} }
@ -40,8 +47,8 @@ export default () => {
} }
setLoading(true); setLoading(true);
fetchFileContent() clearFlashes('files:view');
.then(content => { fetchFileContent().then(content => {
return saveFileContents(uuid, name || hash.replace(/^#/, ''), content); return saveFileContents(uuid, name || hash.replace(/^#/, ''), content);
}) })
.then(() => { .then(() => {
@ -54,13 +61,14 @@ export default () => {
}) })
.catch(error => { .catch(error => {
console.error(error); console.error(error);
addError({ message: httpErrorToHuman(error), key: 'files' }); addError({ message: httpErrorToHuman(error), key: 'files:view' });
}) })
.then(() => setLoading(false)); .then(() => setLoading(false));
}; };
return ( return (
<div className={'mt-10 mb-4'}> <div className={'mt-10 mb-4'}>
<FlashMessageRender byKey={'files:view'} className={'mb-4'}/>
<FileManagerBreadcrumbs withinFileEditor={true} isNewFile={action !== 'edit'}/> <FileManagerBreadcrumbs withinFileEditor={true} isNewFile={action !== 'edit'}/>
<FileNameModal <FileNameModal
visible={modalVisible} visible={modalVisible}
@ -83,13 +91,17 @@ export default () => {
</div> </div>
<div className={'flex justify-end mt-4'}> <div className={'flex justify-end mt-4'}>
{action === 'edit' ? {action === 'edit' ?
<Can action={'file.update'}>
<button className={'btn btn-primary btn-sm'} onClick={() => save()}> <button className={'btn btn-primary btn-sm'} onClick={() => save()}>
Save Content Save Content
</button> </button>
</Can>
: :
<Can action={'file.create'}>
<button className={'btn btn-primary btn-sm'} onClick={() => setModalVisible(true)}> <button className={'btn btn-primary btn-sm'} onClick={() => setModalVisible(true)}>
Create File Create File
</button> </button>
</Can>
} }
</div> </div>
</div> </div>

View File

@ -11,6 +11,7 @@ import FileManagerBreadcrumbs from '@/components/server/files/FileManagerBreadcr
import { FileObject } from '@/api/server/files/loadDirectory'; import { FileObject } from '@/api/server/files/loadDirectory';
import NewDirectoryButton from '@/components/server/files/NewDirectoryButton'; import NewDirectoryButton from '@/components/server/files/NewDirectoryButton';
import { Link } from 'react-router-dom'; import { Link } from 'react-router-dom';
import Can from '@/components/elements/Can';
const sortFiles = (files: FileObject[]): FileObject[] => { const sortFiles = (files: FileObject[]): FileObject[] => {
return files.sort((a, b) => a.name.localeCompare(b.name)) return files.sort((a, b) => a.name.localeCompare(b.name))
@ -34,7 +35,6 @@ export default () => {
console.error(error.message, { error }); console.error(error.message, { error });
addError({ message: httpErrorToHuman(error), key: 'files' }); addError({ message: httpErrorToHuman(error), key: 'files' });
}); });
// eslint-disable-next-line react-hooks/exhaustive-deps
}, [ directory ]); }, [ directory ]);
return ( return (
@ -78,12 +78,17 @@ export default () => {
</React.Fragment> </React.Fragment>
</CSSTransition> </CSSTransition>
} }
<Can action={'file.create'}>
<div className={'flex justify-end mt-8'}> <div className={'flex justify-end mt-8'}>
<NewDirectoryButton/> <NewDirectoryButton/>
<Link to={`/server/${id}/files/new${window.location.hash}`} className={'btn btn-sm btn-primary'}> <Link
to={`/server/${id}/files/new${window.location.hash}`}
className={'btn btn-sm btn-primary'}
>
New File New File
</Link> </Link>
</div> </div>
</Can>
</React.Fragment> </React.Fragment>
} }
</React.Fragment> </React.Fragment>

View File

@ -9,6 +9,7 @@ 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'; import EditScheduleModal from '@/components/server/schedules/EditScheduleModal';
import Can from '@/components/elements/Can';
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!);
@ -35,9 +36,8 @@ export default ({ match, history }: RouteComponentProps) => {
<> <>
{ {
schedules.length === 0 ? schedules.length === 0 ?
<p className={'text-sm text-neutral-400'}> <p className={'text-sm text-center text-neutral-400'}>
There are no schedules configured for this server. Click the button below to get There are no schedules configured for this server.
started.
</p> </p>
: :
schedules.map(schedule => ( schedules.map(schedule => (
@ -54,6 +54,7 @@ export default ({ match, history }: RouteComponentProps) => {
</a> </a>
)) ))
} }
<Can action={'schedule.create'}>
<div className={'mt-8 flex justify-end'}> <div className={'mt-8 flex justify-end'}>
{visible && <EditScheduleModal {visible && <EditScheduleModal
appear={true} appear={true}
@ -63,12 +64,13 @@ export default ({ match, history }: RouteComponentProps) => {
/>} />}
<button <button
type={'button'} type={'button'}
className={'btn btn-lg btn-primary'} className={'btn btn-sm btn-primary'}
onClick={() => setVisible(true)} onClick={() => setVisible(true)}
> >
Create schedule Create schedule
</button> </button>
</div> </div>
</Can>
</> </>
} }
</div> </div>

View File

@ -13,6 +13,7 @@ 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'; import DeleteScheduleButton from '@/components/server/schedules/DeleteScheduleButton';
import Can from '@/components/elements/Can';
interface Params { interface Params {
id: string; id: string;
@ -93,15 +94,17 @@ export default ({ match, history, location: { state } }: RouteComponentProps<Par
</> </>
: :
<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.
button below.
</p> </p>
} }
<div className={'mt-8 flex justify-end'}> <div className={'mt-8 flex justify-end'}>
<Can action={'schedule.delete'}>
<DeleteScheduleButton <DeleteScheduleButton
scheduleId={schedule.id} scheduleId={schedule.id}
onDeleted={() => history.push(`/server/${id}/schedules`)} onDeleted={() => history.push(`/server/${id}/schedules`)}
/> />
</Can>
<Can action={'schedule.update'}>
<button className={'btn btn-primary btn-sm mr-4'} onClick={() => setShowEditModal(true)}> <button className={'btn btn-primary btn-sm mr-4'} onClick={() => setShowEditModal(true)}>
Edit Edit
</button> </button>
@ -111,6 +114,7 @@ export default ({ match, history, location: { state } }: RouteComponentProps<Par
...s!, tasks: [ ...s!.tasks, task ], ...s!, tasks: [ ...s!.tasks, task ],
}))} }))}
/> />
</Can>
</div> </div>
</> </>
} }

View File

@ -13,6 +13,7 @@ import { httpErrorToHuman } from '@/api/http';
import SpinnerOverlay from '@/components/elements/SpinnerOverlay'; import SpinnerOverlay from '@/components/elements/SpinnerOverlay';
import TaskDetailsModal from '@/components/server/schedules/TaskDetailsModal'; import TaskDetailsModal from '@/components/server/schedules/TaskDetailsModal';
import { faPencilAlt } from '@fortawesome/free-solid-svg-icons/faPencilAlt'; import { faPencilAlt } from '@fortawesome/free-solid-svg-icons/faPencilAlt';
import Can from '@/components/elements/Can';
interface Props { interface Props {
schedule: number; schedule: number;
@ -75,6 +76,7 @@ export default ({ schedule, task, onTaskUpdated, onTaskRemoved }: Props) => {
</p> </p>
</div> </div>
} }
<Can action={'schedule.update'}>
<button <button
type={'button'} type={'button'}
aria-label={'Edit scheduled task'} aria-label={'Edit scheduled task'}
@ -83,6 +85,8 @@ export default ({ schedule, task, onTaskUpdated, onTaskRemoved }: Props) => {
> >
<FontAwesomeIcon icon={faPencilAlt}/> <FontAwesomeIcon icon={faPencilAlt}/>
</button> </button>
</Can>
<Can action={'schedule.update'}>
<button <button
type={'button'} type={'button'}
aria-label={'Delete scheduled task'} aria-label={'Delete scheduled task'}
@ -91,6 +95,7 @@ export default ({ schedule, task, onTaskUpdated, onTaskRemoved }: Props) => {
> >
<FontAwesomeIcon icon={faTrashAlt}/> <FontAwesomeIcon icon={faTrashAlt}/>
</button> </button>
</Can>
</div> </div>
); );
}; };

View File

@ -6,6 +6,7 @@ import { ApplicationStore } from '@/state';
import { UserData } from '@/state/user'; import { UserData } from '@/state/user';
import RenameServerBox from '@/components/server/settings/RenameServerBox'; import RenameServerBox from '@/components/server/settings/RenameServerBox';
import FlashMessageRender from '@/components/FlashMessageRender'; import FlashMessageRender from '@/components/FlashMessageRender';
import Can from '@/components/elements/Can';
export default () => { export default () => {
const user = useStoreState<ApplicationStore, UserData>(state => state.user.data!); const user = useStoreState<ApplicationStore, UserData>(state => state.user.data!);
@ -15,7 +16,8 @@ export default () => {
<div className={'my-10 mb-6'}> <div className={'my-10 mb-6'}>
<FlashMessageRender byKey={'settings'} className={'mb-4'}/> <FlashMessageRender byKey={'settings'} className={'mb-4'}/>
<div className={'md:flex'}> <div className={'md:flex'}>
<TitledGreyBox title={'SFTP Details'} className={'w-full md:flex-1 md:mr-6'}> <Can action={'file.sftp'}>
<TitledGreyBox title={'SFTP Details'} className={'w-full md:flex-1 md:max-w-1/2 md:mr-6'}>
<div> <div>
<label className={'input-dark-label'}>Server Address</label> <label className={'input-dark-label'}>Server Address</label>
<input <input
@ -52,9 +54,12 @@ export default () => {
</div> </div>
</div> </div>
</TitledGreyBox> </TitledGreyBox>
<div className={'w-full mt-6 md:flex-1 md:mt-0'}> </Can>
<Can action={'settings.rename'}>
<div className={'w-full mt-6 md:flex-1 md:max-w-1/2 md:mt-0'}>
<RenameServerBox/> <RenameServerBox/>
</div> </div>
</Can>
</div> </div>
</div> </div>
); );

View File

@ -14,6 +14,8 @@ import createOrUpdateSubuser from '@/api/server/users/createOrUpdateSubuser';
import { ServerContext } from '@/state/server'; import { ServerContext } from '@/state/server';
import { httpErrorToHuman } from '@/api/http'; import { httpErrorToHuman } from '@/api/http';
import FlashMessageRender from '@/components/FlashMessageRender'; import FlashMessageRender from '@/components/FlashMessageRender';
import Can from '@/components/elements/Can';
import { usePermissions } from '@/plugins/usePermissions';
type Props = { type Props = {
subuser?: Subuser; subuser?: Subuser;
@ -25,21 +27,32 @@ interface Values {
} }
const PermissionLabel = styled.label` const PermissionLabel = styled.label`
${tw`flex items-center border border-transparent rounded p-2 cursor-pointer`}; ${tw`flex items-center border border-transparent rounded p-2`};
text-transform: none; text-transform: none;
&:not(.disabled) {
${tw`cursor-pointer`};
&:hover { &:hover {
${tw`border-neutral-500 bg-neutral-800`}; ${tw`border-neutral-500 bg-neutral-800`};
} }
}
`; `;
const EditSubuserModal = forwardRef<HTMLHeadingElement, Props>(({ subuser, ...props }, ref) => { const EditSubuserModal = forwardRef<HTMLHeadingElement, Props>(({ subuser, ...props }, ref) => {
const { values, isSubmitting, setFieldValue } = useFormikContext<Values>(); const { values, isSubmitting, setFieldValue } = useFormikContext<Values>();
const [ canEditUser ] = usePermissions([ 'user.update' ]);
const permissions = useStoreState((state: ApplicationStore) => state.permissions.data); const permissions = useStoreState((state: ApplicationStore) => state.permissions.data);
return ( return (
<Modal {...props} showSpinnerOverlay={isSubmitting}> <Modal {...props} showSpinnerOverlay={isSubmitting}>
<h3 ref={ref}>{subuser ? `Modify permissions for ${subuser.email}` : 'Create new subuser'}</h3> <h3 ref={ref}>
{subuser ?
`${canEditUser ? 'Modify' : 'View'} permissions for ${subuser.email}`
:
'Create new subuser'
}
</h3>
<FlashMessageRender byKey={'user:edit'} className={'mt-4'}/> <FlashMessageRender byKey={'user:edit'} className={'mt-4'}/>
{!subuser && {!subuser &&
<div className={'mt-6'}> <div className={'mt-6'}>
@ -50,13 +63,14 @@ const EditSubuserModal = forwardRef<HTMLHeadingElement, Props>(({ subuser, ...pr
/> />
</div> </div>
} }
<div className={'mt-6'}> <div className={'my-6'}>
{Object.keys(permissions).filter(key => key !== 'websocket').map((key, index) => ( {Object.keys(permissions).filter(key => key !== 'websocket').map((key, index) => (
<TitledGreyBox <TitledGreyBox
key={key} key={key}
title={ title={
<div className={'flex items-center'}> <div className={'flex items-center'}>
<p className={'text-sm uppercase flex-1'}>{key}</p> <p className={'text-sm uppercase flex-1'}>{key}</p>
{canEditUser &&
<input <input
type={'checkbox'} type={'checkbox'}
onClick={e => { onClick={e => {
@ -78,6 +92,7 @@ const EditSubuserModal = forwardRef<HTMLHeadingElement, Props>(({ subuser, ...pr
} }
}} }}
/> />
}
</div> </div>
} }
className={index !== 0 ? 'mt-4' : undefined} className={index !== 0 ? 'mt-4' : undefined}
@ -87,9 +102,11 @@ const EditSubuserModal = forwardRef<HTMLHeadingElement, Props>(({ subuser, ...pr
</p> </p>
{Object.keys(permissions[key].keys).map((pkey, index) => ( {Object.keys(permissions[key].keys).map((pkey, index) => (
<PermissionLabel <PermissionLabel
key={`permission_${key}_${pkey}`}
htmlFor={`permission_${key}_${pkey}`} htmlFor={`permission_${key}_${pkey}`}
className={classNames('transition-colors duration-75', { className={classNames('transition-colors duration-75', {
'mt-2': index !== 0, 'mt-2': index !== 0,
disabled: !canEditUser,
})} })}
> >
<div className={'p-2'}> <div className={'p-2'}>
@ -98,6 +115,7 @@ const EditSubuserModal = forwardRef<HTMLHeadingElement, Props>(({ subuser, ...pr
name={'permissions'} name={'permissions'}
value={`${key}.${pkey}`} value={`${key}.${pkey}`}
className={'w-5 h-5 mr-2'} className={'w-5 h-5 mr-2'}
disabled={!canEditUser}
/> />
</div> </div>
<div className={'flex-1'}> <div className={'flex-1'}>
@ -115,11 +133,13 @@ const EditSubuserModal = forwardRef<HTMLHeadingElement, Props>(({ subuser, ...pr
</TitledGreyBox> </TitledGreyBox>
))} ))}
</div> </div>
<div className={'mt-6 pb-6 flex justify-end'}> <Can action={subuser ? 'user.update' : 'user.delete'}>
<div className={'pb-6 flex justify-end'}>
<button className={'btn btn-primary btn-sm'} type={'submit'}> <button className={'btn btn-primary btn-sm'} type={'submit'}>
{subuser ? 'Save' : 'Invite User'} {subuser ? 'Save' : 'Invite User'}
</button> </button>
</div> </div>
</Can>
</Modal> </Modal>
); );
}); });

View File

@ -7,6 +7,7 @@ import EditSubuserModal from '@/components/server/users/EditSubuserModal';
import { faUnlockAlt } from '@fortawesome/free-solid-svg-icons/faUnlockAlt'; import { faUnlockAlt } from '@fortawesome/free-solid-svg-icons/faUnlockAlt';
import { faUserLock } from '@fortawesome/free-solid-svg-icons/faUserLock'; import { faUserLock } from '@fortawesome/free-solid-svg-icons/faUserLock';
import classNames from 'classnames'; import classNames from 'classnames';
import Can from '@/components/elements/Can';
interface Props { interface Props {
subuser: Subuser; subuser: Subuser;
@ -58,7 +59,9 @@ export default ({ subuser }: Props) => {
> >
<FontAwesomeIcon icon={faPencilAlt}/> <FontAwesomeIcon icon={faPencilAlt}/>
</button> </button>
<Can action={'user.delete'}>
<RemoveSubuserButton subuser={subuser}/> <RemoveSubuserButton subuser={subuser}/>
</Can>
</div> </div>
); );
}; };

View File

@ -8,6 +8,7 @@ import UserRow from '@/components/server/users/UserRow';
import FlashMessageRender from '@/components/FlashMessageRender'; import FlashMessageRender from '@/components/FlashMessageRender';
import getServerSubusers from '@/api/server/users/getServerSubusers'; import getServerSubusers from '@/api/server/users/getServerSubusers';
import { httpErrorToHuman } from '@/api/http'; import { httpErrorToHuman } from '@/api/http';
import Can from '@/components/elements/Can';
export default () => { export default () => {
const [ loading, setLoading ] = useState(true); const [ loading, setLoading ] = useState(true);
@ -42,7 +43,7 @@ export default () => {
} }
return ( return (
<div className={'my-10'}> <div className={'mt-10 mb-6'}>
<FlashMessageRender byKey={'users'} className={'mb-4'}/> <FlashMessageRender byKey={'users'} className={'mb-4'}/>
{!subusers.length ? {!subusers.length ?
<p className={'text-center text-sm text-neutral-400'}> <p className={'text-center text-sm text-neutral-400'}>
@ -53,9 +54,11 @@ export default () => {
<UserRow key={subuser.uuid} subuser={subuser}/> <UserRow key={subuser.uuid} subuser={subuser}/>
)) ))
} }
<Can action={'user.create'}>
<div className={'flex justify-end mt-6'}> <div className={'flex justify-end mt-6'}>
<AddSubuserButton/> <AddSubuserButton/>
</div> </div>
</Can>
</div> </div>
); );
}; };

View File

@ -0,0 +1,12 @@
import { useRef } from 'react';
import isEqual from 'lodash-es/isEqual';
export const useDeepMemo = <T, K> (fn: () => T, key: K): T => {
const ref = useRef<{ key: K, value: T }>();
if (!ref.current || !isEqual(key, ref.current.key)) {
ref.current = { key, value: fn() };
}
return ref.current.value;
};

View File

@ -0,0 +1,25 @@
import { ServerContext } from '@/state/server';
import { useDeepMemo } from '@/plugins/useDeepMemo';
export const usePermissions = (action: string | string[]): boolean[] => {
const userPermissions = ServerContext.useStoreState(state => state.server.permissions);
return useDeepMemo(() => {
if (userPermissions[0] === '*') {
return Array(Array.isArray(action) ? action.length : 1).fill(true);
}
return (Array.isArray(action) ? action : [ action ])
.map(permission => (
// Allows checking for any permission matching a name, for example files.*
// will return if the user has any permission under the file.XYZ namespace.
(
permission.endsWith('.*') &&
permission !== 'websocket.*' &&
userPermissions.filter(p => p.startsWith(permission.split('.')[0])).length > 0
) ||
// Otherwise just check if the entire permission exists in the array or not.
userPermissions.indexOf(permission) >= 0
));
}, [ action, userPermissions ]);
};

View File

@ -47,7 +47,7 @@ const ServerRouter = ({ match, location }: RouteComponentProps<{ id: string }>)
<Can action={'user.*'}> <Can action={'user.*'}>
<NavLink to={`${match.url}/users`}>Users</NavLink> <NavLink to={`${match.url}/users`}>Users</NavLink>
</Can> </Can>
<Can action={'settings.*'}> <Can action={['settings.*', 'file.sftp']} matchAny={true}>
<NavLink to={`${match.url}/settings`}>Settings</NavLink> <NavLink to={`${match.url}/settings`}>Settings</NavLink>
</Can> </Can>
</div> </div>

View File

@ -2,7 +2,7 @@ import { action, Action } from 'easy-peasy';
export type SubuserPermission = export type SubuserPermission =
'websocket.*' | 'websocket.*' |
'control.console' | 'control.start' | 'control.stop' | 'control.restart' | 'control.kill' | 'control.console' | 'control.start' | 'control.stop' | 'control.restart' |
'user.create' | 'user.read' | 'user.update' | 'user.delete' | 'user.create' | 'user.read' | 'user.update' | 'user.delete' |
'file.create' | 'file.read' | 'file.update' | 'file.delete' | 'file.archive' | 'file.sftp' | 'file.create' | 'file.read' | 'file.update' | 'file.delete' | 'file.archive' | 'file.sftp' |
'allocation.read' | 'allocation.update' | 'allocation.read' | 'allocation.update' |

View File

@ -4,7 +4,7 @@
transition: opacity 250ms ease; transition: opacity 250ms ease;
& > .modal-container { & > .modal-container {
@apply .relative .w-full .max-w-md .m-auto .flex-col .flex; @apply .relative .w-full .max-w-1/2 .m-auto .flex-col .flex;
/*&.top { /*&.top {
margin-top: 10%; margin-top: 10%;

View File

@ -30,25 +30,25 @@
<div class="form-group"> <div class="form-group">
<label for="email" class="control-label">Email</label> <label for="email" class="control-label">Email</label>
<div> <div>
<input readonly type="email" name="email" value="{{ $user->email }}" class="form-control form-autocomplete-stop"> <input type="email" name="email" value="{{ $user->email }}" class="form-control form-autocomplete-stop">
</div> </div>
</div> </div>
<div class="form-group"> <div class="form-group">
<label for="registered" class="control-label">Username</label> <label for="registered" class="control-label">Username</label>
<div> <div>
<input readonly type="text" name="username" value="{{ $user->username }}" class="form-control form-autocomplete-stop"> <input type="text" name="username" value="{{ $user->username }}" class="form-control form-autocomplete-stop">
</div> </div>
</div> </div>
<div class="form-group"> <div class="form-group">
<label for="registered" class="control-label">Client First Name</label> <label for="registered" class="control-label">Client First Name</label>
<div> <div>
<input readonly type="text" name="name_first" value="{{ $user->name_first }}" class="form-control form-autocomplete-stop"> <input type="text" name="name_first" value="{{ $user->name_first }}" class="form-control form-autocomplete-stop">
</div> </div>
</div> </div>
<div class="form-group"> <div class="form-group">
<label for="registered" class="control-label">Client Last Name</label> <label for="registered" class="control-label">Client Last Name</label>
<div> <div>
<input readonly type="text" name="name_last" value="{{ $user->name_last }}" class="form-control form-autocomplete-stop"> <input type="text" name="name_last" value="{{ $user->name_last }}" class="form-control form-autocomplete-stop">
</div> </div>
</div> </div>
<div class="form-group"> <div class="form-group">
@ -80,7 +80,7 @@
<div class="form-group no-margin-bottom"> <div class="form-group no-margin-bottom">
<label for="password" class="control-label">Password <span class="field-optional"></span></label> <label for="password" class="control-label">Password <span class="field-optional"></span></label>
<div> <div>
<input readonly type="password" id="password" name="password" class="form-control form-autocomplete-stop"> <input type="password" id="password" name="password" class="form-control form-autocomplete-stop">
<p class="text-muted small">Leave blank to keep this user's password the same. User will not receive any notification if password is changed.</p> <p class="text-muted small">Leave blank to keep this user's password the same. User will not receive any notification if password is changed.</p>
</div> </div>
</div> </div>

View File

@ -120,6 +120,52 @@ let colors = {
'green-900': 'hsl(125, 97%, 14%)', 'green-900': 'hsl(125, 97%, 14%)',
}; };
/*
|-------------------------------------------------------------------------------
| Sizes
|-------------------------------------------------------------------------------
|
| Here you can specify the sizes that should be available to all of the height,
| width, padding, and margin utilities.
|
*/
let sizes = {
'auto': 'auto',
'0': '0',
'px': '1px',
'1': '0.25rem',
'2': '0.5rem',
'3': '0.75rem',
'4': '1rem',
'5': '1.25rem',
'6': '1.5rem',
'8': '2rem',
'10': '2.5rem',
'12': '3rem',
'16': '4rem',
'20': '5rem',
'24': '6rem',
'28': '7rem',
'32': '8rem',
'48': '12rem',
'64': '16rem',
'96': '24rem',
'1/2': '50%',
'1/3': '33.33333%',
'2/3': '66.66667%',
'1/4': '25%',
'3/4': '75%',
'1/5': '20%',
'2/5': '40%',
'3/5': '60%',
'4/5': '80%',
'1/6': '16.66667%',
'5/6': '83.33333%',
'full': '100%',
'screen': '100vw',
};
module.exports = { module.exports = {
/* /*
@ -452,38 +498,7 @@ module.exports = {
| |
*/ */
width: { width: { ...sizes },
'auto': 'auto',
'px': '1px',
'1': '0.25rem',
'2': '0.5rem',
'3': '0.75rem',
'4': '1rem',
'5': '1.25rem',
'6': '1.5rem',
'8': '2rem',
'10': '2.5rem',
'12': '3rem',
'16': '4rem',
'24': '6rem',
'32': '8rem',
'48': '12rem',
'64': '16rem',
'96': '24rem',
'1/2': '50%',
'1/3': '33.33333%',
'2/3': '66.66667%',
'1/4': '25%',
'3/4': '75%',
'1/5': '20%',
'2/5': '40%',
'3/5': '60%',
'4/5': '80%',
'1/6': '16.66667%',
'5/6': '83.33333%',
'full': '100%',
'screen': '100vw',
},
/* /*
|----------------------------------------------------------------------------- |-----------------------------------------------------------------------------
@ -500,26 +515,7 @@ module.exports = {
| |
*/ */
height: { height: { ...sizes },
'auto': 'auto',
'px': '1px',
'1': '0.25rem',
'2': '0.5rem',
'3': '0.75rem',
'4': '1rem',
'5': '1.25rem',
'6': '1.5rem',
'8': '2rem',
'10': '2.5rem',
'12': '3rem',
'16': '4rem',
'24': '6rem',
'32': '8rem',
'48': '12rem',
'64': '16rem',
'full': '100%',
'screen': '100vh',
},
/* /*
|----------------------------------------------------------------------------- |-----------------------------------------------------------------------------
@ -535,25 +531,7 @@ module.exports = {
| |
*/ */
minWidth: { minWidth: { ...sizes },
'0': '0',
'1': '0.25rem',
'2': '0.5rem',
'3': '0.75rem',
'4': '1rem',
'5': '1.25rem',
'6': '1.5rem',
'8': '2rem',
'10': '2.5rem',
'12': '3rem',
'16': '4rem',
'24': '6rem',
'32': '8rem',
'48': '12rem',
'64': '16rem',
'96': '24rem',
'full': '100%',
},
/* /*
|----------------------------------------------------------------------------- |-----------------------------------------------------------------------------
@ -569,11 +547,7 @@ module.exports = {
| |
*/ */
minHeight: { minHeight: { ...sizes },
'0': '0',
'full': '100%',
'screen': '100vh',
},
/* /*
|----------------------------------------------------------------------------- |-----------------------------------------------------------------------------
@ -590,19 +564,7 @@ module.exports = {
| |
*/ */
maxWidth: { maxWidth: { ...sizes },
'xxs': '10rem',
'xs': '20rem',
'sm': '30rem',
'md': '40rem',
'lg': '50rem',
'xl': '60rem',
'2xl': '70rem',
'3xl': '80rem',
'4xl': '90rem',
'5xl': '100rem',
'full': '100%',
},
/* /*
|----------------------------------------------------------------------------- |-----------------------------------------------------------------------------
@ -618,10 +580,7 @@ module.exports = {
| |
*/ */
maxHeight: { maxHeight: { ...sizes },
'full': '100%',
'screen': '100vh',
},
/* /*
|----------------------------------------------------------------------------- |-----------------------------------------------------------------------------
@ -638,23 +597,7 @@ module.exports = {
| |
*/ */
padding: { padding: { ...sizes },
'px': '1px',
'0': '0',
'1': '0.25rem',
'2': '0.5rem',
'3': '0.75rem',
'4': '1rem',
'5': '1.25rem',
'6': '1.5rem',
'8': '2rem',
'10': '2.5rem',
'12': '3rem',
'16': '4rem',
'20': '5rem',
'24': '6rem',
'32': '8rem',
},
/* /*
|----------------------------------------------------------------------------- |-----------------------------------------------------------------------------
@ -671,33 +614,7 @@ module.exports = {
| |
*/ */
margin: { margin: { ...sizes },
'n-12': '-3rem',
'n-10': '-2.5rem',
'n-8': '-2rem',
'n-6': '-1.5rem',
'n-4': '-1rem',
'n-3': '-0.75rem',
'n-2': '-0.5rem',
'n-1': '-0.25rem',
'n-px': '-1px',
'auto': 'auto',
'px': '1px',
'0': '0',
'1': '0.25rem',
'2': '0.5rem',
'3': '0.75rem',
'4': '1rem',
'5': '1.25rem',
'6': '1.5rem',
'8': '2rem',
'10': '2.5rem',
'12': '3rem',
'16': '4rem',
'20': '5rem',
'24': '6rem',
'32': '8rem',
},
/* /*
|----------------------------------------------------------------------------- |-----------------------------------------------------------------------------
@ -714,23 +631,7 @@ module.exports = {
| |
*/ */
negativeMargin: { negativeMargin: { ...sizes },
'px': '1px',
'0': '0',
'1': '0.25rem',
'2': '0.5rem',
'3': '0.75rem',
'4': '1rem',
'5': '1.25rem',
'6': '1.5rem',
'8': '2rem',
'10': '2.5rem',
'12': '3rem',
'16': '4rem',
'20': '5rem',
'24': '6rem',
'32': '8rem',
},
/* /*
|----------------------------------------------------------------------------- |-----------------------------------------------------------------------------