mirror of
https://github.com/khoaliber/dockhand.git
synced 2026-03-02 21:19:05 +00:00
1231 lines
44 KiB
Svelte
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, '&')
|
|
.replace(/</g, '<')
|
|
.replace(/>/g, '>')
|
|
.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>
|
|
|