feat(DebugLogsPage): update log management with search, filtering, and pagination features
This commit is contained in:
@@ -3,32 +3,73 @@
|
||||
class="flex flex-col flex-1 overflow-hidden min-w-0 bg-gradient-to-br from-slate-50 via-slate-100 to-white dark:from-zinc-950 dark:via-zinc-900 dark:to-zinc-900"
|
||||
>
|
||||
<div class="flex flex-col h-full overflow-hidden w-full px-4 md:px-8 py-6">
|
||||
<div class="flex items-center justify-between mb-4 w-full max-w-6xl mx-auto">
|
||||
<div class="space-y-1">
|
||||
<div class="text-xs uppercase tracking-wide text-gray-500 dark:text-gray-400">Diagnostics</div>
|
||||
<div class="text-3xl font-semibold text-gray-900 dark:text-white">Debug Logs</div>
|
||||
<div class="flex flex-col mb-4 w-full max-w-6xl mx-auto space-y-4">
|
||||
<div class="flex items-center justify-between">
|
||||
<div class="space-y-1">
|
||||
<div class="text-xs uppercase tracking-wide text-gray-500 dark:text-gray-400">Diagnostics</div>
|
||||
<div class="text-3xl font-semibold text-gray-900 dark:text-white">Debug Logs</div>
|
||||
</div>
|
||||
<div class="flex gap-2">
|
||||
<button type="button" class="secondary-chip px-4 py-2 text-sm" @click="refreshLogs">
|
||||
<MaterialDesignIcon icon-name="refresh" class="w-4 h-4" />
|
||||
Refresh
|
||||
</button>
|
||||
<button type="button" class="primary-chip px-4 py-2 text-sm" @click="copyLogs">
|
||||
<MaterialDesignIcon icon-name="content-copy" class="w-4 h-4" />
|
||||
Copy All
|
||||
</button>
|
||||
</div>
|
||||
</div>
|
||||
<div class="flex gap-2">
|
||||
<button type="button" class="secondary-chip px-4 py-2 text-sm" @click="refreshLogs">
|
||||
<MaterialDesignIcon icon-name="refresh" class="w-4 h-4" />
|
||||
Refresh
|
||||
</button>
|
||||
<button type="button" class="primary-chip px-4 py-2 text-sm" @click="copyLogs">
|
||||
<MaterialDesignIcon icon-name="content-copy" class="w-4 h-4" />
|
||||
Copy All
|
||||
</button>
|
||||
|
||||
<div class="flex flex-wrap gap-3 items-center bg-white/50 dark:bg-zinc-800/50 p-3 rounded-lg border border-gray-200 dark:border-zinc-700">
|
||||
<div class="relative flex-1 min-w-[200px]">
|
||||
<div class="absolute inset-y-0 left-0 pl-3 flex items-center pointer-events-none">
|
||||
<MaterialDesignIcon icon-name="magnify" class="w-4 h-4 text-gray-400" />
|
||||
</div>
|
||||
<input
|
||||
v-model="search"
|
||||
type="text"
|
||||
class="block w-full pl-10 pr-3 py-2 border border-gray-300 dark:border-zinc-600 rounded-md leading-5 bg-white dark:bg-zinc-900 text-gray-900 dark:text-white placeholder-gray-500 focus:outline-none focus:ring-1 focus:ring-blue-500 focus:border-blue-500 sm:text-sm"
|
||||
placeholder="Search logs..."
|
||||
@input="debouncedSearch"
|
||||
/>
|
||||
</div>
|
||||
|
||||
<select
|
||||
v-model="level"
|
||||
class="block pl-3 pr-10 py-2 text-base border-gray-300 dark:border-zinc-600 focus:outline-none focus:ring-blue-500 focus:border-blue-500 sm:text-sm rounded-md bg-white dark:bg-zinc-900 text-gray-900 dark:text-white"
|
||||
@change="refreshLogs"
|
||||
>
|
||||
<option value="">All Levels</option>
|
||||
<option value="DEBUG">Debug</option>
|
||||
<option value="INFO">Info</option>
|
||||
<option value="WARNING">Warning</option>
|
||||
<option value="ERROR">Error</option>
|
||||
<option value="CRITICAL">Critical</option>
|
||||
</select>
|
||||
|
||||
<label class="inline-flex items-center cursor-pointer">
|
||||
<input
|
||||
v-model="is_anomaly"
|
||||
type="checkbox"
|
||||
class="form-checkbox h-4 w-4 text-blue-600 transition duration-150 ease-in-out"
|
||||
@change="refreshLogs"
|
||||
/>
|
||||
<span class="ml-2 text-sm text-gray-700 dark:text-gray-300">Anomalies Only</span>
|
||||
</label>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="flex-1 overflow-hidden glass-card max-w-6xl mx-auto w-full p-0 flex flex-col">
|
||||
<div class="flex-1 overflow-auto p-4 font-mono text-[10px] sm:text-xs leading-relaxed select-text">
|
||||
<div class="flex-1 overflow-hidden glass-card max-w-6xl mx-auto w-full p-0 flex flex-col rounded-sm">
|
||||
<div class="flex-1 overflow-auto p-4 font-mono text-[10px] sm:text-xs leading-relaxed select-text bg-white dark:bg-zinc-950">
|
||||
<div v-if="logs.length === 0" class="text-gray-500 italic text-center py-10">
|
||||
No logs collected yet.
|
||||
{{ loading ? 'Loading logs...' : 'No logs found matching your criteria.' }}
|
||||
</div>
|
||||
<div
|
||||
v-for="(log, index) in logs"
|
||||
:key="index"
|
||||
class="border-b border-gray-100 dark:border-zinc-800 py-1 flex gap-3"
|
||||
class="border-b border-gray-100 dark:border-zinc-900 py-1 flex gap-3 hover:bg-gray-50 dark:hover:bg-zinc-900/50"
|
||||
:class="{'bg-red-50/30 dark:bg-red-900/10': log.is_anomaly}"
|
||||
>
|
||||
<span class="text-gray-400 shrink-0">{{ formatTime(log.timestamp) }}</span>
|
||||
<span :class="levelClass(log.level)" class="w-12 shrink-0 font-bold uppercase">{{
|
||||
@@ -37,7 +78,64 @@
|
||||
<span class="text-blue-500 shrink-0 w-24 overflow-hidden text-ellipsis italic"
|
||||
>[{{ log.module }}]</span
|
||||
>
|
||||
<span class="text-gray-800 dark:text-gray-200 break-words">{{ log.message }}</span>
|
||||
<span class="text-gray-800 dark:text-gray-200 break-words flex-1">
|
||||
{{ log.message }}
|
||||
<span v-if="log.is_anomaly" class="ml-2 inline-flex items-center px-1.5 py-0.5 rounded-full text-[8px] font-medium bg-red-100 text-red-800 dark:bg-red-900 dark:text-red-200 uppercase">
|
||||
<MaterialDesignIcon icon-name="alert-circle" class="w-2.5 h-2.5 mr-1" />
|
||||
{{ log.anomaly_type || 'anomaly' }}
|
||||
</span>
|
||||
</span>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- Pagination -->
|
||||
<div class="px-4 py-3 flex items-center justify-between border-t border-gray-200 dark:border-zinc-800 bg-gray-50 dark:bg-zinc-900/50">
|
||||
<div class="flex-1 flex justify-between sm:hidden">
|
||||
<button
|
||||
class="relative inline-flex items-center px-4 py-2 border border-gray-300 text-sm font-medium rounded-md text-gray-700 bg-white hover:bg-gray-50 disabled:opacity-50"
|
||||
:disabled="offset === 0"
|
||||
@click="prevPage"
|
||||
>
|
||||
Previous
|
||||
</button>
|
||||
<button
|
||||
class="ml-3 relative inline-flex items-center px-4 py-2 border border-gray-300 text-sm font-medium rounded-md text-gray-700 bg-white hover:bg-gray-50 disabled:opacity-50"
|
||||
:disabled="offset + limit >= total"
|
||||
@click="nextPage"
|
||||
>
|
||||
Next
|
||||
</button>
|
||||
</div>
|
||||
<div class="hidden sm:flex-1 sm:flex sm:items-center sm:justify-between">
|
||||
<div>
|
||||
<p class="text-sm text-gray-700 dark:text-gray-400 font-mono">
|
||||
Showing
|
||||
<span class="font-bold">{{ total === 0 ? 0 : offset + 1 }}</span>
|
||||
to
|
||||
<span class="font-bold">{{ Math.min(offset + limit, total) }}</span>
|
||||
of
|
||||
<span class="font-bold">{{ total }}</span>
|
||||
results
|
||||
</p>
|
||||
</div>
|
||||
<div class="flex gap-2">
|
||||
<button
|
||||
class="secondary-chip px-3 py-1 text-xs disabled:opacity-50"
|
||||
:disabled="offset === 0"
|
||||
@click="prevPage"
|
||||
>
|
||||
<MaterialDesignIcon icon-name="chevron-left" class="w-4 h-4" />
|
||||
Previous
|
||||
</button>
|
||||
<button
|
||||
class="secondary-chip px-3 py-1 text-xs disabled:opacity-50"
|
||||
:disabled="offset + limit >= total"
|
||||
@click="nextPage"
|
||||
>
|
||||
Next
|
||||
<MaterialDesignIcon icon-name="chevron-right" class="w-4 h-4" />
|
||||
</button>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
@@ -57,33 +155,80 @@ export default {
|
||||
data() {
|
||||
return {
|
||||
logs: [],
|
||||
total: 0,
|
||||
limit: 100,
|
||||
offset: 0,
|
||||
search: "",
|
||||
level: "",
|
||||
is_anomaly: false,
|
||||
loading: false,
|
||||
updateInterval: null,
|
||||
searchTimeout: null,
|
||||
};
|
||||
},
|
||||
mounted() {
|
||||
this.refreshLogs();
|
||||
this.updateInterval = setInterval(() => {
|
||||
this.refreshLogs();
|
||||
// Only auto-refresh if on page 1 and no search
|
||||
if (this.offset === 0 && !this.search && !this.is_anomaly && !this.level) {
|
||||
this.refreshLogs(true);
|
||||
}
|
||||
}, 5000);
|
||||
},
|
||||
beforeUnmount() {
|
||||
if (this.updateInterval) {
|
||||
clearInterval(this.updateInterval);
|
||||
}
|
||||
if (this.searchTimeout) {
|
||||
clearTimeout(this.searchTimeout);
|
||||
}
|
||||
},
|
||||
methods: {
|
||||
async refreshLogs() {
|
||||
async refreshLogs(silent = false) {
|
||||
if (!silent) this.loading = true;
|
||||
try {
|
||||
const response = await window.axios.get("/api/v1/debug/logs");
|
||||
this.logs = response.data;
|
||||
const params = {
|
||||
limit: this.limit,
|
||||
offset: this.offset,
|
||||
search: this.search || undefined,
|
||||
level: this.level || undefined,
|
||||
is_anomaly: this.is_anomaly ? true : undefined,
|
||||
};
|
||||
const response = await window.axios.get("/api/v1/debug/logs", { params });
|
||||
this.logs = response.data.logs;
|
||||
this.total = response.data.total;
|
||||
} catch (e) {
|
||||
console.log("Failed to fetch logs", e);
|
||||
if (!silent) ToastUtils.error("Failed to fetch logs");
|
||||
} finally {
|
||||
if (!silent) this.loading = false;
|
||||
}
|
||||
},
|
||||
debouncedSearch() {
|
||||
if (this.searchTimeout) clearTimeout(this.searchTimeout);
|
||||
this.searchTimeout = setTimeout(() => {
|
||||
this.offset = 0;
|
||||
this.refreshLogs();
|
||||
}, 500);
|
||||
},
|
||||
prevPage() {
|
||||
if (this.offset >= this.limit) {
|
||||
this.offset -= this.limit;
|
||||
this.refreshLogs();
|
||||
}
|
||||
},
|
||||
nextPage() {
|
||||
if (this.offset + this.limit < this.total) {
|
||||
this.offset += this.limit;
|
||||
this.refreshLogs();
|
||||
}
|
||||
},
|
||||
formatTime(timestamp) {
|
||||
try {
|
||||
const date = new Date(timestamp);
|
||||
return date.toLocaleTimeString();
|
||||
// If timestamp is a number (Unix timestamp from Python), multiply by 1000 for JS
|
||||
const ts = typeof timestamp === 'number' ? timestamp * 1000 : timestamp;
|
||||
const date = new Date(ts);
|
||||
return date.toLocaleString();
|
||||
} catch {
|
||||
return timestamp;
|
||||
}
|
||||
@@ -97,10 +242,10 @@ export default {
|
||||
return "text-gray-400";
|
||||
},
|
||||
async copyLogs() {
|
||||
const logText = this.logs.map((l) => `${l.timestamp} [${l.level}] [${l.module}] ${l.message}`).join("\n");
|
||||
const logText = this.logs.map((l) => `${this.formatTime(l.timestamp)} [${l.level}] [${l.module}] ${l.message}${l.is_anomaly ? ' [ANOMALY:' + l.anomaly_type + ']' : ''}`).join("\n");
|
||||
try {
|
||||
await navigator.clipboard.writeText(logText);
|
||||
ToastUtils.success("Logs copied to clipboard");
|
||||
ToastUtils.success("Logs on this page copied to clipboard");
|
||||
} catch {
|
||||
ToastUtils.error("Failed to copy logs");
|
||||
}
|
||||
@@ -108,3 +253,10 @@ export default {
|
||||
},
|
||||
};
|
||||
</script>
|
||||
|
||||
<style scoped>
|
||||
.glass-card {
|
||||
/* Square corners override if glass-card has them rounded */
|
||||
border-radius: 2px !important;
|
||||
}
|
||||
</style>
|
||||
|
||||
Reference in New Issue
Block a user