Files
dockhand/src/routes/containers/ContainerInspectModal.svelte
T
2026-06-15 14:56:51 +02:00

1746 lines
65 KiB
Svelte

<script lang="ts">
import { onDestroy } from 'svelte';
import { goto } from '$app/navigation';
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, XCircle, Activity, Wifi, Pencil, RefreshCw, X, FolderOpen, Moon, Tags, ExternalLink, Gpu, Globe, Link, Unlink } from 'lucide-svelte';
import * as Select from '$lib/components/ui/select';
import { toast } from 'svelte-sonner';
import * as Tooltip from '$lib/components/ui/tooltip';
import { copyToClipboard } from '$lib/utils/clipboard';
import { parseCustomUrl } from '$lib/utils/custom-url';
import { formatBytes } from '$lib/utils/format';
import { Input } from '$lib/components/ui/input';
import { Label } from '$lib/components/ui/label';
import { currentEnvironment, appendEnvParam, environments } 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';
import { formatHostPortUrl } from '$lib/utils/url';
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);
// Peer containers in the current env — used to resolve "container:<sha>" mode to a friendly name
let peerContainers = $state<Array<{ id: string; name: string }>>([]);
const networkModeLabel = $derived.by(() => {
const raw = containerData?.HostConfig?.NetworkMode || 'default';
if (!raw.startsWith('container:')) return raw;
const ref = raw.slice('container:'.length);
const match = peerContainers.find(c => c.id === ref || c.id.startsWith(ref));
return match ? `container:${match.name}` : raw;
});
// Docker rejects attaching extra networks when the container shares another
// namespace (host / none / container:X / service:X). Hide join controls then.
const isSharedNetworkMode = $derived.by(() => {
const mode = containerData?.HostConfig?.NetworkMode || '';
return mode === 'host' || mode === 'none' || mode.startsWith('container:') || mode.startsWith('service:');
});
// 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<'ok' | 'error' | null>(null);
// Label copy state
let copiedLabel = $state<string | null>(null);
let copyLabelFailed = $state(false);
let labelFilter = $state('');
let copiedAllLabels = $state(false);
async function copyLabel(key: string, value: string) {
const ok = await copyToClipboard(`${key}=${value}`);
if (ok) {
copiedLabel = key;
setTimeout(() => copiedLabel = null, 2000);
} else {
copyLabelFailed = true;
setTimeout(() => copyLabelFailed = false, 2000);
}
}
async function copyAllLabels(entries: [string, string][]) {
if (entries.length === 0) return;
const text = entries.map(([k, v]) => `${k}=${v}`).join('\n');
const ok = await copyToClipboard(text);
if (ok) {
copiedAllLabels = true;
setTimeout(() => copiedAllLabels = false, 2000);
} else {
copyLabelFailed = true;
setTimeout(() => copyLabelFailed = false, 2000);
}
}
// 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;
// Network attach/detach state
interface NetworkListItem {
id: string;
name: string;
driver: string;
}
let availableNetworks = $state<NetworkListItem[]>([]);
let selectedNetwork = $state<string | undefined>(undefined);
let networkConnecting = $state(false);
let networkDisconnecting = $state<string | null>(null);
let networksLoading = $state(false);
const connectedNetworkNames = $derived(
containerData?.NetworkSettings?.Networks
? new Set(Object.keys(containerData.NetworkSettings.Networks))
: new Set<string>()
);
const unconnectedNetworks = $derived(
availableNetworks.filter(n => !connectedNetworkNames.has(n.name))
);
async function fetchNetworks() {
networksLoading = true;
try {
const envId = $currentEnvironment?.id ?? null;
const response = await fetch(appendEnvParam('/api/networks', envId));
if (response.ok) {
availableNetworks = await response.json();
}
} catch (err) {
console.error('Failed to fetch networks:', err);
} finally {
networksLoading = false;
}
}
async function connectToNetwork() {
if (!selectedNetwork || !containerId) return;
networkConnecting = true;
try {
const envId = $currentEnvironment?.id ?? null;
const response = await fetch(appendEnvParam(`/api/networks/${selectedNetwork}/connect`, envId), {
method: 'POST',
headers: { 'Content-Type': 'application/json' },
body: JSON.stringify({ containerId, containerName: displayName })
});
if (response.ok) {
const net = availableNetworks.find(n => n.id === selectedNetwork);
toast.success(`Connected to ${net?.name || 'network'}`);
selectedNetwork = undefined;
await fetchContainerInspect();
} else {
const data = await response.json();
toast.error(data.details || 'Failed to connect to network');
}
} catch (err) {
toast.error('Failed to connect to network');
} finally {
networkConnecting = false;
}
}
async function disconnectFromNetwork(networkId: string, networkName: string) {
networkDisconnecting = networkName;
try {
const envId = $currentEnvironment?.id ?? null;
const response = await fetch(appendEnvParam(`/api/networks/${networkId}/disconnect`, envId), {
method: 'POST',
headers: { 'Content-Type': 'application/json' },
body: JSON.stringify({ containerId, containerName: displayName })
});
if (response.ok) {
toast.success(`Disconnected from ${networkName}`);
await fetchContainerInspect();
} else {
const data = await response.json();
toast.error(data.details || 'Failed to disconnect from network');
}
} catch (err) {
toast.error('Failed to disconnect from network');
} finally {
networkDisconnecting = null;
}
}
// Current environment details for port URL generation
const currentEnvDetails = $derived($environments.find(e => e.id === $currentEnvironment?.id) ?? null);
function extractHostFromUrl(urlString: string): string | null {
if (!urlString) return null;
// Handle tcp:// URLs (Docker remote)
const tcpMatch = urlString.match(/^tcp:\/\/([^:\/]+)/);
if (tcpMatch) return tcpMatch[1];
// Handle http:// or https:// URLs
const httpMatch = urlString.match(/^https?:\/\/([^:\/]+)/);
if (httpMatch) return httpMatch[1];
// Handle host:port format
const hostPortMatch = urlString.match(/^([^:\/]+):\d+/);
if (hostPortMatch) return hostPortMatch[1];
// Just a hostname
return urlString;
}
function getPortUrl(publicPort: number): string | null {
const env = currentEnvDetails;
if (!env) return null;
// Priority 1: Use publicIp if configured
if (env.publicIp) {
return formatHostPortUrl(env.publicIp, publicPort);
}
// Priority 2: Extract from host for direct/hawser-standard
const connectionType = env.connectionType || 'socket';
if (connectionType === 'direct' && env.host) {
const host = extractHostFromUrl(env.host);
if (host) return formatHostPortUrl(host, publicPort);
} else if (connectionType === 'hawser-standard' && env.host) {
const host = extractHostFromUrl(env.host);
if (host) return formatHostPortUrl(host, publicPort);
}
// No public IP available for socket or hawser-edge
return 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();
// One-shot fetch so the Overview's process count tile renders
// immediately, even if the user never opens the Processes tab.
if (!processesData) fetchProcesses();
} else {
stopStatsCollection();
}
});
// Initialize displayName when modal opens
$effect(() => {
if (open) {
displayName = containerName || containerId.slice(0, 12);
}
});
// Fetch available networks when network tab is selected
$effect(() => {
if (open && activeTab === 'network') {
fetchNetworks();
}
});
// 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;
labelFilter = '';
displayName = '';
isEditing = false;
editName = '';
availableNetworks = [];
selectedNetwork = undefined;
}
});
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();
// Fetch peers only when this container shares another container's namespace —
// keeps the dialog snappy when the network mode is bridge/host/none/custom.
if (containerData?.HostConfig?.NetworkMode?.startsWith('container:')) {
try {
const peersRes = await fetch(appendEnvParam('/api/containers', envId));
if (peersRes.ok) {
const list = await peersRes.json();
peerContainers = list.map((c: any) => ({ id: c.id, name: c.name }));
}
} catch {
// Non-fatal — falls back to the raw SHA
}
}
} 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 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) {
const ok = await copyToClipboard(JSON.stringify(containerData, null, 2));
jsonCopied = ok ? 'ok' : 'error';
setTimeout(() => jsonCopied = null, 2000);
}
}
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 w-full h-[calc(100vh-2rem)] 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}
{@const composeStack = containerData?.Config?.Labels?.['com.docker.compose.project']}
{#if composeStack && !loading}
<Tooltip.Root>
<Tooltip.Trigger>
<button
type="button"
onclick={() => {
open = false;
goto(appendEnvParam(`/stacks?search=${encodeURIComponent(composeStack)}`, $currentEnvironment?.id ?? null));
}}
class="cursor-pointer inline-flex items-center"
>
<Badge variant="outline" class="text-xs py-0 px-1.5 hover:bg-primary/10 hover:border-primary/50 transition-colors gap-1">
<Layers class="w-3 h-3" />
{composeStack}
</Badge>
</button>
</Tooltip.Trigger>
<Tooltip.Content>
<p class="text-xs whitespace-nowrap">Open stack "{composeStack}"</p>
</Tooltip.Content>
</Tooltip.Root>
{/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 inspect data"
class="ml-auto mr-6"
>
<Code class="w-4 h-4 mr-1.5" />
Inspect
</Button>
{/if}
</Dialog.Title>
</Dialog.Header>
<div class="flex-1 flex flex-col min-h-[400px]">
{#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 h-auto min-h-10 bg-muted rounded-lg">
<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="labels" onclick={() => showLogs = false}>Labels</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-5 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>
<!-- Processes -->
<div class="p-3 border border-border rounded-lg">
<div class="flex items-center gap-2 mb-2">
<Activity class="w-4 h-4 text-pink-500" />
<span class="text-xs font-medium">Processes</span>
<button
type="button"
class="ml-auto text-sm font-bold hover:text-foreground/80 transition-colors"
onclick={() => activeTab = 'processes'}
title="View process list"
>
{processesData?.Processes?.length ?? '—'}
</button>
</div>
<div class="h-8 flex items-center justify-center text-2xs text-muted-foreground">
{#if processesData?.Processes?.length}
running in container
{:else if processesLoading}
<Loader2 class="w-3 h-3 animate-spin" />
{:else}
{/if}
</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}
</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">{networkModeLabel}</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 -->
<div class="space-y-2">
<h3 class="text-sm font-semibold">Connected networks</h3>
{#if isSharedNetworkMode}
<p class="text-xs text-muted-foreground">
Network namespace is shared via <code class="px-1 py-0.5 rounded bg-muted">{containerData.HostConfig?.NetworkMode}</code> — additional networks cannot be attached.
</p>
{:else if containerData.NetworkSettings?.Networks && Object.keys(containerData.NetworkSettings.Networks).length > 0}
<div class="space-y-2">
{#each Object.entries(containerData.NetworkSettings.Networks) as [networkName, networkData]}
{@const netData = networkData as any}
<div class="p-3 border border-border rounded-lg space-y-2">
<div class="flex items-center justify-between">
<div class="flex items-center gap-2">
<span class="font-medium text-sm">{networkName}</span>
<Badge variant="secondary" class="text-xs">{netData.NetworkID?.slice(0, 12)}</Badge>
</div>
{#if containerData.State?.Running}
<Button
variant="ghost"
size="sm"
class="h-6 px-2 text-xs text-destructive hover:text-destructive hover:bg-destructive/10"
disabled={networkDisconnecting === networkName}
onclick={() => disconnectFromNetwork(netData.NetworkID, networkName)}
>
{#if networkDisconnecting === networkName}
<Loader2 class="w-3 h-3 mr-1 animate-spin" />
{:else}
<Unlink class="w-3 h-3 mr-1" />
{/if}
Leave
</Button>
{/if}
</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>
{:else}
<p class="text-xs text-muted-foreground">No networks connected.</p>
{/if}
<!-- Join network dropdown -->
{#if containerData.State?.Running && !isSharedNetworkMode}
<div class="flex items-center gap-2 pt-1">
<Select.Root type="single" bind:value={selectedNetwork}>
<Select.Trigger class="flex-1 h-8 text-xs">
{#if selectedNetwork}
{@const net = unconnectedNetworks.find(n => n.id === selectedNetwork)}
<span class="flex items-center gap-2">
<Network class="w-3 h-3 text-muted-foreground" />
{net?.name || 'Unknown'}
<Badge variant="outline" class="text-[10px] px-1 py-0">{net?.driver}</Badge>
</span>
{:else}
<span class="text-muted-foreground">
{networksLoading ? 'Loading networks...' : unconnectedNetworks.length > 0 ? 'Join a network...' : 'No networks available'}
</span>
{/if}
</Select.Trigger>
<Select.Content>
{#each unconnectedNetworks as network}
<Select.Item value={network.id}>
<span class="flex items-center gap-2">
<Network class="w-3 h-3 text-muted-foreground" />
{network.name}
<Badge variant="outline" class="text-[10px] px-1 py-0 ml-auto">{network.driver}</Badge>
</span>
</Select.Item>
{/each}
</Select.Content>
</Select.Root>
<Button
size="sm"
class="h-8"
disabled={!selectedNetwork || networkConnecting}
onclick={connectToNetwork}
>
{#if networkConnecting}
<Loader2 class="w-3.5 h-3.5 mr-1 animate-spin" />
{:else}
<Link class="w-3.5 h-3.5 mr-1" />
{/if}
Join
</Button>
</div>
{/if}
</div>
<!-- Ports -->
{#if containerData.NetworkSettings?.Ports && Object.keys(containerData.NetworkSettings.Ports).length > 0}
{@const inspectParsedUrl = parseCustomUrl(containerData.Config?.Labels?.['dockhand.url'])}
<div class="space-y-2">
<h3 class="text-sm font-semibold">Port mappings</h3>
<div class="flex flex-wrap gap-2">
{#if inspectParsedUrl}
<div class="flex items-center gap-2 text-xs p-2 bg-primary/10 rounded">
<a
href={inspectParsedUrl.url}
target="_blank"
rel="noopener noreferrer"
class="inline-flex items-center gap-1 text-primary hover:underline"
title="Open {inspectParsedUrl.url}"
>
<Globe class="w-3 h-3" />
<span>{inspectParsedUrl.name || inspectParsedUrl.url.replace(/^https?:\/\//, '')}</span>
<ExternalLink class="w-3 h-3 opacity-60" />
</a>
</div>
{/if}
{#each Object.entries(containerData.NetworkSettings.Ports) as [containerPort, hostBindings]}
{#if hostBindings && hostBindings.length > 0}
{#each hostBindings as binding}
{@const portParsedOverride = parseCustomUrl(containerData.Config?.Labels?.[`dockhand.port.${binding.HostPort}.url`])}
{@const url = portParsedOverride?.url || getPortUrl(parseInt(binding.HostPort))}
<div class="flex items-center gap-2 text-xs p-2 bg-muted rounded">
{#if url}
<a
href={url}
target="_blank"
rel="noopener noreferrer"
class="inline-flex items-center gap-1 text-primary hover:underline"
title="Open {url}"
>
<code>{portParsedOverride?.name ?? `${binding.HostIp || '0.0.0.0'}:${binding.HostPort}`}</code>
<ExternalLink class="w-3 h-3" />
</a>
{:else}
<code>{binding.HostIp || '0.0.0.0'}:{binding.HostPort}</code>
{/if}
<span class="text-muted-foreground"></span>
<code>{containerPort}</code>
</div>
{/each}
{:else}
<div class="flex items-center gap-2 text-xs p-2 bg-amber-500/10 border border-amber-500/20 rounded">
<code class="text-amber-600 dark:text-amber-400">exposed</code>
<code class="text-amber-600 dark:text-amber-400">{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.divergence?.env?.length > 0}
<div class="flex items-start gap-2 text-xs p-2.5 rounded border border-amber-500/30 bg-amber-500/5 text-amber-700 dark:text-amber-300">
<Info class="w-3.5 h-3.5 shrink-0 mt-0.5" />
<div class="min-w-0">
{containerData.divergence.env.length} env var{containerData.divergence.env.length === 1 ? '' : 's'} differ from the image:
<span class="font-mono">{containerData.divergence.env.join(', ')}</span>.
Values set by you at create time will stay. To reset to the image's current values, Remove &amp; Deploy.
</div>
</div>
{/if}
{#if containerData.Config?.Env && containerData.Config.Env.length > 0}
<div class="space-y-1">
{#each [...containerData.Config.Env].sort((a, b) => a.split('=')[0].localeCompare(b.split('=')[0])) as envVar}
{@const [key, ...valueParts] = envVar.split('=')}
{@const value = valueParts.join('=')}
{@const diverges = containerData.divergence?.env?.includes(key)}
<div class="text-xs p-2 rounded {diverges ? 'bg-amber-500/10 border border-amber-500/30' : 'bg-muted'}">
<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>
<!-- Labels Tab -->
<Tabs.Content value="labels" class="space-y-3 overflow-auto">
{#if containerData.divergence?.labels?.length > 0}
<div class="flex items-start gap-2 text-xs p-2.5 rounded border border-amber-500/30 bg-amber-500/5 text-amber-700 dark:text-amber-300">
<Info class="w-3.5 h-3.5 shrink-0 mt-0.5" />
<div class="min-w-0">
{containerData.divergence.labels.length} label{containerData.divergence.labels.length === 1 ? '' : 's'} differ from the image:
<span class="font-mono">{containerData.divergence.labels.join(', ')}</span>.
Values set by you at create time will stay. To reset to the image's current values, Remove &amp; Deploy.
</div>
</div>
{/if}
{#if containerData.Config?.Labels && Object.keys(containerData.Config.Labels).length > 0}
{@const allLabels = Object.entries(containerData.Config.Labels).sort((a, b) => a[0].localeCompare(b[0]))}
{@const filter = labelFilter.trim().toLowerCase()}
{@const visibleLabels = filter
? allLabels.filter(([k, v]) => k.toLowerCase().includes(filter) || String(v).toLowerCase().includes(filter))
: allLabels}
<div class="flex items-center gap-2">
<Input
type="search"
placeholder="Filter labels..."
bind:value={labelFilter}
class="h-8 text-xs flex-1"
/>
<span class="text-xs text-muted-foreground shrink-0">
{visibleLabels.length === allLabels.length
? `${allLabels.length} label${allLabels.length === 1 ? '' : 's'}`
: `${visibleLabels.length} of ${allLabels.length}`}
</span>
<Button
variant="outline"
size="sm"
onclick={() => copyAllLabels(visibleLabels)}
disabled={visibleLabels.length === 0}
title={copiedAllLabels ? 'Copied!' : 'Copy visible labels as key=value lines'}
>
{#if copiedAllLabels}
<Check class="w-3 h-3 mr-1.5 text-green-500" />
Copied
{:else}
<Copy class="w-3 h-3 mr-1.5" />
Copy all
{/if}
</Button>
</div>
{#if visibleLabels.length > 0}
<div class="space-y-1">
{#each visibleLabels as [key, value]}
{@const diverges = containerData.divergence?.labels?.includes(key)}
<div class="text-xs p-2 rounded flex items-start gap-2 group {diverges ? 'bg-amber-500/10 border border-amber-500/30' : 'bg-muted'}">
<div class="flex-1 min-w-0">
<code class="text-muted-foreground font-medium">{key}</code>
<code class="text-muted-foreground">=</code>
<code class="break-all">{value}</code>
</div>
<button
type="button"
onclick={() => copyLabel(key, value)}
class="shrink-0 p-1 rounded hover:bg-background/50 transition-colors opacity-0 group-hover:opacity-100 {copiedLabel === key ? '!opacity-100' : ''}"
title={copiedLabel === key ? 'Copied!' : 'Copy label'}
>
{#if copiedLabel === key}
<Check class="w-3 h-3 text-green-500" />
{:else}
<Copy class="w-3 h-3 text-muted-foreground" />
{/if}
</button>
</div>
{/each}
</div>
{:else}
<p class="text-sm text-muted-foreground">No labels match "{labelFilter}"</p>
{/if}
{:else}
<p class="text-sm text-muted-foreground">No labels</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}
<!-- GPU / Device Requests -->
{#if containerData.HostConfig?.DeviceRequests?.length > 0 || (containerData.HostConfig?.Runtime && containerData.HostConfig.Runtime !== 'runc')}
<div class="space-y-2">
<h3 class="text-sm font-semibold flex items-center gap-2">
<Gpu class="w-4 h-4" />
GPU
</h3>
<div class="grid grid-cols-2 lg:grid-cols-3 gap-3">
{#if containerData.HostConfig?.Runtime}
<div class="p-3 border border-border rounded-lg">
<p class="text-xs text-muted-foreground mb-1">Runtime</p>
<code class="text-sm">{containerData.HostConfig.Runtime}</code>
</div>
{/if}
{#if containerData.HostConfig?.DeviceRequests?.length > 0}
{@const req = containerData.HostConfig.DeviceRequests[0]}
<div class="p-3 border border-border rounded-lg">
<p class="text-xs text-muted-foreground mb-1">Count</p>
<code class="text-sm">{req.Count === -1 ? 'All' : req.Count}</code>
</div>
{#if req.Driver}
<div class="p-3 border border-border rounded-lg">
<p class="text-xs text-muted-foreground mb-1">Driver</p>
<code class="text-sm">{req.Driver}</code>
</div>
{/if}
{#if req.DeviceIDs?.length > 0}
<div class="p-3 border border-border rounded-lg col-span-full">
<p class="text-xs text-muted-foreground mb-1">Device IDs</p>
<div class="flex flex-wrap gap-1.5">
{#each req.DeviceIDs as id}
<Badge variant="secondary" class="text-2xs">{id}</Badge>
{/each}
</div>
</div>
{/if}
{#if req.Capabilities?.length > 0}
<div class="p-3 border border-border rounded-lg col-span-full">
<p class="text-xs text-muted-foreground mb-1">Capabilities</p>
<div class="flex flex-wrap gap-1.5">
{#each req.Capabilities.flat() as cap}
<Badge variant="outline" class="text-2xs bg-violet-50 text-violet-700 dark:bg-violet-900/30 dark:text-violet-400">{cap}</Badge>
{/each}
</div>
</div>
{/if}
{/if}
</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="flex flex-col overflow-hidden">
{@const healthConfig = containerData.Config?.Healthcheck}
{@const healthState = containerData.State?.Health}
{@const formatNs = (ns: number) => ns ? `${ns / 1e9}s` : '-'}
{#if healthConfig || healthState}
<div class="flex flex-col flex-1 min-h-0 gap-4">
<!-- Healthcheck Configuration -->
{#if healthConfig && healthConfig.Test && healthConfig.Test.length > 0}
<div class="shrink-0">
<h3 class="text-sm font-semibold mb-2">Configuration</h3>
<div class="grid grid-cols-2 gap-3 text-sm">
<div class="col-span-2">
<p class="text-muted-foreground">Command</p>
<code class="text-xs break-all">{healthConfig.Test.join(' ')}</code>
</div>
<div>
<p class="text-muted-foreground">Interval</p>
<code class="text-xs">{formatNs(healthConfig.Interval)}</code>
</div>
<div>
<p class="text-muted-foreground">Timeout</p>
<code class="text-xs">{formatNs(healthConfig.Timeout)}</code>
</div>
<div>
<p class="text-muted-foreground">Retries</p>
<code class="text-xs">{healthConfig.Retries || '-'}</code>
</div>
<div>
<p class="text-muted-foreground">Start period</p>
<code class="text-xs">{formatNs(healthConfig.StartPeriod)}</code>
</div>
</div>
</div>
{/if}
<!-- Runtime Status -->
{#if healthState}
<div class="shrink-0">
<h3 class="text-sm font-semibold mb-2">Status</h3>
<div class="grid grid-cols-2 gap-3 text-sm">
<div>
<p class="text-muted-foreground">Current status</p>
<Badge variant={healthState.Status === 'healthy' ? 'default' : healthState.Status === 'starting' ? 'secondary' : 'destructive'}>
{healthState.Status}
</Badge>
</div>
<div>
<p class="text-muted-foreground">Failing streak</p>
<code class="text-xs">{healthState.FailingStreak || 0}</code>
</div>
</div>
</div>
{#if healthState.Log && healthState.Log.length > 0}
<div class="flex flex-col flex-1 min-h-0">
<h3 class="text-sm font-semibold mb-2 shrink-0">Health check log</h3>
<div class="space-y-1 overflow-y-auto flex-1">
{#each healthState.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}
{:else if healthConfig}
<p class="text-sm text-muted-foreground">Waiting for first health check to complete...</p>
{/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>
<!-- Inspect (raw) modal -->
<Dialog.Root bind:open={showRawJson}>
<Dialog.Content class="max-w-4xl max-h-[90vh] sm:max-h-[80vh] flex flex-col">
<Dialog.Header class="shrink-0">
<Dialog.Title class="flex items-center gap-2">
<Code class="w-5 h-5" />
Inspect
<Button
variant="outline"
size="sm"
onclick={copyJson}
title={jsonCopied === 'ok' ? 'Copied!' : 'Copy to clipboard'}
>
{#if jsonCopied === 'error'}
<Tooltip.Root open>
<Tooltip.Trigger>
<XCircle class="w-4 h-4 mr-1.5 text-red-500" />
</Tooltip.Trigger>
<Tooltip.Content>Copy requires HTTPS</Tooltip.Content>
</Tooltip.Root>
<span class="text-red-500">Failed</span>
{:else if jsonCopied === 'ok'}
<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>