시스템 모니터
This commit is contained in:
213
frontend/components/DashboardControl.vue
Normal file
213
frontend/components/DashboardControl.vue
Normal file
@@ -0,0 +1,213 @@
|
||||
<template>
|
||||
<div class="control-panel">
|
||||
<div class="control-row">
|
||||
<span class="control-label">조회 간격:</span>
|
||||
<div class="interval-buttons">
|
||||
<button
|
||||
v-for="min in [1, 2, 3, 4, 5]"
|
||||
:key="min"
|
||||
:class="['interval-btn', { active: autoRefresh && interval === min }]"
|
||||
@click="selectInterval(min)"
|
||||
>
|
||||
{{ min }}분
|
||||
</button>
|
||||
</div>
|
||||
|
||||
<div class="control-divider"></div>
|
||||
|
||||
<button
|
||||
:class="['auto-refresh-btn', { active: autoRefresh }]"
|
||||
@click="toggleAutoRefresh"
|
||||
>
|
||||
{{ autoRefresh ? '⏸ 자동갱신 ON' : '▶ 자동갱신 OFF' }}
|
||||
</button>
|
||||
|
||||
<div class="last-fetch-info">
|
||||
마지막 조회: <span class="last-fetch-time">{{ relativeTime }}</span>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="control-row">
|
||||
<span class="control-label">특정시간:</span>
|
||||
<input
|
||||
type="datetime-local"
|
||||
v-model="selectedDatetime"
|
||||
class="datetime-input"
|
||||
:disabled="autoRefresh || fetchState !== 'idle'"
|
||||
:class="{ disabled: autoRefresh || fetchState !== 'idle' }"
|
||||
/>
|
||||
<button
|
||||
class="refresh-btn"
|
||||
:disabled="autoRefresh || fetchState !== 'idle'"
|
||||
:class="{
|
||||
disabled: autoRefresh,
|
||||
loading: fetchState === 'loading',
|
||||
success: fetchState === 'success'
|
||||
}"
|
||||
@click="doRefresh"
|
||||
>
|
||||
<span v-if="fetchState === 'loading'" class="loading-spinner"></span>
|
||||
{{ buttonText }}
|
||||
</button>
|
||||
<span v-if="autoRefresh" class="hint-text">자동갱신 OFF 시 특정시간 조회 가능</span>
|
||||
</div>
|
||||
</div>
|
||||
</template>
|
||||
|
||||
<script setup lang="ts">
|
||||
const props = defineProps<{
|
||||
interval: number
|
||||
autoRefresh: boolean
|
||||
lastFetchTime: Date | null
|
||||
fetchState: 'idle' | 'loading' | 'success'
|
||||
}>()
|
||||
|
||||
const emit = defineEmits<{
|
||||
(e: 'update:interval', value: number): void
|
||||
(e: 'update:autoRefresh', value: boolean): void
|
||||
(e: 'fetchAt', datetime: string): void
|
||||
(e: 'refresh'): void
|
||||
}>()
|
||||
|
||||
// props destructure for template
|
||||
const { interval, autoRefresh, fetchState } = toRefs(props)
|
||||
|
||||
const selectedDatetime = ref('')
|
||||
const relativeTime = ref('-')
|
||||
|
||||
const buttonText = computed(() => {
|
||||
switch (props.fetchState) {
|
||||
case 'loading': return '조회 중...'
|
||||
case 'success': return '✓ 완료'
|
||||
default: return '조회'
|
||||
}
|
||||
})
|
||||
|
||||
function getCurrentDatetimeLocal(): string {
|
||||
const now = new Date()
|
||||
const y = now.getFullYear()
|
||||
const m = String(now.getMonth() + 1).padStart(2, '0')
|
||||
const d = String(now.getDate()).padStart(2, '0')
|
||||
const h = String(now.getHours()).padStart(2, '0')
|
||||
const min = String(now.getMinutes()).padStart(2, '0')
|
||||
return `${y}-${m}-${d}T${h}:${min}`
|
||||
}
|
||||
|
||||
function getRelativeTime(date: Date | null): string {
|
||||
if (!date) return '-'
|
||||
|
||||
const now = new Date()
|
||||
const diff = Math.floor((now.getTime() - date.getTime()) / 1000)
|
||||
|
||||
if (diff < 5) return '방금 전'
|
||||
if (diff < 60) return `${diff}초 전`
|
||||
if (diff < 3600) return `${Math.floor(diff / 60)}분 전`
|
||||
if (diff < 86400) return `${Math.floor(diff / 3600)}시간 전`
|
||||
return `${Math.floor(diff / 86400)}일 전`
|
||||
}
|
||||
|
||||
function updateRelativeTime() {
|
||||
relativeTime.value = getRelativeTime(props.lastFetchTime)
|
||||
}
|
||||
|
||||
let timeTimer: ReturnType<typeof window.setInterval> | null = null
|
||||
|
||||
onMounted(() => {
|
||||
updateRelativeTime()
|
||||
timeTimer = window.setInterval(updateRelativeTime, 1000)
|
||||
})
|
||||
|
||||
onUnmounted(() => {
|
||||
if (timeTimer) {
|
||||
window.clearInterval(timeTimer)
|
||||
}
|
||||
})
|
||||
|
||||
watch(() => props.lastFetchTime, () => {
|
||||
updateRelativeTime()
|
||||
})
|
||||
|
||||
// 조회간격 선택 → 자동갱신 ON + 특정시간 초기화
|
||||
function selectInterval(min: number) {
|
||||
selectedDatetime.value = ''
|
||||
emit('update:interval', min)
|
||||
if (!props.autoRefresh) {
|
||||
emit('update:autoRefresh', true)
|
||||
}
|
||||
emit('refresh')
|
||||
}
|
||||
|
||||
function toggleAutoRefresh() {
|
||||
const newValue = !props.autoRefresh
|
||||
emit('update:autoRefresh', newValue)
|
||||
|
||||
if (newValue) {
|
||||
// 자동갱신 ON 시 특정시간 초기화 + 즉시 조회
|
||||
selectedDatetime.value = ''
|
||||
emit('refresh')
|
||||
} else {
|
||||
// 자동갱신 OFF 시 현재시간을 기본값으로 설정
|
||||
selectedDatetime.value = getCurrentDatetimeLocal()
|
||||
}
|
||||
}
|
||||
|
||||
function doRefresh() {
|
||||
// 자동갱신 OFF일 때만 동작
|
||||
if (props.autoRefresh || props.fetchState !== 'idle') return
|
||||
|
||||
// 특정시간 입력된 경우 → 해당 시간 조회
|
||||
if (selectedDatetime.value) {
|
||||
emit('fetchAt', selectedDatetime.value.replace('T', ' '))
|
||||
} else {
|
||||
// 특정시간 미입력 → 현재 시점 조회
|
||||
emit('refresh')
|
||||
}
|
||||
}
|
||||
</script>
|
||||
|
||||
<style scoped>
|
||||
.datetime-input.disabled {
|
||||
opacity: 0.5;
|
||||
cursor: not-allowed;
|
||||
}
|
||||
|
||||
.refresh-btn.disabled {
|
||||
opacity: 0.5;
|
||||
cursor: not-allowed;
|
||||
}
|
||||
|
||||
.refresh-btn.loading {
|
||||
cursor: wait;
|
||||
background: var(--btn-primary-bg);
|
||||
}
|
||||
|
||||
.refresh-btn.success {
|
||||
background: var(--success-border);
|
||||
border-color: var(--success-border);
|
||||
}
|
||||
|
||||
.loading-spinner {
|
||||
display: inline-block;
|
||||
width: 12px;
|
||||
height: 12px;
|
||||
border: 2px solid #fff;
|
||||
border-top-color: transparent;
|
||||
border-radius: 50%;
|
||||
animation: spin 0.8s linear infinite;
|
||||
margin-right: 6px;
|
||||
vertical-align: middle;
|
||||
}
|
||||
|
||||
@keyframes spin {
|
||||
to {
|
||||
transform: rotate(360deg);
|
||||
}
|
||||
}
|
||||
|
||||
.hint-text {
|
||||
font-size: 12px;
|
||||
color: var(--text-dim);
|
||||
font-style: italic;
|
||||
margin-left: 8px;
|
||||
}
|
||||
</style>
|
||||
84
frontend/components/NetworkPortlet.vue
Normal file
84
frontend/components/NetworkPortlet.vue
Normal file
@@ -0,0 +1,84 @@
|
||||
<template>
|
||||
<div :class="['network-card', statusClass]" @click="goToList">
|
||||
<div class="card-icon">{{ icon }}</div>
|
||||
<div class="card-title">{{ title }}</div>
|
||||
<div class="card-status" v-if="status && status.last_checked_at">
|
||||
<span class="status-icon">{{ status.is_healthy ? '✅' : '❌' }}</span>
|
||||
<span class="status-text">{{ status.is_healthy ? '정상' : '오류' }}</span>
|
||||
</div>
|
||||
<div class="card-status" v-else>
|
||||
<span class="status-icon">⚪</span>
|
||||
<span class="status-text">대기</span>
|
||||
</div>
|
||||
<div class="card-time" v-if="status && status.last_checked_at">
|
||||
{{ formatTimeAgo(status.last_checked_at) }}
|
||||
</div>
|
||||
</div>
|
||||
</template>
|
||||
|
||||
<script setup lang="ts">
|
||||
const props = defineProps<{
|
||||
type: 'pubnet' | 'privnet'
|
||||
title: string
|
||||
icon: string
|
||||
status: {
|
||||
is_healthy: number
|
||||
last_checked_at: string | null
|
||||
last_target_name: string | null
|
||||
} | null
|
||||
}>()
|
||||
|
||||
const router = useRouter()
|
||||
|
||||
const statusClass = computed(() => {
|
||||
if (!props.status || !props.status.last_checked_at) return 'pending'
|
||||
return props.status.is_healthy ? 'healthy' : 'unhealthy'
|
||||
})
|
||||
|
||||
function goToList() {
|
||||
router.push(`/network/${props.type}`)
|
||||
}
|
||||
|
||||
function formatTimeAgo(datetime: string | null): string {
|
||||
if (!datetime) return '-'
|
||||
const now = new Date()
|
||||
const then = new Date(datetime.replace(' ', 'T') + '+09:00')
|
||||
const diff = Math.floor((now.getTime() - then.getTime()) / 1000)
|
||||
if (diff < 60) return `${diff}초 전`
|
||||
if (diff < 3600) return `${Math.floor(diff / 60)}분 전`
|
||||
if (diff < 86400) return `${Math.floor(diff / 3600)}시간 전`
|
||||
return `${Math.floor(diff / 86400)}일 전`
|
||||
}
|
||||
</script>
|
||||
|
||||
<style scoped>
|
||||
.network-card {
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
align-items: center;
|
||||
padding: 16px 12px;
|
||||
background: var(--bg-secondary);
|
||||
border: 1px solid var(--border-color);
|
||||
border-radius: 12px;
|
||||
cursor: pointer;
|
||||
transition: all 0.2s;
|
||||
text-align: center;
|
||||
}
|
||||
.network-card:hover {
|
||||
background: var(--bg-tertiary, #f8fafc);
|
||||
transform: translateY(-2px);
|
||||
box-shadow: 0 4px 12px rgba(0,0,0,0.08);
|
||||
}
|
||||
.network-card.healthy { border-top: 4px solid #22c55e; }
|
||||
.network-card.unhealthy { border-top: 4px solid #ef4444; }
|
||||
.network-card.pending { border-top: 4px solid #9ca3af; }
|
||||
|
||||
.card-icon { font-size: 28px; margin-bottom: 8px; }
|
||||
.card-title { font-size: 14px; font-weight: 600; color: var(--text-primary); margin-bottom: 10px; }
|
||||
|
||||
.card-status { display: flex; align-items: center; justify-content: center; gap: 6px; margin-bottom: 4px; }
|
||||
.status-icon { font-size: 16px; }
|
||||
.status-text { font-size: 15px; font-weight: 500; color: var(--text-secondary); }
|
||||
|
||||
.card-time { font-size: 12px; color: var(--text-muted); }
|
||||
</style>
|
||||
319
frontend/components/ServerPortlet.vue
Normal file
319
frontend/components/ServerPortlet.vue
Normal file
@@ -0,0 +1,319 @@
|
||||
<template>
|
||||
<div class="server-portlet">
|
||||
<div class="portlet-header">
|
||||
<h2 class="portlet-title">🖥️ 서버 현황</h2>
|
||||
<div class="summary-badges">
|
||||
<span class="badge">
|
||||
서버 {{ summary.servers.total }}대
|
||||
<span class="level-counts">
|
||||
<span class="lv">🟢{{ summary.servers.normal }}</span>
|
||||
<span class="lv">🟡{{ summary.servers.warning }}</span>
|
||||
<span class="lv">🟠{{ summary.servers.critical }}</span>
|
||||
<span class="lv">🔴{{ summary.servers.danger }}</span>
|
||||
<span class="lv">⚫{{ summary.servers.offline }}</span>
|
||||
</span>
|
||||
</span>
|
||||
<span class="divider">|</span>
|
||||
<span class="badge">
|
||||
컨테이너 {{ summary.containers.total }}개
|
||||
<span class="level-counts">
|
||||
<span class="lv">🟢{{ summary.containers.normal }}</span>
|
||||
<span class="lv">🟡{{ summary.containers.warning }}</span>
|
||||
<span class="lv">🟠{{ summary.containers.critical }}</span>
|
||||
<span class="lv">🔴{{ (summary.containers.danger || 0) + summary.containers.stopped }}</span>
|
||||
</span>
|
||||
</span>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="server-grid">
|
||||
<!-- 서버 유닛들 -->
|
||||
<div
|
||||
v-for="server in servers"
|
||||
:key="server.target_id"
|
||||
:class="['server-unit', server.level]"
|
||||
>
|
||||
<!-- 서버 정보 (왼쪽) -->
|
||||
<div class="server-info" @dblclick="goToServerStatus(server.target_id)">
|
||||
<div class="server-name">
|
||||
<span class="level-icon">{{ levelIcon(server.level) }}</span>
|
||||
<span class="name">{{ server.server_name }}</span>
|
||||
<span class="container-count" v-if="server.level !== 'offline'">📦{{ server.container_summary.total }}</span>
|
||||
</div>
|
||||
|
||||
<template v-if="server.level !== 'offline'">
|
||||
<div class="metric-row">
|
||||
<span class="metric-label">CPU</span>
|
||||
<div class="progress-bar">
|
||||
<div :class="['progress-fill', server.cpu_level]" :style="{ width: (server.cpu_percent || 0) + '%' }"></div>
|
||||
</div>
|
||||
<span :class="['metric-value', server.cpu_level]">{{ server.cpu_percent?.toFixed(0) || '-' }}</span>
|
||||
</div>
|
||||
<div class="metric-row">
|
||||
<span class="metric-label">MEM</span>
|
||||
<div class="progress-bar">
|
||||
<div :class="['progress-fill', server.memory_level]" :style="{ width: (server.memory_percent || 0) + '%' }"></div>
|
||||
</div>
|
||||
<span :class="['metric-value', server.memory_level]">{{ server.memory_percent?.toFixed(0) || '-' }}</span>
|
||||
</div>
|
||||
<div class="metric-row">
|
||||
<span class="metric-label">DISK</span>
|
||||
<div class="progress-bar">
|
||||
<div :class="['progress-fill', server.disk_level]" :style="{ width: (server.disk_percent || 0) + '%' }"></div>
|
||||
</div>
|
||||
<span :class="['metric-value', server.disk_level]">{{ server.disk_percent?.toFixed(0) || '-' }}</span>
|
||||
</div>
|
||||
</template>
|
||||
|
||||
<template v-else>
|
||||
<div class="offline-info">
|
||||
<div class="offline-text">오프라인</div>
|
||||
<div class="offline-time">{{ formatTimeAgo(server.last_collected) }}</div>
|
||||
</div>
|
||||
</template>
|
||||
</div>
|
||||
|
||||
<!-- 컨테이너 영역 (오른쪽) -->
|
||||
<div class="container-area" v-if="server.level !== 'offline'">
|
||||
<div
|
||||
v-for="container in sortContainers(server.containers)"
|
||||
:key="server.target_id + '-' + container.name"
|
||||
:class="['container-card', container.level]"
|
||||
@click="goToServerStatus(server.target_id, container.name)"
|
||||
>
|
||||
<div class="card-header">
|
||||
<div class="card-name">
|
||||
<span class="card-level">{{ levelIcon(container.level) }}</span>
|
||||
<span class="name">{{ container.name }}</span>
|
||||
</div>
|
||||
<span class="card-uptime" v-if="container.status === 'running'">{{ container.uptime || '-' }}</span>
|
||||
</div>
|
||||
<template v-if="container.status === 'running'">
|
||||
<div class="card-metrics">
|
||||
<div class="card-metric">
|
||||
<span class="label">CPU</span>
|
||||
<div class="mini-bar">
|
||||
<div :class="['mini-fill', getContainerCpuLevel(container)]" :style="{ width: (container.cpu_percent || 0) + '%' }"></div>
|
||||
</div>
|
||||
<span class="value">{{ container.cpu_percent?.toFixed(0) || '-' }}%</span>
|
||||
</div>
|
||||
<div class="card-metric">
|
||||
<span class="label">MEM</span>
|
||||
<div class="mini-bar">
|
||||
<div :class="['mini-fill', getContainerMemLevel(container)]" :style="{ width: getMemPercent(container) + '%' }"></div>
|
||||
</div>
|
||||
<span class="value">{{ formatMemoryShort(container.memory_usage) }}</span>
|
||||
</div>
|
||||
<div class="card-metric">
|
||||
<span class="label">↓RX</span>
|
||||
<span class="value net">{{ formatNetworkShort(container.network_rx) }}</span>
|
||||
</div>
|
||||
<div class="card-metric">
|
||||
<span class="label">↑TX</span>
|
||||
<span class="value net">{{ formatNetworkShort(container.network_tx) }}</span>
|
||||
</div>
|
||||
</div>
|
||||
</template>
|
||||
<template v-else>
|
||||
<div class="card-stopped">{{ container.status }}</div>
|
||||
</template>
|
||||
</div>
|
||||
|
||||
<!-- 컨테이너 없음 -->
|
||||
<div class="no-container" v-if="server.containers.length === 0">
|
||||
<span>컨테이너 없음</span>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</template>
|
||||
|
||||
<script setup lang="ts">
|
||||
interface ContainerStatus {
|
||||
name: string
|
||||
status: string
|
||||
level: string
|
||||
cpu_percent: number | null
|
||||
memory_usage: number | null
|
||||
memory_limit: number | null
|
||||
uptime: string | null
|
||||
network_rx: number | null
|
||||
network_tx: number | null
|
||||
}
|
||||
|
||||
interface ServerStatus {
|
||||
target_id: number
|
||||
server_name: string
|
||||
level: string
|
||||
cpu_percent: number | null
|
||||
cpu_level: string
|
||||
memory_percent: number | null
|
||||
memory_level: string
|
||||
disk_percent: number | null
|
||||
disk_level: string
|
||||
last_collected: string | null
|
||||
containers: ContainerStatus[]
|
||||
container_summary: { total: number; normal: number; warning: number; critical: number; stopped: number }
|
||||
}
|
||||
|
||||
interface Summary {
|
||||
servers: { total: number; normal: number; warning: number; critical: number; danger: number; offline: number }
|
||||
containers: { total: number; normal: number; warning: number; critical: number; danger: number; stopped: number }
|
||||
}
|
||||
|
||||
const props = defineProps<{
|
||||
servers: ServerStatus[]
|
||||
summary: Summary
|
||||
}>()
|
||||
|
||||
const emit = defineEmits<{
|
||||
(e: 'navigate', path: string): void
|
||||
}>()
|
||||
|
||||
const levelPriority: Record<string, number> = { stopped: 3, critical: 2, danger: 2, warning: 1, normal: 0 }
|
||||
|
||||
function sortContainers(containers: ContainerStatus[]) {
|
||||
return [...containers].sort((a, b) => a.name.localeCompare(b.name))
|
||||
}
|
||||
|
||||
function goToServerStatus(targetId: number, containerName?: string) {
|
||||
let path = `/server/history?target=${targetId}`
|
||||
if (containerName) path += `&container=${containerName}`
|
||||
emit('navigate', path)
|
||||
}
|
||||
|
||||
function levelIcon(level: string): string {
|
||||
const icons: Record<string, string> = { normal: '🟢', warning: '🟡', critical: '🟠', danger: '🔴', offline: '⚫', stopped: '🔴' }
|
||||
return icons[level] || '⚪'
|
||||
}
|
||||
|
||||
function getContainerCpuLevel(c: ContainerStatus): string {
|
||||
if (c.cpu_percent === null) return 'normal'
|
||||
if (c.cpu_percent >= 95) return 'danger'
|
||||
if (c.cpu_percent >= 90) return 'critical'
|
||||
if (c.cpu_percent >= 80) return 'warning'
|
||||
return 'normal'
|
||||
}
|
||||
|
||||
function getContainerMemLevel(c: ContainerStatus): string {
|
||||
const pct = getMemPercent(c)
|
||||
if (pct >= 95) return 'danger'
|
||||
if (pct >= 90) return 'critical'
|
||||
if (pct >= 80) return 'warning'
|
||||
return 'normal'
|
||||
}
|
||||
|
||||
function getMemPercent(c: ContainerStatus): number {
|
||||
if (!c.memory_limit || !c.memory_usage) return 0
|
||||
return (c.memory_usage / c.memory_limit) * 100
|
||||
}
|
||||
|
||||
function formatMemoryShort(bytes: number | null): string {
|
||||
if (bytes === null || bytes === undefined) return '-'
|
||||
if (bytes < 1024 * 1024) return `${(bytes / 1024).toFixed(0)}K`
|
||||
if (bytes < 1024 * 1024 * 1024) return `${(bytes / 1024 / 1024).toFixed(0)}M`
|
||||
return `${(bytes / 1024 / 1024 / 1024).toFixed(1)}G`
|
||||
}
|
||||
|
||||
function formatNetworkShort(bytes: number | null): string {
|
||||
if (bytes === null || bytes === undefined) return '-'
|
||||
if (bytes < 1024) return `${bytes}B`
|
||||
if (bytes < 1024 * 1024) return `${(bytes / 1024).toFixed(0)}K`
|
||||
if (bytes < 1024 * 1024 * 1024) return `${(bytes / 1024 / 1024).toFixed(1)}M`
|
||||
return `${(bytes / 1024 / 1024 / 1024).toFixed(2)}G`
|
||||
}
|
||||
|
||||
function formatTimeAgo(datetime: string | null): string {
|
||||
if (!datetime) return '-'
|
||||
const now = new Date()
|
||||
const then = new Date(datetime.replace(' ', 'T') + '+09:00')
|
||||
const diff = Math.floor((now.getTime() - then.getTime()) / 1000)
|
||||
if (diff < 60) return `${diff}초 전`
|
||||
if (diff < 3600) return `${Math.floor(diff / 60)}분 전`
|
||||
if (diff < 86400) return `${Math.floor(diff / 3600)}시간 전`
|
||||
return `${Math.floor(diff / 86400)}일 전`
|
||||
}
|
||||
</script>
|
||||
|
||||
<style scoped>
|
||||
.server-portlet { background: var(--bg-secondary); border: 1px solid var(--border-color); border-radius: 12px; overflow: hidden; }
|
||||
|
||||
.portlet-header { display: flex; justify-content: space-between; align-items: center; padding: 14px 18px; border-bottom: 1px solid var(--border-color); background: var(--bg-tertiary, #f8fafc); }
|
||||
.portlet-title { margin: 0; font-size: 18px; font-weight: 600; color: var(--text-primary); }
|
||||
.summary-badges { display: flex; align-items: center; gap: 12px; font-size: 14px; color: var(--text-secondary); }
|
||||
.divider { color: var(--border-color); }
|
||||
.level-counts { margin-left: 8px; }
|
||||
.level-counts .lv { margin-right: 6px; }
|
||||
|
||||
/* 서버 그리드 - flex-wrap */
|
||||
.server-grid { display: flex; flex-wrap: wrap; gap: 12px; padding: 16px; align-items: flex-start; }
|
||||
|
||||
/* 서버 유닛 */
|
||||
.server-unit { display: flex; background: var(--bg-primary); border: 1px solid var(--border-color); border-radius: 10px; overflow: hidden; }
|
||||
.server-unit.warning { border-left: 3px solid #eab308; }
|
||||
.server-unit.critical { border-left: 3px solid #f97316; }
|
||||
.server-unit.danger { border-left: 3px solid #ef4444; }
|
||||
.server-unit.offline { border-left: 3px solid #6b7280; opacity: 0.7; }
|
||||
|
||||
/* 서버 정보 (왼쪽) */
|
||||
.server-info { width: 150px; min-width: 150px; padding: 12px; border-right: 1px solid var(--border-color); cursor: pointer; }
|
||||
.server-info:hover { background: var(--bg-tertiary, #f8fafc); }
|
||||
|
||||
.server-name { display: flex; align-items: center; gap: 6px; margin-bottom: 10px; }
|
||||
.server-name .level-icon { font-size: 12px; flex-shrink: 0; }
|
||||
.server-name .name { font-size: 15px; font-weight: 600; color: var(--text-primary); overflow: hidden; text-overflow: ellipsis; white-space: nowrap; flex: 1; }
|
||||
.server-name .container-count { font-size: 12px; color: var(--text-muted); flex-shrink: 0; }
|
||||
|
||||
/* 서버 메트릭 */
|
||||
.metric-row { display: flex; align-items: center; gap: 6px; margin-bottom: 6px; }
|
||||
.metric-label { font-size: 11px; font-weight: 500; color: var(--text-muted); width: 28px; }
|
||||
.progress-bar { flex: 1; height: 6px; background: var(--bg-tertiary, #e5e7eb); border-radius: 3px; overflow: hidden; }
|
||||
.progress-fill { height: 100%; border-radius: 3px; transition: width 0.3s; }
|
||||
.progress-fill.normal { background: #22c55e; }
|
||||
.progress-fill.warning { background: #eab308; }
|
||||
.progress-fill.critical { background: #f97316; }
|
||||
.progress-fill.danger { background: #ef4444; }
|
||||
.metric-value { font-size: 12px; font-weight: 600; width: 24px; text-align: right; }
|
||||
.metric-value.normal { color: #16a34a; }
|
||||
.metric-value.warning { color: #ca8a04; }
|
||||
.metric-value.critical { color: #ea580c; }
|
||||
.metric-value.danger { color: #dc2626; }
|
||||
|
||||
.offline-info { text-align: center; padding: 16px 0; color: var(--text-muted); }
|
||||
.offline-text { font-size: 14px; margin-bottom: 4px; }
|
||||
.offline-time { font-size: 12px; opacity: 0.7; }
|
||||
|
||||
/* 컨테이너 영역 */
|
||||
.container-area { display: flex; flex-wrap: wrap; gap: 8px; padding: 10px; align-content: flex-start; min-width: 100px; }
|
||||
|
||||
/* 컨테이너 카드 */
|
||||
.container-card { width: 200px; padding: 10px; border-radius: 8px; border: 1px solid var(--border-color); background: var(--bg-secondary); cursor: pointer; transition: all 0.15s; overflow: hidden; }
|
||||
.container-card:hover { transform: translateY(-1px); box-shadow: 0 2px 8px rgba(0,0,0,0.08); }
|
||||
.container-card.normal { background: #f0fdf4; border-color: #86efac; }
|
||||
.container-card.warning { background: #fefce8; border-color: #fde047; }
|
||||
.container-card.critical { background: #fff7ed; border-color: #fdba74; }
|
||||
.container-card.danger { background: #fef2f2; border-color: #fca5a5; }
|
||||
.container-card.stopped { background: #fef2f2; border-color: #fca5a5; }
|
||||
|
||||
.card-header { display: flex; align-items: center; justify-content: space-between; margin-bottom: 8px; overflow: hidden; }
|
||||
.card-name { display: flex; align-items: center; gap: 4px; flex: 1; min-width: 0; overflow: hidden; }
|
||||
.card-name .card-level { font-size: 10px; flex-shrink: 0; }
|
||||
.card-name .name { font-size: 13px; font-weight: 600; color: var(--text-primary); overflow: hidden; text-overflow: ellipsis; white-space: nowrap; }
|
||||
.card-uptime { font-size: 11px; color: var(--text-muted); flex-shrink: 0; margin-left: 8px; white-space: nowrap; }
|
||||
|
||||
.card-metrics { display: flex; flex-wrap: wrap; gap: 4px 8px; }
|
||||
.card-metric { display: flex; align-items: center; gap: 4px; width: calc(50% - 4px); overflow: hidden; }
|
||||
.card-metric .label { font-size: 10px; color: var(--text-muted); width: 22px; flex-shrink: 0; }
|
||||
.mini-bar { flex: 1; height: 5px; background: rgba(0,0,0,0.1); border-radius: 2px; overflow: hidden; min-width: 20px; }
|
||||
.mini-fill { height: 100%; border-radius: 2px; }
|
||||
.mini-fill.normal { background: #22c55e; }
|
||||
.mini-fill.warning { background: #eab308; }
|
||||
.mini-fill.critical { background: #f97316; }
|
||||
.mini-fill.danger { background: #ef4444; }
|
||||
.card-metric .value { font-size: 11px; font-weight: 500; color: var(--text-secondary); width: 36px; text-align: right; flex-shrink: 0; overflow: hidden; text-overflow: ellipsis; }
|
||||
|
||||
.card-stopped { font-size: 12px; color: var(--text-muted); font-style: italic; text-align: center; padding: 6px 0; }
|
||||
|
||||
.no-container { font-size: 13px; color: var(--text-muted); padding: 12px; display: flex; align-items: center; justify-content: center; }
|
||||
</style>
|
||||
74
frontend/components/SidebarNav.vue
Normal file
74
frontend/components/SidebarNav.vue
Normal file
@@ -0,0 +1,74 @@
|
||||
<template>
|
||||
<aside class="sidebar">
|
||||
<div class="sidebar-header">
|
||||
<div class="sidebar-logo">
|
||||
<span class="logo-icon">📡</span>
|
||||
<span>OSOLIT Monitor</span>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<nav class="sidebar-nav">
|
||||
<NuxtLink to="/" class="nav-item" :class="{ active: route.path === '/' }">
|
||||
<span class="icon">📊</span>
|
||||
<span>대시보드</span>
|
||||
</NuxtLink>
|
||||
|
||||
<div class="nav-group-title">네트워크</div>
|
||||
|
||||
<NuxtLink to="/network/pubnet" class="nav-item nav-sub-item" :class="{ active: route.path === '/network/pubnet' }">
|
||||
<span class="icon">🌐</span>
|
||||
<span>Public Network</span>
|
||||
</NuxtLink>
|
||||
|
||||
<NuxtLink to="/network/privnet" class="nav-item nav-sub-item" :class="{ active: route.path === '/network/privnet' }">
|
||||
<span class="icon">🔒</span>
|
||||
<span>Private Network</span>
|
||||
</NuxtLink>
|
||||
|
||||
<div class="nav-group-title">서버</div>
|
||||
|
||||
<NuxtLink to="/server/list" class="nav-item nav-sub-item" :class="{ active: route.path === '/server/list' }">
|
||||
<span class="icon">🖥️</span>
|
||||
<span>Server Targets</span>
|
||||
</NuxtLink>
|
||||
|
||||
<NuxtLink to="/server/history" class="nav-item nav-sub-item" :class="{ active: route.path === '/server/history' }">
|
||||
<span class="icon">📈</span>
|
||||
<span>Server Status</span>
|
||||
</NuxtLink>
|
||||
|
||||
<div class="nav-group-title">이상감지</div>
|
||||
|
||||
<NuxtLink to="/anomaly/short-term" class="nav-item nav-sub-item" :class="{ active: route.path === '/anomaly/short-term' }">
|
||||
<span class="icon">⚡</span>
|
||||
<span>단기 변화율</span>
|
||||
</NuxtLink>
|
||||
|
||||
<NuxtLink to="/anomaly/zscore" class="nav-item nav-sub-item" :class="{ active: route.path === '/anomaly/zscore' }">
|
||||
<span class="icon">📊</span>
|
||||
<span>Z-Score 분석</span>
|
||||
</NuxtLink>
|
||||
|
||||
<NuxtLink to="/anomaly/baseline" class="nav-item nav-sub-item" :class="{ active: route.path === '/anomaly/baseline' }">
|
||||
<span class="icon">🕐</span>
|
||||
<span>시간대별 베이스라인</span>
|
||||
</NuxtLink>
|
||||
|
||||
<NuxtLink to="/anomaly/trend" class="nav-item nav-sub-item" :class="{ active: route.path === '/anomaly/trend' }">
|
||||
<span class="icon">📉</span>
|
||||
<span>추세 분석</span>
|
||||
</NuxtLink>
|
||||
|
||||
<div class="nav-group-title">설정</div>
|
||||
|
||||
<NuxtLink to="/settings/thresholds" class="nav-item nav-sub-item" :class="{ active: route.path === '/settings/thresholds' }">
|
||||
<span class="icon">⚙️</span>
|
||||
<span>임계값 설정</span>
|
||||
</NuxtLink>
|
||||
</nav>
|
||||
</aside>
|
||||
</template>
|
||||
|
||||
<script setup lang="ts">
|
||||
const route = useRoute()
|
||||
</script>
|
||||
112
frontend/components/ThemeToggle.vue
Normal file
112
frontend/components/ThemeToggle.vue
Normal file
@@ -0,0 +1,112 @@
|
||||
<template>
|
||||
<div
|
||||
class="theme-switch"
|
||||
:class="{ dark: theme === 'dark' }"
|
||||
@click="toggleTheme"
|
||||
:title="theme === 'dark' ? '라이트 모드로 전환' : '다크 모드로 전환'"
|
||||
>
|
||||
<div class="switch-track">
|
||||
<span class="switch-icon sun">☀️</span>
|
||||
<span class="switch-icon moon">🌙</span>
|
||||
</div>
|
||||
<div class="switch-thumb"></div>
|
||||
</div>
|
||||
</template>
|
||||
|
||||
<script setup lang="ts">
|
||||
const { theme, toggleTheme, initTheme } = useTheme()
|
||||
|
||||
onMounted(() => {
|
||||
initTheme()
|
||||
})
|
||||
</script>
|
||||
|
||||
<style scoped>
|
||||
.theme-switch {
|
||||
position: relative;
|
||||
width: 56px;
|
||||
height: 28px;
|
||||
background: #e0e0e0;
|
||||
border: 2px solid #ccc;
|
||||
border-radius: 14px;
|
||||
cursor: pointer;
|
||||
transition: background 0.3s ease, border-color 0.3s ease;
|
||||
box-shadow: inset 0 1px 3px rgba(0, 0, 0, 0.1);
|
||||
}
|
||||
|
||||
.theme-switch.dark {
|
||||
background: #3a3a3a;
|
||||
border-color: #666;
|
||||
}
|
||||
|
||||
.switch-track {
|
||||
position: absolute;
|
||||
top: 0;
|
||||
left: 0;
|
||||
right: 0;
|
||||
bottom: 0;
|
||||
display: flex;
|
||||
justify-content: space-between;
|
||||
align-items: center;
|
||||
padding: 0 6px;
|
||||
}
|
||||
|
||||
.switch-icon {
|
||||
font-size: 14px;
|
||||
transition: opacity 0.3s ease;
|
||||
}
|
||||
|
||||
.switch-icon.sun {
|
||||
opacity: 0.4;
|
||||
}
|
||||
|
||||
.switch-icon.moon {
|
||||
opacity: 0.4;
|
||||
}
|
||||
|
||||
.theme-switch.dark .switch-icon.sun {
|
||||
opacity: 0.4;
|
||||
}
|
||||
|
||||
.theme-switch.dark .switch-icon.moon {
|
||||
opacity: 0.4;
|
||||
}
|
||||
|
||||
.switch-thumb {
|
||||
position: absolute;
|
||||
top: 2px;
|
||||
left: 2px;
|
||||
width: 20px;
|
||||
height: 20px;
|
||||
background: #1a1a1a;
|
||||
border-radius: 50%;
|
||||
box-shadow: 0 2px 4px rgba(0, 0, 0, 0.2);
|
||||
transition: transform 0.3s cubic-bezier(0.4, 0, 0.2, 1), background 0.3s ease;
|
||||
}
|
||||
|
||||
.theme-switch.dark .switch-thumb {
|
||||
transform: translateX(28px);
|
||||
background: #fff;
|
||||
}
|
||||
|
||||
/* 호버 효과 */
|
||||
.theme-switch:hover {
|
||||
border-color: #999;
|
||||
}
|
||||
|
||||
.theme-switch.dark:hover {
|
||||
border-color: #888;
|
||||
}
|
||||
|
||||
.theme-switch:hover .switch-thumb {
|
||||
box-shadow: 0 2px 6px rgba(0, 0, 0, 0.3);
|
||||
}
|
||||
|
||||
.theme-switch:active .switch-thumb {
|
||||
width: 24px;
|
||||
}
|
||||
|
||||
.theme-switch.dark:active .switch-thumb {
|
||||
transform: translateX(24px);
|
||||
}
|
||||
</style>
|
||||
Reference in New Issue
Block a user