Files
dockhand/routes/terminal/[id]/+page.svelte
Jarek Krochmalski 62e3c6439e Initial commit
2025-12-28 21:16:03 +01:00

252 lines
6.4 KiB
Svelte

<script lang="ts">
import { page } from '$app/stores';
import { onMount, onDestroy } from 'svelte';
import { Terminal as TerminalIcon } from 'lucide-svelte';
// Dynamic imports for browser-only xterm
let Terminal: any;
let FitAddon: any;
let WebLinksAddon: any;
let xtermLoaded = $state(false);
let terminalRef: HTMLDivElement | undefined;
let terminal: any = null;
let fitAddon: any = null;
let ws: WebSocket | null = null;
let connected = $state(false);
let error = $state<string | null>(null);
let containerName = $state('');
// Get params from URL
let containerId = $derived($page.params.id);
let shell = $derived($page.url.searchParams.get('shell') || '/bin/bash');
let user = $derived($page.url.searchParams.get('user') || 'root');
let name = $derived($page.url.searchParams.get('name') || 'Container');
function initTerminal() {
if (!terminalRef || terminal || !xtermLoaded) return;
containerName = name;
terminal = new Terminal({
cursorBlink: true,
fontFamily: 'Menlo, Monaco, "Courier New", monospace',
fontSize: 14,
theme: {
background: '#0c0c0c',
foreground: '#cccccc',
cursor: '#ffffff',
cursorAccent: '#000000',
selectionBackground: '#264f78',
black: '#0c0c0c',
red: '#c50f1f',
green: '#13a10e',
yellow: '#c19c00',
blue: '#0037da',
magenta: '#881798',
cyan: '#3a96dd',
white: '#cccccc',
brightBlack: '#767676',
brightRed: '#e74856',
brightGreen: '#16c60c',
brightYellow: '#f9f1a5',
brightBlue: '#3b78ff',
brightMagenta: '#b4009e',
brightCyan: '#61d6d6',
brightWhite: '#f2f2f2'
}
});
fitAddon = new FitAddon();
terminal.loadAddon(fitAddon);
terminal.loadAddon(new WebLinksAddon());
terminal.open(terminalRef);
fitAddon.fit();
// Handle terminal input
terminal.onData((data: string) => {
if (ws && ws.readyState === WebSocket.OPEN) {
ws.send(JSON.stringify({ type: 'input', data }));
}
});
// Handle resize
terminal.onResize(({ cols, rows }: { cols: number; rows: number }) => {
if (ws && ws.readyState === WebSocket.OPEN) {
ws.send(JSON.stringify({ type: 'resize', cols, rows }));
}
});
// Connect to container
connect();
}
function connect() {
if (!terminal) return;
error = null;
const protocol = window.location.protocol === 'https:' ? 'wss:' : 'ws:';
const wsUrl = `${protocol}//${window.location.host}/api/containers/${containerId}/exec?shell=${encodeURIComponent(shell)}&user=${encodeURIComponent(user)}`;
terminal.writeln(`\x1b[90mConnecting to ${name}...\x1b[0m`);
terminal.writeln(`\x1b[90mShell: ${shell}, User: ${user || 'default'}\x1b[0m`);
terminal.writeln('');
ws = new WebSocket(wsUrl);
ws.onopen = () => {
connected = true;
document.title = `Terminal - ${name}`;
terminal?.focus();
// Send initial resize
if (fitAddon && terminal) {
const dims = fitAddon.proposeDimensions();
if (dims) {
ws?.send(JSON.stringify({ type: 'resize', cols: dims.cols, rows: dims.rows }));
}
}
};
ws.onmessage = (event) => {
try {
const msg = JSON.parse(event.data);
if (msg.type === 'output') {
terminal?.write(msg.data);
} else if (msg.type === 'error') {
error = msg.message;
terminal?.writeln(`\x1b[31mError: ${msg.message}\x1b[0m`);
} else if (msg.type === 'exit') {
terminal?.writeln('\x1b[90m\r\nSession ended.\x1b[0m');
connected = false;
// Close the window after a brief delay so user sees the message
setTimeout(() => {
window.close();
}, 500);
}
} catch (e) {
terminal?.write(event.data);
}
};
ws.onerror = (e) => {
console.error('WebSocket error:', e);
error = 'Connection error';
terminal?.writeln('\x1b[31mConnection error\x1b[0m');
};
ws.onclose = () => {
connected = false;
terminal?.writeln('\x1b[90mDisconnected.\x1b[0m');
};
}
function disconnect() {
if (ws) {
ws.close();
ws = null;
}
}
function cleanup() {
disconnect();
if (terminal) {
terminal.dispose();
terminal = null;
}
fitAddon = null;
}
// Handle window resize
function handleResize() {
if (fitAddon && terminal) {
fitAddon.fit();
}
}
onMount(async () => {
window.addEventListener('resize', handleResize);
// Dynamically load xterm modules (browser only)
const xtermModule = await import('@xterm/xterm');
const fitModule = await import('@xterm/addon-fit');
const webLinksModule = await import('@xterm/addon-web-links');
// Handle both ESM and CommonJS exports
Terminal = xtermModule.Terminal || xtermModule.default?.Terminal;
FitAddon = fitModule.FitAddon || fitModule.default?.FitAddon;
WebLinksAddon = webLinksModule.WebLinksAddon || webLinksModule.default?.WebLinksAddon;
// Load CSS
await import('@xterm/xterm/css/xterm.css');
xtermLoaded = true;
// Initialize terminal after xterm is loaded
setTimeout(() => {
initTerminal();
}, 100);
});
onDestroy(() => {
window.removeEventListener('resize', handleResize);
cleanup();
});
</script>
<svelte:head>
<title>Terminal - {containerName || 'Loading...'}</title>
</svelte:head>
<div class="h-screen w-screen flex flex-col bg-[#0c0c0c]">
<!-- Header -->
<div class="flex items-center justify-between px-4 py-2 bg-zinc-900 border-b border-zinc-800 flex-shrink-0">
<div class="flex items-center gap-2">
<TerminalIcon class="w-4 h-4 text-zinc-400" />
<span class="text-sm text-zinc-200 font-medium">{containerName}</span>
{#if connected}
<span class="inline-flex items-center gap-1 text-xs text-green-500">
<span class="w-2 h-2 rounded-full bg-green-500 animate-pulse"></span>
Connected
</span>
{:else if error}
<span class="text-xs text-red-500">{error}</span>
{:else}
<span class="text-xs text-zinc-500">Connecting...</span>
{/if}
</div>
<div class="flex items-center gap-2 text-xs text-zinc-500">
<span>Shell: {shell}</span>
<span>|</span>
<span>User: {user}</span>
</div>
</div>
<!-- Terminal -->
<div class="flex-1 p-2 overflow-hidden">
{#if xtermLoaded}
<div bind:this={terminalRef} class="h-full w-full"></div>
{:else}
<div class="h-full w-full flex items-center justify-center">
<span class="text-zinc-500">Loading terminal...</span>
</div>
{/if}
</div>
</div>
<style>
:global(body) {
margin: 0;
padding: 0;
overflow: hidden;
}
:global(.xterm) {
height: 100%;
padding: 8px;
}
:global(.xterm-viewport) {
overflow-y: auto !important;
}
</style>