* feat: card alignment + fix modals * feat: change admon title in restore alert modal * fix: lint * feat: backups queue api into api-client * feat: impl backup queue api endpoints into frontend * feat: ack fix * feat: bulk actions * feat: bulk delete impl * fix: lint * fix: align error states * fix: transition group * feat: ready for qa * fix: lint * feat: qa * feat: stacked admonitions component * fix: issues with stacking * feat: hook up admonition stacking + fix app csp for staging kyros nodes * fix: logs.vue * qa: close stack on admonitions click * fix: all problems with stacked admonitions * qa: admonition cleanup and copy overhaul draft * fix: qa issues padding * fix: padding bug * feat: qa * fix: intercom in app csp bug * fix: positioning intercom * feat: loading overlay on top of console + admon consistency changes * feat: scroll indicator fade in backup delete modal + admon timestamp fix * feat: move action bar behind modal * fix: lint + i18n * fix: server ping spam on filter (cache but clear on unmount) * fix: 1 admon fade in flicker issue * chore: temp staging undo * qa: changes * fix: lint * chore: revert staging to use staging * fix: scoping
399 lines
11 KiB
Vue
399 lines
11 KiB
Vue
<template>
|
|
<div
|
|
class="flex min-h-0 flex-1 flex-col gap-4"
|
|
:class="
|
|
isFullscreen ? `fixed inset-0 z-[15] bg-surface-1 p-6 py-8 ${isApp ? 'pt-12' : ''}` : ''
|
|
"
|
|
>
|
|
<CollapsibleAdmonition
|
|
v-if="ctx.crashAnalysis?.value && !isFullscreen"
|
|
type="critical"
|
|
:header="crashHeader"
|
|
:items="crashItems"
|
|
dismissible
|
|
@dismiss="ctx.onDismissCrash?.()"
|
|
/>
|
|
|
|
<div class="flex items-center gap-2">
|
|
<StyledInput
|
|
v-model="searchQuery"
|
|
:icon="SearchIcon"
|
|
placeholder="Search logs"
|
|
wrapper-class="flex-1"
|
|
input-class="!h-10"
|
|
clearable
|
|
/>
|
|
<div v-if="ctx.logSources?.value && ctx.activeLogSourceIndex" class="w-[220px]">
|
|
<Combobox
|
|
:model-value="ctx.activeLogSourceIndex.value"
|
|
:options="logSourceOptions"
|
|
@update:model-value="(v) => (ctx.activeLogSourceIndex!.value = v)"
|
|
/>
|
|
</div>
|
|
</div>
|
|
|
|
<div class="flex items-center justify-between">
|
|
<ConsoleFilterPills
|
|
v-model="activeFilters"
|
|
:present-levels="presentLevels"
|
|
@toggle="handleFilterToggle"
|
|
/>
|
|
<ConsoleActionButtons
|
|
:show-clear="isLiveSource"
|
|
:has-logs="hasLogs"
|
|
:share-disabled="resolvedShareDisabled"
|
|
:sharing="isSharing"
|
|
:fullscreen="isFullscreen"
|
|
:show-delete="showDelete"
|
|
:delete-disabled="resolvedDeleteDisabled"
|
|
:delete-disabled-tooltip="ctx.deleteDisabledTooltip"
|
|
@clear="handleClear"
|
|
@share="handleShare"
|
|
@toggle-fullscreen="toggleFullscreen"
|
|
@delete="handleDelete"
|
|
/>
|
|
</div>
|
|
|
|
<BaseTerminal
|
|
ref="terminalRef"
|
|
class="min-h-0 flex-1"
|
|
:show-input="resolvedShowInput"
|
|
:disable-input="resolvedInputDisabled"
|
|
:fullscreen="isFullscreen"
|
|
:empty-state-type="ctx.emptyStateType"
|
|
:loading="resolvedLoading"
|
|
@command="handleCommand"
|
|
@ready="handleTerminalReady"
|
|
/>
|
|
</div>
|
|
<ShareModal ref="shareModal" header="Share Logs" link :social-buttons="false" />
|
|
<NewModal ref="deleteModal" header="Delete log file" :fade="'danger'" max-width="500px">
|
|
<div class="flex flex-col gap-6">
|
|
<Admonition type="critical" header="This is irreversible">
|
|
Deleting this log file cannot be undone. Are you sure you want to continue?
|
|
</Admonition>
|
|
</div>
|
|
<template #actions>
|
|
<div class="flex justify-end gap-2">
|
|
<ButtonStyled type="outlined">
|
|
<button class="!border !border-surface-4" @click="deleteModal?.hide()">
|
|
<XIcon />
|
|
Cancel
|
|
</button>
|
|
</ButtonStyled>
|
|
<ButtonStyled color="red">
|
|
<button :disabled="isDeleting" @click="confirmDelete">
|
|
<TrashIcon />
|
|
Delete
|
|
</button>
|
|
</ButtonStyled>
|
|
</div>
|
|
</template>
|
|
</NewModal>
|
|
</template>
|
|
|
|
<script setup lang="ts">
|
|
import { SearchIcon, TrashIcon, XIcon } from '@modrinth/assets'
|
|
import type { Terminal } from '@xterm/xterm'
|
|
import { computed, isRef, nextTick, onBeforeUnmount, ref, watch } from 'vue'
|
|
|
|
import Admonition from '#ui/components/base/Admonition.vue'
|
|
import BaseTerminal from '#ui/components/base/BaseTerminal.vue'
|
|
import ButtonStyled from '#ui/components/base/ButtonStyled.vue'
|
|
import type { CollapsibleAdmonitionItem } from '#ui/components/base/CollapsibleAdmonition.vue'
|
|
import CollapsibleAdmonition from '#ui/components/base/CollapsibleAdmonition.vue'
|
|
import Combobox from '#ui/components/base/Combobox.vue'
|
|
import StyledInput from '#ui/components/base/StyledInput.vue'
|
|
import NewModal from '#ui/components/modal/NewModal.vue'
|
|
import ShareModal from '#ui/components/modal/ShareModal.vue'
|
|
import { injectModrinthClient } from '#ui/providers'
|
|
import { injectModalBehavior } from '#ui/providers/modal-behavior'
|
|
import { injectNotificationManager } from '#ui/providers/web-notifications.ts'
|
|
|
|
import ConsoleActionButtons from './components/ConsoleActionButtons.vue'
|
|
import ConsoleFilterPills from './components/ConsoleFilterPills.vue'
|
|
import {
|
|
clearSearchHighlights,
|
|
colorize,
|
|
getHighlightVersion,
|
|
highlightAppendedRange,
|
|
rewriteTerminal,
|
|
useConsoleFilters,
|
|
} from './composables'
|
|
import type { ConditionalLevel } from './composables/console-filtering'
|
|
import { injectConsoleManager } from './providers'
|
|
import type { LogLevel, LogLine } from './types'
|
|
|
|
const ctx = injectConsoleManager()
|
|
const client = injectModrinthClient()
|
|
const modalBehavior = injectModalBehavior()
|
|
const { addNotification } = injectNotificationManager()
|
|
|
|
const crashHeader = computed(() => {
|
|
const problems = ctx.crashAnalysis?.value?.analysis.problems ?? []
|
|
const count = problems.length
|
|
return `${count} problem${count !== 1 ? 's' : ''} detected`
|
|
})
|
|
|
|
const crashItems = computed<CollapsibleAdmonitionItem[]>(() => {
|
|
const problems = ctx.crashAnalysis?.value?.analysis.problems ?? []
|
|
return problems.map((p) => ({
|
|
title: p.message,
|
|
descriptions: p.solutions.map((s) => s.message),
|
|
}))
|
|
})
|
|
|
|
const terminalRef = ref<InstanceType<typeof BaseTerminal> | null>(null)
|
|
const shareModal = ref<InstanceType<typeof ShareModal> | null>(null)
|
|
const deleteModal = ref<InstanceType<typeof NewModal> | null>(null)
|
|
const isDeleting = ref(false)
|
|
const searchQuery = ref('')
|
|
const isFullscreen = ref(false)
|
|
const isApp =
|
|
typeof window !== 'undefined' && !!(window as Record<string, unknown>).__TAURI_INTERNALS__
|
|
const isSharing = ref(false)
|
|
const { activeFilters, toggleFilter, buildFilterPredicate } = useConsoleFilters()
|
|
const hasLogs = computed(() => ctx.logLines.value.length > 0)
|
|
const presentLevels = computed(() => {
|
|
const levels = new Set<ConditionalLevel>()
|
|
for (const line of ctx.logLines.value) {
|
|
if (line.level === 'debug') levels.add('debug')
|
|
if (line.level === 'trace') levels.add('trace')
|
|
if (levels.size === 2) break
|
|
}
|
|
return levels
|
|
})
|
|
const isLiveSource = computed(() => {
|
|
const sources = ctx.logSources?.value
|
|
const index = ctx.activeLogSourceIndex?.value
|
|
if (!sources || index === undefined) return true
|
|
return sources[index]?.live ?? true
|
|
})
|
|
const logSourceOptions = computed(() =>
|
|
(ctx.logSources?.value ?? []).map((s, i) => ({ value: i, label: s.name })),
|
|
)
|
|
|
|
function buildCombinedPredicate(): ((line: LogLine) => boolean) | null {
|
|
const levelPred = buildFilterPredicate()
|
|
const query = searchQuery.value.trim().toLowerCase()
|
|
if (!levelPred && !query) return null
|
|
return (line: LogLine) => {
|
|
if (levelPred && !levelPred(line)) return false
|
|
if (query && !line.text.toLowerCase().includes(query)) return false
|
|
return true
|
|
}
|
|
}
|
|
|
|
onBeforeUnmount(() => {
|
|
if (isFullscreen.value) {
|
|
document.body.style.overflow = ''
|
|
modalBehavior?.onHide?.()
|
|
}
|
|
})
|
|
|
|
let lastWrittenIndex = 0
|
|
let searchDebounce: ReturnType<typeof setTimeout> | null = null
|
|
|
|
const resolvedShowInput = computed(() => {
|
|
const v = ctx.showCommandInput
|
|
if (v === undefined) return false
|
|
if (typeof v === 'boolean') return v
|
|
return isRef(v) ? v.value : v
|
|
})
|
|
|
|
const resolvedDisableInput = computed(() => {
|
|
const v = ctx.disableCommandInput
|
|
if (!v) return false
|
|
return isRef(v) ? v.value : v
|
|
})
|
|
|
|
// needs historical log start/end flags on ws to be properly useful
|
|
const resolvedLoading = computed(() => {
|
|
const v = ctx.loading
|
|
if (!v) return false
|
|
return v.value
|
|
})
|
|
|
|
const resolvedInputDisabled = computed(() => resolvedDisableInput.value || resolvedLoading.value)
|
|
|
|
const resolvedShareDisabled = computed(() => {
|
|
const v = ctx.shareDisabled
|
|
if (!v) return false
|
|
return isRef(v) ? v.value : v
|
|
})
|
|
|
|
const showDelete = computed(() => !isLiveSource.value && ctx.onDelete != null)
|
|
|
|
const resolvedDeleteDisabled = computed(() => {
|
|
const v = ctx.deleteDisabled
|
|
if (!v) return false
|
|
return isRef(v) ? v.value : v
|
|
})
|
|
|
|
function handleTerminalReady(_terminal: Terminal) {
|
|
rewriteFiltered()
|
|
}
|
|
|
|
function handleFilterToggle(value: LogLevel | 'all') {
|
|
toggleFilter(value)
|
|
rewriteFiltered()
|
|
}
|
|
|
|
function activeSearchQuery(): string {
|
|
return searchQuery.value.trim().toLowerCase()
|
|
}
|
|
|
|
function rewriteFiltered() {
|
|
const term = terminalRef.value?.terminal
|
|
if (!term) return
|
|
const lines = ctx.logLines.value
|
|
if (resolvedLoading.value && lines.length === 0 && isLiveSource.value) {
|
|
terminalRef.value?.clearEmptyState()
|
|
lastWrittenIndex = 0
|
|
return
|
|
}
|
|
if (lines.length === 0 && isLiveSource.value) {
|
|
writeEmptyState()
|
|
return
|
|
}
|
|
terminalRef.value?.clearEmptyState()
|
|
const predicate = buildCombinedPredicate()
|
|
rewriteTerminal(term, lines, predicate, activeSearchQuery())
|
|
lastWrittenIndex = lines.length
|
|
}
|
|
|
|
function toggleFullscreen() {
|
|
isFullscreen.value = !isFullscreen.value
|
|
if (isFullscreen.value) {
|
|
document.body.style.overflow = 'hidden'
|
|
modalBehavior?.onShow?.()
|
|
} else {
|
|
document.body.style.overflow = ''
|
|
modalBehavior?.onHide?.()
|
|
}
|
|
nextTick(() => {
|
|
terminalRef.value?.fit()
|
|
})
|
|
}
|
|
|
|
function writeEmptyState() {
|
|
terminalRef.value?.writeEmptyState()
|
|
lastWrittenIndex = 0
|
|
}
|
|
|
|
watch(ctx.logLines, (lines, oldLines) => {
|
|
const term = terminalRef.value?.terminal
|
|
if (!term) return
|
|
|
|
if (lines.length === 0 && isLiveSource.value) {
|
|
if (resolvedLoading.value) {
|
|
terminalRef.value?.clearEmptyState()
|
|
lastWrittenIndex = 0
|
|
return
|
|
}
|
|
|
|
writeEmptyState()
|
|
return
|
|
}
|
|
|
|
if (
|
|
terminalRef.value?.showingEmptyState ||
|
|
lines !== oldLines ||
|
|
lines.length < lastWrittenIndex
|
|
) {
|
|
terminalRef.value?.clearEmptyState()
|
|
rewriteFiltered()
|
|
return
|
|
}
|
|
|
|
const predicate = buildCombinedPredicate()
|
|
const newLines: string[] = []
|
|
for (let i = lastWrittenIndex; i < lines.length; i++) {
|
|
if (!predicate || predicate(lines[i])) {
|
|
newLines.push(colorize(lines[i]))
|
|
}
|
|
}
|
|
if (newLines.length > 0) {
|
|
const buffer = term.buffer.active
|
|
const onFreshLine = buffer.cursorX === 0
|
|
const data = onFreshLine ? newLines.join('\r\n') : '\r\n' + newLines.join('\r\n')
|
|
const fromRow = buffer.baseY + buffer.cursorY
|
|
const version = getHighlightVersion(term)
|
|
term.write(data, () => {
|
|
highlightAppendedRange(term, fromRow, version)
|
|
})
|
|
}
|
|
lastWrittenIndex = lines.length
|
|
})
|
|
|
|
watch(searchQuery, () => {
|
|
if (searchDebounce) clearTimeout(searchDebounce)
|
|
searchDebounce = setTimeout(() => {
|
|
rewriteFiltered()
|
|
}, 200)
|
|
})
|
|
|
|
watch(resolvedLoading, (loading) => {
|
|
if (!loading) {
|
|
rewriteFiltered()
|
|
}
|
|
})
|
|
|
|
function handleCommand(cmd: string) {
|
|
ctx.sendCommand?.(cmd)
|
|
}
|
|
|
|
function handleClear() {
|
|
const term = terminalRef.value?.terminal
|
|
if (term) clearSearchHighlights(term)
|
|
terminalRef.value?.reset()
|
|
lastWrittenIndex = 0
|
|
ctx.onClear?.()
|
|
}
|
|
|
|
function handleDelete() {
|
|
deleteModal.value?.show()
|
|
}
|
|
|
|
async function confirmDelete() {
|
|
if (!ctx.onDelete) return
|
|
isDeleting.value = true
|
|
try {
|
|
await ctx.onDelete()
|
|
deleteModal.value?.hide()
|
|
} catch (err) {
|
|
console.error('Failed to delete log file:', err)
|
|
addNotification({
|
|
type: 'error',
|
|
title: 'Failed to delete log file',
|
|
text: typeof err === 'string' ? err : 'Unknown error.',
|
|
})
|
|
} finally {
|
|
isDeleting.value = false
|
|
}
|
|
}
|
|
|
|
async function handleShare() {
|
|
const predicate = buildCombinedPredicate()
|
|
const lines = predicate ? ctx.logLines.value.filter(predicate) : ctx.logLines.value
|
|
const content = lines.map((l) => l.text).join('\n')
|
|
|
|
isSharing.value = true
|
|
try {
|
|
const data = await client.mclogs.logs_v1.create(content)
|
|
if (data.url) {
|
|
shareModal.value?.show(data.url)
|
|
}
|
|
} catch (err) {
|
|
console.error('Failed to share logs:', err)
|
|
addNotification({
|
|
type: 'error',
|
|
title: 'Failed to share logs',
|
|
text: typeof err === 'string' ? err : 'Unknown error.',
|
|
})
|
|
} finally {
|
|
isSharing.value = false
|
|
}
|
|
}
|
|
</script>
|