Files
dockhand/routes/containers/ContainerInspectModal.svelte
Jarek Krochmalski 62e3c6439e Initial commit
2025-12-28 21:16:03 +01:00

1231 lines
44 KiB
Svelte

<script lang="ts">
import { onDestroy } from 'svelte';
import * as Dialog from '$lib/components/ui/dialog';
import * as Tabs from '$lib/components/ui/tabs';
import { Button } from '$lib/components/ui/button';
import { Badge } from '$lib/components/ui/badge';
import { Loader2, Box, Info, Layers, Cpu, MemoryStick, HardDrive, Network, Shield, Settings2, Code, Copy, Check, Activity, Wifi, Pencil, RefreshCw, X, FolderOpen, Moon } from 'lucide-svelte';
import { Input } from '$lib/components/ui/input';
import { Label } from '$lib/components/ui/label';
import { currentEnvironment, appendEnvParam } from '$lib/stores/environment';
import ImageLayersView from '../images/ImageLayersView.svelte';
import LogsPanel from '../logs/LogsPanel.svelte';
import FileBrowserPanel from './FileBrowserPanel.svelte';
import { formatDateTime } from '$lib/stores/settings';
interface Props {
open: boolean;
containerId: string;
containerName?: string;
onRename?: (newName: string) => void;
}
let { open = $bindable(), containerId, containerName, onRename }: Props = $props();
// Rename state
let isEditing = $state(false);
let editName = $state('');
let renaming = $state(false);
let displayName = $state('');
let loading = $state(true);
let error = $state('');
let containerData = $state<any>(null);
// Active tab state for layers visibility
let activeTab = $state('overview');
// Logs panel state
let showLogs = $state(false);
// Raw JSON modal state
let showRawJson = $state(false);
let jsonCopied = $state(false);
// Processes state
interface ProcessesData {
Titles: string[];
Processes: string[][];
}
let processesData = $state<ProcessesData | null>(null);
let processesLoading = $state(false);
let processesError = $state('');
let processesInterval: ReturnType<typeof setInterval> | null = null;
let processesAutoRefresh = $state(true);
// Stats state
interface ContainerStat {
cpuPercent: number;
memoryUsage: number;
memoryLimit: number;
memoryPercent: number;
networkRx: number;
networkTx: number;
blockRead: number;
blockWrite: number;
timestamp: number;
}
let currentStats = $state<ContainerStat | null>(null);
let cpuHistory = $state<number[]>([]);
let memoryHistory = $state<number[]>([]);
let statsInterval: ReturnType<typeof setInterval> | null = null;
const MAX_HISTORY = 30;
let lastStatsUpdate = $state<number>(0);
let isLiveConnected = $state(false);
let editInputRef: HTMLInputElement | null = null;
function startEditing() {
editName = displayName;
isEditing = true;
// Focus after DOM updates
setTimeout(() => {
editInputRef?.focus();
editInputRef?.select();
}, 0);
}
function cancelEditing() {
isEditing = false;
editName = '';
}
async function saveRename() {
if (!editName.trim() || editName === displayName) {
cancelEditing();
return;
}
renaming = true;
try {
const envId = $currentEnvironment?.id ?? null;
const response = await fetch(appendEnvParam(`/api/containers/${containerId}/rename`, envId), {
method: 'POST',
headers: { 'Content-Type': 'application/json' },
body: JSON.stringify({ name: editName.trim() })
});
if (response.ok) {
displayName = editName.trim();
isEditing = false;
if (onRename) {
onRename(editName.trim());
}
} else {
const data = await response.json();
console.error('Failed to rename container:', data.error);
}
} catch (error) {
console.error('Failed to rename container:', error);
} finally {
renaming = false;
}
}
// Track previous containerId to avoid re-fetching
let lastFetchedId = $state('');
// Fetch container data when modal opens
$effect(() => {
if (open && containerId && containerId !== lastFetchedId) {
lastFetchedId = containerId;
fetchContainerInspect();
}
});
// Start/stop stats collection based on container state (separate effect)
$effect(() => {
if (open && containerData?.State?.Running) {
startStatsCollection();
} else {
stopStatsCollection();
}
});
// Initialize displayName when modal opens
$effect(() => {
if (open) {
displayName = containerName || containerId.slice(0, 12);
}
});
// Reset when modal closes
$effect(() => {
if (!open) {
showLogs = false;
activeTab = 'overview';
stopStatsCollection();
stopProcessesCollection();
cpuHistory = [];
memoryHistory = [];
currentStats = null;
processesData = null;
containerData = null;
loading = true;
error = '';
lastFetchedId = '';
isLiveConnected = false;
lastStatsUpdate = 0;
displayName = '';
isEditing = false;
editName = '';
}
});
async function fetchContainerInspect() {
loading = true;
error = '';
try {
const envId = $currentEnvironment?.id ?? null;
const response = await fetch(appendEnvParam(`/api/containers/${containerId}/inspect`, envId));
if (!response.ok) {
throw new Error('Failed to fetch container details');
}
containerData = await response.json();
} catch (err: any) {
error = err.message || 'Failed to load container details';
console.error('Failed to fetch container inspect:', err);
} finally {
loading = false;
}
}
async function fetchStats() {
if (!containerId || !containerData?.State?.Running) return;
try {
const envId = $currentEnvironment?.id ?? null;
const response = await fetch(appendEnvParam(`/api/containers/${containerId}/stats`, envId));
if (response.ok) {
const stats = await response.json();
if (!stats.error) {
currentStats = stats;
cpuHistory = [...cpuHistory.slice(-(MAX_HISTORY - 1)), stats.cpuPercent];
memoryHistory = [...memoryHistory.slice(-(MAX_HISTORY - 1)), stats.memoryPercent];
lastStatsUpdate = Date.now();
isLiveConnected = true;
} else {
isLiveConnected = false;
}
} else {
isLiveConnected = false;
}
} catch (err) {
isLiveConnected = false;
}
}
function startStatsCollection() {
if (statsInterval) return;
fetchStats();
statsInterval = setInterval(fetchStats, 2000);
}
function stopStatsCollection() {
if (statsInterval) {
clearInterval(statsInterval);
statsInterval = null;
}
}
async function fetchProcesses() {
if (!containerId || !containerData?.State?.Running) return;
// Only show loading spinner on first fetch
if (!processesData) {
processesLoading = true;
}
processesError = '';
try {
const envId = $currentEnvironment?.id ?? null;
const response = await fetch(appendEnvParam(`/api/containers/${containerId}/top`, envId));
if (response.ok) {
const data = await response.json();
if (!data.error) {
processesData = data;
} else {
processesError = data.error;
}
} else {
processesError = 'Failed to fetch processes';
}
} catch (err: any) {
processesError = err.message || 'Failed to fetch processes';
} finally {
processesLoading = false;
}
}
function startProcessesCollection() {
if (processesInterval) return;
fetchProcesses();
processesInterval = setInterval(fetchProcesses, 2000);
}
function stopProcessesCollection() {
if (processesInterval) {
clearInterval(processesInterval);
processesInterval = null;
}
}
function toggleProcessesAutoRefresh() {
processesAutoRefresh = !processesAutoRefresh;
if (processesAutoRefresh) {
startProcessesCollection();
} else {
stopProcessesCollection();
}
}
onDestroy(() => {
stopStatsCollection();
stopProcessesCollection();
});
function formatDate(dateString: string): string {
if (!dateString) return 'N/A';
return formatDateTime(dateString);
}
function formatBytes(bytes: number): string {
if (!bytes || bytes === 0) return '0 B';
const k = 1024;
const sizes = ['B', 'KB', 'MB', 'GB', 'TB'];
const i = Math.floor(Math.log(bytes) / Math.log(k));
return `${(bytes / Math.pow(k, i)).toFixed(i > 1 ? 2 : 0)} ${sizes[i]}`;
}
function formatMemory(bytes: number): string {
if (!bytes) return 'unlimited';
const mb = bytes / (1024 * 1024);
if (mb < 1024) return `${mb.toFixed(0)} MB`;
return `${(mb / 1024).toFixed(2)} GB`;
}
function getStateColor(state: string): 'default' | 'secondary' | 'destructive' | 'outline' {
switch (state.toLowerCase()) {
case 'running': return 'default';
case 'paused': return 'secondary';
case 'exited': return 'destructive';
default: return 'outline';
}
}
// Sparkline path generator
function generateSparklinePath(data: number[], width: number, height: number): string {
if (data.length < 2) return '';
const max = Math.max(...data, 1);
const min = 0;
const range = max - min || 1;
const stepX = width / (data.length - 1);
const points = data.map((value, i) => {
const x = i * stepX;
const y = height - ((value - min) / range) * height;
return `${x},${y}`;
});
return `M ${points.join(' L ')}`;
}
function generateAreaPath(data: number[], width: number, height: number): string {
if (data.length < 2) return '';
const max = Math.max(...data, 1);
const min = 0;
const range = max - min || 1;
const stepX = width / (data.length - 1);
const points = data.map((value, i) => {
const x = i * stepX;
const y = height - ((value - min) / range) * height;
return `${x},${y}`;
});
return `M 0,${height} L ${points.join(' L ')} L ${width},${height} Z`;
}
async function copyJson() {
if (containerData) {
try {
await navigator.clipboard.writeText(JSON.stringify(containerData, null, 2));
jsonCopied = true;
setTimeout(() => jsonCopied = false, 2000);
} catch (err) {
console.error('Failed to copy:', err);
}
}
}
function syntaxHighlight(json: string): string {
return json
.replace(/&/g, '&amp;')
.replace(/</g, '&lt;')
.replace(/>/g, '&gt;')
.replace(/("(\\u[a-zA-Z0-9]{4}|\\[^u]|[^\\"])*"(\s*:)?|\b(true|false|null)\b|-?\d+(?:\.\d*)?(?:[eE][+\-]?\d+)?)/g, (match) => {
let cls = 'text-orange-500'; // number
if (/^"/.test(match)) {
if (/:$/.test(match)) {
cls = 'text-blue-500'; // key
} else {
cls = 'text-green-500'; // string
}
} else if (/true|false/.test(match)) {
cls = 'text-purple-500'; // boolean
} else if (/null/.test(match)) {
cls = 'text-red-500'; // null
}
return `<span class="${cls}">${match}</span>`;
});
}
const formattedJson = $derived(
containerData ? syntaxHighlight(JSON.stringify(containerData, null, 2)) : ''
);
const jsonLines = $derived(formattedJson.split('\n'));
</script>
<Dialog.Root bind:open>
<Dialog.Content class="max-w-6xl h-[90vh] flex flex-col">
<Dialog.Header class="shrink-0">
<Dialog.Title class="flex items-center gap-2">
<Box class="w-5 h-5" />
Container details:
{#if isEditing}
<input
type="text"
bind:value={editName}
bind:this={editInputRef}
class="text-muted-foreground font-normal bg-muted border border-input rounded px-2 py-0.5 text-sm outline-none focus:ring-1 focus:ring-ring"
onkeydown={(e) => {
if (e.key === 'Enter') saveRename();
if (e.key === 'Escape') cancelEditing();
}}
disabled={renaming}
/>
<button
type="button"
onclick={saveRename}
title="Save"
disabled={renaming}
class="p-1 rounded hover:bg-muted transition-colors"
>
{#if renaming}
<RefreshCw class="w-3.5 h-3.5 text-muted-foreground animate-spin" />
{:else}
<Check class="w-3.5 h-3.5 text-green-500 hover:text-green-600" />
{/if}
</button>
<button
type="button"
onclick={cancelEditing}
title="Cancel"
disabled={renaming}
class="p-1 rounded hover:bg-muted transition-colors"
>
<X class="w-3.5 h-3.5 text-muted-foreground hover:text-foreground" />
</button>
{:else}
<span class="text-muted-foreground font-normal">{displayName || containerId.slice(0, 12)}</span>
<button
type="button"
onclick={startEditing}
title="Rename container"
class="p-0.5 rounded hover:bg-muted transition-colors ml-0.5"
>
<Pencil class="w-3 h-3 text-muted-foreground hover:text-foreground" />
</button>
{/if}
{#if containerData?.State?.Running && !loading}
<span class="inline-flex items-center gap-1.5 ml-2 text-xs {isLiveConnected ? 'text-emerald-500' : 'text-muted-foreground'}" title={isLiveConnected ? 'Receiving live updates' : 'Connection lost'}>
<Wifi class="w-3.5 h-3.5 {isLiveConnected ? 'animate-pulse' : ''}" />
{isLiveConnected ? 'Live' : 'Offline'}
</span>
{/if}
{#if containerData && !loading}
<Button
variant="outline"
size="sm"
onclick={() => showRawJson = true}
title="View raw JSON"
class="ml-auto mr-6"
>
<Code class="w-4 h-4 mr-1.5" />
JSON
</Button>
{/if}
</Dialog.Title>
</Dialog.Header>
<div class="flex-1 flex flex-col min-h-0">
{#if loading}
<div class="flex items-center justify-center py-8">
<Loader2 class="w-6 h-6 animate-spin text-muted-foreground" />
</div>
{:else if error}
<div class="text-sm text-red-600 dark:text-red-400 p-3 bg-red-50 dark:bg-red-950 rounded">
{error}
</div>
{:else if containerData}
<Tabs.Root bind:value={activeTab} class="w-full h-full flex flex-col">
<Tabs.List class="w-full justify-start shrink-0 flex-wrap">
<Tabs.Trigger value="overview" onclick={() => showLogs = false}>Overview</Tabs.Trigger>
<Tabs.Trigger value="logs" onclick={() => showLogs = true}>Logs</Tabs.Trigger>
<Tabs.Trigger value="layers" onclick={() => showLogs = false}>Layers</Tabs.Trigger>
<Tabs.Trigger value="processes" onclick={() => { showLogs = false; if (processesAutoRefresh) startProcessesCollection(); else fetchProcesses(); }}>Processes</Tabs.Trigger>
<Tabs.Trigger value="network" onclick={() => showLogs = false}>Network</Tabs.Trigger>
<Tabs.Trigger value="mounts" onclick={() => showLogs = false}>Mounts</Tabs.Trigger>
<Tabs.Trigger value="files" onclick={() => showLogs = false}>Files</Tabs.Trigger>
<Tabs.Trigger value="env" onclick={() => showLogs = false}>Environment</Tabs.Trigger>
<Tabs.Trigger value="security" onclick={() => showLogs = false}>Security</Tabs.Trigger>
<Tabs.Trigger value="resources" onclick={() => showLogs = false}>Resources</Tabs.Trigger>
<Tabs.Trigger value="health" onclick={() => showLogs = false}>Health</Tabs.Trigger>
</Tabs.List>
<!-- Overview Tab -->
<Tabs.Content value="overview" class="space-y-4 overflow-auto">
<!-- Real-time Stats (only for running containers) -->
{#if containerData.State?.Running}
<div class="grid grid-cols-2 lg:grid-cols-4 gap-3">
<!-- CPU -->
<div class="p-3 border border-border rounded-lg">
<div class="flex items-center gap-2 mb-2">
<Cpu class="w-4 h-4 text-blue-500" />
<span class="text-xs font-medium">CPU</span>
<span class="ml-auto text-sm font-bold">{currentStats?.cpuPercent?.toFixed(1) ?? '—'}%</span>
</div>
{#if cpuHistory.length >= 2}
<svg class="w-full h-8" viewBox="0 0 120 32" preserveAspectRatio="none">
<path
d={generateAreaPath(cpuHistory, 120, 32)}
fill="rgba(59, 130, 246, 0.2)"
/>
<path
d={generateSparklinePath(cpuHistory, 120, 32)}
fill="none"
stroke="rgb(59, 130, 246)"
stroke-width="1.5"
/>
</svg>
{:else}
<div class="h-8 flex items-center justify-center text-xs text-muted-foreground">Loading...</div>
{/if}
</div>
<!-- Memory -->
<div class="p-3 border border-border rounded-lg">
<div class="flex items-center gap-2 mb-2">
<MemoryStick class="w-4 h-4 text-green-500" />
<span class="text-xs font-medium">Memory</span>
<span class="ml-auto text-sm font-bold">{currentStats?.memoryPercent?.toFixed(1) ?? '—'}%</span>
</div>
{#if memoryHistory.length >= 2}
<svg class="w-full h-8" viewBox="0 0 120 32" preserveAspectRatio="none">
<path
d={generateAreaPath(memoryHistory, 120, 32)}
fill="rgba(34, 197, 94, 0.2)"
/>
<path
d={generateSparklinePath(memoryHistory, 120, 32)}
fill="none"
stroke="rgb(34, 197, 94)"
stroke-width="1.5"
/>
</svg>
{:else}
<div class="h-8 flex items-center justify-center text-xs text-muted-foreground">Loading...</div>
{/if}
<div class="text-2xs text-muted-foreground mt-1">
{formatBytes(currentStats?.memoryUsage ?? 0)} / {formatBytes(currentStats?.memoryLimit ?? 0)}
</div>
</div>
<!-- Network I/O -->
<div class="p-3 border border-border rounded-lg">
<div class="flex items-center gap-2 mb-2">
<Network class="w-4 h-4 text-purple-500" />
<span class="text-xs font-medium">Network I/O</span>
</div>
<div class="space-y-1 text-xs">
<div class="flex justify-between">
<span class="text-muted-foreground">RX:</span>
<span class="font-mono">{formatBytes(currentStats?.networkRx ?? 0)}</span>
</div>
<div class="flex justify-between">
<span class="text-muted-foreground">TX:</span>
<span class="font-mono">{formatBytes(currentStats?.networkTx ?? 0)}</span>
</div>
</div>
</div>
<!-- Block I/O -->
<div class="p-3 border border-border rounded-lg">
<div class="flex items-center gap-2 mb-2">
<HardDrive class="w-4 h-4 text-orange-500" />
<span class="text-xs font-medium">Disk I/O</span>
</div>
<div class="space-y-1 text-xs">
<div class="flex justify-between">
<span class="text-muted-foreground">Read:</span>
<span class="font-mono">{formatBytes(currentStats?.blockRead ?? 0)}</span>
</div>
<div class="flex justify-between">
<span class="text-muted-foreground">Write:</span>
<span class="font-mono">{formatBytes(currentStats?.blockWrite ?? 0)}</span>
</div>
</div>
</div>
</div>
{/if}
<!-- Status & Basic Info combined -->
<div class="grid grid-cols-1 lg:grid-cols-2 gap-4">
<!-- Status -->
<div class="space-y-3">
<h3 class="text-sm font-semibold flex items-center gap-2">
<Info class="w-4 h-4" />
Status
</h3>
<div class="grid grid-cols-2 gap-2 text-sm">
<div>
<p class="text-muted-foreground text-xs">State</p>
<Badge variant={getStateColor(containerData.State?.Status || 'unknown')}>
{containerData.State?.Status || 'unknown'}
</Badge>
</div>
<div>
<p class="text-muted-foreground text-xs">Restart Policy</p>
<Badge variant="outline">{containerData.HostConfig?.RestartPolicy?.Name || 'no'}</Badge>
</div>
<div>
<p class="text-muted-foreground text-xs">Exit Code</p>
<code class="text-xs">{containerData.State?.ExitCode ?? 'N/A'}</code>
</div>
<div>
<p class="text-muted-foreground text-xs">Restart Count</p>
<code class="text-xs">{containerData.RestartCount ?? 0}</code>
</div>
</div>
</div>
<!-- Basic Info -->
<div class="space-y-3">
<h3 class="text-sm font-semibold">Basic information</h3>
<div class="grid grid-cols-2 gap-2 text-sm">
<div>
<p class="text-muted-foreground text-xs">ID</p>
<code class="text-xs">{containerData.Id?.slice(0, 12)}</code>
</div>
<div>
<p class="text-muted-foreground text-xs">Platform</p>
<p class="text-xs">{containerData.Platform || 'N/A'}</p>
</div>
<div>
<p class="text-muted-foreground text-xs">Created</p>
<p class="text-xs">{formatDate(containerData.Created)}</p>
</div>
<div>
<p class="text-muted-foreground text-xs">Started</p>
<p class="text-xs">{formatDate(containerData.State?.StartedAt)}</p>
</div>
</div>
</div>
</div>
<!-- Image -->
<div class="space-y-2">
<h3 class="text-sm font-semibold">Image</h3>
<div class="flex items-center gap-2 p-2 bg-muted rounded">
<code class="text-xs break-all flex-1">{containerData.Config?.Image || 'N/A'}</code>
</div>
</div>
<!-- Command -->
{#if containerData.Path || containerData.Args}
<div class="space-y-2">
<h3 class="text-sm font-semibold">Command</h3>
<div class="p-2 bg-muted rounded">
<code class="text-xs break-all">
{containerData.Path || ''} {containerData.Args?.join(' ') || ''}
</code>
</div>
</div>
{/if}
<!-- Labels (collapsible) -->
{#if containerData.Config?.Labels && Object.keys(containerData.Config.Labels).length > 0}
<details class="group">
<summary class="text-sm font-semibold cursor-pointer hover:text-primary">
Labels ({Object.keys(containerData.Config.Labels).length})
</summary>
<div class="space-y-1 mt-2 max-h-32 overflow-y-auto">
{#each Object.entries(containerData.Config.Labels) as [key, value]}
<div class="text-xs p-2 bg-muted rounded">
<code class="text-muted-foreground">{key}</code>
<code class="text-muted-foreground">=</code>
<code class="break-all">{value}</code>
</div>
{/each}
</div>
</details>
{/if}
</Tabs.Content>
<!-- Processes Tab -->
<Tabs.Content value="processes" class="overflow-auto data-[state=inactive]:hidden">
{#if !containerData.State?.Running}
<div class="flex items-center gap-2 text-sm text-muted-foreground py-8 justify-center">
<Moon class="w-5 h-5" />
<span>Container is not running</span>
</div>
{:else if processesLoading}
<div class="flex items-center justify-center py-8">
<Loader2 class="w-6 h-6 animate-spin text-muted-foreground" />
</div>
{:else if processesError}
<div class="text-sm text-red-600 dark:text-red-400 p-3 bg-red-50 dark:bg-red-950 rounded">
{processesError}
</div>
{:else if processesData && processesData.Processes?.length > 0}
<div class="border border-border rounded-lg overflow-auto max-h-[60vh]">
<table class="w-full text-xs">
<thead class="sticky top-0 bg-muted z-10">
<tr class="border-b border-border">
<th class="text-left p-2 font-medium text-muted-foreground">#</th>
{#each processesData.Titles as title}
<th class="text-left p-2 font-medium text-muted-foreground">{title}</th>
{/each}
</tr>
</thead>
<tbody>
{#each processesData.Processes as process, i}
<tr class="border-b border-border hover:bg-muted/50">
<td class="p-2 text-muted-foreground">{i + 1}</td>
{#each process as cell}
<td class="p-2 font-mono">{cell}</td>
{/each}
</tr>
{/each}
</tbody>
</table>
</div>
<div class="text-xs text-muted-foreground pt-2">
{processesData.Processes.length} process(es)
</div>
{:else}
<p class="text-sm text-muted-foreground">No processes found</p>
{/if}
</Tabs.Content>
<!-- Logs Tab -->
<Tabs.Content value="logs" class="flex-1 min-h-0">
<LogsPanel
containerId={containerId}
containerName={containerName || containerId.slice(0, 12)}
visible={showLogs}
envId={$currentEnvironment?.id ?? null}
fillHeight={true}
showCloseButton={false}
onClose={() => showLogs = false}
/>
</Tabs.Content>
<!-- Layers Tab -->
<Tabs.Content value="layers" class="overflow-auto">
{#if containerData?.Image}
<ImageLayersView
imageId={containerData.Image}
imageName={containerData.Config?.Image || containerData.Image}
visible={activeTab === 'layers'}
/>
{:else}
<p class="text-sm text-muted-foreground py-8 text-center">No image information available</p>
{/if}
</Tabs.Content>
<!-- Network Tab -->
<Tabs.Content value="network" class="space-y-4 overflow-auto">
<!-- Network Mode -->
<div class="space-y-2">
<h3 class="text-sm font-semibold">Network mode</h3>
<Badge variant="outline">{containerData.HostConfig?.NetworkMode || 'default'}</Badge>
</div>
<!-- DNS Settings -->
{#if containerData.HostConfig?.Dns?.length > 0 || containerData.HostConfig?.DnsSearch?.length > 0 || containerData.HostConfig?.DnsOptions?.length > 0}
<div class="space-y-2">
<h3 class="text-sm font-semibold">DNS configuration</h3>
<div class="grid grid-cols-1 lg:grid-cols-3 gap-3">
{#if containerData.HostConfig?.Dns?.length > 0}
<div class="p-2 bg-muted rounded">
<p class="text-xs text-muted-foreground mb-1">DNS Servers</p>
{#each containerData.HostConfig.Dns as dns}
<code class="text-xs block">{dns}</code>
{/each}
</div>
{/if}
{#if containerData.HostConfig?.DnsSearch?.length > 0}
<div class="p-2 bg-muted rounded">
<p class="text-xs text-muted-foreground mb-1">DNS Search</p>
{#each containerData.HostConfig.DnsSearch as search}
<code class="text-xs block">{search}</code>
{/each}
</div>
{/if}
{#if containerData.HostConfig?.DnsOptions?.length > 0}
<div class="p-2 bg-muted rounded">
<p class="text-xs text-muted-foreground mb-1">DNS Options</p>
{#each containerData.HostConfig.DnsOptions as opt}
<code class="text-xs block">{opt}</code>
{/each}
</div>
{/if}
</div>
</div>
{/if}
<!-- Extra Hosts -->
{#if containerData.HostConfig?.ExtraHosts?.length > 0}
<div class="space-y-2">
<h3 class="text-sm font-semibold">Extra hosts</h3>
<div class="space-y-1">
{#each containerData.HostConfig.ExtraHosts as host}
<div class="text-xs p-2 bg-muted rounded">
<code>{host}</code>
</div>
{/each}
</div>
</div>
{/if}
<!-- Networks -->
{#if containerData.NetworkSettings?.Networks && Object.keys(containerData.NetworkSettings.Networks).length > 0}
<div class="space-y-2">
<h3 class="text-sm font-semibold">Connected networks</h3>
<div class="space-y-2">
{#each Object.entries(containerData.NetworkSettings.Networks) as [networkName, networkData]}
<div class="p-3 border border-border rounded-lg space-y-2">
<div class="flex items-center justify-between">
<span class="font-medium text-sm">{networkName}</span>
<Badge variant="secondary" class="text-xs">{networkData.NetworkID?.slice(0, 12)}</Badge>
</div>
<div class="grid grid-cols-2 lg:grid-cols-4 gap-2 text-xs">
{#if networkData.IPAddress}
<div>
<p class="text-muted-foreground">IPv4</p>
<code>{networkData.IPAddress}</code>
</div>
{/if}
{#if networkData.GlobalIPv6Address}
<div>
<p class="text-muted-foreground">IPv6</p>
<code>{networkData.GlobalIPv6Address}</code>
</div>
{/if}
{#if networkData.MacAddress}
<div>
<p class="text-muted-foreground">MAC</p>
<code>{networkData.MacAddress}</code>
</div>
{/if}
{#if networkData.Gateway}
<div>
<p class="text-muted-foreground">Gateway</p>
<code>{networkData.Gateway}</code>
</div>
{/if}
{#if networkData.Aliases?.length > 0}
<div class="col-span-2">
<p class="text-muted-foreground">Aliases</p>
<code>{networkData.Aliases.join(', ')}</code>
</div>
{/if}
</div>
</div>
{/each}
</div>
</div>
{/if}
<!-- Ports -->
{#if containerData.NetworkSettings?.Ports && Object.keys(containerData.NetworkSettings.Ports).length > 0}
<div class="space-y-2">
<h3 class="text-sm font-semibold">Port mappings</h3>
<div class="flex flex-wrap gap-2">
{#each Object.entries(containerData.NetworkSettings.Ports) as [containerPort, hostBindings]}
{#if hostBindings && hostBindings.length > 0}
{#each hostBindings as binding}
<div class="flex items-center gap-2 text-xs p-2 bg-muted rounded">
<code>{binding.HostIp || '0.0.0.0'}:{binding.HostPort}</code>
<span class="text-muted-foreground"></span>
<code>{containerPort}</code>
</div>
{/each}
{:else}
<div class="flex items-center gap-2 text-xs p-2 bg-muted rounded">
<code class="text-muted-foreground">exposed</code>
<code>{containerPort}</code>
</div>
{/if}
{/each}
</div>
</div>
{/if}
</Tabs.Content>
<!-- Mounts Tab -->
<Tabs.Content value="mounts" class="space-y-4 overflow-auto">
{#if containerData.Mounts && containerData.Mounts.length > 0}
<div class="space-y-2">
{#each containerData.Mounts as mount}
<div class="p-3 border border-border rounded-lg space-y-2">
<div class="flex items-center justify-between">
<Badge variant="outline" class="text-xs">{mount.Type}</Badge>
<Badge variant={mount.RW ? 'default' : 'secondary'} class="text-xs">
{mount.RW ? 'Read/Write' : 'Read-Only'}
</Badge>
</div>
<div class="grid grid-cols-1 lg:grid-cols-2 gap-2 text-xs">
<div>
<p class="text-muted-foreground">Source</p>
<code class="break-all">{mount.Source || mount.Name || 'N/A'}</code>
</div>
<div>
<p class="text-muted-foreground">Destination</p>
<code class="break-all">{mount.Destination}</code>
</div>
{#if mount.Driver}
<div>
<p class="text-muted-foreground">Driver</p>
<code>{mount.Driver}</code>
</div>
{/if}
{#if mount.Propagation}
<div>
<p class="text-muted-foreground">Propagation</p>
<code>{mount.Propagation}</code>
</div>
{/if}
</div>
</div>
{/each}
</div>
{:else}
<p class="text-sm text-muted-foreground">No mounts configured</p>
{/if}
</Tabs.Content>
<!-- Files Tab -->
<Tabs.Content value="files" class="flex-1 min-h-0">
{#if containerData.State?.Running && !containerData.State?.Paused}
<FileBrowserPanel
containerId={containerId}
envId={$currentEnvironment?.id ?? undefined}
/>
{:else if containerData.State?.Paused}
<div class="flex items-center gap-2 text-sm text-muted-foreground py-8 justify-center">
<Moon class="w-5 h-5" />
<span>Container is paused</span>
</div>
{:else}
<div class="flex items-center gap-2 text-sm text-muted-foreground py-8 justify-center">
<Moon class="w-5 h-5" />
<span>Container is not running</span>
</div>
{/if}
</Tabs.Content>
<!-- Environment Tab -->
<Tabs.Content value="env" class="space-y-4 overflow-auto">
{#if containerData.Config?.Env && containerData.Config.Env.length > 0}
<div class="space-y-1">
{#each containerData.Config.Env as envVar}
{@const [key, ...valueParts] = envVar.split('=')}
{@const value = valueParts.join('=')}
<div class="text-xs p-2 bg-muted rounded">
<code class="text-muted-foreground font-medium">{key}</code>
<code class="text-muted-foreground">=</code>
<code class="break-all">{value}</code>
</div>
{/each}
</div>
{:else}
<p class="text-sm text-muted-foreground">No environment variables</p>
{/if}
</Tabs.Content>
<!-- Security Tab -->
<Tabs.Content value="security" class="space-y-4 overflow-auto">
<!-- Privileged & User -->
<div class="grid grid-cols-2 lg:grid-cols-4 gap-3">
<div class="p-3 border border-border rounded-lg">
<p class="text-xs text-muted-foreground mb-1">Privileged</p>
<Badge variant={containerData.HostConfig?.Privileged ? 'destructive' : 'secondary'}>
{containerData.HostConfig?.Privileged ? 'Yes' : 'No'}
</Badge>
</div>
<div class="p-3 border border-border rounded-lg">
<p class="text-xs text-muted-foreground mb-1">Read-only Root</p>
<Badge variant={containerData.HostConfig?.ReadonlyRootfs ? 'default' : 'outline'}>
{containerData.HostConfig?.ReadonlyRootfs ? 'Yes' : 'No'}
</Badge>
</div>
<div class="p-3 border border-border rounded-lg">
<p class="text-xs text-muted-foreground mb-1">User</p>
<code class="text-xs">{containerData.Config?.User || 'root'}</code>
</div>
<div class="p-3 border border-border rounded-lg">
<p class="text-xs text-muted-foreground mb-1">User Namespace</p>
<code class="text-xs">{containerData.HostConfig?.UsernsMode || 'host'}</code>
</div>
</div>
<!-- Security Options -->
{#if containerData.HostConfig?.SecurityOpt?.length > 0}
<div class="space-y-2">
<h3 class="text-sm font-semibold">Security options</h3>
<div class="space-y-1">
{#each containerData.HostConfig.SecurityOpt as opt}
<div class="text-xs p-2 bg-muted rounded">
<code>{opt}</code>
</div>
{/each}
</div>
</div>
{/if}
<!-- AppArmor / Seccomp -->
<div class="grid grid-cols-1 lg:grid-cols-2 gap-3">
{#if containerData.AppArmorProfile !== undefined}
<div class="p-3 border border-border rounded-lg">
<p class="text-xs text-muted-foreground mb-1">AppArmor Profile</p>
<code class="text-xs">{containerData.AppArmorProfile || 'unconfined'}</code>
</div>
{/if}
{#if containerData.HostConfig?.SecurityOpt?.some((o: string) => o.startsWith('seccomp'))}
<div class="p-3 border border-border rounded-lg">
<p class="text-xs text-muted-foreground mb-1">Seccomp</p>
<code class="text-xs">
{containerData.HostConfig.SecurityOpt.find((o: string) => o.startsWith('seccomp'))?.split('=')[1] || 'default'}
</code>
</div>
{/if}
</div>
<!-- Capabilities -->
<div class="grid grid-cols-1 lg:grid-cols-2 gap-4">
{#if containerData.HostConfig?.CapAdd?.length > 0}
<div class="space-y-2">
<h3 class="text-sm font-semibold text-green-600 dark:text-green-400">Added capabilities</h3>
<div class="flex flex-wrap gap-1">
{#each containerData.HostConfig.CapAdd as cap}
<Badge variant="outline" class="text-xs bg-green-500/10">{cap}</Badge>
{/each}
</div>
</div>
{/if}
{#if containerData.HostConfig?.CapDrop?.length > 0}
<div class="space-y-2">
<h3 class="text-sm font-semibold text-red-600 dark:text-red-400">Dropped capabilities</h3>
<div class="flex flex-wrap gap-1">
{#each containerData.HostConfig.CapDrop as cap}
<Badge variant="outline" class="text-xs bg-red-500/10">{cap}</Badge>
{/each}
</div>
</div>
{/if}
</div>
{#if !containerData.HostConfig?.CapAdd?.length && !containerData.HostConfig?.CapDrop?.length && !containerData.HostConfig?.SecurityOpt?.length}
<p class="text-sm text-muted-foreground">Default security settings</p>
{/if}
</Tabs.Content>
<!-- Resources Tab -->
<Tabs.Content value="resources" class="space-y-4 overflow-auto">
<!-- CPU & Memory Limits -->
<div class="space-y-2">
<h3 class="text-sm font-semibold flex items-center gap-2">
<Settings2 class="w-4 h-4" />
Resource limits
</h3>
<div class="grid grid-cols-2 lg:grid-cols-4 gap-3">
<div class="p-3 border border-border rounded-lg">
<p class="text-xs text-muted-foreground mb-1">CPU Shares</p>
<code class="text-sm">{containerData.HostConfig?.CpuShares || 'default'}</code>
</div>
<div class="p-3 border border-border rounded-lg">
<p class="text-xs text-muted-foreground mb-1">CPUs</p>
<code class="text-sm">{containerData.HostConfig?.NanoCpus ? (containerData.HostConfig.NanoCpus / 1e9).toFixed(2) : 'unlimited'}</code>
</div>
<div class="p-3 border border-border rounded-lg">
<p class="text-xs text-muted-foreground mb-1">Memory</p>
<code class="text-sm">{formatMemory(containerData.HostConfig?.Memory)}</code>
</div>
<div class="p-3 border border-border rounded-lg">
<p class="text-xs text-muted-foreground mb-1">Memory Swap</p>
<code class="text-sm">{formatMemory(containerData.HostConfig?.MemorySwap)}</code>
</div>
<div class="p-3 border border-border rounded-lg">
<p class="text-xs text-muted-foreground mb-1">Memory Reservation</p>
<code class="text-sm">{formatMemory(containerData.HostConfig?.MemoryReservation)}</code>
</div>
<div class="p-3 border border-border rounded-lg">
<p class="text-xs text-muted-foreground mb-1">PIDs Limit</p>
<code class="text-sm">{containerData.HostConfig?.PidsLimit ?? 'unlimited'}</code>
</div>
<div class="p-3 border border-border rounded-lg">
<p class="text-xs text-muted-foreground mb-1">OOM Kill</p>
<Badge variant={containerData.HostConfig?.OomKillDisable ? 'destructive' : 'default'}>
{containerData.HostConfig?.OomKillDisable ? 'Disabled' : 'Enabled'}
</Badge>
</div>
<div class="p-3 border border-border rounded-lg">
<p class="text-xs text-muted-foreground mb-1">CPU Period/Quota</p>
<code class="text-sm">
{containerData.HostConfig?.CpuPeriod || 0}/{containerData.HostConfig?.CpuQuota || 0}
</code>
</div>
</div>
</div>
<!-- Ulimits -->
{#if containerData.HostConfig?.Ulimits?.length > 0}
<div class="space-y-2">
<h3 class="text-sm font-semibold">Ulimits</h3>
<div class="grid grid-cols-1 lg:grid-cols-2 gap-2">
{#each containerData.HostConfig.Ulimits as ulimit}
<div class="flex justify-between text-xs p-2 bg-muted rounded">
<code class="text-muted-foreground">{ulimit.Name}</code>
<code>soft={ulimit.Soft} hard={ulimit.Hard}</code>
</div>
{/each}
</div>
</div>
{/if}
<!-- Devices -->
{#if containerData.HostConfig?.Devices?.length > 0}
<div class="space-y-2">
<h3 class="text-sm font-semibold">Devices</h3>
<div class="space-y-1">
{#each containerData.HostConfig.Devices as device}
<div class="text-xs p-2 bg-muted rounded flex gap-2">
<code class="text-muted-foreground">{device.PathOnHost}</code>
<span class="text-muted-foreground"></span>
<code>{device.PathInContainer}</code>
{#if device.CgroupPermissions}
<Badge variant="outline" class="text-2xs">{device.CgroupPermissions}</Badge>
{/if}
</div>
{/each}
</div>
</div>
{/if}
<!-- Cgroup -->
<div class="space-y-2">
<h3 class="text-sm font-semibold">Cgroup settings</h3>
<div class="grid grid-cols-2 lg:grid-cols-3 gap-3">
<div class="p-2 bg-muted rounded">
<p class="text-xs text-muted-foreground">Cgroup</p>
<code class="text-xs">{containerData.HostConfig?.Cgroup || 'default'}</code>
</div>
<div class="p-2 bg-muted rounded">
<p class="text-xs text-muted-foreground">Cgroup Parent</p>
<code class="text-xs">{containerData.HostConfig?.CgroupParent || 'default'}</code>
</div>
<div class="p-2 bg-muted rounded">
<p class="text-xs text-muted-foreground">Cgroupns Mode</p>
<code class="text-xs">{containerData.HostConfig?.CgroupnsMode || 'host'}</code>
</div>
</div>
</div>
</Tabs.Content>
<!-- Health Tab -->
<Tabs.Content value="health" class="space-y-4 overflow-auto">
{#if containerData.State?.Health}
<div class="space-y-3">
<div class="grid grid-cols-2 gap-3 text-sm">
<div>
<p class="text-muted-foreground">Status</p>
<Badge variant={containerData.State.Health.Status === 'healthy' ? 'default' : 'destructive'}>
{containerData.State.Health.Status}
</Badge>
</div>
<div>
<p class="text-muted-foreground">Failing Streak</p>
<code class="text-xs">{containerData.State.Health.FailingStreak || 0}</code>
</div>
</div>
{#if containerData.State.Health.Log && containerData.State.Health.Log.length > 0}
<div class="space-y-2">
<h3 class="text-sm font-semibold">Health check log</h3>
<div class="space-y-1 max-h-64 overflow-y-auto">
{#each containerData.State.Health.Log.slice(-5) as log}
<div class="p-2 border border-border rounded text-xs space-y-1">
<div class="flex justify-between items-center">
<Badge variant={log.ExitCode === 0 ? 'default' : 'destructive'} class="text-xs">
Exit: {log.ExitCode}
</Badge>
<span class="text-muted-foreground">{formatDate(log.End)}</span>
</div>
{#if log.Output}
<code class="block text-xs bg-muted p-1 rounded break-all">{log.Output.trim()}</code>
{/if}
</div>
{/each}
</div>
</div>
{/if}
</div>
{:else}
<p class="text-sm text-muted-foreground">No health check configured</p>
{/if}
</Tabs.Content>
</Tabs.Root>
{/if}
</div>
<Dialog.Footer class="shrink-0">
<Button variant="outline" onclick={() => (open = false)}>Close</Button>
</Dialog.Footer>
</Dialog.Content>
</Dialog.Root>
<!-- Raw JSON Modal -->
<Dialog.Root bind:open={showRawJson}>
<Dialog.Content class="max-w-4xl h-[80vh] flex flex-col">
<Dialog.Header class="shrink-0">
<Dialog.Title class="flex items-center gap-2">
<Code class="w-5 h-5" />
Raw JSON
<Button
variant="outline"
size="sm"
onclick={copyJson}
title={jsonCopied ? 'Copied!' : 'Copy to clipboard'}
>
{#if jsonCopied}
<Check class="w-4 h-4 mr-1.5 text-green-500" />
<span class="text-green-500">Copied!</span>
{:else}
<Copy class="w-4 h-4 mr-1.5" />
Copy
{/if}
</Button>
</Dialog.Title>
</Dialog.Header>
<div class="flex-1 overflow-auto min-h-0">
<div class="bg-gray-100 dark:bg-zinc-900 rounded-lg text-xs font-mono overflow-auto h-full">
<table class="w-full">
<tbody>
{#each jsonLines as line, i}
<tr class="hover:bg-gray-200/50 dark:hover:bg-zinc-800/50">
<td class="text-right text-gray-400 dark:text-zinc-500 select-none px-3 py-0 border-r border-gray-300 dark:border-zinc-700 sticky left-0 bg-gray-100 dark:bg-zinc-900">{i + 1}</td>
<td class="px-3 py-0 whitespace-pre text-gray-900 dark:text-gray-100">{@html line || ' '}</td>
</tr>
{/each}
</tbody>
</table>
</div>
</div>
<Dialog.Footer class="shrink-0">
<Button variant="outline" onclick={() => showRawJson = false}>Close</Button>
</Dialog.Footer>
</Dialog.Content>
</Dialog.Root>