teemux 1.4.0 → 1.5.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/teemux.js +137 -17
- package/dist/teemux.js.map +1 -1
- package/package.json +1 -1
- package/src/LogServer.ts +163 -14
- package/src/teemux.ts +5 -5
- package/src/testing/runWithTeemux.ts +5 -5
package/dist/teemux.js
CHANGED
|
@@ -177,7 +177,7 @@ var LogServer = class {
|
|
|
177
177
|
port;
|
|
178
178
|
server = null;
|
|
179
179
|
tailSize;
|
|
180
|
-
constructor(port, tailSize =
|
|
180
|
+
constructor(port, tailSize = 1e4) {
|
|
181
181
|
this.port = port;
|
|
182
182
|
this.tailSize = tailSize;
|
|
183
183
|
}
|
|
@@ -191,6 +191,33 @@ var LogServer = class {
|
|
|
191
191
|
start() {
|
|
192
192
|
return new Promise((resolve, reject) => {
|
|
193
193
|
this.server = http.createServer((request, response) => {
|
|
194
|
+
if (request.method === "GET" && request.url?.startsWith("/search")) {
|
|
195
|
+
const url = new URL(request.url, `http://${request.headers.host}`);
|
|
196
|
+
const includeParameter = url.searchParams.get("include");
|
|
197
|
+
const includes = includeParameter ? includeParameter.split(",").map((term) => term.trim()).filter(Boolean) : [];
|
|
198
|
+
const excludeParameter = url.searchParams.get("exclude");
|
|
199
|
+
const excludes = excludeParameter ? excludeParameter.split(",").map((pattern) => pattern.trim()).filter(Boolean) : [];
|
|
200
|
+
const limit = Math.min(Number.parseInt(url.searchParams.get("limit") ?? "1000", 10), 1e3);
|
|
201
|
+
const sortedBuffer = this.buffer.toSorted((a, b) => a.timestamp - b.timestamp);
|
|
202
|
+
const results = [];
|
|
203
|
+
for (const entry of sortedBuffer) if (matchesFilters(entry.line, includes, excludes)) {
|
|
204
|
+
let html = this.ansiConverter.toHtml(entry.line);
|
|
205
|
+
html = highlightJson(html);
|
|
206
|
+
html = linkifyUrls(html);
|
|
207
|
+
results.push({
|
|
208
|
+
html,
|
|
209
|
+
raw: stripAnsi(entry.line)
|
|
210
|
+
});
|
|
211
|
+
if (results.length >= limit) break;
|
|
212
|
+
}
|
|
213
|
+
response.writeHead(200, {
|
|
214
|
+
"Access-Control-Allow-Origin": "*",
|
|
215
|
+
"Cache-Control": "no-cache",
|
|
216
|
+
"Content-Type": "application/json; charset=utf-8"
|
|
217
|
+
});
|
|
218
|
+
response.end(JSON.stringify(results));
|
|
219
|
+
return;
|
|
220
|
+
}
|
|
194
221
|
if (request.method === "GET" && request.url?.startsWith("/")) {
|
|
195
222
|
const url = new URL(request.url, `http://${request.headers.host}`);
|
|
196
223
|
const includeParameter = url.searchParams.get("include");
|
|
@@ -207,7 +234,8 @@ var LogServer = class {
|
|
|
207
234
|
"X-Content-Type-Options": "nosniff"
|
|
208
235
|
});
|
|
209
236
|
response.write(this.getHtmlHeader());
|
|
210
|
-
|
|
237
|
+
const initialLogs = sortedBuffer.slice(-1e3);
|
|
238
|
+
for (const entry of initialLogs) response.write(this.getHtmlLine(entry.line));
|
|
211
239
|
} else {
|
|
212
240
|
const filteredBuffer = sortedBuffer.filter((entry) => matchesFilters(entry.line, includes, excludes));
|
|
213
241
|
response.writeHead(200, {
|
|
@@ -447,7 +475,7 @@ var LogServer = class {
|
|
|
447
475
|
const highlightInput = document.getElementById('highlight');
|
|
448
476
|
const tailBtn = document.getElementById('tail-btn');
|
|
449
477
|
const params = new URLSearchParams(window.location.search);
|
|
450
|
-
const tailSize = ${this.tailSize};
|
|
478
|
+
const tailSize = Math.min(${this.tailSize}, 1000);
|
|
451
479
|
|
|
452
480
|
includeInput.value = params.get('include') || '';
|
|
453
481
|
excludeInput.value = params.get('exclude') || '';
|
|
@@ -504,7 +532,7 @@ var LogServer = class {
|
|
|
504
532
|
return result;
|
|
505
533
|
};
|
|
506
534
|
|
|
507
|
-
const
|
|
535
|
+
const applyFiltersLocal = () => {
|
|
508
536
|
const includes = includeInput.value.split(',').map(s => s.trim()).filter(Boolean);
|
|
509
537
|
const excludes = excludeInput.value.split(',').map(s => s.trim()).filter(Boolean);
|
|
510
538
|
const highlights = highlightInput.value.split(',').map(s => s.trim()).filter(Boolean);
|
|
@@ -525,6 +553,15 @@ var LogServer = class {
|
|
|
525
553
|
contentEl.innerHTML = html;
|
|
526
554
|
}
|
|
527
555
|
});
|
|
556
|
+
};
|
|
557
|
+
|
|
558
|
+
let lastSearchQuery = '';
|
|
559
|
+
let searchController = null;
|
|
560
|
+
|
|
561
|
+
const applyFilters = async () => {
|
|
562
|
+
const includes = includeInput.value.split(',').map(s => s.trim()).filter(Boolean);
|
|
563
|
+
const excludes = excludeInput.value.split(',').map(s => s.trim()).filter(Boolean);
|
|
564
|
+
const highlights = highlightInput.value.split(',').map(s => s.trim()).filter(Boolean);
|
|
528
565
|
|
|
529
566
|
// Update URL without reload
|
|
530
567
|
const newParams = new URLSearchParams();
|
|
@@ -534,10 +571,93 @@ var LogServer = class {
|
|
|
534
571
|
const newUrl = newParams.toString() ? '?' + newParams.toString() : window.location.pathname;
|
|
535
572
|
history.replaceState(null, '', newUrl);
|
|
536
573
|
|
|
537
|
-
//
|
|
538
|
-
|
|
539
|
-
|
|
540
|
-
|
|
574
|
+
// Build search query string for comparison
|
|
575
|
+
const searchQuery = includeInput.value + '|' + excludeInput.value;
|
|
576
|
+
|
|
577
|
+
// If only highlight changed, just re-apply local highlighting
|
|
578
|
+
if (searchQuery === lastSearchQuery) {
|
|
579
|
+
applyFiltersLocal();
|
|
580
|
+
return;
|
|
581
|
+
}
|
|
582
|
+
|
|
583
|
+
lastSearchQuery = searchQuery;
|
|
584
|
+
|
|
585
|
+
// Cancel any pending search request
|
|
586
|
+
if (searchController) {
|
|
587
|
+
searchController.abort();
|
|
588
|
+
}
|
|
589
|
+
|
|
590
|
+
// If no filters, just apply local filtering (show all)
|
|
591
|
+
if (includes.length === 0 && excludes.length === 0) {
|
|
592
|
+
applyFiltersLocal();
|
|
593
|
+
container.scrollTop = container.scrollHeight;
|
|
594
|
+
tailing = true;
|
|
595
|
+
updateTailButton();
|
|
596
|
+
return;
|
|
597
|
+
}
|
|
598
|
+
|
|
599
|
+
// Fetch matching logs from server
|
|
600
|
+
searchController = new AbortController();
|
|
601
|
+
const searchParams = new URLSearchParams();
|
|
602
|
+
if (includeInput.value) searchParams.set('include', includeInput.value);
|
|
603
|
+
if (excludeInput.value) searchParams.set('exclude', excludeInput.value);
|
|
604
|
+
searchParams.set('limit', '1000');
|
|
605
|
+
|
|
606
|
+
try {
|
|
607
|
+
const response = await fetch('/search?' + searchParams.toString(), {
|
|
608
|
+
signal: searchController.signal
|
|
609
|
+
});
|
|
610
|
+
const results = await response.json();
|
|
611
|
+
|
|
612
|
+
// Clear non-pinned lines
|
|
613
|
+
document.querySelectorAll('.line').forEach(line => {
|
|
614
|
+
if (!pinnedIds.has(line.dataset.id)) {
|
|
615
|
+
line.remove();
|
|
616
|
+
}
|
|
617
|
+
});
|
|
618
|
+
|
|
619
|
+
// Add search results
|
|
620
|
+
for (const item of results) {
|
|
621
|
+
const id = 'line-' + (lineCounter++);
|
|
622
|
+
const div = document.createElement('div');
|
|
623
|
+
div.className = 'line';
|
|
624
|
+
div.dataset.id = id;
|
|
625
|
+
div.dataset.raw = item.raw;
|
|
626
|
+
div.dataset.html = item.html;
|
|
627
|
+
|
|
628
|
+
let displayHtml = item.html;
|
|
629
|
+
displayHtml = highlightTerms(displayHtml, includes, 'filter');
|
|
630
|
+
displayHtml = highlightTerms(displayHtml, highlights);
|
|
631
|
+
|
|
632
|
+
div.innerHTML = '<span class="line-content">' + displayHtml + '</span><span class="pin-btn" title="Pin">' + pinIcon + '</span>';
|
|
633
|
+
|
|
634
|
+
// Pin button handler
|
|
635
|
+
div.querySelector('.pin-btn').addEventListener('click', (e) => {
|
|
636
|
+
e.stopPropagation();
|
|
637
|
+
if (pinnedIds.has(id)) {
|
|
638
|
+
pinnedIds.delete(id);
|
|
639
|
+
div.classList.remove('pinned');
|
|
640
|
+
} else {
|
|
641
|
+
pinnedIds.add(id);
|
|
642
|
+
div.classList.add('pinned');
|
|
643
|
+
}
|
|
644
|
+
applyFiltersLocal();
|
|
645
|
+
});
|
|
646
|
+
|
|
647
|
+
container.appendChild(div);
|
|
648
|
+
}
|
|
649
|
+
|
|
650
|
+
// Jump to bottom and resume tailing
|
|
651
|
+
container.scrollTop = container.scrollHeight;
|
|
652
|
+
tailing = true;
|
|
653
|
+
updateTailButton();
|
|
654
|
+
} catch (e) {
|
|
655
|
+
if (e.name !== 'AbortError') {
|
|
656
|
+
console.error('Search failed:', e);
|
|
657
|
+
// Fallback to local filtering
|
|
658
|
+
applyFiltersLocal();
|
|
659
|
+
}
|
|
660
|
+
}
|
|
541
661
|
};
|
|
542
662
|
|
|
543
663
|
const trimBuffer = () => {
|
|
@@ -580,7 +700,7 @@ var LogServer = class {
|
|
|
580
700
|
pinnedIds.add(id);
|
|
581
701
|
div.classList.add('pinned');
|
|
582
702
|
}
|
|
583
|
-
|
|
703
|
+
applyFiltersLocal();
|
|
584
704
|
});
|
|
585
705
|
|
|
586
706
|
const matches = matchesFilters(raw, includes, excludes);
|
|
@@ -609,9 +729,9 @@ var LogServer = class {
|
|
|
609
729
|
debounceTimer = setTimeout(fn, delay);
|
|
610
730
|
};
|
|
611
731
|
|
|
612
|
-
includeInput.addEventListener('input', () => debounce(applyFilters,
|
|
613
|
-
excludeInput.addEventListener('input', () => debounce(applyFilters,
|
|
614
|
-
highlightInput.addEventListener('input', () => debounce(applyFilters,
|
|
732
|
+
includeInput.addEventListener('input', () => debounce(applyFilters, 300));
|
|
733
|
+
excludeInput.addEventListener('input', () => debounce(applyFilters, 300));
|
|
734
|
+
highlightInput.addEventListener('input', () => debounce(applyFilters, 150));
|
|
615
735
|
<\/script>
|
|
616
736
|
`;
|
|
617
737
|
}
|
|
@@ -838,10 +958,10 @@ const main = async () => {
|
|
|
838
958
|
default: 8336,
|
|
839
959
|
description: "Port for the log aggregation server",
|
|
840
960
|
type: "number"
|
|
841
|
-
}).option("
|
|
842
|
-
alias: "
|
|
843
|
-
default:
|
|
844
|
-
description: "Number of log lines to keep in buffer",
|
|
961
|
+
}).option("buffer", {
|
|
962
|
+
alias: "b",
|
|
963
|
+
default: 1e4,
|
|
964
|
+
description: "Number of log lines to keep in server buffer",
|
|
845
965
|
type: "number"
|
|
846
966
|
}).help().parse();
|
|
847
967
|
const command = argv._;
|
|
@@ -852,7 +972,7 @@ const main = async () => {
|
|
|
852
972
|
}
|
|
853
973
|
const name = argv.name ?? command[0] ?? "unknown";
|
|
854
974
|
const port = argv.port;
|
|
855
|
-
const server = new LogServer(port, argv.
|
|
975
|
+
const server = new LogServer(port, argv.buffer);
|
|
856
976
|
let isServer = false;
|
|
857
977
|
const maxRetries = 3;
|
|
858
978
|
for (let attempt = 0; attempt < maxRetries; attempt++) try {
|
package/dist/teemux.js.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"teemux.js","names":["RESET","DIM","RED"],"sources":["../src/utils/stripHtmlTags.ts","../src/utils/unescapeHtml.ts","../src/utils/highlightJson.ts","../src/utils/linkifyUrls.ts","../src/utils/stripAnsi.ts","../src/utils/matchesFilters.ts","../src/LogServer.ts","../src/teemux.ts"],"sourcesContent":["/**\n * Strip HTML tags from a string, leaving only text content.\n */\nexport const stripHtmlTags = (html: string): string => {\n return html.replaceAll(/<[^>]*>/gu, '');\n};\n","/**\n * Unescape HTML entities back to their original characters.\n */\nexport const unescapeHtml = (text: string): string => {\n return text\n .replaceAll('"', '\"')\n .replaceAll('&', '&')\n .replaceAll('<', '<')\n .replaceAll('>', '>')\n .replaceAll(''', \"'\")\n .replaceAll(''', \"'\");\n};\n","import { stripHtmlTags } from './stripHtmlTags.js';\nimport { unescapeHtml } from './unescapeHtml.js';\n\n/**\n * Apply syntax highlighting to JSON text that uses HTML-escaped quotes (").\n * Uses placeholder technique to avoid double-wrapping strings.\n */\nexport const highlightJsonText = (text: string): string => {\n // First, extract and mark all JSON strings with placeholders\n const strings: string[] = [];\n let result = text.replaceAll(\n /"((?:(?!").)*)"/gu,\n (_match, content) => {\n strings.push(content as string);\n return `\\u0000STR${strings.length - 1}\\u0000`;\n },\n );\n\n // Booleans and null\n result = result.replaceAll(\n /\\b(true|false|null)\\b/gu,\n '<span class=\"json-bool\">$1</span>',\n );\n\n // Numbers\n result = result.replaceAll(\n /(?<!\\w)(-?\\d+(?:\\.\\d+)?(?:[eE][+-]?\\d+)?)\\b/gu,\n '<span class=\"json-number\">$1</span>',\n );\n\n // Restore strings with appropriate highlighting\n result = result.replaceAll(\n /\\0STR(\\d+)\\0(\\s*:)?/gu,\n (_match, index, colon) => {\n const content = strings[Number.parseInt(index as string, 10)];\n if (colon) {\n // This is a key\n return `<span class=\"json-key\">"${content}"</span>${colon}`;\n }\n\n // This is a value\n return `<span class=\"json-string\">"${content}"</span>`;\n },\n );\n\n return result;\n};\n\n/**\n * Process HTML text, applying JSON highlighting only to text outside of HTML tags.\n */\nexport const syntaxHighlightJson = (html: string): string => {\n let result = '';\n let index = 0;\n\n while (index < html.length) {\n if (html[index] === '<') {\n // Find end of tag\n const tagEnd = html.indexOf('>', index);\n if (tagEnd === -1) {\n result += html.slice(index);\n break;\n }\n\n result += html.slice(index, tagEnd + 1);\n index = tagEnd + 1;\n } else {\n // Find next tag or end of string\n const nextTag = html.indexOf('<', index);\n const textEnd = nextTag === -1 ? html.length : nextTag;\n const text = html.slice(index, textEnd);\n\n // Highlight JSON syntax in this text segment\n result += highlightJsonText(text);\n index = textEnd;\n }\n }\n\n return result;\n};\n\n/**\n * Detect if the content (after prefix) is valid JSON and apply syntax highlighting.\n * Returns the original HTML if not valid JSON.\n */\nexport const highlightJson = (html: string): string => {\n // Extract the text content (strip HTML tags) to check if it's JSON\n const textContent = stripHtmlTags(html);\n\n // Unescape HTML entities for JSON parsing\n const unescaped = unescapeHtml(textContent);\n\n // Find where the actual log content starts (after the prefix like [name])\n const prefixMatch = /^\\[[\\w-]+\\]\\s*/u.exec(unescaped);\n const prefix = prefixMatch?.[0] ?? '';\n const content = unescaped.slice(prefix.length).trim();\n\n // Check if the content is valid JSON\n if (!content.startsWith('{') && !content.startsWith('[')) {\n return html;\n }\n\n try {\n JSON.parse(content);\n } catch {\n return html;\n }\n\n // It's valid JSON - now highlight it\n // Find the position after the prefix span in the HTML\n const prefixHtmlMatch = /^<span[^>]*>\\[[^\\]]+\\]<\\/span>\\s*/u.exec(html);\n const htmlPrefix = prefixHtmlMatch?.[0] ?? '';\n const jsonHtml = html.slice(htmlPrefix.length);\n\n // Apply syntax highlighting to the JSON portion\n const highlighted = syntaxHighlightJson(jsonHtml);\n\n return htmlPrefix + highlighted;\n};\n","/**\n * Convert URLs in HTML text to clickable anchor tags.\n * Supports http://, https://, and file:// URLs.\n * Avoids double-linking URLs that are already in href attributes.\n */\nexport const linkifyUrls = (html: string): string => {\n // Match URLs that are not already inside href attributes\n // Supports http://, https://, and file:// URLs\n // Exclude common delimiters and HTML entities (" & etc)\n const urlRegex = /(?<!href=[\"'])(?:https?|file):\\/\\/[^\\s<>\"'{}&]+/gu;\n\n return html.replaceAll(urlRegex, (url) => {\n // Remove trailing punctuation that's likely not part of the URL\n const cleanUrl = url.replace(/[.,;:!?)\\]]+$/u, '');\n const trailing = url.slice(cleanUrl.length);\n\n // Escape HTML entities in the URL for the href attribute\n const escapedHref = cleanUrl\n .replaceAll('&', '&')\n .replaceAll('\"', '"');\n\n return `<a href=\"${escapedHref}\" target=\"_blank\" rel=\"noopener\">${cleanUrl}</a>${trailing}`;\n });\n};\n","/**\n * Strip ANSI escape codes from text.\n * Removes color codes and other terminal formatting sequences.\n */\nexport const stripAnsi = (text: string): string => {\n // eslint-disable-next-line no-control-regex\n return text.replaceAll(/\\u001B\\[[\\d;]*m/gu, '');\n};\n","import { stripAnsi } from './stripAnsi.js';\n\n/**\n * Convert a glob pattern (with * wildcards) to a RegExp.\n * - `*` matches any characters (zero or more)\n * - All other characters are escaped for literal matching\n */\nconst globToRegex = (pattern: string): RegExp => {\n // Escape regex special characters except *\n const escaped = pattern.replaceAll(/[$()+.?[\\\\\\]^{|}]/gu, '\\\\$&');\n // Convert * to .*\n const regexPattern = escaped.replaceAll('*', '.*');\n return new RegExp(regexPattern, 'iu');\n};\n\n/**\n * Check if text matches a pattern (supports * glob wildcards).\n * If no wildcards, does a simple substring match for better performance.\n */\nconst matchesPattern = (text: string, pattern: string): boolean => {\n if (pattern.includes('*')) {\n return globToRegex(pattern).test(text);\n }\n\n return text.includes(pattern.toLowerCase());\n};\n\n/**\n * Check if a line matches the given filter criteria.\n * @param line - The line to check (may contain ANSI codes)\n * @param includes - Patterns where ANY match includes the line (OR logic), case-insensitive. Supports * wildcards.\n * @param excludes - Patterns where ANY match excludes the line (OR logic), case-insensitive. Supports * wildcards.\n * @returns true if the line should be included, false if filtered out\n */\nexport const matchesFilters = (\n line: string,\n includes: string[],\n excludes: string[],\n): boolean => {\n const plainText = stripAnsi(line).toLowerCase();\n\n // Any include must match (OR logic) - case insensitive\n if (includes.length > 0) {\n const anyIncludeMatches = includes.some((pattern) =>\n matchesPattern(plainText, pattern),\n );\n\n if (!anyIncludeMatches) {\n return false;\n }\n }\n\n // None of the excludes should match (OR logic for exclusion) - case insensitive\n if (excludes.length > 0) {\n const anyExcludeMatches = excludes.some((pattern) =>\n matchesPattern(plainText, pattern),\n );\n\n if (anyExcludeMatches) {\n return false;\n }\n }\n\n return true;\n};\n","import { highlightJson } from './utils/highlightJson.js';\nimport { linkifyUrls } from './utils/linkifyUrls.js';\nimport { matchesFilters } from './utils/matchesFilters.js';\nimport { stripAnsi } from './utils/stripAnsi.js';\nimport Convert from 'ansi-to-html';\nimport http from 'node:http';\nimport { performance } from 'node:perf_hooks';\nimport { URL } from 'node:url';\n\nconst COLORS = [\n '\\u001B[36m',\n '\\u001B[33m',\n '\\u001B[32m',\n '\\u001B[35m',\n '\\u001B[34m',\n '\\u001B[91m',\n '\\u001B[92m',\n '\\u001B[93m',\n];\nconst RESET = '\\u001B[0m';\nconst DIM = '\\u001B[90m';\nconst RED = '\\u001B[91m';\nconst HOST = '0.0.0.0';\n\ntype BufferedLog = {\n line: string;\n timestamp: number;\n};\n\ntype EventPayload = {\n code?: number;\n event: 'exit' | 'start';\n name: string;\n pid: number;\n timestamp: number;\n};\n\ntype LogPayload = {\n line: string;\n name: string;\n timestamp: number;\n type: LogType;\n};\n\ntype LogType = 'stderr' | 'stdout';\n\ntype StreamClient = {\n excludes: string[];\n includes: string[];\n isBrowser: boolean;\n response: http.ServerResponse;\n};\n\nexport class LogServer {\n private ansiConverter = new Convert({ escapeXML: true, newline: true });\n\n private buffer: BufferedLog[] = [];\n\n private clients = new Set<StreamClient>();\n\n private colorIndex = 0;\n\n private colorMap = new Map<string, string>();\n\n private port: number;\n\n private server: http.Server | null = null;\n\n private tailSize: number;\n\n constructor(port: number, tailSize: number = 1_000) {\n this.port = port;\n this.tailSize = tailSize;\n }\n\n getPort(): number {\n if (this.server) {\n const address = this.server.address();\n if (address && typeof address === 'object') {\n return address.port;\n }\n }\n\n return this.port;\n }\n\n start(): Promise<void> {\n return new Promise((resolve, reject) => {\n this.server = http.createServer((request, response) => {\n // Handle streaming GET request\n if (request.method === 'GET' && request.url?.startsWith('/')) {\n const url = new URL(request.url, `http://${request.headers.host}`);\n const includeParameter = url.searchParams.get('include');\n const includes = includeParameter\n ? includeParameter\n .split(',')\n .map((term) => term.trim())\n .filter(Boolean)\n : [];\n const excludeParameter = url.searchParams.get('exclude');\n const excludes = excludeParameter\n ? excludeParameter\n .split(',')\n .map((pattern) => pattern.trim())\n .filter(Boolean)\n : [];\n\n const userAgent = request.headers['user-agent'] ?? '';\n const isBrowser = userAgent.includes('Mozilla');\n\n // Sort buffer by timestamp\n const sortedBuffer = this.buffer.toSorted(\n (a, b) => a.timestamp - b.timestamp,\n );\n\n if (isBrowser) {\n // Browser: send all logs, filtering is done client-side\n response.writeHead(200, {\n 'Cache-Control': 'no-cache',\n Connection: 'keep-alive',\n 'Content-Type': 'text/html; charset=utf-8',\n 'X-Content-Type-Options': 'nosniff',\n });\n\n // Send HTML header with styling\n response.write(this.getHtmlHeader());\n\n // Send all buffered logs as HTML\n for (const entry of sortedBuffer) {\n response.write(this.getHtmlLine(entry.line));\n }\n } else {\n // Non-browser (curl, etc): apply server-side filtering\n const filteredBuffer = sortedBuffer.filter((entry) =>\n matchesFilters(entry.line, includes, excludes),\n );\n\n response.writeHead(200, {\n 'Cache-Control': 'no-cache',\n Connection: 'keep-alive',\n 'Content-Type': 'text/plain; charset=utf-8',\n 'X-Content-Type-Options': 'nosniff',\n });\n\n // Send filtered logs as plain text (strip ANSI)\n for (const entry of filteredBuffer) {\n response.write(stripAnsi(entry.line) + '\\n');\n }\n }\n\n // Add to clients for streaming\n const client: StreamClient = {\n excludes,\n includes,\n isBrowser,\n response,\n };\n\n this.clients.add(client);\n\n request.on('close', () => {\n this.clients.delete(client);\n });\n\n return;\n }\n\n let body = '';\n\n request.on('data', (chunk: Buffer) => {\n body += chunk.toString();\n });\n request.on('end', () => {\n if (request.method === 'POST' && request.url === '/log') {\n try {\n const { line, name, timestamp, type } = JSON.parse(\n body,\n ) as LogPayload;\n\n this.broadcastLog(name, line, type, timestamp);\n } catch {\n // Ignore parse errors\n }\n\n response.writeHead(200);\n response.end();\n } else if (request.method === 'POST' && request.url === '/event') {\n try {\n const { code, event, name, pid, timestamp } = JSON.parse(\n body,\n ) as EventPayload;\n\n if (event === 'start') {\n this.broadcastEvent(name, `● started (pid ${pid})`, timestamp);\n } else if (event === 'exit') {\n this.broadcastEvent(name, `○ exited (code ${code})`, timestamp);\n }\n } catch {\n // Ignore parse errors\n }\n\n response.writeHead(200);\n response.end();\n } else if (request.method === 'POST' && request.url === '/inject') {\n // Test injection endpoint\n try {\n const data = JSON.parse(body) as {\n event?: 'exit' | 'start';\n message: string;\n name: string;\n pid?: number;\n };\n const timestamp = performance.timeOrigin + performance.now();\n\n if (data.event === 'start') {\n this.broadcastEvent(\n data.name,\n `● started (pid ${data.pid ?? 0})`,\n timestamp,\n );\n } else if (data.event === 'exit') {\n this.broadcastEvent(data.name, `○ exited (code 0)`, timestamp);\n } else {\n this.broadcastLog(data.name, data.message, 'stdout', timestamp);\n }\n } catch {\n // Ignore parse errors\n }\n\n response.writeHead(200);\n response.end();\n } else {\n response.writeHead(200);\n response.end();\n }\n });\n });\n\n this.server.once('error', (error: NodeJS.ErrnoException) => {\n reject(error);\n });\n\n this.server.listen(this.port, '0.0.0.0', () => {\n // eslint-disable-next-line no-console\n console.log(\n `${DIM}[teemux] aggregating logs on http://${HOST}:${this.port}${RESET}`,\n );\n resolve();\n });\n });\n }\n\n stop(): Promise<void> {\n return new Promise((resolve) => {\n // Close all client connections\n for (const client of this.clients) {\n client.response.end();\n }\n\n this.clients.clear();\n\n if (this.server) {\n this.server.close(() => {\n this.server = null;\n resolve();\n });\n } else {\n resolve();\n }\n });\n }\n\n private broadcastEvent(\n name: string,\n message: string,\n timestamp: number,\n ): void {\n const color = this.getColor(name);\n const forWeb = `${DIM}${color}[${name}]${RESET} ${DIM}${message}${RESET}`;\n\n this.sendToClients(forWeb, timestamp);\n }\n\n private broadcastLog(\n name: string,\n line: string,\n type: LogType,\n timestamp: number,\n ): void {\n const color = this.getColor(name);\n const errorPrefix = type === 'stderr' ? `${RED}[ERR]${RESET} ` : '';\n const forWeb = `${color}[${name}]${RESET} ${errorPrefix}${line}`;\n\n this.sendToClients(forWeb, timestamp);\n }\n\n private getColor(name: string): string {\n if (!this.colorMap.has(name)) {\n this.colorMap.set(name, COLORS[this.colorIndex++ % COLORS.length]);\n }\n\n return this.colorMap.get(name) ?? COLORS[0];\n }\n\n private getHtmlHeader(): string {\n return `<!DOCTYPE html>\n<html>\n<head>\n <meta charset=\"utf-8\">\n <title>teemux</title>\n <style>\n * { box-sizing: border-box; }\n html, body {\n height: 100%;\n margin: 0;\n overflow: hidden;\n }\n body {\n background: #1e1e1e;\n color: #d4d4d4;\n font-family: 'SF Mono', 'Monaco', 'Inconsolata', 'Fira Code', monospace;\n font-size: 12px;\n line-height: 1.3;\n display: flex;\n flex-direction: column;\n }\n #filter-bar {\n flex-shrink: 0;\n display: flex;\n gap: 8px;\n padding: 8px 12px;\n background: #252526;\n border-bottom: 1px solid #3c3c3c;\n }\n #filter-bar label {\n display: flex;\n align-items: center;\n gap: 6px;\n color: #888;\n }\n #filter-bar input {\n background: #1e1e1e;\n border: 1px solid #3c3c3c;\n border-radius: 3px;\n color: #d4d4d4;\n font-family: inherit;\n font-size: 12px;\n padding: 4px 8px;\n width: 200px;\n }\n #filter-bar input:focus {\n outline: none;\n border-color: #007acc;\n }\n #container {\n flex: 1;\n overflow-y: auto;\n padding: 8px 12px;\n }\n .line {\n white-space: pre-wrap;\n word-break: break-all;\n padding: 1px 4px;\n margin: 0 -4px;\n border-radius: 2px;\n position: relative;\n display: flex;\n align-items: flex-start;\n }\n .line:hover {\n background: rgba(255, 255, 255, 0.05);\n }\n .line.pinned {\n background: rgba(255, 204, 0, 0.1);\n border-left: 2px solid #fc0;\n margin-left: -6px;\n padding-left: 6px;\n }\n .line-content {\n flex: 1;\n }\n .pin-btn {\n opacity: 0;\n cursor: pointer;\n padding: 0 4px;\n color: #888;\n flex-shrink: 0;\n transition: opacity 0.15s;\n }\n .line:hover .pin-btn {\n opacity: 0.5;\n }\n .pin-btn:hover {\n opacity: 1 !important;\n color: #fc0;\n }\n .line.pinned .pin-btn {\n opacity: 1;\n color: #fc0;\n }\n a { color: #4fc1ff; text-decoration: underline; }\n a:hover { text-decoration: none; }\n mark { background: #623800; color: inherit; border-radius: 2px; }\n mark.filter { background: #264f00; }\n .json-key { color: #9cdcfe; }\n .json-string { color: #ce9178; }\n .json-number { color: #b5cea8; }\n .json-bool { color: #569cd6; }\n .json-null { color: #569cd6; }\n #tail-btn {\n position: fixed;\n bottom: 20px;\n right: 20px;\n background: #007acc;\n color: #fff;\n border: none;\n border-radius: 4px;\n padding: 8px 16px;\n font-family: inherit;\n font-size: 12px;\n cursor: pointer;\n display: none;\n align-items: center;\n gap: 6px;\n box-shadow: 0 2px 8px rgba(0, 0, 0, 0.3);\n transition: background 0.15s;\n }\n #tail-btn:hover {\n background: #0098ff;\n }\n #tail-btn svg {\n flex-shrink: 0;\n }\n </style>\n</head>\n<body>\n <div id=\"filter-bar\">\n <label>Include: <input type=\"text\" id=\"include\" placeholder=\"error*,warn* (OR, * = wildcard)\"></label>\n <label>Exclude: <input type=\"text\" id=\"exclude\" placeholder=\"health*,debug (OR, * = wildcard)\"></label>\n <label>Highlight: <input type=\"text\" id=\"highlight\" placeholder=\"term1,term2\"></label>\n </div>\n <div id=\"container\"></div>\n <button id=\"tail-btn\" title=\"Jump to bottom and follow new logs\">\n <svg xmlns=\"http://www.w3.org/2000/svg\" width=\"14\" height=\"14\" viewBox=\"0 0 24 24\" fill=\"none\" stroke=\"currentColor\" stroke-width=\"2\" stroke-linecap=\"round\" stroke-linejoin=\"round\"><path d=\"M12 5v14\"/><path d=\"m19 12-7 7-7-7\"/></svg>\n Tail\n </button>\n <script>\n const container = document.getElementById('container');\n const includeInput = document.getElementById('include');\n const excludeInput = document.getElementById('exclude');\n const highlightInput = document.getElementById('highlight');\n const tailBtn = document.getElementById('tail-btn');\n const params = new URLSearchParams(window.location.search);\n const tailSize = ${this.tailSize};\n \n includeInput.value = params.get('include') || '';\n excludeInput.value = params.get('exclude') || '';\n highlightInput.value = params.get('highlight') || '';\n \n let tailing = true;\n let pinnedIds = new Set();\n \n const updateTailButton = () => {\n tailBtn.style.display = tailing ? 'none' : 'flex';\n };\n \n // Lucide pin icon SVG\n const pinIcon = '<svg xmlns=\"http://www.w3.org/2000/svg\" width=\"14\" height=\"14\" viewBox=\"0 0 24 24\" fill=\"none\" stroke=\"currentColor\" stroke-width=\"2\" stroke-linecap=\"round\" stroke-linejoin=\"round\"><path d=\"M12 17v5\"/><path d=\"M9 10.76a2 2 0 0 1-1.11 1.79l-1.78.9A2 2 0 0 0 5 15.24V16a1 1 0 0 0 1 1h12a1 1 0 0 0 1-1v-.76a2 2 0 0 0-1.11-1.79l-1.78-.9A2 2 0 0 1 15 10.76V7a1 1 0 0 1 1-1 2 2 0 0 0 0-4H8a2 2 0 0 0 0 4 1 1 0 0 1 1 1z\"/></svg>';\n \n const stripAnsi = (str) => str.replace(/\\\\u001B\\\\[[\\\\d;]*m/g, '');\n \n const globToRegex = (pattern) => {\n const escaped = pattern.replace(/([.+?^\\${}()|[\\\\]\\\\\\\\])/g, '\\\\\\\\$1');\n const regexPattern = escaped.replace(/\\\\*/g, '.*');\n return new RegExp(regexPattern, 'i');\n };\n \n const matchesPattern = (text, pattern) => {\n if (pattern.includes('*')) {\n return globToRegex(pattern).test(text);\n }\n return text.includes(pattern.toLowerCase());\n };\n \n const matchesFilters = (text, includes, excludes) => {\n const plain = stripAnsi(text).toLowerCase();\n if (includes.length > 0) {\n const anyMatch = includes.some(p => matchesPattern(plain, p));\n if (!anyMatch) return false;\n }\n if (excludes.length > 0) {\n const anyMatch = excludes.some(p => matchesPattern(plain, p));\n if (anyMatch) return false;\n }\n return true;\n };\n \n const highlightTerms = (html, terms, className = '') => {\n if (!terms.length) return html;\n let result = html;\n for (const term of terms) {\n if (!term) continue;\n const escaped = term.replace(/([.*+?^\\${}()|[\\\\]\\\\\\\\])/g, '\\\\\\\\$1');\n const regex = new RegExp('(?![^<]*>)(' + escaped + ')', 'gi');\n const cls = className ? ' class=\"' + className + '\"' : '';\n result = result.replace(regex, '<mark' + cls + '>$1</mark>');\n }\n return result;\n };\n \n const applyFilters = () => {\n const includes = includeInput.value.split(',').map(s => s.trim()).filter(Boolean);\n const excludes = excludeInput.value.split(',').map(s => s.trim()).filter(Boolean);\n const highlights = highlightInput.value.split(',').map(s => s.trim()).filter(Boolean);\n \n document.querySelectorAll('.line').forEach(line => {\n const id = line.dataset.id;\n const isPinned = pinnedIds.has(id);\n const text = line.dataset.raw;\n const matches = matchesFilters(text, includes, excludes);\n line.style.display = (matches || isPinned) ? '' : 'none';\n \n // Re-apply highlighting\n const contentEl = line.querySelector('.line-content');\n if (contentEl) {\n let html = line.dataset.html;\n html = highlightTerms(html, includes, 'filter');\n html = highlightTerms(html, highlights);\n contentEl.innerHTML = html;\n }\n });\n \n // Update URL without reload\n const newParams = new URLSearchParams();\n if (includeInput.value) newParams.set('include', includeInput.value);\n if (excludeInput.value) newParams.set('exclude', excludeInput.value);\n if (highlightInput.value) newParams.set('highlight', highlightInput.value);\n const newUrl = newParams.toString() ? '?' + newParams.toString() : window.location.pathname;\n history.replaceState(null, '', newUrl);\n \n // Jump to bottom and resume tailing after filter change\n container.scrollTop = container.scrollHeight;\n tailing = true;\n updateTailButton();\n };\n \n const trimBuffer = () => {\n const lines = container.querySelectorAll('.line');\n const unpinnedLines = Array.from(lines).filter(l => !pinnedIds.has(l.dataset.id));\n const excess = unpinnedLines.length - tailSize;\n if (excess > 0) {\n for (let i = 0; i < excess; i++) {\n unpinnedLines[i].remove();\n }\n }\n };\n \n let lineCounter = 0;\n const addLine = (html, raw) => {\n const id = 'line-' + (lineCounter++);\n const includes = includeInput.value.split(',').map(s => s.trim()).filter(Boolean);\n const excludes = excludeInput.value.split(',').map(s => s.trim()).filter(Boolean);\n const highlights = highlightInput.value.split(',').map(s => s.trim()).filter(Boolean);\n \n const div = document.createElement('div');\n div.className = 'line';\n div.dataset.id = id;\n div.dataset.raw = raw;\n div.dataset.html = html;\n \n let displayHtml = html;\n displayHtml = highlightTerms(displayHtml, includes, 'filter');\n displayHtml = highlightTerms(displayHtml, highlights);\n \n div.innerHTML = '<span class=\"line-content\">' + displayHtml + '</span><span class=\"pin-btn\" title=\"Pin\">' + pinIcon + '</span>';\n \n // Pin button handler\n div.querySelector('.pin-btn').addEventListener('click', (e) => {\n e.stopPropagation();\n if (pinnedIds.has(id)) {\n pinnedIds.delete(id);\n div.classList.remove('pinned');\n } else {\n pinnedIds.add(id);\n div.classList.add('pinned');\n }\n applyFilters();\n });\n \n const matches = matchesFilters(raw, includes, excludes);\n div.style.display = matches ? '' : 'none';\n \n container.appendChild(div);\n trimBuffer();\n if (tailing) container.scrollTop = container.scrollHeight;\n };\n \n container.addEventListener('scroll', () => {\n const atBottom = container.scrollHeight - container.scrollTop - container.clientHeight < 50;\n tailing = atBottom;\n updateTailButton();\n });\n \n tailBtn.addEventListener('click', () => {\n container.scrollTop = container.scrollHeight;\n tailing = true;\n updateTailButton();\n });\n \n let debounceTimer;\n const debounce = (fn, delay) => {\n clearTimeout(debounceTimer);\n debounceTimer = setTimeout(fn, delay);\n };\n \n includeInput.addEventListener('input', () => debounce(applyFilters, 50));\n excludeInput.addEventListener('input', () => debounce(applyFilters, 50));\n highlightInput.addEventListener('input', () => debounce(applyFilters, 50));\n </script>\n`;\n }\n\n private getHtmlLine(line: string): string {\n let html = this.ansiConverter.toHtml(line);\n html = highlightJson(html);\n html = linkifyUrls(html);\n const escaped = html.replaceAll('\\\\', '\\\\\\\\').replaceAll(\"'\", \"\\\\'\");\n const raw = stripAnsi(line).replaceAll('\\\\', '\\\\\\\\').replaceAll(\"'\", \"\\\\'\");\n return `<script>addLine('${escaped}', '${raw}')</script>\\n`;\n }\n\n private sendToClients(forWeb: string, timestamp: number): void {\n // Add to buffer\n this.buffer.push({ line: forWeb, timestamp });\n\n // Trim buffer to tail size\n if (this.buffer.length > this.tailSize) {\n this.buffer.shift();\n }\n\n // Send to all connected clients\n for (const client of this.clients) {\n if (client.isBrowser) {\n client.response.write(this.getHtmlLine(forWeb));\n } else {\n // Server-side filtering for non-browser clients\n if (!matchesFilters(forWeb, client.includes, client.excludes)) {\n continue;\n }\n\n client.response.write(stripAnsi(forWeb) + '\\n');\n }\n }\n\n // Note: Each client prints its own logs locally, so server doesn't need to\n }\n}\n","#!/usr/bin/env node\n\nimport { LogServer } from './LogServer.js';\nimport { spawn } from 'node:child_process';\nimport http from 'node:http';\nimport { performance } from 'node:perf_hooks';\nimport readline from 'node:readline';\nimport yargs from 'yargs';\nimport { hideBin } from 'yargs/helpers';\n\n// High-precision timestamp (milliseconds with microsecond precision)\nconst getTimestamp = (): number => performance.timeOrigin + performance.now();\n\nconst RESET = '\\u001B[0m';\nconst DIM = '\\u001B[90m';\nconst RED = '\\u001B[91m';\n\n// Leader monitoring configuration\nconst LEADER_CHECK_INTERVAL = 2_000; // Check every 2 seconds\nconst MAX_PROMOTION_RETRIES = 3;\n\ntype LogType = 'stderr' | 'stdout';\n\nclass LogClient {\n private name: string;\n\n private port: number;\n\n private queue: Array<{ line: string; timestamp: number; type: LogType }> = [];\n\n private sending = false;\n\n constructor(name: string, port: number) {\n this.name = name;\n this.port = port;\n }\n\n async event(\n event: 'exit' | 'start',\n pid: number,\n code?: number,\n ): Promise<void> {\n await this.send('/event', {\n code,\n event,\n name: this.name,\n pid,\n timestamp: getTimestamp(),\n });\n }\n\n async flush(): Promise<void> {\n if (this.sending || this.queue.length === 0) {\n return;\n }\n\n this.sending = true;\n\n while (this.queue.length > 0) {\n const item = this.queue.shift();\n\n if (!item) {\n continue;\n }\n\n const success = await this.send('/log', {\n line: item.line,\n name: this.name,\n timestamp: item.timestamp,\n type: item.type,\n });\n\n if (!success) {\n // Fallback to local output if server unreachable\n // eslint-disable-next-line no-console\n console.log(`[${this.name}] ${item.line}`);\n }\n }\n\n this.sending = false;\n }\n\n log(line: string, type: LogType = 'stdout'): void {\n // Always output locally\n const errorPrefix = type === 'stderr' ? `${RED}[ERR]${RESET} ` : '';\n\n // eslint-disable-next-line no-console\n console.log(`${errorPrefix}${line}`);\n\n // Capture timestamp immediately when log is received\n this.queue.push({ line, timestamp: getTimestamp(), type });\n void this.flush();\n }\n\n private async send(endpoint: string, data: object): Promise<boolean> {\n return new Promise((resolve) => {\n const postData = JSON.stringify(data);\n const request = http.request(\n {\n headers: {\n 'Content-Length': Buffer.byteLength(postData),\n 'Content-Type': 'application/json',\n },\n hostname: '127.0.0.1',\n method: 'POST',\n path: endpoint,\n port: this.port,\n timeout: 1_000,\n },\n (response) => {\n response.resume();\n response.on('end', () => resolve(true));\n },\n );\n\n request.on('error', () => resolve(false));\n request.on('timeout', () => {\n request.destroy();\n resolve(false);\n });\n request.write(postData);\n request.end();\n });\n }\n}\n\nconst runProcess = async (\n name: string,\n command: string[],\n client: LogClient,\n): Promise<number> => {\n const [cmd, ...args] = command;\n\n const child = spawn(cmd, args, {\n env: {\n ...process.env,\n FORCE_COLOR: '1',\n },\n shell: process.platform === 'win32',\n stdio: ['inherit', 'pipe', 'pipe'],\n });\n\n const pid = child.pid ?? 0;\n\n await client.event('start', pid);\n\n let rlStdout: null | readline.Interface = null;\n let rlStderr: null | readline.Interface = null;\n\n if (child.stdout) {\n rlStdout = readline.createInterface({ input: child.stdout });\n\n rlStdout.on('line', (line) => client.log(line, 'stdout'));\n }\n\n if (child.stderr) {\n rlStderr = readline.createInterface({ input: child.stderr });\n\n rlStderr.on('line', (line) => client.log(line, 'stderr'));\n }\n\n // Track signal count for force-kill on second signal\n let signalCount = 0;\n\n const onSignal = (): void => {\n signalCount++;\n\n if (signalCount >= 2 && child.pid && !child.killed) {\n // Second signal: force kill\n child.kill('SIGKILL');\n }\n };\n\n process.on('SIGINT', onSignal);\n process.on('SIGTERM', onSignal);\n process.on('SIGHUP', onSignal);\n\n return new Promise((resolve) => {\n child.on('close', async (code) => {\n // Clean up readline interfaces\n rlStdout?.close();\n rlStderr?.close();\n\n // Remove signal handlers\n process.off('SIGINT', onSignal);\n process.off('SIGTERM', onSignal);\n process.off('SIGHUP', onSignal);\n\n await client.flush();\n await client.event('exit', pid, code ?? 0);\n resolve(code ?? 0);\n });\n });\n};\n\nconst sleep = (ms: number): Promise<void> =>\n new Promise((resolve) => {\n setTimeout(resolve, ms);\n });\n\nconst checkServerReady = async (port: number): Promise<boolean> => {\n return new Promise((resolve) => {\n const request = http.request(\n {\n hostname: '127.0.0.1',\n method: 'GET',\n path: '/',\n port,\n timeout: 200,\n },\n (response) => {\n response.resume();\n resolve(true);\n },\n );\n\n request.on('error', () => resolve(false));\n request.on('timeout', () => {\n request.destroy();\n resolve(false);\n });\n request.end();\n });\n};\n\nconst waitForServer = async (\n port: number,\n maxAttempts = 50,\n): Promise<boolean> => {\n for (let index = 0; index < maxAttempts; index++) {\n if (await checkServerReady(port)) {\n return true;\n }\n\n // Exponential backoff: 10ms, 20ms, 40ms, ... capped at 200ms\n const delay = Math.min(10 * 2 ** index, 200);\n\n await sleep(delay);\n }\n\n return false;\n};\n\nconst tryBecomeLeader = async (server: LogServer): Promise<boolean> => {\n for (let attempt = 0; attempt < MAX_PROMOTION_RETRIES; attempt++) {\n try {\n await server.start();\n return true;\n } catch (error) {\n if ((error as NodeJS.ErrnoException).code !== 'EADDRINUSE') {\n throw error;\n }\n\n // Check if another server took over\n if (await checkServerReady(server.getPort())) {\n // Another process became leader\n return false;\n }\n\n // Port in use but server not responding - might be starting up\n // Add random jitter to avoid thundering herd\n const jitter = Math.random() * 100;\n\n await sleep(50 + jitter);\n }\n }\n\n return false;\n};\n\nconst startLeaderMonitoring = (\n server: LogServer,\n port: number,\n): { stop: () => void } => {\n let isRunning = true;\n let timeoutId: null | ReturnType<typeof setTimeout> = null;\n\n const checkAndPromote = async (): Promise<void> => {\n if (!isRunning) {\n return;\n }\n\n const serverAlive = await checkServerReady(port);\n\n if (!serverAlive && isRunning) {\n // Leader might be dead, try to become leader\n // Add random jitter to prevent all clients from trying simultaneously\n const jitter = Math.random() * 500;\n\n await sleep(jitter);\n\n // Double-check server is still down after jitter\n if (isRunning && !(await checkServerReady(port))) {\n const promoted = await tryBecomeLeader(server);\n\n if (promoted) {\n // eslint-disable-next-line no-console\n console.log(\n `${DIM}[teemux] promoted to leader, now aggregating logs${RESET}`,\n );\n // Stop monitoring - we're now the leader\n // eslint-disable-next-line require-atomic-updates -- safe: only modified here or in stop()\n isRunning = false;\n return;\n }\n }\n }\n\n // Schedule next check\n if (isRunning) {\n timeoutId = setTimeout(() => {\n void checkAndPromote();\n }, LEADER_CHECK_INTERVAL);\n }\n };\n\n // Start monitoring after initial delay\n timeoutId = setTimeout(() => {\n void checkAndPromote();\n }, LEADER_CHECK_INTERVAL);\n\n return {\n stop: () => {\n isRunning = false;\n\n if (timeoutId) {\n clearTimeout(timeoutId);\n }\n },\n };\n};\n\nconst main = async (): Promise<void> => {\n const argv = await yargs(hideBin(process.argv))\n .env('TEEMUX')\n .usage('Usage: $0 --name <name> -- <command> [args...]')\n .option('name', {\n alias: 'n',\n description:\n 'Name to identify this process in logs (defaults to command)',\n type: 'string',\n })\n .option('port', {\n alias: 'p',\n default: 8_336,\n description: 'Port for the log aggregation server',\n type: 'number',\n })\n .option('tail', {\n alias: 't',\n default: 1_000,\n description: 'Number of log lines to keep in buffer',\n type: 'number',\n })\n .help()\n .parse();\n\n const command = argv._ as string[];\n\n if (command.length === 0) {\n // eslint-disable-next-line no-console\n console.error('No command specified');\n // eslint-disable-next-line no-console\n console.error('Usage: teemux --name <name> -- <command> [args...]');\n process.exit(1);\n }\n\n const name = argv.name ?? command[0] ?? 'unknown';\n const port = argv.port;\n\n const server = new LogServer(port, argv.tail);\n\n // Try to become server with retries - if port is taken, become client\n let isServer = false;\n const maxRetries = 3;\n\n for (let attempt = 0; attempt < maxRetries; attempt++) {\n try {\n await server.start();\n isServer = true;\n break;\n } catch (error) {\n if ((error as NodeJS.ErrnoException).code !== 'EADDRINUSE') {\n throw error;\n }\n\n // Check if another server is actually running\n if (await checkServerReady(port)) {\n // Server exists, we're a client\n break;\n }\n\n // Port in use but server not responding - might be starting up\n // Add random jitter to avoid thundering herd\n const jitter = Math.random() * 100;\n\n await sleep(50 + jitter);\n }\n }\n\n // If we're not the server, wait for it to be ready and start monitoring\n let leaderMonitor: null | { stop: () => void } = null;\n\n if (!isServer) {\n const serverReady = await waitForServer(port);\n\n if (!serverReady) {\n // eslint-disable-next-line no-console\n console.error(\n '[teemux] Could not connect to server. Is another instance running?',\n );\n }\n\n // Start monitoring for leader failover\n leaderMonitor = startLeaderMonitoring(server, port);\n }\n\n const client = new LogClient(name, port);\n\n // Cleanup function for graceful shutdown\n const cleanup = async (): Promise<void> => {\n leaderMonitor?.stop();\n\n if (isServer) {\n await server.stop();\n }\n };\n\n // Run the process\n const exitCode = await runProcess(name, command, client);\n\n // Stop leader monitoring if running\n await cleanup();\n\n process.exit(exitCode);\n};\n\nmain().catch((error: unknown) => {\n // eslint-disable-next-line no-console\n console.error('Fatal error:', error);\n process.exit(1);\n});\n"],"mappings":"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"}
|
|
1
|
+
{"version":3,"file":"teemux.js","names":["RESET","DIM","RED"],"sources":["../src/utils/stripHtmlTags.ts","../src/utils/unescapeHtml.ts","../src/utils/highlightJson.ts","../src/utils/linkifyUrls.ts","../src/utils/stripAnsi.ts","../src/utils/matchesFilters.ts","../src/LogServer.ts","../src/teemux.ts"],"sourcesContent":["/**\n * Strip HTML tags from a string, leaving only text content.\n */\nexport const stripHtmlTags = (html: string): string => {\n return html.replaceAll(/<[^>]*>/gu, '');\n};\n","/**\n * Unescape HTML entities back to their original characters.\n */\nexport const unescapeHtml = (text: string): string => {\n return text\n .replaceAll('"', '\"')\n .replaceAll('&', '&')\n .replaceAll('<', '<')\n .replaceAll('>', '>')\n .replaceAll(''', \"'\")\n .replaceAll(''', \"'\");\n};\n","import { stripHtmlTags } from './stripHtmlTags.js';\nimport { unescapeHtml } from './unescapeHtml.js';\n\n/**\n * Apply syntax highlighting to JSON text that uses HTML-escaped quotes (").\n * Uses placeholder technique to avoid double-wrapping strings.\n */\nexport const highlightJsonText = (text: string): string => {\n // First, extract and mark all JSON strings with placeholders\n const strings: string[] = [];\n let result = text.replaceAll(\n /"((?:(?!").)*)"/gu,\n (_match, content) => {\n strings.push(content as string);\n return `\\u0000STR${strings.length - 1}\\u0000`;\n },\n );\n\n // Booleans and null\n result = result.replaceAll(\n /\\b(true|false|null)\\b/gu,\n '<span class=\"json-bool\">$1</span>',\n );\n\n // Numbers\n result = result.replaceAll(\n /(?<!\\w)(-?\\d+(?:\\.\\d+)?(?:[eE][+-]?\\d+)?)\\b/gu,\n '<span class=\"json-number\">$1</span>',\n );\n\n // Restore strings with appropriate highlighting\n result = result.replaceAll(\n /\\0STR(\\d+)\\0(\\s*:)?/gu,\n (_match, index, colon) => {\n const content = strings[Number.parseInt(index as string, 10)];\n if (colon) {\n // This is a key\n return `<span class=\"json-key\">"${content}"</span>${colon}`;\n }\n\n // This is a value\n return `<span class=\"json-string\">"${content}"</span>`;\n },\n );\n\n return result;\n};\n\n/**\n * Process HTML text, applying JSON highlighting only to text outside of HTML tags.\n */\nexport const syntaxHighlightJson = (html: string): string => {\n let result = '';\n let index = 0;\n\n while (index < html.length) {\n if (html[index] === '<') {\n // Find end of tag\n const tagEnd = html.indexOf('>', index);\n if (tagEnd === -1) {\n result += html.slice(index);\n break;\n }\n\n result += html.slice(index, tagEnd + 1);\n index = tagEnd + 1;\n } else {\n // Find next tag or end of string\n const nextTag = html.indexOf('<', index);\n const textEnd = nextTag === -1 ? html.length : nextTag;\n const text = html.slice(index, textEnd);\n\n // Highlight JSON syntax in this text segment\n result += highlightJsonText(text);\n index = textEnd;\n }\n }\n\n return result;\n};\n\n/**\n * Detect if the content (after prefix) is valid JSON and apply syntax highlighting.\n * Returns the original HTML if not valid JSON.\n */\nexport const highlightJson = (html: string): string => {\n // Extract the text content (strip HTML tags) to check if it's JSON\n const textContent = stripHtmlTags(html);\n\n // Unescape HTML entities for JSON parsing\n const unescaped = unescapeHtml(textContent);\n\n // Find where the actual log content starts (after the prefix like [name])\n const prefixMatch = /^\\[[\\w-]+\\]\\s*/u.exec(unescaped);\n const prefix = prefixMatch?.[0] ?? '';\n const content = unescaped.slice(prefix.length).trim();\n\n // Check if the content is valid JSON\n if (!content.startsWith('{') && !content.startsWith('[')) {\n return html;\n }\n\n try {\n JSON.parse(content);\n } catch {\n return html;\n }\n\n // It's valid JSON - now highlight it\n // Find the position after the prefix span in the HTML\n const prefixHtmlMatch = /^<span[^>]*>\\[[^\\]]+\\]<\\/span>\\s*/u.exec(html);\n const htmlPrefix = prefixHtmlMatch?.[0] ?? '';\n const jsonHtml = html.slice(htmlPrefix.length);\n\n // Apply syntax highlighting to the JSON portion\n const highlighted = syntaxHighlightJson(jsonHtml);\n\n return htmlPrefix + highlighted;\n};\n","/**\n * Convert URLs in HTML text to clickable anchor tags.\n * Supports http://, https://, and file:// URLs.\n * Avoids double-linking URLs that are already in href attributes.\n */\nexport const linkifyUrls = (html: string): string => {\n // Match URLs that are not already inside href attributes\n // Supports http://, https://, and file:// URLs\n // Exclude common delimiters and HTML entities (" & etc)\n const urlRegex = /(?<!href=[\"'])(?:https?|file):\\/\\/[^\\s<>\"'{}&]+/gu;\n\n return html.replaceAll(urlRegex, (url) => {\n // Remove trailing punctuation that's likely not part of the URL\n const cleanUrl = url.replace(/[.,;:!?)\\]]+$/u, '');\n const trailing = url.slice(cleanUrl.length);\n\n // Escape HTML entities in the URL for the href attribute\n const escapedHref = cleanUrl\n .replaceAll('&', '&')\n .replaceAll('\"', '"');\n\n return `<a href=\"${escapedHref}\" target=\"_blank\" rel=\"noopener\">${cleanUrl}</a>${trailing}`;\n });\n};\n","/**\n * Strip ANSI escape codes from text.\n * Removes color codes and other terminal formatting sequences.\n */\nexport const stripAnsi = (text: string): string => {\n // eslint-disable-next-line no-control-regex\n return text.replaceAll(/\\u001B\\[[\\d;]*m/gu, '');\n};\n","import { stripAnsi } from './stripAnsi.js';\n\n/**\n * Convert a glob pattern (with * wildcards) to a RegExp.\n * - `*` matches any characters (zero or more)\n * - All other characters are escaped for literal matching\n */\nconst globToRegex = (pattern: string): RegExp => {\n // Escape regex special characters except *\n const escaped = pattern.replaceAll(/[$()+.?[\\\\\\]^{|}]/gu, '\\\\$&');\n // Convert * to .*\n const regexPattern = escaped.replaceAll('*', '.*');\n return new RegExp(regexPattern, 'iu');\n};\n\n/**\n * Check if text matches a pattern (supports * glob wildcards).\n * If no wildcards, does a simple substring match for better performance.\n */\nconst matchesPattern = (text: string, pattern: string): boolean => {\n if (pattern.includes('*')) {\n return globToRegex(pattern).test(text);\n }\n\n return text.includes(pattern.toLowerCase());\n};\n\n/**\n * Check if a line matches the given filter criteria.\n * @param line - The line to check (may contain ANSI codes)\n * @param includes - Patterns where ANY match includes the line (OR logic), case-insensitive. Supports * wildcards.\n * @param excludes - Patterns where ANY match excludes the line (OR logic), case-insensitive. Supports * wildcards.\n * @returns true if the line should be included, false if filtered out\n */\nexport const matchesFilters = (\n line: string,\n includes: string[],\n excludes: string[],\n): boolean => {\n const plainText = stripAnsi(line).toLowerCase();\n\n // Any include must match (OR logic) - case insensitive\n if (includes.length > 0) {\n const anyIncludeMatches = includes.some((pattern) =>\n matchesPattern(plainText, pattern),\n );\n\n if (!anyIncludeMatches) {\n return false;\n }\n }\n\n // None of the excludes should match (OR logic for exclusion) - case insensitive\n if (excludes.length > 0) {\n const anyExcludeMatches = excludes.some((pattern) =>\n matchesPattern(plainText, pattern),\n );\n\n if (anyExcludeMatches) {\n return false;\n }\n }\n\n return true;\n};\n","import { highlightJson } from './utils/highlightJson.js';\nimport { linkifyUrls } from './utils/linkifyUrls.js';\nimport { matchesFilters } from './utils/matchesFilters.js';\nimport { stripAnsi } from './utils/stripAnsi.js';\nimport Convert from 'ansi-to-html';\nimport http from 'node:http';\nimport { performance } from 'node:perf_hooks';\nimport { URL } from 'node:url';\n\nconst COLORS = [\n '\\u001B[36m',\n '\\u001B[33m',\n '\\u001B[32m',\n '\\u001B[35m',\n '\\u001B[34m',\n '\\u001B[91m',\n '\\u001B[92m',\n '\\u001B[93m',\n];\nconst RESET = '\\u001B[0m';\nconst DIM = '\\u001B[90m';\nconst RED = '\\u001B[91m';\nconst HOST = '0.0.0.0';\n\ntype BufferedLog = {\n line: string;\n timestamp: number;\n};\n\ntype EventPayload = {\n code?: number;\n event: 'exit' | 'start';\n name: string;\n pid: number;\n timestamp: number;\n};\n\ntype LogPayload = {\n line: string;\n name: string;\n timestamp: number;\n type: LogType;\n};\n\ntype LogType = 'stderr' | 'stdout';\n\ntype StreamClient = {\n excludes: string[];\n includes: string[];\n isBrowser: boolean;\n response: http.ServerResponse;\n};\n\nexport class LogServer {\n private ansiConverter = new Convert({ escapeXML: true, newline: true });\n\n private buffer: BufferedLog[] = [];\n\n private clients = new Set<StreamClient>();\n\n private colorIndex = 0;\n\n private colorMap = new Map<string, string>();\n\n private port: number;\n\n private server: http.Server | null = null;\n\n private tailSize: number;\n\n constructor(port: number, tailSize: number = 10_000) {\n this.port = port;\n this.tailSize = tailSize;\n }\n\n getPort(): number {\n if (this.server) {\n const address = this.server.address();\n if (address && typeof address === 'object') {\n return address.port;\n }\n }\n\n return this.port;\n }\n\n start(): Promise<void> {\n return new Promise((resolve, reject) => {\n this.server = http.createServer((request, response) => {\n // Handle search endpoint - returns matching logs as JSON\n if (request.method === 'GET' && request.url?.startsWith('/search')) {\n const url = new URL(request.url, `http://${request.headers.host}`);\n const includeParameter = url.searchParams.get('include');\n const includes = includeParameter\n ? includeParameter\n .split(',')\n .map((term) => term.trim())\n .filter(Boolean)\n : [];\n const excludeParameter = url.searchParams.get('exclude');\n const excludes = excludeParameter\n ? excludeParameter\n .split(',')\n .map((pattern) => pattern.trim())\n .filter(Boolean)\n : [];\n const limit = Math.min(\n Number.parseInt(url.searchParams.get('limit') ?? '1000', 10),\n 1_000,\n );\n\n // Sort buffer by timestamp\n const sortedBuffer = this.buffer.toSorted(\n (a, b) => a.timestamp - b.timestamp,\n );\n\n // Filter and limit results\n const results: Array<{ html: string; raw: string }> = [];\n\n for (const entry of sortedBuffer) {\n if (matchesFilters(entry.line, includes, excludes)) {\n let html = this.ansiConverter.toHtml(entry.line);\n html = highlightJson(html);\n html = linkifyUrls(html);\n results.push({\n html,\n raw: stripAnsi(entry.line),\n });\n\n if (results.length >= limit) {\n break;\n }\n }\n }\n\n response.writeHead(200, {\n 'Access-Control-Allow-Origin': '*',\n 'Cache-Control': 'no-cache',\n 'Content-Type': 'application/json; charset=utf-8',\n });\n response.end(JSON.stringify(results));\n return;\n }\n\n // Handle streaming GET request\n if (request.method === 'GET' && request.url?.startsWith('/')) {\n const url = new URL(request.url, `http://${request.headers.host}`);\n const includeParameter = url.searchParams.get('include');\n const includes = includeParameter\n ? includeParameter\n .split(',')\n .map((term) => term.trim())\n .filter(Boolean)\n : [];\n const excludeParameter = url.searchParams.get('exclude');\n const excludes = excludeParameter\n ? excludeParameter\n .split(',')\n .map((pattern) => pattern.trim())\n .filter(Boolean)\n : [];\n\n const userAgent = request.headers['user-agent'] ?? '';\n const isBrowser = userAgent.includes('Mozilla');\n\n // Sort buffer by timestamp\n const sortedBuffer = this.buffer.toSorted(\n (a, b) => a.timestamp - b.timestamp,\n );\n\n if (isBrowser) {\n // Browser: send initial batch (limited), more available via /search\n response.writeHead(200, {\n 'Cache-Control': 'no-cache',\n Connection: 'keep-alive',\n 'Content-Type': 'text/html; charset=utf-8',\n 'X-Content-Type-Options': 'nosniff',\n });\n\n // Send HTML header with styling\n response.write(this.getHtmlHeader());\n\n // Send last 1000 logs initially (browser can fetch more via /search)\n const initialLogs = sortedBuffer.slice(-1_000);\n\n for (const entry of initialLogs) {\n response.write(this.getHtmlLine(entry.line));\n }\n } else {\n // Non-browser (curl, etc): apply server-side filtering\n const filteredBuffer = sortedBuffer.filter((entry) =>\n matchesFilters(entry.line, includes, excludes),\n );\n\n response.writeHead(200, {\n 'Cache-Control': 'no-cache',\n Connection: 'keep-alive',\n 'Content-Type': 'text/plain; charset=utf-8',\n 'X-Content-Type-Options': 'nosniff',\n });\n\n // Send filtered logs as plain text (strip ANSI)\n for (const entry of filteredBuffer) {\n response.write(stripAnsi(entry.line) + '\\n');\n }\n }\n\n // Add to clients for streaming\n const client: StreamClient = {\n excludes,\n includes,\n isBrowser,\n response,\n };\n\n this.clients.add(client);\n\n request.on('close', () => {\n this.clients.delete(client);\n });\n\n return;\n }\n\n let body = '';\n\n request.on('data', (chunk: Buffer) => {\n body += chunk.toString();\n });\n request.on('end', () => {\n if (request.method === 'POST' && request.url === '/log') {\n try {\n const { line, name, timestamp, type } = JSON.parse(\n body,\n ) as LogPayload;\n\n this.broadcastLog(name, line, type, timestamp);\n } catch {\n // Ignore parse errors\n }\n\n response.writeHead(200);\n response.end();\n } else if (request.method === 'POST' && request.url === '/event') {\n try {\n const { code, event, name, pid, timestamp } = JSON.parse(\n body,\n ) as EventPayload;\n\n if (event === 'start') {\n this.broadcastEvent(name, `● started (pid ${pid})`, timestamp);\n } else if (event === 'exit') {\n this.broadcastEvent(name, `○ exited (code ${code})`, timestamp);\n }\n } catch {\n // Ignore parse errors\n }\n\n response.writeHead(200);\n response.end();\n } else if (request.method === 'POST' && request.url === '/inject') {\n // Test injection endpoint\n try {\n const data = JSON.parse(body) as {\n event?: 'exit' | 'start';\n message: string;\n name: string;\n pid?: number;\n };\n const timestamp = performance.timeOrigin + performance.now();\n\n if (data.event === 'start') {\n this.broadcastEvent(\n data.name,\n `● started (pid ${data.pid ?? 0})`,\n timestamp,\n );\n } else if (data.event === 'exit') {\n this.broadcastEvent(data.name, `○ exited (code 0)`, timestamp);\n } else {\n this.broadcastLog(data.name, data.message, 'stdout', timestamp);\n }\n } catch {\n // Ignore parse errors\n }\n\n response.writeHead(200);\n response.end();\n } else {\n response.writeHead(200);\n response.end();\n }\n });\n });\n\n this.server.once('error', (error: NodeJS.ErrnoException) => {\n reject(error);\n });\n\n this.server.listen(this.port, '0.0.0.0', () => {\n // eslint-disable-next-line no-console\n console.log(\n `${DIM}[teemux] aggregating logs on http://${HOST}:${this.port}${RESET}`,\n );\n resolve();\n });\n });\n }\n\n stop(): Promise<void> {\n return new Promise((resolve) => {\n // Close all client connections\n for (const client of this.clients) {\n client.response.end();\n }\n\n this.clients.clear();\n\n if (this.server) {\n this.server.close(() => {\n this.server = null;\n resolve();\n });\n } else {\n resolve();\n }\n });\n }\n\n private broadcastEvent(\n name: string,\n message: string,\n timestamp: number,\n ): void {\n const color = this.getColor(name);\n const forWeb = `${DIM}${color}[${name}]${RESET} ${DIM}${message}${RESET}`;\n\n this.sendToClients(forWeb, timestamp);\n }\n\n private broadcastLog(\n name: string,\n line: string,\n type: LogType,\n timestamp: number,\n ): void {\n const color = this.getColor(name);\n const errorPrefix = type === 'stderr' ? `${RED}[ERR]${RESET} ` : '';\n const forWeb = `${color}[${name}]${RESET} ${errorPrefix}${line}`;\n\n this.sendToClients(forWeb, timestamp);\n }\n\n private getColor(name: string): string {\n if (!this.colorMap.has(name)) {\n this.colorMap.set(name, COLORS[this.colorIndex++ % COLORS.length]);\n }\n\n return this.colorMap.get(name) ?? COLORS[0];\n }\n\n private getHtmlHeader(): string {\n return `<!DOCTYPE html>\n<html>\n<head>\n <meta charset=\"utf-8\">\n <title>teemux</title>\n <style>\n * { box-sizing: border-box; }\n html, body {\n height: 100%;\n margin: 0;\n overflow: hidden;\n }\n body {\n background: #1e1e1e;\n color: #d4d4d4;\n font-family: 'SF Mono', 'Monaco', 'Inconsolata', 'Fira Code', monospace;\n font-size: 12px;\n line-height: 1.3;\n display: flex;\n flex-direction: column;\n }\n #filter-bar {\n flex-shrink: 0;\n display: flex;\n gap: 8px;\n padding: 8px 12px;\n background: #252526;\n border-bottom: 1px solid #3c3c3c;\n }\n #filter-bar label {\n display: flex;\n align-items: center;\n gap: 6px;\n color: #888;\n }\n #filter-bar input {\n background: #1e1e1e;\n border: 1px solid #3c3c3c;\n border-radius: 3px;\n color: #d4d4d4;\n font-family: inherit;\n font-size: 12px;\n padding: 4px 8px;\n width: 200px;\n }\n #filter-bar input:focus {\n outline: none;\n border-color: #007acc;\n }\n #container {\n flex: 1;\n overflow-y: auto;\n padding: 8px 12px;\n }\n .line {\n white-space: pre-wrap;\n word-break: break-all;\n padding: 1px 4px;\n margin: 0 -4px;\n border-radius: 2px;\n position: relative;\n display: flex;\n align-items: flex-start;\n }\n .line:hover {\n background: rgba(255, 255, 255, 0.05);\n }\n .line.pinned {\n background: rgba(255, 204, 0, 0.1);\n border-left: 2px solid #fc0;\n margin-left: -6px;\n padding-left: 6px;\n }\n .line-content {\n flex: 1;\n }\n .pin-btn {\n opacity: 0;\n cursor: pointer;\n padding: 0 4px;\n color: #888;\n flex-shrink: 0;\n transition: opacity 0.15s;\n }\n .line:hover .pin-btn {\n opacity: 0.5;\n }\n .pin-btn:hover {\n opacity: 1 !important;\n color: #fc0;\n }\n .line.pinned .pin-btn {\n opacity: 1;\n color: #fc0;\n }\n a { color: #4fc1ff; text-decoration: underline; }\n a:hover { text-decoration: none; }\n mark { background: #623800; color: inherit; border-radius: 2px; }\n mark.filter { background: #264f00; }\n .json-key { color: #9cdcfe; }\n .json-string { color: #ce9178; }\n .json-number { color: #b5cea8; }\n .json-bool { color: #569cd6; }\n .json-null { color: #569cd6; }\n #tail-btn {\n position: fixed;\n bottom: 20px;\n right: 20px;\n background: #007acc;\n color: #fff;\n border: none;\n border-radius: 4px;\n padding: 8px 16px;\n font-family: inherit;\n font-size: 12px;\n cursor: pointer;\n display: none;\n align-items: center;\n gap: 6px;\n box-shadow: 0 2px 8px rgba(0, 0, 0, 0.3);\n transition: background 0.15s;\n }\n #tail-btn:hover {\n background: #0098ff;\n }\n #tail-btn svg {\n flex-shrink: 0;\n }\n </style>\n</head>\n<body>\n <div id=\"filter-bar\">\n <label>Include: <input type=\"text\" id=\"include\" placeholder=\"error*,warn* (OR, * = wildcard)\"></label>\n <label>Exclude: <input type=\"text\" id=\"exclude\" placeholder=\"health*,debug (OR, * = wildcard)\"></label>\n <label>Highlight: <input type=\"text\" id=\"highlight\" placeholder=\"term1,term2\"></label>\n </div>\n <div id=\"container\"></div>\n <button id=\"tail-btn\" title=\"Jump to bottom and follow new logs\">\n <svg xmlns=\"http://www.w3.org/2000/svg\" width=\"14\" height=\"14\" viewBox=\"0 0 24 24\" fill=\"none\" stroke=\"currentColor\" stroke-width=\"2\" stroke-linecap=\"round\" stroke-linejoin=\"round\"><path d=\"M12 5v14\"/><path d=\"m19 12-7 7-7-7\"/></svg>\n Tail\n </button>\n <script>\n const container = document.getElementById('container');\n const includeInput = document.getElementById('include');\n const excludeInput = document.getElementById('exclude');\n const highlightInput = document.getElementById('highlight');\n const tailBtn = document.getElementById('tail-btn');\n const params = new URLSearchParams(window.location.search);\n const tailSize = Math.min(${this.tailSize}, 1000);\n \n includeInput.value = params.get('include') || '';\n excludeInput.value = params.get('exclude') || '';\n highlightInput.value = params.get('highlight') || '';\n \n let tailing = true;\n let pinnedIds = new Set();\n \n const updateTailButton = () => {\n tailBtn.style.display = tailing ? 'none' : 'flex';\n };\n \n // Lucide pin icon SVG\n const pinIcon = '<svg xmlns=\"http://www.w3.org/2000/svg\" width=\"14\" height=\"14\" viewBox=\"0 0 24 24\" fill=\"none\" stroke=\"currentColor\" stroke-width=\"2\" stroke-linecap=\"round\" stroke-linejoin=\"round\"><path d=\"M12 17v5\"/><path d=\"M9 10.76a2 2 0 0 1-1.11 1.79l-1.78.9A2 2 0 0 0 5 15.24V16a1 1 0 0 0 1 1h12a1 1 0 0 0 1-1v-.76a2 2 0 0 0-1.11-1.79l-1.78-.9A2 2 0 0 1 15 10.76V7a1 1 0 0 1 1-1 2 2 0 0 0 0-4H8a2 2 0 0 0 0 4 1 1 0 0 1 1 1z\"/></svg>';\n \n const stripAnsi = (str) => str.replace(/\\\\u001B\\\\[[\\\\d;]*m/g, '');\n \n const globToRegex = (pattern) => {\n const escaped = pattern.replace(/([.+?^\\${}()|[\\\\]\\\\\\\\])/g, '\\\\\\\\$1');\n const regexPattern = escaped.replace(/\\\\*/g, '.*');\n return new RegExp(regexPattern, 'i');\n };\n \n const matchesPattern = (text, pattern) => {\n if (pattern.includes('*')) {\n return globToRegex(pattern).test(text);\n }\n return text.includes(pattern.toLowerCase());\n };\n \n const matchesFilters = (text, includes, excludes) => {\n const plain = stripAnsi(text).toLowerCase();\n if (includes.length > 0) {\n const anyMatch = includes.some(p => matchesPattern(plain, p));\n if (!anyMatch) return false;\n }\n if (excludes.length > 0) {\n const anyMatch = excludes.some(p => matchesPattern(plain, p));\n if (anyMatch) return false;\n }\n return true;\n };\n \n const highlightTerms = (html, terms, className = '') => {\n if (!terms.length) return html;\n let result = html;\n for (const term of terms) {\n if (!term) continue;\n const escaped = term.replace(/([.*+?^\\${}()|[\\\\]\\\\\\\\])/g, '\\\\\\\\$1');\n const regex = new RegExp('(?![^<]*>)(' + escaped + ')', 'gi');\n const cls = className ? ' class=\"' + className + '\"' : '';\n result = result.replace(regex, '<mark' + cls + '>$1</mark>');\n }\n return result;\n };\n \n const applyFiltersLocal = () => {\n const includes = includeInput.value.split(',').map(s => s.trim()).filter(Boolean);\n const excludes = excludeInput.value.split(',').map(s => s.trim()).filter(Boolean);\n const highlights = highlightInput.value.split(',').map(s => s.trim()).filter(Boolean);\n \n document.querySelectorAll('.line').forEach(line => {\n const id = line.dataset.id;\n const isPinned = pinnedIds.has(id);\n const text = line.dataset.raw;\n const matches = matchesFilters(text, includes, excludes);\n line.style.display = (matches || isPinned) ? '' : 'none';\n \n // Re-apply highlighting\n const contentEl = line.querySelector('.line-content');\n if (contentEl) {\n let html = line.dataset.html;\n html = highlightTerms(html, includes, 'filter');\n html = highlightTerms(html, highlights);\n contentEl.innerHTML = html;\n }\n });\n };\n \n let lastSearchQuery = '';\n let searchController = null;\n \n const applyFilters = async () => {\n const includes = includeInput.value.split(',').map(s => s.trim()).filter(Boolean);\n const excludes = excludeInput.value.split(',').map(s => s.trim()).filter(Boolean);\n const highlights = highlightInput.value.split(',').map(s => s.trim()).filter(Boolean);\n \n // Update URL without reload\n const newParams = new URLSearchParams();\n if (includeInput.value) newParams.set('include', includeInput.value);\n if (excludeInput.value) newParams.set('exclude', excludeInput.value);\n if (highlightInput.value) newParams.set('highlight', highlightInput.value);\n const newUrl = newParams.toString() ? '?' + newParams.toString() : window.location.pathname;\n history.replaceState(null, '', newUrl);\n \n // Build search query string for comparison\n const searchQuery = includeInput.value + '|' + excludeInput.value;\n \n // If only highlight changed, just re-apply local highlighting\n if (searchQuery === lastSearchQuery) {\n applyFiltersLocal();\n return;\n }\n \n lastSearchQuery = searchQuery;\n \n // Cancel any pending search request\n if (searchController) {\n searchController.abort();\n }\n \n // If no filters, just apply local filtering (show all)\n if (includes.length === 0 && excludes.length === 0) {\n applyFiltersLocal();\n container.scrollTop = container.scrollHeight;\n tailing = true;\n updateTailButton();\n return;\n }\n \n // Fetch matching logs from server\n searchController = new AbortController();\n const searchParams = new URLSearchParams();\n if (includeInput.value) searchParams.set('include', includeInput.value);\n if (excludeInput.value) searchParams.set('exclude', excludeInput.value);\n searchParams.set('limit', '1000');\n \n try {\n const response = await fetch('/search?' + searchParams.toString(), {\n signal: searchController.signal\n });\n const results = await response.json();\n \n // Clear non-pinned lines\n document.querySelectorAll('.line').forEach(line => {\n if (!pinnedIds.has(line.dataset.id)) {\n line.remove();\n }\n });\n \n // Add search results\n for (const item of results) {\n const id = 'line-' + (lineCounter++);\n const div = document.createElement('div');\n div.className = 'line';\n div.dataset.id = id;\n div.dataset.raw = item.raw;\n div.dataset.html = item.html;\n \n let displayHtml = item.html;\n displayHtml = highlightTerms(displayHtml, includes, 'filter');\n displayHtml = highlightTerms(displayHtml, highlights);\n \n div.innerHTML = '<span class=\"line-content\">' + displayHtml + '</span><span class=\"pin-btn\" title=\"Pin\">' + pinIcon + '</span>';\n \n // Pin button handler\n div.querySelector('.pin-btn').addEventListener('click', (e) => {\n e.stopPropagation();\n if (pinnedIds.has(id)) {\n pinnedIds.delete(id);\n div.classList.remove('pinned');\n } else {\n pinnedIds.add(id);\n div.classList.add('pinned');\n }\n applyFiltersLocal();\n });\n \n container.appendChild(div);\n }\n \n // Jump to bottom and resume tailing\n container.scrollTop = container.scrollHeight;\n tailing = true;\n updateTailButton();\n } catch (e) {\n if (e.name !== 'AbortError') {\n console.error('Search failed:', e);\n // Fallback to local filtering\n applyFiltersLocal();\n }\n }\n };\n \n const trimBuffer = () => {\n const lines = container.querySelectorAll('.line');\n const unpinnedLines = Array.from(lines).filter(l => !pinnedIds.has(l.dataset.id));\n const excess = unpinnedLines.length - tailSize;\n if (excess > 0) {\n for (let i = 0; i < excess; i++) {\n unpinnedLines[i].remove();\n }\n }\n };\n \n let lineCounter = 0;\n const addLine = (html, raw) => {\n const id = 'line-' + (lineCounter++);\n const includes = includeInput.value.split(',').map(s => s.trim()).filter(Boolean);\n const excludes = excludeInput.value.split(',').map(s => s.trim()).filter(Boolean);\n const highlights = highlightInput.value.split(',').map(s => s.trim()).filter(Boolean);\n \n const div = document.createElement('div');\n div.className = 'line';\n div.dataset.id = id;\n div.dataset.raw = raw;\n div.dataset.html = html;\n \n let displayHtml = html;\n displayHtml = highlightTerms(displayHtml, includes, 'filter');\n displayHtml = highlightTerms(displayHtml, highlights);\n \n div.innerHTML = '<span class=\"line-content\">' + displayHtml + '</span><span class=\"pin-btn\" title=\"Pin\">' + pinIcon + '</span>';\n \n // Pin button handler\n div.querySelector('.pin-btn').addEventListener('click', (e) => {\n e.stopPropagation();\n if (pinnedIds.has(id)) {\n pinnedIds.delete(id);\n div.classList.remove('pinned');\n } else {\n pinnedIds.add(id);\n div.classList.add('pinned');\n }\n applyFiltersLocal();\n });\n \n const matches = matchesFilters(raw, includes, excludes);\n div.style.display = matches ? '' : 'none';\n \n container.appendChild(div);\n trimBuffer();\n if (tailing) container.scrollTop = container.scrollHeight;\n };\n \n container.addEventListener('scroll', () => {\n const atBottom = container.scrollHeight - container.scrollTop - container.clientHeight < 50;\n tailing = atBottom;\n updateTailButton();\n });\n \n tailBtn.addEventListener('click', () => {\n container.scrollTop = container.scrollHeight;\n tailing = true;\n updateTailButton();\n });\n \n let debounceTimer;\n const debounce = (fn, delay) => {\n clearTimeout(debounceTimer);\n debounceTimer = setTimeout(fn, delay);\n };\n \n includeInput.addEventListener('input', () => debounce(applyFilters, 300));\n excludeInput.addEventListener('input', () => debounce(applyFilters, 300));\n highlightInput.addEventListener('input', () => debounce(applyFilters, 150));\n </script>\n`;\n }\n\n private getHtmlLine(line: string): string {\n let html = this.ansiConverter.toHtml(line);\n html = highlightJson(html);\n html = linkifyUrls(html);\n const escaped = html.replaceAll('\\\\', '\\\\\\\\').replaceAll(\"'\", \"\\\\'\");\n const raw = stripAnsi(line).replaceAll('\\\\', '\\\\\\\\').replaceAll(\"'\", \"\\\\'\");\n return `<script>addLine('${escaped}', '${raw}')</script>\\n`;\n }\n\n private sendToClients(forWeb: string, timestamp: number): void {\n // Add to buffer\n this.buffer.push({ line: forWeb, timestamp });\n\n // Trim buffer to tail size\n if (this.buffer.length > this.tailSize) {\n this.buffer.shift();\n }\n\n // Send to all connected clients\n for (const client of this.clients) {\n if (client.isBrowser) {\n client.response.write(this.getHtmlLine(forWeb));\n } else {\n // Server-side filtering for non-browser clients\n if (!matchesFilters(forWeb, client.includes, client.excludes)) {\n continue;\n }\n\n client.response.write(stripAnsi(forWeb) + '\\n');\n }\n }\n\n // Note: Each client prints its own logs locally, so server doesn't need to\n }\n}\n","#!/usr/bin/env node\n\nimport { LogServer } from './LogServer.js';\nimport { spawn } from 'node:child_process';\nimport http from 'node:http';\nimport { performance } from 'node:perf_hooks';\nimport readline from 'node:readline';\nimport yargs from 'yargs';\nimport { hideBin } from 'yargs/helpers';\n\n// High-precision timestamp (milliseconds with microsecond precision)\nconst getTimestamp = (): number => performance.timeOrigin + performance.now();\n\nconst RESET = '\\u001B[0m';\nconst DIM = '\\u001B[90m';\nconst RED = '\\u001B[91m';\n\n// Leader monitoring configuration\nconst LEADER_CHECK_INTERVAL = 2_000; // Check every 2 seconds\nconst MAX_PROMOTION_RETRIES = 3;\n\ntype LogType = 'stderr' | 'stdout';\n\nclass LogClient {\n private name: string;\n\n private port: number;\n\n private queue: Array<{ line: string; timestamp: number; type: LogType }> = [];\n\n private sending = false;\n\n constructor(name: string, port: number) {\n this.name = name;\n this.port = port;\n }\n\n async event(\n event: 'exit' | 'start',\n pid: number,\n code?: number,\n ): Promise<void> {\n await this.send('/event', {\n code,\n event,\n name: this.name,\n pid,\n timestamp: getTimestamp(),\n });\n }\n\n async flush(): Promise<void> {\n if (this.sending || this.queue.length === 0) {\n return;\n }\n\n this.sending = true;\n\n while (this.queue.length > 0) {\n const item = this.queue.shift();\n\n if (!item) {\n continue;\n }\n\n const success = await this.send('/log', {\n line: item.line,\n name: this.name,\n timestamp: item.timestamp,\n type: item.type,\n });\n\n if (!success) {\n // Fallback to local output if server unreachable\n // eslint-disable-next-line no-console\n console.log(`[${this.name}] ${item.line}`);\n }\n }\n\n this.sending = false;\n }\n\n log(line: string, type: LogType = 'stdout'): void {\n // Always output locally\n const errorPrefix = type === 'stderr' ? `${RED}[ERR]${RESET} ` : '';\n\n // eslint-disable-next-line no-console\n console.log(`${errorPrefix}${line}`);\n\n // Capture timestamp immediately when log is received\n this.queue.push({ line, timestamp: getTimestamp(), type });\n void this.flush();\n }\n\n private async send(endpoint: string, data: object): Promise<boolean> {\n return new Promise((resolve) => {\n const postData = JSON.stringify(data);\n const request = http.request(\n {\n headers: {\n 'Content-Length': Buffer.byteLength(postData),\n 'Content-Type': 'application/json',\n },\n hostname: '127.0.0.1',\n method: 'POST',\n path: endpoint,\n port: this.port,\n timeout: 1_000,\n },\n (response) => {\n response.resume();\n response.on('end', () => resolve(true));\n },\n );\n\n request.on('error', () => resolve(false));\n request.on('timeout', () => {\n request.destroy();\n resolve(false);\n });\n request.write(postData);\n request.end();\n });\n }\n}\n\nconst runProcess = async (\n name: string,\n command: string[],\n client: LogClient,\n): Promise<number> => {\n const [cmd, ...args] = command;\n\n const child = spawn(cmd, args, {\n env: {\n ...process.env,\n FORCE_COLOR: '1',\n },\n shell: process.platform === 'win32',\n stdio: ['inherit', 'pipe', 'pipe'],\n });\n\n const pid = child.pid ?? 0;\n\n await client.event('start', pid);\n\n let rlStdout: null | readline.Interface = null;\n let rlStderr: null | readline.Interface = null;\n\n if (child.stdout) {\n rlStdout = readline.createInterface({ input: child.stdout });\n\n rlStdout.on('line', (line) => client.log(line, 'stdout'));\n }\n\n if (child.stderr) {\n rlStderr = readline.createInterface({ input: child.stderr });\n\n rlStderr.on('line', (line) => client.log(line, 'stderr'));\n }\n\n // Track signal count for force-kill on second signal\n let signalCount = 0;\n\n const onSignal = (): void => {\n signalCount++;\n\n if (signalCount >= 2 && child.pid && !child.killed) {\n // Second signal: force kill\n child.kill('SIGKILL');\n }\n };\n\n process.on('SIGINT', onSignal);\n process.on('SIGTERM', onSignal);\n process.on('SIGHUP', onSignal);\n\n return new Promise((resolve) => {\n child.on('close', async (code) => {\n // Clean up readline interfaces\n rlStdout?.close();\n rlStderr?.close();\n\n // Remove signal handlers\n process.off('SIGINT', onSignal);\n process.off('SIGTERM', onSignal);\n process.off('SIGHUP', onSignal);\n\n await client.flush();\n await client.event('exit', pid, code ?? 0);\n resolve(code ?? 0);\n });\n });\n};\n\nconst sleep = (ms: number): Promise<void> =>\n new Promise((resolve) => {\n setTimeout(resolve, ms);\n });\n\nconst checkServerReady = async (port: number): Promise<boolean> => {\n return new Promise((resolve) => {\n const request = http.request(\n {\n hostname: '127.0.0.1',\n method: 'GET',\n path: '/',\n port,\n timeout: 200,\n },\n (response) => {\n response.resume();\n resolve(true);\n },\n );\n\n request.on('error', () => resolve(false));\n request.on('timeout', () => {\n request.destroy();\n resolve(false);\n });\n request.end();\n });\n};\n\nconst waitForServer = async (\n port: number,\n maxAttempts = 50,\n): Promise<boolean> => {\n for (let index = 0; index < maxAttempts; index++) {\n if (await checkServerReady(port)) {\n return true;\n }\n\n // Exponential backoff: 10ms, 20ms, 40ms, ... capped at 200ms\n const delay = Math.min(10 * 2 ** index, 200);\n\n await sleep(delay);\n }\n\n return false;\n};\n\nconst tryBecomeLeader = async (server: LogServer): Promise<boolean> => {\n for (let attempt = 0; attempt < MAX_PROMOTION_RETRIES; attempt++) {\n try {\n await server.start();\n return true;\n } catch (error) {\n if ((error as NodeJS.ErrnoException).code !== 'EADDRINUSE') {\n throw error;\n }\n\n // Check if another server took over\n if (await checkServerReady(server.getPort())) {\n // Another process became leader\n return false;\n }\n\n // Port in use but server not responding - might be starting up\n // Add random jitter to avoid thundering herd\n const jitter = Math.random() * 100;\n\n await sleep(50 + jitter);\n }\n }\n\n return false;\n};\n\nconst startLeaderMonitoring = (\n server: LogServer,\n port: number,\n): { stop: () => void } => {\n let isRunning = true;\n let timeoutId: null | ReturnType<typeof setTimeout> = null;\n\n const checkAndPromote = async (): Promise<void> => {\n if (!isRunning) {\n return;\n }\n\n const serverAlive = await checkServerReady(port);\n\n if (!serverAlive && isRunning) {\n // Leader might be dead, try to become leader\n // Add random jitter to prevent all clients from trying simultaneously\n const jitter = Math.random() * 500;\n\n await sleep(jitter);\n\n // Double-check server is still down after jitter\n if (isRunning && !(await checkServerReady(port))) {\n const promoted = await tryBecomeLeader(server);\n\n if (promoted) {\n // eslint-disable-next-line no-console\n console.log(\n `${DIM}[teemux] promoted to leader, now aggregating logs${RESET}`,\n );\n // Stop monitoring - we're now the leader\n // eslint-disable-next-line require-atomic-updates -- safe: only modified here or in stop()\n isRunning = false;\n return;\n }\n }\n }\n\n // Schedule next check\n if (isRunning) {\n timeoutId = setTimeout(() => {\n void checkAndPromote();\n }, LEADER_CHECK_INTERVAL);\n }\n };\n\n // Start monitoring after initial delay\n timeoutId = setTimeout(() => {\n void checkAndPromote();\n }, LEADER_CHECK_INTERVAL);\n\n return {\n stop: () => {\n isRunning = false;\n\n if (timeoutId) {\n clearTimeout(timeoutId);\n }\n },\n };\n};\n\nconst main = async (): Promise<void> => {\n const argv = await yargs(hideBin(process.argv))\n .env('TEEMUX')\n .usage('Usage: $0 --name <name> -- <command> [args...]')\n .option('name', {\n alias: 'n',\n description:\n 'Name to identify this process in logs (defaults to command)',\n type: 'string',\n })\n .option('port', {\n alias: 'p',\n default: 8_336,\n description: 'Port for the log aggregation server',\n type: 'number',\n })\n .option('buffer', {\n alias: 'b',\n default: 10_000,\n description: 'Number of log lines to keep in server buffer',\n type: 'number',\n })\n .help()\n .parse();\n\n const command = argv._ as string[];\n\n if (command.length === 0) {\n // eslint-disable-next-line no-console\n console.error('No command specified');\n // eslint-disable-next-line no-console\n console.error('Usage: teemux --name <name> -- <command> [args...]');\n process.exit(1);\n }\n\n const name = argv.name ?? command[0] ?? 'unknown';\n const port = argv.port;\n\n const server = new LogServer(port, argv.buffer);\n\n // Try to become server with retries - if port is taken, become client\n let isServer = false;\n const maxRetries = 3;\n\n for (let attempt = 0; attempt < maxRetries; attempt++) {\n try {\n await server.start();\n isServer = true;\n break;\n } catch (error) {\n if ((error as NodeJS.ErrnoException).code !== 'EADDRINUSE') {\n throw error;\n }\n\n // Check if another server is actually running\n if (await checkServerReady(port)) {\n // Server exists, we're a client\n break;\n }\n\n // Port in use but server not responding - might be starting up\n // Add random jitter to avoid thundering herd\n const jitter = Math.random() * 100;\n\n await sleep(50 + jitter);\n }\n }\n\n // If we're not the server, wait for it to be ready and start monitoring\n let leaderMonitor: null | { stop: () => void } = null;\n\n if (!isServer) {\n const serverReady = await waitForServer(port);\n\n if (!serverReady) {\n // eslint-disable-next-line no-console\n console.error(\n '[teemux] Could not connect to server. Is another instance running?',\n );\n }\n\n // Start monitoring for leader failover\n leaderMonitor = startLeaderMonitoring(server, port);\n }\n\n const client = new LogClient(name, port);\n\n // Cleanup function for graceful shutdown\n const cleanup = async (): Promise<void> => {\n leaderMonitor?.stop();\n\n if (isServer) {\n await server.stop();\n }\n };\n\n // Run the process\n const exitCode = await runProcess(name, command, client);\n\n // Stop leader monitoring if running\n await cleanup();\n\n process.exit(exitCode);\n};\n\nmain().catch((error: unknown) => {\n // eslint-disable-next-line no-console\n console.error('Fatal error:', error);\n process.exit(1);\n});\n"],"mappings":"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"}
|
package/package.json
CHANGED
package/src/LogServer.ts
CHANGED
|
@@ -68,7 +68,7 @@ export class LogServer {
|
|
|
68
68
|
|
|
69
69
|
private tailSize: number;
|
|
70
70
|
|
|
71
|
-
constructor(port: number, tailSize: number =
|
|
71
|
+
constructor(port: number, tailSize: number = 10_000) {
|
|
72
72
|
this.port = port;
|
|
73
73
|
this.tailSize = tailSize;
|
|
74
74
|
}
|
|
@@ -87,6 +87,61 @@ export class LogServer {
|
|
|
87
87
|
start(): Promise<void> {
|
|
88
88
|
return new Promise((resolve, reject) => {
|
|
89
89
|
this.server = http.createServer((request, response) => {
|
|
90
|
+
// Handle search endpoint - returns matching logs as JSON
|
|
91
|
+
if (request.method === 'GET' && request.url?.startsWith('/search')) {
|
|
92
|
+
const url = new URL(request.url, `http://${request.headers.host}`);
|
|
93
|
+
const includeParameter = url.searchParams.get('include');
|
|
94
|
+
const includes = includeParameter
|
|
95
|
+
? includeParameter
|
|
96
|
+
.split(',')
|
|
97
|
+
.map((term) => term.trim())
|
|
98
|
+
.filter(Boolean)
|
|
99
|
+
: [];
|
|
100
|
+
const excludeParameter = url.searchParams.get('exclude');
|
|
101
|
+
const excludes = excludeParameter
|
|
102
|
+
? excludeParameter
|
|
103
|
+
.split(',')
|
|
104
|
+
.map((pattern) => pattern.trim())
|
|
105
|
+
.filter(Boolean)
|
|
106
|
+
: [];
|
|
107
|
+
const limit = Math.min(
|
|
108
|
+
Number.parseInt(url.searchParams.get('limit') ?? '1000', 10),
|
|
109
|
+
1_000,
|
|
110
|
+
);
|
|
111
|
+
|
|
112
|
+
// Sort buffer by timestamp
|
|
113
|
+
const sortedBuffer = this.buffer.toSorted(
|
|
114
|
+
(a, b) => a.timestamp - b.timestamp,
|
|
115
|
+
);
|
|
116
|
+
|
|
117
|
+
// Filter and limit results
|
|
118
|
+
const results: Array<{ html: string; raw: string }> = [];
|
|
119
|
+
|
|
120
|
+
for (const entry of sortedBuffer) {
|
|
121
|
+
if (matchesFilters(entry.line, includes, excludes)) {
|
|
122
|
+
let html = this.ansiConverter.toHtml(entry.line);
|
|
123
|
+
html = highlightJson(html);
|
|
124
|
+
html = linkifyUrls(html);
|
|
125
|
+
results.push({
|
|
126
|
+
html,
|
|
127
|
+
raw: stripAnsi(entry.line),
|
|
128
|
+
});
|
|
129
|
+
|
|
130
|
+
if (results.length >= limit) {
|
|
131
|
+
break;
|
|
132
|
+
}
|
|
133
|
+
}
|
|
134
|
+
}
|
|
135
|
+
|
|
136
|
+
response.writeHead(200, {
|
|
137
|
+
'Access-Control-Allow-Origin': '*',
|
|
138
|
+
'Cache-Control': 'no-cache',
|
|
139
|
+
'Content-Type': 'application/json; charset=utf-8',
|
|
140
|
+
});
|
|
141
|
+
response.end(JSON.stringify(results));
|
|
142
|
+
return;
|
|
143
|
+
}
|
|
144
|
+
|
|
90
145
|
// Handle streaming GET request
|
|
91
146
|
if (request.method === 'GET' && request.url?.startsWith('/')) {
|
|
92
147
|
const url = new URL(request.url, `http://${request.headers.host}`);
|
|
@@ -114,7 +169,7 @@ export class LogServer {
|
|
|
114
169
|
);
|
|
115
170
|
|
|
116
171
|
if (isBrowser) {
|
|
117
|
-
// Browser: send
|
|
172
|
+
// Browser: send initial batch (limited), more available via /search
|
|
118
173
|
response.writeHead(200, {
|
|
119
174
|
'Cache-Control': 'no-cache',
|
|
120
175
|
Connection: 'keep-alive',
|
|
@@ -125,8 +180,10 @@ export class LogServer {
|
|
|
125
180
|
// Send HTML header with styling
|
|
126
181
|
response.write(this.getHtmlHeader());
|
|
127
182
|
|
|
128
|
-
// Send
|
|
129
|
-
|
|
183
|
+
// Send last 1000 logs initially (browser can fetch more via /search)
|
|
184
|
+
const initialLogs = sortedBuffer.slice(-1_000);
|
|
185
|
+
|
|
186
|
+
for (const entry of initialLogs) {
|
|
130
187
|
response.write(this.getHtmlLine(entry.line));
|
|
131
188
|
}
|
|
132
189
|
} else {
|
|
@@ -451,7 +508,7 @@ export class LogServer {
|
|
|
451
508
|
const highlightInput = document.getElementById('highlight');
|
|
452
509
|
const tailBtn = document.getElementById('tail-btn');
|
|
453
510
|
const params = new URLSearchParams(window.location.search);
|
|
454
|
-
const tailSize = ${this.tailSize};
|
|
511
|
+
const tailSize = Math.min(${this.tailSize}, 1000);
|
|
455
512
|
|
|
456
513
|
includeInput.value = params.get('include') || '';
|
|
457
514
|
excludeInput.value = params.get('exclude') || '';
|
|
@@ -508,7 +565,7 @@ export class LogServer {
|
|
|
508
565
|
return result;
|
|
509
566
|
};
|
|
510
567
|
|
|
511
|
-
const
|
|
568
|
+
const applyFiltersLocal = () => {
|
|
512
569
|
const includes = includeInput.value.split(',').map(s => s.trim()).filter(Boolean);
|
|
513
570
|
const excludes = excludeInput.value.split(',').map(s => s.trim()).filter(Boolean);
|
|
514
571
|
const highlights = highlightInput.value.split(',').map(s => s.trim()).filter(Boolean);
|
|
@@ -529,6 +586,15 @@ export class LogServer {
|
|
|
529
586
|
contentEl.innerHTML = html;
|
|
530
587
|
}
|
|
531
588
|
});
|
|
589
|
+
};
|
|
590
|
+
|
|
591
|
+
let lastSearchQuery = '';
|
|
592
|
+
let searchController = null;
|
|
593
|
+
|
|
594
|
+
const applyFilters = async () => {
|
|
595
|
+
const includes = includeInput.value.split(',').map(s => s.trim()).filter(Boolean);
|
|
596
|
+
const excludes = excludeInput.value.split(',').map(s => s.trim()).filter(Boolean);
|
|
597
|
+
const highlights = highlightInput.value.split(',').map(s => s.trim()).filter(Boolean);
|
|
532
598
|
|
|
533
599
|
// Update URL without reload
|
|
534
600
|
const newParams = new URLSearchParams();
|
|
@@ -538,10 +604,93 @@ export class LogServer {
|
|
|
538
604
|
const newUrl = newParams.toString() ? '?' + newParams.toString() : window.location.pathname;
|
|
539
605
|
history.replaceState(null, '', newUrl);
|
|
540
606
|
|
|
541
|
-
//
|
|
542
|
-
|
|
543
|
-
|
|
544
|
-
|
|
607
|
+
// Build search query string for comparison
|
|
608
|
+
const searchQuery = includeInput.value + '|' + excludeInput.value;
|
|
609
|
+
|
|
610
|
+
// If only highlight changed, just re-apply local highlighting
|
|
611
|
+
if (searchQuery === lastSearchQuery) {
|
|
612
|
+
applyFiltersLocal();
|
|
613
|
+
return;
|
|
614
|
+
}
|
|
615
|
+
|
|
616
|
+
lastSearchQuery = searchQuery;
|
|
617
|
+
|
|
618
|
+
// Cancel any pending search request
|
|
619
|
+
if (searchController) {
|
|
620
|
+
searchController.abort();
|
|
621
|
+
}
|
|
622
|
+
|
|
623
|
+
// If no filters, just apply local filtering (show all)
|
|
624
|
+
if (includes.length === 0 && excludes.length === 0) {
|
|
625
|
+
applyFiltersLocal();
|
|
626
|
+
container.scrollTop = container.scrollHeight;
|
|
627
|
+
tailing = true;
|
|
628
|
+
updateTailButton();
|
|
629
|
+
return;
|
|
630
|
+
}
|
|
631
|
+
|
|
632
|
+
// Fetch matching logs from server
|
|
633
|
+
searchController = new AbortController();
|
|
634
|
+
const searchParams = new URLSearchParams();
|
|
635
|
+
if (includeInput.value) searchParams.set('include', includeInput.value);
|
|
636
|
+
if (excludeInput.value) searchParams.set('exclude', excludeInput.value);
|
|
637
|
+
searchParams.set('limit', '1000');
|
|
638
|
+
|
|
639
|
+
try {
|
|
640
|
+
const response = await fetch('/search?' + searchParams.toString(), {
|
|
641
|
+
signal: searchController.signal
|
|
642
|
+
});
|
|
643
|
+
const results = await response.json();
|
|
644
|
+
|
|
645
|
+
// Clear non-pinned lines
|
|
646
|
+
document.querySelectorAll('.line').forEach(line => {
|
|
647
|
+
if (!pinnedIds.has(line.dataset.id)) {
|
|
648
|
+
line.remove();
|
|
649
|
+
}
|
|
650
|
+
});
|
|
651
|
+
|
|
652
|
+
// Add search results
|
|
653
|
+
for (const item of results) {
|
|
654
|
+
const id = 'line-' + (lineCounter++);
|
|
655
|
+
const div = document.createElement('div');
|
|
656
|
+
div.className = 'line';
|
|
657
|
+
div.dataset.id = id;
|
|
658
|
+
div.dataset.raw = item.raw;
|
|
659
|
+
div.dataset.html = item.html;
|
|
660
|
+
|
|
661
|
+
let displayHtml = item.html;
|
|
662
|
+
displayHtml = highlightTerms(displayHtml, includes, 'filter');
|
|
663
|
+
displayHtml = highlightTerms(displayHtml, highlights);
|
|
664
|
+
|
|
665
|
+
div.innerHTML = '<span class="line-content">' + displayHtml + '</span><span class="pin-btn" title="Pin">' + pinIcon + '</span>';
|
|
666
|
+
|
|
667
|
+
// Pin button handler
|
|
668
|
+
div.querySelector('.pin-btn').addEventListener('click', (e) => {
|
|
669
|
+
e.stopPropagation();
|
|
670
|
+
if (pinnedIds.has(id)) {
|
|
671
|
+
pinnedIds.delete(id);
|
|
672
|
+
div.classList.remove('pinned');
|
|
673
|
+
} else {
|
|
674
|
+
pinnedIds.add(id);
|
|
675
|
+
div.classList.add('pinned');
|
|
676
|
+
}
|
|
677
|
+
applyFiltersLocal();
|
|
678
|
+
});
|
|
679
|
+
|
|
680
|
+
container.appendChild(div);
|
|
681
|
+
}
|
|
682
|
+
|
|
683
|
+
// Jump to bottom and resume tailing
|
|
684
|
+
container.scrollTop = container.scrollHeight;
|
|
685
|
+
tailing = true;
|
|
686
|
+
updateTailButton();
|
|
687
|
+
} catch (e) {
|
|
688
|
+
if (e.name !== 'AbortError') {
|
|
689
|
+
console.error('Search failed:', e);
|
|
690
|
+
// Fallback to local filtering
|
|
691
|
+
applyFiltersLocal();
|
|
692
|
+
}
|
|
693
|
+
}
|
|
545
694
|
};
|
|
546
695
|
|
|
547
696
|
const trimBuffer = () => {
|
|
@@ -584,7 +733,7 @@ export class LogServer {
|
|
|
584
733
|
pinnedIds.add(id);
|
|
585
734
|
div.classList.add('pinned');
|
|
586
735
|
}
|
|
587
|
-
|
|
736
|
+
applyFiltersLocal();
|
|
588
737
|
});
|
|
589
738
|
|
|
590
739
|
const matches = matchesFilters(raw, includes, excludes);
|
|
@@ -613,9 +762,9 @@ export class LogServer {
|
|
|
613
762
|
debounceTimer = setTimeout(fn, delay);
|
|
614
763
|
};
|
|
615
764
|
|
|
616
|
-
includeInput.addEventListener('input', () => debounce(applyFilters,
|
|
617
|
-
excludeInput.addEventListener('input', () => debounce(applyFilters,
|
|
618
|
-
highlightInput.addEventListener('input', () => debounce(applyFilters,
|
|
765
|
+
includeInput.addEventListener('input', () => debounce(applyFilters, 300));
|
|
766
|
+
excludeInput.addEventListener('input', () => debounce(applyFilters, 300));
|
|
767
|
+
highlightInput.addEventListener('input', () => debounce(applyFilters, 150));
|
|
619
768
|
</script>
|
|
620
769
|
`;
|
|
621
770
|
}
|
package/src/teemux.ts
CHANGED
|
@@ -346,10 +346,10 @@ const main = async (): Promise<void> => {
|
|
|
346
346
|
description: 'Port for the log aggregation server',
|
|
347
347
|
type: 'number',
|
|
348
348
|
})
|
|
349
|
-
.option('
|
|
350
|
-
alias: '
|
|
351
|
-
default:
|
|
352
|
-
description: 'Number of log lines to keep in buffer',
|
|
349
|
+
.option('buffer', {
|
|
350
|
+
alias: 'b',
|
|
351
|
+
default: 10_000,
|
|
352
|
+
description: 'Number of log lines to keep in server buffer',
|
|
353
353
|
type: 'number',
|
|
354
354
|
})
|
|
355
355
|
.help()
|
|
@@ -368,7 +368,7 @@ const main = async (): Promise<void> => {
|
|
|
368
368
|
const name = argv.name ?? command[0] ?? 'unknown';
|
|
369
369
|
const port = argv.port;
|
|
370
370
|
|
|
371
|
-
const server = new LogServer(port, argv.
|
|
371
|
+
const server = new LogServer(port, argv.buffer);
|
|
372
372
|
|
|
373
373
|
// Try to become server with retries - if port is taken, become client
|
|
374
374
|
let isServer = false;
|
|
@@ -26,13 +26,13 @@ export type TeemuxContext = {
|
|
|
26
26
|
|
|
27
27
|
export type TeemuxOptions = {
|
|
28
28
|
/**
|
|
29
|
-
*
|
|
29
|
+
* Number of log lines to keep in the server buffer.
|
|
30
30
|
*/
|
|
31
|
-
|
|
31
|
+
buffer?: number;
|
|
32
32
|
/**
|
|
33
|
-
*
|
|
33
|
+
* Port to run on. If 0 or undefined, auto-assigns an available port.
|
|
34
34
|
*/
|
|
35
|
-
|
|
35
|
+
port?: number;
|
|
36
36
|
};
|
|
37
37
|
|
|
38
38
|
const postJson = (
|
|
@@ -83,7 +83,7 @@ export const runWithTeemux = async (
|
|
|
83
83
|
options: TeemuxOptions,
|
|
84
84
|
callback: (context: TeemuxContext) => Promise<void>,
|
|
85
85
|
): Promise<void> => {
|
|
86
|
-
const server = new LogServer(options.port ?? 0, options.
|
|
86
|
+
const server = new LogServer(options.port ?? 0, options.buffer ?? 10_000);
|
|
87
87
|
|
|
88
88
|
await server.start();
|
|
89
89
|
|