Merge branch 'develop' into cputhreads
This commit is contained in:
commit
78d6e59fc5
1
.php_cs
1
.php_cs
|
@ -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,
|
||||||
|
|
|
@ -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';
|
||||||
|
|
|
@ -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.',
|
|
||||||
],
|
],
|
||||||
],
|
],
|
||||||
|
|
||||||
|
|
|
@ -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.
|
||||||
*
|
*
|
||||||
|
|
|
@ -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);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|
|
@ -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->repository->withoutFreshModel()->update($server->id, [
|
||||||
}
|
'installed' => Server::STATUS_INSTALLING,
|
||||||
|
]);
|
||||||
|
|
||||||
$this->database->beginTransaction();
|
|
||||||
$this->repository->withoutFreshModel()->update($server->id, [
|
|
||||||
'installed' => 0,
|
|
||||||
], 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();
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -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": {
|
||||||
|
|
File diff suppressed because it is too large
Load Diff
|
@ -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
|
||||||
}
|
}
|
||||||
</>
|
</>
|
||||||
);
|
);
|
||||||
|
|
|
@ -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>
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
|
@ -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,28 +110,36 @@ export default () => {
|
||||||
{cpu.toFixed(2)} %
|
{cpu.toFixed(2)} %
|
||||||
</p>
|
</p>
|
||||||
</TitledGreyBox>
|
</TitledGreyBox>
|
||||||
<div className={'grey-box justify-center'}>
|
<Can action={[ 'control.start', 'control.stop', 'control.restart' ]} matchAny={true}>
|
||||||
<button
|
<div className={'grey-box justify-center'}>
|
||||||
className={'btn btn-secondary btn-xs mr-2'}
|
<Can action={'control.start'}>
|
||||||
disabled={status !== 'offline'}
|
<button
|
||||||
onClick={e => {
|
className={'btn btn-secondary btn-xs mr-2'}
|
||||||
e.preventDefault();
|
disabled={status !== 'offline'}
|
||||||
sendPowerCommand('start');
|
onClick={e => {
|
||||||
}}
|
e.preventDefault();
|
||||||
>
|
sendPowerCommand('start');
|
||||||
Start
|
}}
|
||||||
</button>
|
>
|
||||||
<button
|
Start
|
||||||
className={'btn btn-secondary btn-xs mr-2'}
|
</button>
|
||||||
onClick={e => {
|
</Can>
|
||||||
e.preventDefault();
|
<Can action={'control.restart'}>
|
||||||
sendPowerCommand('restart');
|
<button
|
||||||
}}
|
className={'btn btn-secondary btn-xs mr-2'}
|
||||||
>
|
onClick={e => {
|
||||||
Restart
|
e.preventDefault();
|
||||||
</button>
|
sendPowerCommand('restart');
|
||||||
<StopOrKillButton onPress={action => sendPowerCommand(action)}/>
|
}}
|
||||||
</div>
|
>
|
||||||
|
Restart
|
||||||
|
</button>
|
||||||
|
</Can>
|
||||||
|
<Can action={'control.stop'}>
|
||||||
|
<StopOrKillButton onPress={action => sendPowerCommand(action)}/>
|
||||||
|
</Can>
|
||||||
|
</div>
|
||||||
|
</Can>
|
||||||
</div>
|
</div>
|
||||||
<div className={'flex-1 mx-4 mr-4'}>
|
<div className={'flex-1 mx-4 mr-4'}>
|
||||||
<SuspenseSpinner>
|
<SuspenseSpinner>
|
||||||
|
|
|
@ -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;
|
||||||
|
@ -24,10 +24,10 @@ interface Props {
|
||||||
}
|
}
|
||||||
|
|
||||||
export default ({ databaseId, className, onDelete }: Props) => {
|
export default ({ databaseId, className, onDelete }: Props) => {
|
||||||
const [visible, setVisible] = useState(false);
|
const [ visible, setVisible ] = useState(false);
|
||||||
const database = ServerContext.useStoreState(state => state.databases.items.find(item => item.id === databaseId));
|
const database = ServerContext.useStoreState(state => state.databases.items.find(item => item.id === databaseId));
|
||||||
const appendDatabase = ServerContext.useStoreActions(actions => actions.databases.appendDatabase);
|
const appendDatabase = ServerContext.useStoreActions(actions => actions.databases.appendDatabase);
|
||||||
const [connectionVisible, setConnectionVisible] = useState(false);
|
const [ connectionVisible, setConnectionVisible ] = useState(false);
|
||||||
const { addFlash, clearFlashes } = useStoreActions((actions: Actions<ApplicationStore>) => actions.flashes);
|
const { addFlash, clearFlashes } = useStoreActions((actions: Actions<ApplicationStore>) => actions.flashes);
|
||||||
const server = ServerContext.useStoreState(state => state.server.data!);
|
const server = ServerContext.useStoreState(state => state.server.data!);
|
||||||
|
|
||||||
|
@ -38,7 +38,7 @@ export default ({ databaseId, className, onDelete }: Props) => {
|
||||||
const schema = object().shape({
|
const schema = object().shape({
|
||||||
confirm: string()
|
confirm: string()
|
||||||
.required('The database name must be provided.')
|
.required('The database name must be provided.')
|
||||||
.oneOf([database.name.split('_', 2)[1], database.name], 'The database name must be provided.'),
|
.oneOf([ database.name.split('_', 2)[1], database.name ], 'The database name must be provided.'),
|
||||||
});
|
});
|
||||||
|
|
||||||
const submit = (values: { confirm: string }, { setSubmitting }: FormikHelpers<{ confirm: string }>) => {
|
const submit = (values: { confirm: string }, { setSubmitting }: FormikHelpers<{ confirm: string }>) => {
|
||||||
|
@ -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>
|
||||||
<div>
|
<Can action={'database.view_password'}>
|
||||||
<label className={'input-dark-label'}>Password</label>
|
<div>
|
||||||
<input type={'text'} className={'input-dark'} readOnly={true} value={database.password}/>
|
<label className={'input-dark-label'}>Password</label>
|
||||||
</div>
|
<input type={'text'} className={'input-dark'} readOnly={true} value={database.password}/>
|
||||||
|
</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'}>
|
||||||
<RotatePasswordButton databaseId={database.id} onUpdate={appendDatabase}/>
|
<Can action={'database.update'}>
|
||||||
|
<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>
|
||||||
<button className={'btn btn-sm btn-secondary btn-red'} onClick={() => setVisible(true)}>
|
<Can action={'database.delete'}>
|
||||||
<FontAwesomeIcon icon={faTrashAlt} fixedWidth={true}/>
|
<button className={'btn btn-sm btn-secondary btn-red'} onClick={() => setVisible(true)}>
|
||||||
</button>
|
<FontAwesomeIcon icon={faTrashAlt} fixedWidth={true}/>
|
||||||
|
</button>
|
||||||
|
</Can>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</React.Fragment>
|
</React.Fragment>
|
||||||
|
|
|
@ -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>
|
||||||
}
|
}
|
||||||
{server.featureLimits.databases > 0 &&
|
<Can action={'database.create'}>
|
||||||
<div className={'mt-6 flex justify-end'}>
|
{server.featureLimits.databases > 0 &&
|
||||||
<CreateDatabaseButton onCreated={appendDatabase}/>
|
<div className={'mt-6 flex justify-end'}>
|
||||||
</div>
|
<CreateDatabaseButton onCreated={appendDatabase}/>
|
||||||
}
|
</div>
|
||||||
</React.Fragment>
|
}
|
||||||
|
</Can>
|
||||||
|
</>
|
||||||
</CSSTransition>
|
</CSSTransition>
|
||||||
}
|
}
|
||||||
</div>
|
</div>
|
||||||
|
|
|
@ -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,30 +119,37 @@ 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'}
|
||||||
>
|
>
|
||||||
<div
|
<Can action={'file.update'}>
|
||||||
onClick={() => setModal('rename')}
|
<div
|
||||||
className={'hover:text-neutral-700 p-2 flex items-center hover:bg-neutral-100 rounded'}
|
onClick={() => setModal('rename')}
|
||||||
>
|
className={'hover:text-neutral-700 p-2 flex items-center hover:bg-neutral-100 rounded'}
|
||||||
<FontAwesomeIcon icon={faPencilAlt} className={'text-xs'}/>
|
>
|
||||||
<span className={'ml-2'}>Rename</span>
|
<FontAwesomeIcon icon={faPencilAlt} className={'text-xs'}/>
|
||||||
</div>
|
<span className={'ml-2'}>Rename</span>
|
||||||
<div
|
</div>
|
||||||
onClick={() => setModal('move')}
|
<div
|
||||||
className={'hover:text-neutral-700 p-2 flex items-center hover:bg-neutral-100 rounded'}
|
onClick={() => setModal('move')}
|
||||||
>
|
className={'hover:text-neutral-700 p-2 flex items-center hover:bg-neutral-100 rounded'}
|
||||||
<FontAwesomeIcon icon={faLevelUpAlt} className={'text-xs'}/>
|
>
|
||||||
<span className={'ml-2'}>Move</span>
|
<FontAwesomeIcon icon={faLevelUpAlt} className={'text-xs'}/>
|
||||||
</div>
|
<span className={'ml-2'}>Move</span>
|
||||||
<div
|
</div>
|
||||||
onClick={() => doCopy()}
|
</Can>
|
||||||
className={'hover:text-neutral-700 p-2 flex items-center hover:bg-neutral-100 rounded'}
|
<Can action={'file.create'}>
|
||||||
>
|
<div
|
||||||
<FontAwesomeIcon icon={faCopy} className={'text-xs'}/>
|
onClick={() => doCopy()}
|
||||||
<span className={'ml-2'}>Copy</span>
|
className={'hover:text-neutral-700 p-2 flex items-center hover:bg-neutral-100 rounded'}
|
||||||
</div>
|
>
|
||||||
|
<FontAwesomeIcon icon={faCopy} className={'text-xs'}/>
|
||||||
|
<span className={'ml-2'}>Copy</span>
|
||||||
|
</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,13 +157,15 @@ 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>
|
||||||
<div
|
<Can action={'file.delete'}>
|
||||||
onClick={() => doDeletion()}
|
<div
|
||||||
className={'hover:text-red-700 p-2 flex items-center hover:bg-red-100 rounded'}
|
onClick={() => doDeletion()}
|
||||||
>
|
className={'hover:text-red-700 p-2 flex items-center hover:bg-red-100 rounded'}
|
||||||
<FontAwesomeIcon icon={faTrashAlt} className={'text-xs'}/>
|
>
|
||||||
<span className={'ml-2'}>Delete</span>
|
<FontAwesomeIcon icon={faTrashAlt} className={'text-xs'}/>
|
||||||
</div>
|
<span className={'ml-2'}>Delete</span>
|
||||||
|
</div>
|
||||||
|
</Can>
|
||||||
</div>
|
</div>
|
||||||
</CSSTransition>
|
</CSSTransition>
|
||||||
</div>
|
</div>
|
||||||
|
|
|
@ -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,10 +47,10 @@ 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(() => {
|
||||||
if (name) {
|
if (name) {
|
||||||
history.push(`/server/${id}/files/edit#/${name}`);
|
history.push(`/server/${id}/files/edit#/${name}`);
|
||||||
|
@ -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' ?
|
||||||
<button className={'btn btn-primary btn-sm'} onClick={() => save()}>
|
<Can action={'file.update'}>
|
||||||
Save Content
|
<button className={'btn btn-primary btn-sm'} onClick={() => save()}>
|
||||||
</button>
|
Save Content
|
||||||
|
</button>
|
||||||
|
</Can>
|
||||||
:
|
:
|
||||||
<button className={'btn btn-primary btn-sm'} onClick={() => setModalVisible(true)}>
|
<Can action={'file.create'}>
|
||||||
Create File
|
<button className={'btn btn-primary btn-sm'} onClick={() => setModalVisible(true)}>
|
||||||
</button>
|
Create File
|
||||||
|
</button>
|
||||||
|
</Can>
|
||||||
}
|
}
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
|
@ -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>
|
||||||
}
|
}
|
||||||
<div className={'flex justify-end mt-8'}>
|
<Can action={'file.create'}>
|
||||||
<NewDirectoryButton/>
|
<div className={'flex justify-end mt-8'}>
|
||||||
<Link to={`/server/${id}/files/new${window.location.hash}`} className={'btn btn-sm btn-primary'}>
|
<NewDirectoryButton/>
|
||||||
New File
|
<Link
|
||||||
</Link>
|
to={`/server/${id}/files/new${window.location.hash}`}
|
||||||
</div>
|
className={'btn btn-sm btn-primary'}
|
||||||
|
>
|
||||||
|
New File
|
||||||
|
</Link>
|
||||||
|
</div>
|
||||||
|
</Can>
|
||||||
</React.Fragment>
|
</React.Fragment>
|
||||||
}
|
}
|
||||||
</React.Fragment>
|
</React.Fragment>
|
||||||
|
|
|
@ -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,21 +54,23 @@ export default ({ match, history }: RouteComponentProps) => {
|
||||||
</a>
|
</a>
|
||||||
))
|
))
|
||||||
}
|
}
|
||||||
<div className={'mt-8 flex justify-end'}>
|
<Can action={'schedule.create'}>
|
||||||
{visible && <EditScheduleModal
|
<div className={'mt-8 flex justify-end'}>
|
||||||
appear={true}
|
{visible && <EditScheduleModal
|
||||||
visible={true}
|
appear={true}
|
||||||
onScheduleUpdated={schedule => setSchedules(s => [...(s || []), schedule])}
|
visible={true}
|
||||||
onDismissed={() => setVisible(false)}
|
onScheduleUpdated={schedule => setSchedules(s => [ ...(s || []), schedule ])}
|
||||||
/>}
|
onDismissed={() => setVisible(false)}
|
||||||
<button
|
/>}
|
||||||
type={'button'}
|
<button
|
||||||
className={'btn btn-lg btn-primary'}
|
type={'button'}
|
||||||
onClick={() => setVisible(true)}
|
className={'btn btn-sm btn-primary'}
|
||||||
>
|
onClick={() => setVisible(true)}
|
||||||
Create schedule
|
>
|
||||||
</button>
|
Create schedule
|
||||||
</div>
|
</button>
|
||||||
|
</div>
|
||||||
|
</Can>
|
||||||
</>
|
</>
|
||||||
}
|
}
|
||||||
</div>
|
</div>
|
||||||
|
|
|
@ -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,24 +94,27 @@ 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'}>
|
||||||
<DeleteScheduleButton
|
<Can action={'schedule.delete'}>
|
||||||
scheduleId={schedule.id}
|
<DeleteScheduleButton
|
||||||
onDeleted={() => history.push(`/server/${id}/schedules`)}
|
scheduleId={schedule.id}
|
||||||
/>
|
onDeleted={() => history.push(`/server/${id}/schedules`)}
|
||||||
<button className={'btn btn-primary btn-sm mr-4'} onClick={() => setShowEditModal(true)}>
|
/>
|
||||||
Edit
|
</Can>
|
||||||
</button>
|
<Can action={'schedule.update'}>
|
||||||
<NewTaskButton
|
<button className={'btn btn-primary btn-sm mr-4'} onClick={() => setShowEditModal(true)}>
|
||||||
scheduleId={schedule.id}
|
Edit
|
||||||
onTaskAdded={task => setSchedule(s => ({
|
</button>
|
||||||
...s!, tasks: [ ...s!.tasks, task ],
|
<NewTaskButton
|
||||||
}))}
|
scheduleId={schedule.id}
|
||||||
/>
|
onTaskAdded={task => setSchedule(s => ({
|
||||||
|
...s!, tasks: [ ...s!.tasks, task ],
|
||||||
|
}))}
|
||||||
|
/>
|
||||||
|
</Can>
|
||||||
</div>
|
</div>
|
||||||
</>
|
</>
|
||||||
}
|
}
|
||||||
|
|
|
@ -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,22 +76,26 @@ export default ({ schedule, task, onTaskUpdated, onTaskRemoved }: Props) => {
|
||||||
</p>
|
</p>
|
||||||
</div>
|
</div>
|
||||||
}
|
}
|
||||||
<button
|
<Can action={'schedule.update'}>
|
||||||
type={'button'}
|
<button
|
||||||
aria-label={'Edit scheduled task'}
|
type={'button'}
|
||||||
className={'block text-sm p-2 text-neutral-500 hover:text-neutral-100 transition-colors duration-150 mr-4'}
|
aria-label={'Edit scheduled task'}
|
||||||
onClick={() => setIsEditing(true)}
|
className={'block text-sm p-2 text-neutral-500 hover:text-neutral-100 transition-colors duration-150 mr-4'}
|
||||||
>
|
onClick={() => setIsEditing(true)}
|
||||||
<FontAwesomeIcon icon={faPencilAlt}/>
|
>
|
||||||
</button>
|
<FontAwesomeIcon icon={faPencilAlt}/>
|
||||||
<button
|
</button>
|
||||||
type={'button'}
|
</Can>
|
||||||
aria-label={'Delete scheduled task'}
|
<Can action={'schedule.update'}>
|
||||||
className={'block text-sm p-2 text-neutral-500 hover:text-red-600 transition-colors duration-150'}
|
<button
|
||||||
onClick={() => setVisible(true)}
|
type={'button'}
|
||||||
>
|
aria-label={'Delete scheduled task'}
|
||||||
<FontAwesomeIcon icon={faTrashAlt}/>
|
className={'block text-sm p-2 text-neutral-500 hover:text-red-600 transition-colors duration-150'}
|
||||||
</button>
|
onClick={() => setVisible(true)}
|
||||||
|
>
|
||||||
|
<FontAwesomeIcon icon={faTrashAlt}/>
|
||||||
|
</button>
|
||||||
|
</Can>
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
|
@ -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,46 +16,50 @@ 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'}>
|
||||||
<div>
|
<TitledGreyBox title={'SFTP Details'} className={'w-full md:flex-1 md:max-w-1/2 md:mr-6'}>
|
||||||
<label className={'input-dark-label'}>Server Address</label>
|
<div>
|
||||||
<input
|
<label className={'input-dark-label'}>Server Address</label>
|
||||||
type={'text'}
|
<input
|
||||||
className={'input-dark'}
|
type={'text'}
|
||||||
value={`sftp://${server.sftpDetails.ip}:${server.sftpDetails.port}`}
|
className={'input-dark'}
|
||||||
readOnly={true}
|
value={`sftp://${server.sftpDetails.ip}:${server.sftpDetails.port}`}
|
||||||
/>
|
readOnly={true}
|
||||||
</div>
|
/>
|
||||||
<div className={'mt-6'}>
|
</div>
|
||||||
<label className={'input-dark-label'}>Username</label>
|
<div className={'mt-6'}>
|
||||||
<input
|
<label className={'input-dark-label'}>Username</label>
|
||||||
type={'text'}
|
<input
|
||||||
className={'input-dark'}
|
type={'text'}
|
||||||
value={`${user.username}.${server.id}`}
|
className={'input-dark'}
|
||||||
readOnly={true}
|
value={`${user.username}.${server.id}`}
|
||||||
/>
|
readOnly={true}
|
||||||
</div>
|
/>
|
||||||
<div className={'mt-6 flex items-center'}>
|
</div>
|
||||||
<div className={'flex-1'}>
|
<div className={'mt-6 flex items-center'}>
|
||||||
<div className={'border-l-4 border-cyan-500 p-3'}>
|
<div className={'flex-1'}>
|
||||||
<p className={'text-xs text-neutral-200'}>
|
<div className={'border-l-4 border-cyan-500 p-3'}>
|
||||||
Your SFTP password is the same as the password you use to access this panel.
|
<p className={'text-xs text-neutral-200'}>
|
||||||
</p>
|
Your SFTP password is the same as the password you use to access this panel.
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div className={'ml-4'}>
|
||||||
|
<a
|
||||||
|
href={`sftp://${user.username}.${server.id}@${server.sftpDetails.ip}:${server.sftpDetails.port}`}
|
||||||
|
className={'btn btn-sm btn-secondary'}
|
||||||
|
>
|
||||||
|
Launch SFTP
|
||||||
|
</a>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div className={'ml-4'}>
|
</TitledGreyBox>
|
||||||
<a
|
</Can>
|
||||||
href={`sftp://${user.username}.${server.id}@${server.sftpDetails.ip}:${server.sftpDetails.port}`}
|
<Can action={'settings.rename'}>
|
||||||
className={'btn btn-sm btn-secondary'}
|
<div className={'w-full mt-6 md:flex-1 md:max-w-1/2 md:mt-0'}>
|
||||||
>
|
<RenameServerBox/>
|
||||||
Launch SFTP
|
|
||||||
</a>
|
|
||||||
</div>
|
|
||||||
</div>
|
</div>
|
||||||
</TitledGreyBox>
|
</Can>
|
||||||
<div className={'w-full mt-6 md:flex-1 md:mt-0'}>
|
|
||||||
<RenameServerBox/>
|
|
||||||
</div>
|
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
|
|
|
@ -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;
|
||||||
|
|
||||||
&:hover {
|
&:not(.disabled) {
|
||||||
${tw`border-neutral-500 bg-neutral-800`};
|
${tw`cursor-pointer`};
|
||||||
|
|
||||||
|
&:hover {
|
||||||
|
${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'}>
|
||||||
<button className={'btn btn-primary btn-sm'} type={'submit'}>
|
<div className={'pb-6 flex justify-end'}>
|
||||||
{subuser ? 'Save' : 'Invite User'}
|
<button className={'btn btn-primary btn-sm'} type={'submit'}>
|
||||||
</button>
|
{subuser ? 'Save' : 'Invite User'}
|
||||||
</div>
|
</button>
|
||||||
|
</div>
|
||||||
|
</Can>
|
||||||
</Modal>
|
</Modal>
|
||||||
);
|
);
|
||||||
});
|
});
|
||||||
|
|
|
@ -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>
|
||||||
<RemoveSubuserButton subuser={subuser}/>
|
<Can action={'user.delete'}>
|
||||||
|
<RemoveSubuserButton subuser={subuser}/>
|
||||||
|
</Can>
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
|
@ -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}/>
|
||||||
))
|
))
|
||||||
}
|
}
|
||||||
<div className={'flex justify-end mt-6'}>
|
<Can action={'user.create'}>
|
||||||
<AddSubuserButton/>
|
<div className={'flex justify-end mt-6'}>
|
||||||
</div>
|
<AddSubuserButton/>
|
||||||
|
</div>
|
||||||
|
</Can>
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
|
@ -5,7 +5,7 @@ export function bytesToHuman (bytes: number): string {
|
||||||
|
|
||||||
const i = Math.floor(Math.log(bytes) / Math.log(1000));
|
const i = Math.floor(Math.log(bytes) / Math.log(1000));
|
||||||
// @ts-ignore
|
// @ts-ignore
|
||||||
return `${(bytes / Math.pow(1000, i)).toFixed(2) * 1} ${['Bytes', 'kB', 'MB', 'GB', 'TB'][i]}`;
|
return `${(bytes / Math.pow(1000, i)).toFixed(2) * 1} ${[ 'Bytes', 'kB', 'MB', 'GB', 'TB' ][i]}`;
|
||||||
}
|
}
|
||||||
|
|
||||||
export const bytesToMegabytes = (bytes: number) => Math.floor(bytes / 1000 / 1000);
|
export const bytesToMegabytes = (bytes: number) => Math.floor(bytes / 1000 / 1000);
|
||||||
|
|
|
@ -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;
|
||||||
|
};
|
|
@ -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 ]);
|
||||||
|
};
|
|
@ -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>
|
||||||
|
|
|
@ -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' |
|
||||||
|
|
|
@ -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%;
|
||||||
|
|
|
@ -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>
|
||||||
|
|
209
tailwind.js
209
tailwind.js
|
@ -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',
|
|
||||||
},
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|-----------------------------------------------------------------------------
|
|-----------------------------------------------------------------------------
|
||||||
|
|
Loading…
Reference in New Issue