Renames the app/ingest/ package to app/glean/ and updates all references across Python modules, shell scripts, Vue components, tests, and documentation. Intentionally preserved: - SQLite column name ingest_time (avoids schema migration) - RetrievedEntry.ingest_time field (maps to the column above) - Any public-facing JSON keys that reference ingest_time Changes by category: - app/ingest/ → app/glean/ (full package move, all parsers) - app/tasks/ingest_scheduler.py → app/tasks/glean_scheduler.py - scripts/ingest_corpus.py → scripts/glean_corpus.py - tests/test_ingest_*.py → tests/test_glean_*.py - Docstrings, log messages, comments: ingest → glean - Env var: TURNSTONE_INGEST_INTERVAL → TURNSTONE_GLEAN_INTERVAL - Shell scripts: glean.log, glean_corpus.py references - README.md: multi-source ingest → multi-source glean - .env.example: updated env var name - patterns/: new diagnostic patterns from 2026-05-20 SSH incident (service_crash_loop, pkg_daemon_restart, ssh_forward_conflict) - SourcesView.vue: pipeline label updated - All test import paths updated to app.glean.* 285 tests passing.
133 lines
5.5 KiB
Vue
133 lines
5.5 KiB
Vue
<template>
|
|
<div class="flex flex-col md:flex-row md:h-[calc(100vh-49px)]">
|
|
|
|
<!-- Mobile filter toggle -->
|
|
<div class="md:hidden border-b border-surface-border px-4 py-2">
|
|
<button
|
|
@click="sidebarOpen = !sidebarOpen"
|
|
class="text-xs text-text-dim hover:text-text-primary transition-colors"
|
|
>{{ sidebarOpen ? 'Hide filters ▲' : 'Filters ▼' }}</button>
|
|
</div>
|
|
|
|
<!-- Sidebar: filters -->
|
|
<aside :class="[
|
|
'bg-surface-raised p-4 flex flex-col gap-5 overflow-y-auto',
|
|
'md:w-56 md:shrink-0 md:border-r md:border-surface-border',
|
|
sidebarOpen ? 'flex border-b border-surface-border' : 'hidden md:flex',
|
|
]">
|
|
<div>
|
|
<h3 class="text-text-dim text-xs uppercase tracking-widest mb-2">Source</h3>
|
|
<div class="flex flex-col gap-1">
|
|
<button
|
|
class="text-left px-2 py-1 rounded text-sm transition-colors"
|
|
:class="!store.sourceFilter ? 'text-accent bg-accent-muted' : 'text-text-muted hover:text-text-primary hover:bg-surface-border'"
|
|
@click="store.sourceFilter = undefined"
|
|
>All sources</button>
|
|
<button
|
|
v-for="src in store.sources"
|
|
:key="src.source_id"
|
|
class="text-left px-2 py-1 rounded text-xs transition-colors truncate"
|
|
:class="store.sourceFilter === src.source_id ? 'text-accent bg-accent-muted' : 'text-text-muted hover:text-text-primary hover:bg-surface-border'"
|
|
:title="src.source_id"
|
|
@click="store.sourceFilter = src.source_id"
|
|
>
|
|
{{ src.source_id }}
|
|
<span v-if="src.error_count" class="text-sev-error ml-1">{{ src.error_count }}e</span>
|
|
</button>
|
|
</div>
|
|
</div>
|
|
|
|
<div>
|
|
<h3 class="text-text-dim text-xs uppercase tracking-widest mb-2">Severity</h3>
|
|
<div class="flex flex-col gap-1">
|
|
<button
|
|
class="text-left px-2 py-1 rounded text-sm transition-colors"
|
|
:class="!store.severityFilter ? 'text-accent bg-accent-muted' : 'text-text-muted hover:text-text-primary hover:bg-surface-border'"
|
|
@click="store.severityFilter = undefined"
|
|
>All</button>
|
|
<button
|
|
v-for="sev in store.severityOptions"
|
|
:key="sev"
|
|
class="text-left px-2 py-1 rounded text-xs transition-colors"
|
|
:class="store.severityFilter === sev ? 'text-accent bg-accent-muted' : 'text-text-muted hover:text-text-primary hover:bg-surface-border'"
|
|
@click="store.severityFilter = sev"
|
|
>{{ sev }}</button>
|
|
</div>
|
|
</div>
|
|
|
|
<div>
|
|
<h3 class="text-text-dim text-xs uppercase tracking-widest mb-2">Limit</h3>
|
|
<select
|
|
v-model.number="store.limit"
|
|
class="w-full bg-surface border border-surface-border rounded px-2 py-1 text-sm text-text-primary"
|
|
>
|
|
<option v-for="n in [20, 50, 100, 200]" :key="n" :value="n">{{ n }}</option>
|
|
</select>
|
|
</div>
|
|
|
|
<button
|
|
class="mt-auto text-text-dim text-xs hover:text-text-muted transition-colors"
|
|
@click="store.clearFilters()"
|
|
>Clear filters</button>
|
|
</aside>
|
|
|
|
<!-- Main: search + results -->
|
|
<main class="flex-1 flex flex-col min-w-0 min-h-0">
|
|
<!-- Search bar -->
|
|
<div class="border-b border-surface-border p-3 sm:p-4 flex gap-2 sm:gap-3">
|
|
<input
|
|
v-model="store.query"
|
|
type="text"
|
|
placeholder="Search logs…"
|
|
class="flex-1 bg-surface-raised border border-surface-border rounded px-3 sm:px-4 py-2 text-sm text-text-primary placeholder-text-dim focus:outline-none focus:border-accent transition-colors"
|
|
@keydown.enter="store.runSearch()"
|
|
/>
|
|
<button
|
|
class="px-4 sm:px-5 py-2 rounded bg-accent text-white text-sm font-semibold hover:bg-blue-400 transition-colors disabled:opacity-50"
|
|
:disabled="store.loading || !store.query.trim()"
|
|
@click="store.runSearch()"
|
|
>
|
|
<span v-if="store.loading">…</span>
|
|
<span v-else>Search</span>
|
|
</button>
|
|
</div>
|
|
|
|
<!-- Results header -->
|
|
<div v-if="store.hasResults || store.error" class="border-b border-surface-border px-4 py-2 flex items-center gap-3 text-xs text-text-dim">
|
|
<span v-if="store.error" class="text-sev-error">{{ store.error }}</span>
|
|
<span v-else>{{ store.total }} result{{ store.total !== 1 ? 's' : '' }}</span>
|
|
</div>
|
|
|
|
<!-- Empty / zero states -->
|
|
<div v-if="!store.loading && !store.hasResults && !store.error" class="flex-1 flex flex-col items-center justify-center text-text-dim gap-3 px-4">
|
|
<div v-if="!store.query" class="text-center">
|
|
<p class="text-lg mb-1">Search your log corpus</p>
|
|
<p class="text-sm">Type a query and press Enter or click Search.</p>
|
|
</div>
|
|
<div v-else class="text-center">
|
|
<p class="text-base mb-1">No results for "{{ store.query }}"</p>
|
|
<p class="text-sm">Try broader terms or check the Sources tab to confirm data is gleaned.</p>
|
|
</div>
|
|
</div>
|
|
|
|
<!-- Results list -->
|
|
<div v-else class="flex-1 overflow-y-auto">
|
|
<LogEntryRow
|
|
v-for="entry in store.results"
|
|
:key="entry.entry_id"
|
|
:entry="entry"
|
|
/>
|
|
</div>
|
|
</main>
|
|
</div>
|
|
</template>
|
|
|
|
<script setup lang="ts">
|
|
import { ref, onMounted } from 'vue'
|
|
import { useSearchStore } from '@/stores/search'
|
|
import LogEntryRow from '@/components/LogEntryRow.vue'
|
|
|
|
const store = useSearchStore()
|
|
const sidebarOpen = ref(false)
|
|
onMounted(() => store.loadSources())
|
|
</script>
|