Files
dockhand/routes/settings/auth/roles/RoleModal.svelte
Jarek Krochmalski 62e3c6439e Initial commit
2025-12-28 21:16:03 +01:00

633 lines
22 KiB
Svelte

<script lang="ts">
import { Button } from '$lib/components/ui/button';
import * as Dialog from '$lib/components/ui/dialog';
import { Label } from '$lib/components/ui/label';
import { Input } from '$lib/components/ui/input';
import { Checkbox } from '$lib/components/ui/checkbox';
import { TogglePill } from '$lib/components/ui/toggle-pill';
import { Shield, Pencil, Plus, Check, RefreshCw, Box, Image, HardDrive, Cable, Layers, Globe, Download, Bell, Sliders, Settings, Users, Eye, SquarePlus, Play, Square, RotateCcw, Trash2, Terminal, ScrollText, Search, Upload, Plug, Unplug, Copy, GitBranch, KeyRound, Building2, Container, TriangleAlert, ClipboardList, Activity, Timer } from 'lucide-svelte';
import { getIconComponent } from '$lib/utils/icons';
import * as Alert from '$lib/components/ui/alert';
import { focusFirstInput } from '$lib/utils';
export interface Role {
id: number;
name: string;
description?: string;
is_system: boolean;
permissions: any;
environmentIds?: number[] | null;
created_at: string;
}
interface Environment {
id: number;
name: string;
icon?: string;
}
interface Props {
open: boolean;
role?: Role | null;
copyFrom?: Role | null;
environments?: Environment[];
onClose: () => void;
onSaved: () => void;
}
let { open = $bindable(), role = null, copyFrom = null, environments = [], onClose, onSaved }: Props = $props();
const isEditing = $derived(role !== null);
const isCopying = $derived(copyFrom !== null);
// Form state
let formName = $state('');
let formDescription = $state('');
let formError = $state('');
let formErrors = $state<{ name?: string }>({});
let formSaving = $state(false);
let formAllEnvironments = $state(false); // true = applies to all envs, false = specific envs
let formEnvironmentIds = $state<number[]>([]); // selected env IDs when not all
let formPermissions = $state<{
containers: string[];
images: string[];
volumes: string[];
networks: string[];
stacks: string[];
environments: string[];
registries: string[];
notifications: string[];
configsets: string[];
settings: string[];
users: string[];
git: string[];
license: string[];
audit_logs: string[];
activity: string[];
schedules: string[];
}>({
containers: [],
images: [],
volumes: [],
networks: [],
stacks: [],
environments: [],
registries: [],
notifications: [],
configsets: [],
settings: [],
users: [],
git: [],
license: [],
audit_logs: [],
activity: [],
schedules: []
});
// Permission definitions - separated into system and environment categories
const systemPermissions = {
users: [
{ key: 'view', label: 'View users' },
{ key: 'create', label: 'Create users' },
{ key: 'edit', label: 'Edit users' },
{ key: 'delete', label: 'Delete users' }
],
settings: [
{ key: 'view', label: 'View settings' },
{ key: 'edit', label: 'Edit settings' }
],
environments: [
{ key: 'view', label: 'View environments' },
{ key: 'create', label: 'Create environments' },
{ key: 'edit', label: 'Edit environments' },
{ key: 'delete', label: 'Delete environments' }
],
registries: [
{ key: 'view', label: 'View registries' },
{ key: 'create', label: 'Create registries' },
{ key: 'edit', label: 'Edit registries' },
{ key: 'delete', label: 'Delete registries' }
],
notifications: [
{ key: 'view', label: 'View notifications' },
{ key: 'create', label: 'Create notifications' },
{ key: 'edit', label: 'Edit notifications' },
{ key: 'delete', label: 'Delete notifications' },
{ key: 'test', label: 'Test notifications' }
],
configsets: [
{ key: 'view', label: 'View config sets' },
{ key: 'create', label: 'Create config sets' },
{ key: 'edit', label: 'Edit config sets' },
{ key: 'delete', label: 'Delete config sets' }
],
git: [
{ key: 'view', label: 'View credentials & repos' },
{ key: 'create', label: 'Create credentials & repos' },
{ key: 'edit', label: 'Edit credentials & repos' },
{ key: 'delete', label: 'Delete credentials & repos' }
],
license: [
{ key: 'manage', label: 'Manage license' }
],
audit_logs: [
{ key: 'view', label: 'View audit logs' }
],
schedules: [
{ key: 'view', label: 'View schedules' }
]
};
const environmentPermissions = {
activity: [
{ key: 'view', label: 'View activity' }
],
containers: [
{ key: 'view', label: 'View containers' },
{ key: 'create', label: 'Create containers' },
{ key: 'start', label: 'Start containers' },
{ key: 'stop', label: 'Stop containers' },
{ key: 'restart', label: 'Restart containers' },
{ key: 'remove', label: 'Remove containers' },
{ key: 'exec', label: 'Execute terminal' },
{ key: 'logs', label: 'View logs' },
{ key: 'inspect', label: 'Inspect containers' }
],
images: [
{ key: 'view', label: 'View images' },
{ key: 'pull', label: 'Pull images' },
{ key: 'push', label: 'Push images' },
{ key: 'remove', label: 'Remove images' },
{ key: 'build', label: 'Build images' },
{ key: 'inspect', label: 'Inspect images' }
],
volumes: [
{ key: 'view', label: 'View volumes' },
{ key: 'create', label: 'Create volumes' },
{ key: 'remove', label: 'Remove volumes' },
{ key: 'inspect', label: 'Inspect volumes' }
],
networks: [
{ key: 'view', label: 'View networks' },
{ key: 'create', label: 'Create networks' },
{ key: 'remove', label: 'Remove networks' },
{ key: 'inspect', label: 'Inspect networks' },
{ key: 'connect', label: 'Connect containers' },
{ key: 'disconnect', label: 'Disconnect containers' }
],
stacks: [
{ key: 'view', label: 'View stacks' },
{ key: 'create', label: 'Create stacks' },
{ key: 'start', label: 'Start stacks' },
{ key: 'stop', label: 'Stop stacks' },
{ key: 'remove', label: 'Remove stacks' },
{ key: 'edit', label: 'Edit stacks' }
]
};
const categoryIcons: Record<string, typeof Box> = {
containers: Container,
images: Image,
volumes: HardDrive,
networks: Cable,
stacks: Layers,
environments: Globe,
registries: Download,
notifications: Bell,
configsets: Sliders,
settings: Settings,
users: Users,
git: GitBranch,
license: KeyRound,
audit_logs: ClipboardList,
activity: Activity,
schedules: Timer
};
const categoryColorsSolid: Record<string, string> = {
containers: 'bg-blue-100 dark:bg-blue-950 text-blue-700 dark:text-blue-400 border-blue-300 dark:border-blue-800',
images: 'bg-purple-100 dark:bg-purple-950 text-purple-700 dark:text-purple-400 border-purple-300 dark:border-purple-800',
volumes: 'bg-amber-100 dark:bg-amber-950 text-amber-700 dark:text-amber-400 border-amber-300 dark:border-amber-800',
networks: 'bg-green-100 dark:bg-green-950 text-green-700 dark:text-green-400 border-green-300 dark:border-green-800',
stacks: 'bg-cyan-100 dark:bg-cyan-950 text-cyan-700 dark:text-cyan-400 border-cyan-300 dark:border-cyan-800',
environments: 'bg-indigo-100 dark:bg-indigo-950 text-indigo-700 dark:text-indigo-400 border-indigo-300 dark:border-indigo-800',
registries: 'bg-pink-100 dark:bg-pink-950 text-pink-700 dark:text-pink-400 border-pink-300 dark:border-pink-800',
notifications: 'bg-orange-100 dark:bg-orange-950 text-orange-700 dark:text-orange-400 border-orange-300 dark:border-orange-800',
configsets: 'bg-teal-100 dark:bg-teal-950 text-teal-700 dark:text-teal-400 border-teal-300 dark:border-teal-800',
settings: 'bg-slate-100 dark:bg-slate-900 text-slate-700 dark:text-slate-400 border-slate-300 dark:border-slate-700',
users: 'bg-rose-100 dark:bg-rose-950 text-rose-700 dark:text-rose-400 border-rose-300 dark:border-rose-800',
git: 'bg-violet-100 dark:bg-violet-950 text-violet-700 dark:text-violet-400 border-violet-300 dark:border-violet-800',
license: 'bg-yellow-100 dark:bg-yellow-950 text-yellow-700 dark:text-yellow-400 border-yellow-300 dark:border-yellow-800',
audit_logs: 'bg-stone-100 dark:bg-stone-950 text-stone-700 dark:text-stone-400 border-stone-300 dark:border-stone-800',
activity: 'bg-emerald-100 dark:bg-emerald-950 text-emerald-700 dark:text-emerald-400 border-emerald-300 dark:border-emerald-800',
schedules: 'bg-sky-100 dark:bg-sky-950 text-sky-700 dark:text-sky-400 border-sky-300 dark:border-sky-800'
};
const permissionIcons: Record<string, typeof Eye> = {
view: Eye,
create: SquarePlus,
start: Play,
stop: Square,
restart: RotateCcw,
remove: Trash2,
delete: Trash2,
exec: Terminal,
logs: ScrollText,
inspect: Search,
pull: Download,
push: Upload,
build: Box,
connect: Plug,
disconnect: Unplug,
edit: Pencil,
test: Play,
manage: Settings
};
function resetForm() {
formName = '';
formDescription = '';
formError = '';
formErrors = {};
formSaving = false;
formAllEnvironments = false;
formEnvironmentIds = [];
formPermissions = {
containers: [],
images: [],
volumes: [],
networks: [],
stacks: [],
environments: [],
registries: [],
notifications: [],
configsets: [],
settings: [],
users: [],
git: [],
license: [],
audit_logs: [],
activity: [],
schedules: []
};
}
// Initialize form when role changes or modal opens
$effect(() => {
if (open) {
if (role) {
// Editing existing role
formName = role.name;
formDescription = role.description || '';
// Environment scope: null = all environments
formAllEnvironments = role.environmentIds === null || role.environmentIds === undefined;
formEnvironmentIds = role.environmentIds ? [...role.environmentIds] : [];
formPermissions = {
containers: [...(role.permissions.containers || [])],
images: [...(role.permissions.images || [])],
volumes: [...(role.permissions.volumes || [])],
networks: [...(role.permissions.networks || [])],
stacks: [...(role.permissions.stacks || [])],
environments: [...(role.permissions.environments || [])],
registries: [...(role.permissions.registries || [])],
notifications: [...(role.permissions.notifications || [])],
configsets: [...(role.permissions.configsets || [])],
settings: [...(role.permissions.settings || [])],
users: [...(role.permissions.users || [])],
git: [...(role.permissions.git || [])],
license: [...(role.permissions.license || [])],
audit_logs: [...(role.permissions.audit_logs || [])],
activity: [...(role.permissions.activity || [])],
schedules: [...(role.permissions.schedules || [])]
};
formError = '';
formErrors = {};
formSaving = false;
} else if (copyFrom) {
// Copying from existing role - new role with pre-filled permissions
formName = `${copyFrom.name} (copy)`;
formDescription = copyFrom.description || '';
// Copy environment scope from source role
formAllEnvironments = copyFrom.environmentIds === null || copyFrom.environmentIds === undefined;
formEnvironmentIds = copyFrom.environmentIds ? [...copyFrom.environmentIds] : [];
formPermissions = {
containers: [...(copyFrom.permissions.containers || [])],
images: [...(copyFrom.permissions.images || [])],
volumes: [...(copyFrom.permissions.volumes || [])],
networks: [...(copyFrom.permissions.networks || [])],
stacks: [...(copyFrom.permissions.stacks || [])],
environments: [...(copyFrom.permissions.environments || [])],
registries: [...(copyFrom.permissions.registries || [])],
notifications: [...(copyFrom.permissions.notifications || [])],
configsets: [...(copyFrom.permissions.configsets || [])],
settings: [...(copyFrom.permissions.settings || [])],
users: [...(copyFrom.permissions.users || [])],
git: [...(copyFrom.permissions.git || [])],
license: [...(copyFrom.permissions.license || [])],
audit_logs: [...(copyFrom.permissions.audit_logs || [])],
activity: [...(copyFrom.permissions.activity || [])],
schedules: [...(copyFrom.permissions.schedules || [])]
};
formError = '';
formErrors = {};
formSaving = false;
} else {
resetForm();
}
}
});
function togglePermission(category: keyof typeof formPermissions, permission: string) {
const current = formPermissions[category];
if (current.includes(permission)) {
formPermissions[category] = current.filter(p => p !== permission);
} else {
formPermissions[category] = [...current, permission];
}
}
function toggleAllPermissions(category: keyof typeof formPermissions, enable: boolean, definitions: { key: string; label: string }[]) {
if (enable) {
formPermissions[category] = definitions.map(p => p.key);
} else {
formPermissions[category] = [];
}
}
function toggleEnvironment(envId: number) {
if (formEnvironmentIds.includes(envId)) {
formEnvironmentIds = formEnvironmentIds.filter(id => id !== envId);
} else {
formEnvironmentIds = [...formEnvironmentIds, envId];
}
}
async function save() {
formErrors = {};
if (!formName.trim()) {
formErrors.name = 'Role name is required';
return;
}
formSaving = true;
formError = '';
try {
const url = isEditing ? `/api/roles/${role!.id}` : '/api/roles';
const method = isEditing ? 'PUT' : 'POST';
// null = all environments, array = specific environments
const environmentIds = formAllEnvironments ? null : formEnvironmentIds;
const response = await fetch(url, {
method,
headers: { 'Content-Type': 'application/json' },
body: JSON.stringify({
name: formName.trim(),
description: formDescription.trim() || (isEditing ? null : undefined),
permissions: formPermissions,
environmentIds
})
});
if (response.ok) {
open = false;
onSaved();
} else {
const data = await response.json();
if (data.error?.includes('already exists')) {
formErrors.name = 'Role name already exists';
} else {
formError = data.error || `Failed to ${isEditing ? 'update' : 'create'} role`;
}
}
} catch {
formError = `Failed to ${isEditing ? 'update' : 'create'} role`;
} finally {
formSaving = false;
}
}
function handleClose() {
open = false;
onClose();
}
</script>
<Dialog.Root bind:open onOpenChange={(o) => { if (o) { formError = ''; formErrors = {}; focusFirstInput(); } }}>
<Dialog.Content class="max-w-6xl max-h-[85vh] flex flex-col overflow-hidden">
<Dialog.Header class="flex-shrink-0">
<Dialog.Title class="flex items-center gap-2">
{#if isEditing}
<Pencil class="w-5 h-5" />
Edit role
{:else if isCopying}
<Copy class="w-5 h-5" />
Copy role
{:else}
<Shield class="w-5 h-5" />
Create role
{/if}
</Dialog.Title>
<Dialog.Description>
{#if isEditing}
Update role permissions
{:else if isCopying}
Create a new role based on "{copyFrom?.name}"
{:else}
Define a new role with specific permissions
{/if}
</Dialog.Description>
</Dialog.Header>
{#if formError}
<Alert.Root variant="destructive" class="flex-shrink-0">
<TriangleAlert class="h-4 w-4" />
<Alert.Description>{formError}</Alert.Description>
</Alert.Root>
{/if}
<div class="flex-shrink-0 grid grid-cols-2 gap-4 py-4">
<div class="space-y-2">
<Label>Role name</Label>
<Input
bind:value={formName}
placeholder="Developer"
class={formErrors.name ? 'border-destructive focus-visible:ring-destructive' : ''}
oninput={() => formErrors.name = undefined}
/>
{#if formErrors.name}
<p class="text-xs text-destructive">{formErrors.name}</p>
{/if}
</div>
<div class="space-y-2">
<Label>Description (optional)</Label>
<Input
bind:value={formDescription}
placeholder="Access to development resources"
/>
</div>
</div>
<!-- Vertically stacked permissions layout -->
<div class="flex-1 flex flex-col gap-4 min-h-0 overflow-y-auto pr-1">
<!-- System Permissions Section -->
<div class="flex-shrink-0 border rounded-lg">
<div class="px-4 py-3 border-b bg-muted/30">
<div class="flex items-center gap-2">
<Building2 class="w-4 h-4" />
<span class="font-medium text-sm">System permissions</span>
<span class="text-xs text-muted-foreground">(always global)</span>
</div>
</div>
<div class="p-3 grid grid-cols-2 lg:grid-cols-4 gap-3">
{#each Object.entries(systemPermissions) as [category, permissions]}
{@const IconComponent = categoryIcons[category]}
<div class="relative border rounded-md pt-5 pb-3 px-3">
<!-- Category pill on border -->
<div class="absolute -top-2.5 left-3 inline-flex items-center gap-1.5 px-2 py-0.5 rounded border {categoryColorsSolid[category] || 'bg-gray-100 dark:bg-gray-900 text-gray-700 dark:text-gray-400 border-gray-300 dark:border-gray-700'}">
<IconComponent class="w-3.5 h-3.5" />
<span class="text-xs font-medium capitalize">{category}</span>
</div>
<!-- Select all / Clear links -->
<div class="absolute -top-2 right-3 flex gap-2 bg-background px-1">
<button
type="button"
class="text-xs text-primary hover:underline"
onclick={() => toggleAllPermissions(category as keyof typeof formPermissions, true, permissions)}
>
All
</button>
<span class="text-muted-foreground">|</span>
<button
type="button"
class="text-xs text-muted-foreground hover:underline"
onclick={() => toggleAllPermissions(category as keyof typeof formPermissions, false, permissions)}
>
Clear
</button>
</div>
<div class="flex flex-col gap-1.5">
{#each permissions as permission}
{@const PermIcon = permissionIcons[permission.key]}
<label class="flex items-center gap-1.5 cursor-pointer">
<Checkbox
checked={formPermissions[category as keyof typeof formPermissions].includes(permission.key)}
onCheckedChange={() => togglePermission(category as keyof typeof formPermissions, permission.key)}
/>
{#if PermIcon}
<PermIcon class="w-3 h-3 text-muted-foreground" />
{/if}
<span class="text-xs truncate">{permission.label}</span>
</label>
{/each}
</div>
</div>
{/each}
</div>
</div>
<!-- Environment Permissions Section -->
<div class="flex-shrink-0 border rounded-lg">
<div class="px-4 py-3 border-b bg-muted/30">
<div class="flex items-center justify-between">
<div class="flex items-center gap-2">
<Globe class="w-4 h-4" />
<span class="font-medium text-sm">Environment permissions</span>
</div>
{#if environments.length > 0}
<div class="flex items-center gap-2">
<span class="text-xs text-muted-foreground">All environments (incl. new)</span>
<TogglePill bind:checked={formAllEnvironments} />
</div>
{/if}
</div>
<!-- Environment selector -->
{#if environments.length > 0}
{#if !formAllEnvironments}
<div class="mt-3 grid grid-cols-3 sm:grid-cols-4 lg:grid-cols-6 gap-2">
{#each environments as env}
{@const EnvIcon = getIconComponent(env.icon || 'globe')}
<label class="flex items-center gap-2 p-2 border rounded-md cursor-pointer hover:bg-muted/50 transition-colors text-xs {formEnvironmentIds.includes(env.id) ? 'border-primary bg-primary/5' : ''}">
<Checkbox
checked={formEnvironmentIds.includes(env.id)}
onCheckedChange={() => toggleEnvironment(env.id)}
/>
<EnvIcon class="w-3.5 h-3.5 flex-shrink-0 text-muted-foreground" />
<span class="truncate">{env.name}</span>
</label>
{/each}
</div>
{#if formEnvironmentIds.length === 0}
<p class="text-xs text-amber-600 mt-2">Select at least one environment for these permissions to be effective.</p>
{/if}
{:else}
<p class="text-xs text-muted-foreground mt-1">Permissions apply to all environments, including future ones.</p>
{/if}
{:else}
<p class="text-xs text-muted-foreground mt-1">Permissions apply to all environments.</p>
{/if}
</div>
<div class="p-3 grid grid-cols-2 lg:grid-cols-5 gap-3">
{#each Object.entries(environmentPermissions) as [category, permissions]}
{@const IconComponent = categoryIcons[category]}
<div class="relative border rounded-md pt-5 pb-3 px-3">
<!-- Category pill on border -->
<div class="absolute -top-2.5 left-3 inline-flex items-center gap-1.5 px-2 py-0.5 rounded border {categoryColorsSolid[category] || 'bg-gray-100 dark:bg-gray-900 text-gray-700 dark:text-gray-400 border-gray-300 dark:border-gray-700'}">
<IconComponent class="w-3.5 h-3.5" />
<span class="text-xs font-medium capitalize">{category}</span>
</div>
<!-- Select all / Clear links -->
<div class="absolute -top-2 right-3 flex gap-2 bg-background px-1">
<button
type="button"
class="text-xs text-primary hover:underline"
onclick={() => toggleAllPermissions(category as keyof typeof formPermissions, true, permissions)}
>
All
</button>
<span class="text-muted-foreground">|</span>
<button
type="button"
class="text-xs text-muted-foreground hover:underline"
onclick={() => toggleAllPermissions(category as keyof typeof formPermissions, false, permissions)}
>
Clear
</button>
</div>
<div class="flex flex-col gap-1.5">
{#each permissions as permission}
{@const PermIcon = permissionIcons[permission.key]}
<label class="flex items-center gap-1.5 cursor-pointer">
<Checkbox
checked={formPermissions[category as keyof typeof formPermissions].includes(permission.key)}
onCheckedChange={() => togglePermission(category as keyof typeof formPermissions, permission.key)}
/>
{#if PermIcon}
<PermIcon class="w-3 h-3 text-muted-foreground" />
{/if}
<span class="text-xs truncate">{permission.label}</span>
</label>
{/each}
</div>
</div>
{/each}
</div>
</div>
</div>
<Dialog.Footer class="flex-shrink-0 pt-4">
<Button variant="outline" onclick={handleClose}>Cancel</Button>
<Button onclick={save} disabled={formSaving}>
{#if formSaving}
<RefreshCw class="w-4 h-4 mr-1 animate-spin" />
{:else if isEditing}
<Check class="w-4 h-4 mr-1" />
{:else}
<Plus class="w-4 h-4 mr-1" />
{/if}
{isEditing ? 'Save' : 'Create role'}
</Button>
</Dialog.Footer>
</Dialog.Content>
</Dialog.Root>