743 lines
24 KiB
Vue
743 lines
24 KiB
Vue
<template>
|
|
<div>
|
|
<LayoutsBreadcrumb />
|
|
|
|
<!-- Page Info Card -->
|
|
<rs-card class="mb-6">
|
|
<template #header>
|
|
<div class="flex items-center">
|
|
<Icon class="mr-2 text-primary" name="ic:outline-monitor"></Icon>
|
|
<h1 class="text-xl font-bold text-primary">Real-Time Monitoring</h1>
|
|
</div>
|
|
</template>
|
|
<template #body>
|
|
<p class="text-gray-600">
|
|
Live monitoring of notification system performance with real-time alerts and system health indicators.
|
|
Track ongoing activities, monitor system load, and receive immediate notifications about issues.
|
|
</p>
|
|
</template>
|
|
</rs-card>
|
|
|
|
<!-- System Status Overview -->
|
|
<div class="grid grid-cols-1 lg:grid-cols-2 xl:grid-cols-4 gap-6 mb-6">
|
|
<rs-card
|
|
v-for="(status, index) in systemStatus"
|
|
:key="index"
|
|
class="transition-all duration-300 hover:shadow-lg"
|
|
>
|
|
<div class="pt-5 pb-3 px-5 flex items-center gap-4">
|
|
<div
|
|
class="p-5 flex justify-center items-center rounded-2xl transition-all duration-300"
|
|
:class="status.status === 'healthy' ? 'bg-green-100' :
|
|
status.status === 'warning' ? 'bg-yellow-100' : 'bg-red-100'"
|
|
>
|
|
<Icon
|
|
class="text-3xl"
|
|
:class="status.status === 'healthy' ? 'text-green-600' :
|
|
status.status === 'warning' ? 'text-yellow-600' : 'text-red-600'"
|
|
:name="status.icon"
|
|
/>
|
|
</div>
|
|
<div class="flex-1 truncate">
|
|
<span class="block font-bold text-2xl leading-tight text-primary">
|
|
{{ status.value }}
|
|
</span>
|
|
<span class="text-sm font-medium text-gray-600">
|
|
{{ status.title }}
|
|
</span>
|
|
<div class="flex items-center mt-1">
|
|
<div
|
|
class="w-2 h-2 rounded-full mr-2"
|
|
:class="status.status === 'healthy' ? 'bg-green-500' :
|
|
status.status === 'warning' ? 'bg-yellow-500' : 'bg-red-500'"
|
|
></div>
|
|
<span
|
|
class="text-xs font-medium capitalize"
|
|
:class="status.status === 'healthy' ? 'text-green-600' :
|
|
status.status === 'warning' ? 'text-yellow-600' : 'text-red-600'"
|
|
>
|
|
{{ status.status }}
|
|
</span>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</rs-card>
|
|
</div>
|
|
|
|
<!-- Real-Time Controls -->
|
|
<rs-card class="mb-6">
|
|
<template #body>
|
|
<div class="flex items-center justify-between">
|
|
<div class="flex items-center gap-4">
|
|
<h3 class="text-lg font-semibold text-primary">Monitoring Controls</h3>
|
|
<div class="flex items-center gap-2">
|
|
<div
|
|
class="w-3 h-3 rounded-full animate-pulse"
|
|
:class="isMonitoring ? 'bg-green-500' : 'bg-gray-400'"
|
|
></div>
|
|
<span class="text-sm font-medium">
|
|
{{ isMonitoring ? 'Live' : 'Paused' }}
|
|
</span>
|
|
</div>
|
|
</div>
|
|
<div class="flex items-center gap-2">
|
|
<FormKit
|
|
type="select"
|
|
v-model="refreshInterval"
|
|
:options="refreshOptions"
|
|
outer-class="mb-0"
|
|
@input="updateRefreshInterval"
|
|
/>
|
|
<rs-button
|
|
:variant="isMonitoring ? 'danger-outline' : 'primary'"
|
|
size="sm"
|
|
@click="toggleMonitoring"
|
|
>
|
|
<Icon
|
|
:name="isMonitoring ? 'ic:outline-pause' : 'ic:outline-play-arrow'"
|
|
class="mr-1"
|
|
/>
|
|
{{ isMonitoring ? 'Pause' : 'Start' }}
|
|
</rs-button>
|
|
<rs-button variant="primary-outline" size="sm" @click="refreshData">
|
|
<Icon name="ic:outline-refresh" class="mr-1"/> Refresh
|
|
</rs-button>
|
|
</div>
|
|
</div>
|
|
</template>
|
|
</rs-card>
|
|
|
|
<!-- System Performance Dashboard -->
|
|
<rs-card class="mb-6">
|
|
<template #header>
|
|
<div class="flex items-center justify-between">
|
|
<div class="flex items-center">
|
|
<Icon name="ic:outline-speed" class="mr-2 text-primary"/>
|
|
<h3 class="text-lg font-semibold text-primary">System Performance</h3>
|
|
</div>
|
|
<rs-button variant="outline" size="sm" @click="exportPerformanceData">
|
|
<Icon name="ic:outline-file-download" class="mr-1"/> Export Data
|
|
</rs-button>
|
|
</div>
|
|
</template>
|
|
<template #body>
|
|
<div class="grid grid-cols-1 lg:grid-cols-3 gap-6 mb-6">
|
|
<!-- CPU Usage -->
|
|
<div class="text-center">
|
|
<div class="relative mx-auto w-32 h-32 mb-4">
|
|
<div class="w-full h-full bg-gray-200 rounded-full flex items-center justify-center">
|
|
<div class="text-center">
|
|
<div class="text-2xl font-bold text-primary">{{ performanceMetrics.cpu }}%</div>
|
|
<div class="text-xs text-gray-600">CPU</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
|
|
<!-- Memory Usage -->
|
|
<div class="text-center">
|
|
<div class="relative mx-auto w-32 h-32 mb-4">
|
|
<div class="w-full h-full bg-gray-200 rounded-full flex items-center justify-center">
|
|
<div class="text-center">
|
|
<div class="text-2xl font-bold text-primary">{{ performanceMetrics.memory }}%</div>
|
|
<div class="text-xs text-gray-600">Memory</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
|
|
<!-- Queue Load -->
|
|
<div class="text-center">
|
|
<div class="relative mx-auto w-32 h-32 mb-4">
|
|
<div class="w-full h-full bg-gray-200 rounded-full flex items-center justify-center">
|
|
<div class="text-center">
|
|
<div class="text-2xl font-bold text-primary">{{ performanceMetrics.queueLoad }}%</div>
|
|
<div class="text-xs text-gray-600">Queue Load</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
|
|
<!-- Performance Chart Placeholder -->
|
|
<div class="h-64 bg-gray-100 dark:bg-gray-800 flex items-center justify-center rounded">
|
|
<div class="text-center">
|
|
<Icon name="ic:outline-show-chart" class="text-4xl text-gray-400 mb-2"/>
|
|
<p class="text-gray-500">Real-time Performance Chart</p>
|
|
<p class="text-sm text-gray-400 mt-1">Implementation pending for live performance metrics</p>
|
|
</div>
|
|
</div>
|
|
</template>
|
|
</rs-card>
|
|
|
|
<!-- Live Activity & Alerts Grid -->
|
|
<div class="grid grid-cols-1 lg:grid-cols-2 gap-6 mb-6">
|
|
<!-- Live Activity Feed -->
|
|
<rs-card>
|
|
<template #header>
|
|
<div class="flex items-center justify-between">
|
|
<div class="flex items-center">
|
|
<Icon name="ic:outline-notifications-active" class="mr-2 text-primary"/>
|
|
<h3 class="text-lg font-semibold text-primary">Live Activity Feed</h3>
|
|
</div>
|
|
<div class="flex items-center gap-2">
|
|
<rs-button variant="outline" size="sm" @click="clearActivityFeed">
|
|
<Icon name="ic:outline-clear" class="mr-1"/> Clear
|
|
</rs-button>
|
|
</div>
|
|
</div>
|
|
</template>
|
|
<template #body>
|
|
<div class="h-96 overflow-y-auto space-y-3">
|
|
<div
|
|
v-for="(activity, index) in liveActivityFeed"
|
|
:key="index"
|
|
class="flex items-start p-3 bg-gray-50 dark:bg-gray-800 rounded-lg transition-all duration-300 hover:bg-gray-100 dark:hover:bg-gray-700"
|
|
>
|
|
<div
|
|
class="w-3 h-3 rounded-full mr-3 mt-2 flex-shrink-0"
|
|
:class="{
|
|
'bg-green-500': activity.type === 'success',
|
|
'bg-blue-500': activity.type === 'info',
|
|
'bg-yellow-500': activity.type === 'warning',
|
|
'bg-red-500': activity.type === 'error',
|
|
}"
|
|
></div>
|
|
<div class="flex-1 min-w-0">
|
|
<p class="font-medium text-sm">{{ activity.action }}</p>
|
|
<p class="text-xs text-gray-600 mt-1">{{ activity.details }}</p>
|
|
<div class="flex items-center mt-2 text-xs text-gray-500">
|
|
<Icon name="ic:outline-access-time" class="mr-1"/>
|
|
<span>{{ activity.timestamp }}</span>
|
|
<span class="mx-2">•</span>
|
|
<span>{{ activity.source }}</span>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
|
|
<div v-if="liveActivityFeed.length === 0" class="text-center py-8 text-gray-500">
|
|
<Icon name="ic:outline-wifi-tethering" class="text-3xl mb-2 mx-auto"/>
|
|
<p>Waiting for live activity...</p>
|
|
</div>
|
|
</div>
|
|
</template>
|
|
</rs-card>
|
|
|
|
<!-- Error Alerts -->
|
|
<rs-card>
|
|
<template #header>
|
|
<div class="flex items-center justify-between">
|
|
<div class="flex items-center">
|
|
<Icon name="ic:outline-warning" class="mr-2 text-primary"/>
|
|
<h3 class="text-lg font-semibold text-primary">Error Alerts</h3>
|
|
</div>
|
|
<div class="flex items-center gap-2">
|
|
<rs-badge
|
|
:variant="errorAlerts.filter(a => a.severity === 'critical').length > 0 ? 'danger' : 'secondary'"
|
|
size="sm"
|
|
>
|
|
{{ errorAlerts.length }} Active
|
|
</rs-badge>
|
|
<rs-button variant="outline" size="sm" @click="acknowledgeAllAlerts">
|
|
<Icon name="ic:outline-check" class="mr-1"/> Acknowledge All
|
|
</rs-button>
|
|
</div>
|
|
</div>
|
|
</template>
|
|
<template #body>
|
|
<div class="h-96 overflow-y-auto space-y-3">
|
|
<div
|
|
v-for="(alert, index) in errorAlerts"
|
|
:key="index"
|
|
class="p-3 rounded-lg border-l-4 transition-all duration-300 hover:shadow-sm"
|
|
:class="{
|
|
'bg-red-50 border-red-400': alert.severity === 'critical',
|
|
'bg-yellow-50 border-yellow-400': alert.severity === 'warning',
|
|
'bg-blue-50 border-blue-400': alert.severity === 'info',
|
|
}"
|
|
>
|
|
<div class="flex items-start justify-between">
|
|
<div class="flex-1">
|
|
<div class="flex items-center">
|
|
<Icon
|
|
:name="alert.severity === 'critical' ? 'ic:outline-error' :
|
|
alert.severity === 'warning' ? 'ic:outline-warning' : 'ic:outline-info'"
|
|
:class="{
|
|
'text-red-600': alert.severity === 'critical',
|
|
'text-yellow-600': alert.severity === 'warning',
|
|
'text-blue-600': alert.severity === 'info',
|
|
}"
|
|
class="mr-2"
|
|
/>
|
|
<span class="font-medium text-sm">{{ alert.title }}</span>
|
|
</div>
|
|
<p class="text-xs text-gray-600 mt-1">{{ alert.description }}</p>
|
|
<div class="flex items-center mt-2 text-xs text-gray-500">
|
|
<span>{{ alert.timestamp }}</span>
|
|
<span class="mx-2">•</span>
|
|
<span>{{ alert.component }}</span>
|
|
</div>
|
|
</div>
|
|
<rs-button
|
|
variant="outline"
|
|
size="sm"
|
|
@click="acknowledgeAlert(index)"
|
|
class="ml-2"
|
|
>
|
|
<Icon name="ic:outline-check" />
|
|
</rs-button>
|
|
</div>
|
|
</div>
|
|
|
|
<div v-if="errorAlerts.length === 0" class="text-center py-8 text-gray-500">
|
|
<Icon name="ic:outline-check-circle" class="text-3xl mb-2 mx-auto text-green-500"/>
|
|
<p>No active alerts</p>
|
|
<p class="text-sm text-gray-400 mt-1">All systems operating normally</p>
|
|
</div>
|
|
</div>
|
|
</template>
|
|
</rs-card>
|
|
</div>
|
|
|
|
<!-- Queue Status -->
|
|
<rs-card class="mb-6">
|
|
<template #header>
|
|
<div class="flex items-center justify-between">
|
|
<div class="flex items-center">
|
|
<Icon name="ic:outline-queue" class="mr-2 text-primary"/>
|
|
<h3 class="text-lg font-semibold text-primary">Queue Status</h3>
|
|
</div>
|
|
<rs-button variant="outline" size="sm" @click="navigateTo('/notification/queue-scheduler/monitor')">
|
|
View Queue Monitor
|
|
</rs-button>
|
|
</div>
|
|
</template>
|
|
<template #body>
|
|
<div class="grid grid-cols-1 md:grid-cols-2 lg:grid-cols-4 gap-4">
|
|
<div
|
|
v-for="queue in queueStatus"
|
|
:key="queue.name"
|
|
class="p-4 bg-gray-50 dark:bg-gray-800 rounded-lg"
|
|
>
|
|
<div class="flex items-center justify-between mb-2">
|
|
<span class="font-medium">{{ queue.name }}</span>
|
|
<rs-badge
|
|
:variant="queue.status === 'active' ? 'success' :
|
|
queue.status === 'warning' ? 'warning' : 'danger'"
|
|
size="sm"
|
|
>
|
|
{{ queue.status }}
|
|
</rs-badge>
|
|
</div>
|
|
<div class="text-2xl font-bold text-primary mb-1">{{ queue.count }}</div>
|
|
<div class="text-sm text-gray-600">{{ queue.description }}</div>
|
|
<div class="w-full bg-gray-200 rounded-full h-2 mt-2">
|
|
<div
|
|
class="h-2 rounded-full transition-all duration-300"
|
|
:class="queue.status === 'active' ? 'bg-green-500' :
|
|
queue.status === 'warning' ? 'bg-yellow-500' : 'bg-red-500'"
|
|
:style="{ width: queue.utilization + '%' }"
|
|
></div>
|
|
</div>
|
|
<div class="text-xs text-gray-500 mt-1">{{ queue.utilization }}% utilized</div>
|
|
</div>
|
|
</div>
|
|
</template>
|
|
</rs-card>
|
|
|
|
<!-- Recent Logs -->
|
|
<rs-card>
|
|
<template #header>
|
|
<div class="flex items-center justify-between">
|
|
<div class="flex items-center">
|
|
<Icon name="ic:outline-history" class="mr-2 text-primary"/>
|
|
<h3 class="text-lg font-semibold text-primary">Recent Activity Logs</h3>
|
|
</div>
|
|
<rs-button variant="outline" size="sm" @click="navigateTo('/notification/log-audit/logs')">
|
|
View All Logs
|
|
</rs-button>
|
|
</div>
|
|
</template>
|
|
<template #body>
|
|
<div class="space-y-3">
|
|
<div
|
|
v-for="(log, index) in recentLogs"
|
|
:key="index"
|
|
class="flex items-center justify-between p-3 bg-gray-50 dark:bg-gray-800 rounded-lg"
|
|
>
|
|
<div class="flex items-center">
|
|
<div
|
|
class="w-3 h-3 rounded-full mr-3"
|
|
:class="{
|
|
'bg-green-500': log.status === 'sent' || log.status === 'created',
|
|
'bg-yellow-500': log.status === 'queued',
|
|
'bg-red-500': log.status === 'failed',
|
|
'bg-blue-500': log.status === 'opened',
|
|
}"
|
|
></div>
|
|
<div>
|
|
<p class="font-medium">{{ log.action }}</p>
|
|
<p class="text-sm text-gray-600">{{ log.description }}</p>
|
|
</div>
|
|
</div>
|
|
<div class="text-right">
|
|
<p class="text-sm font-medium capitalize">{{ log.status }}</p>
|
|
<p class="text-xs text-gray-500">{{ log.time }}</p>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</template>
|
|
</rs-card>
|
|
</div>
|
|
</template>
|
|
|
|
<script setup>
|
|
definePageMeta({
|
|
title: "Real-Time Monitoring",
|
|
middleware: ["auth"],
|
|
requiresAuth: true,
|
|
breadcrumb: [
|
|
{
|
|
name: "Dashboard",
|
|
path: "/dashboard",
|
|
},
|
|
{
|
|
name: "Notification",
|
|
path: "/notification",
|
|
},
|
|
{
|
|
name: "Logs & Audit Trail",
|
|
path: "/notification/log-audit",
|
|
},
|
|
{
|
|
name: "Monitoring",
|
|
path: "/notification/log-audit/monitoring",
|
|
type: "current"
|
|
},
|
|
],
|
|
});
|
|
|
|
import { ref, computed, onMounted, onUnmounted } from 'vue'
|
|
|
|
// Monitoring state
|
|
const isMonitoring = ref(true)
|
|
const refreshInterval = ref('5s')
|
|
const refreshIntervalId = ref(null)
|
|
|
|
const refreshOptions = [
|
|
{ label: '1 second', value: '1s' },
|
|
{ label: '5 seconds', value: '5s' },
|
|
{ label: '10 seconds', value: '10s' },
|
|
{ label: '30 seconds', value: '30s' },
|
|
{ label: '1 minute', value: '1m' },
|
|
]
|
|
|
|
// System status data
|
|
const systemStatus = ref([
|
|
{
|
|
title: "System Health",
|
|
value: "Healthy",
|
|
icon: "ic:outline-favorite",
|
|
status: "healthy"
|
|
},
|
|
{
|
|
title: "Throughput",
|
|
value: "12,847/min",
|
|
icon: "ic:outline-speed",
|
|
status: "healthy"
|
|
},
|
|
{
|
|
title: "Error Rate",
|
|
value: "0.02%",
|
|
icon: "ic:outline-error-outline",
|
|
status: "healthy"
|
|
},
|
|
{
|
|
title: "Response Time",
|
|
value: "145ms",
|
|
icon: "ic:outline-timer",
|
|
status: "warning"
|
|
},
|
|
])
|
|
|
|
// Performance metrics
|
|
const performanceMetrics = ref({
|
|
cpu: 23,
|
|
memory: 67,
|
|
queueLoad: 45
|
|
})
|
|
|
|
// Live activity feed
|
|
const liveActivityFeed = ref([
|
|
{
|
|
action: "Email Batch Processed",
|
|
details: "Successfully sent 1,250 welcome emails to new users",
|
|
timestamp: "Just now",
|
|
source: "Email Service",
|
|
type: "success"
|
|
},
|
|
{
|
|
action: "SMS Delivery Completed",
|
|
details: "OTP messages delivered to 89 recipients",
|
|
timestamp: "2 seconds ago",
|
|
source: "SMS Gateway",
|
|
type: "success"
|
|
},
|
|
{
|
|
action: "Push Notification Sent",
|
|
details: "Order confirmation sent to 456 devices",
|
|
timestamp: "8 seconds ago",
|
|
source: "Push Service",
|
|
type: "info"
|
|
},
|
|
{
|
|
action: "Webhook Timeout",
|
|
details: "Webhook to api.example.com timed out after 30 seconds",
|
|
timestamp: "15 seconds ago",
|
|
source: "Webhook Service",
|
|
type: "warning"
|
|
},
|
|
{
|
|
action: "Rate Limit Triggered",
|
|
details: "SMS rate limit reached for provider Twilio",
|
|
timestamp: "32 seconds ago",
|
|
source: "Rate Limiter",
|
|
type: "warning"
|
|
}
|
|
])
|
|
|
|
// Error alerts
|
|
const errorAlerts = ref([
|
|
{
|
|
title: "Database Connection Pool Exhausted",
|
|
description: "Connection pool has reached maximum capacity. New connections are being queued.",
|
|
timestamp: "2 minutes ago",
|
|
component: "Database",
|
|
severity: "critical"
|
|
},
|
|
{
|
|
title: "Email Provider Rate Limit",
|
|
description: "SendGrid API rate limit reached. Email delivery is temporarily throttled.",
|
|
timestamp: "5 minutes ago",
|
|
component: "Email Service",
|
|
severity: "warning"
|
|
},
|
|
{
|
|
title: "High Memory Usage",
|
|
description: "System memory usage has exceeded 85% threshold.",
|
|
timestamp: "8 minutes ago",
|
|
component: "System",
|
|
severity: "warning"
|
|
}
|
|
])
|
|
|
|
// Queue status
|
|
const queueStatus = ref([
|
|
{
|
|
name: "Email Queue",
|
|
count: "1,247",
|
|
description: "Pending emails",
|
|
status: "active",
|
|
utilization: 78
|
|
},
|
|
{
|
|
name: "SMS Queue",
|
|
count: "89",
|
|
description: "Pending SMS",
|
|
status: "active",
|
|
utilization: 23
|
|
},
|
|
{
|
|
name: "Push Queue",
|
|
count: "3,456",
|
|
description: "Pending push notifications",
|
|
status: "warning",
|
|
utilization: 92
|
|
},
|
|
{
|
|
name: "Webhook Queue",
|
|
count: "12",
|
|
description: "Pending webhooks",
|
|
status: "active",
|
|
utilization: 8
|
|
}
|
|
])
|
|
|
|
// Recent logs
|
|
const recentLogs = ref([
|
|
{
|
|
action: "Email Campaign Sent",
|
|
description: "Newsletter campaign to 50k subscribers",
|
|
status: "sent",
|
|
time: "1 minute ago"
|
|
},
|
|
{
|
|
action: "SMS Batch Queued",
|
|
description: "OTP messages for authentication",
|
|
status: "queued",
|
|
time: "3 minutes ago"
|
|
},
|
|
{
|
|
action: "Push Notification Failed",
|
|
description: "Failed to deliver to iOS devices",
|
|
status: "failed",
|
|
time: "5 minutes ago"
|
|
},
|
|
{
|
|
action: "Email Opened",
|
|
description: "User opened promotional email",
|
|
status: "opened",
|
|
time: "7 minutes ago"
|
|
},
|
|
{
|
|
action: "Webhook Created",
|
|
description: "New webhook endpoint configured",
|
|
status: "created",
|
|
time: "10 minutes ago"
|
|
}
|
|
])
|
|
|
|
// Methods
|
|
const toggleMonitoring = () => {
|
|
isMonitoring.value = !isMonitoring.value
|
|
if (isMonitoring.value) {
|
|
startMonitoring()
|
|
} else {
|
|
stopMonitoring()
|
|
}
|
|
}
|
|
|
|
const updateRefreshInterval = () => {
|
|
if (isMonitoring.value) {
|
|
stopMonitoring()
|
|
startMonitoring()
|
|
}
|
|
}
|
|
|
|
const startMonitoring = () => {
|
|
const intervalMs = {
|
|
'1s': 1000,
|
|
'5s': 5000,
|
|
'10s': 10000,
|
|
'30s': 30000,
|
|
'1m': 60000
|
|
}[refreshInterval.value] || 5000
|
|
|
|
refreshIntervalId.value = setInterval(() => {
|
|
// Simulate real-time updates
|
|
simulateActivityUpdate()
|
|
updatePerformanceMetrics()
|
|
}, intervalMs)
|
|
}
|
|
|
|
const stopMonitoring = () => {
|
|
if (refreshIntervalId.value) {
|
|
clearInterval(refreshIntervalId.value)
|
|
refreshIntervalId.value = null
|
|
}
|
|
}
|
|
|
|
const simulateActivityUpdate = () => {
|
|
// Add a new activity to the feed
|
|
const activities = [
|
|
{ action: "Email Sent", details: "Marketing email delivered successfully", type: "success", source: "Email Service" },
|
|
{ action: "SMS Delivered", details: "OTP message sent to user", type: "success", source: "SMS Gateway" },
|
|
{ action: "Push Failed", details: "Device token expired", type: "error", source: "Push Service" },
|
|
{ action: "Webhook Called", details: "Order confirmation webhook triggered", type: "info", source: "Webhook Service" }
|
|
]
|
|
|
|
const newActivity = {
|
|
...activities[Math.floor(Math.random() * activities.length)],
|
|
timestamp: "Just now"
|
|
}
|
|
|
|
liveActivityFeed.value.unshift(newActivity)
|
|
if (liveActivityFeed.value.length > 10) {
|
|
liveActivityFeed.value.pop()
|
|
}
|
|
}
|
|
|
|
const updatePerformanceMetrics = () => {
|
|
// Simulate metric updates
|
|
performanceMetrics.value.cpu = Math.max(10, Math.min(90, performanceMetrics.value.cpu + (Math.random() - 0.5) * 10))
|
|
performanceMetrics.value.memory = Math.max(20, Math.min(95, performanceMetrics.value.memory + (Math.random() - 0.5) * 5))
|
|
performanceMetrics.value.queueLoad = Math.max(0, Math.min(100, performanceMetrics.value.queueLoad + (Math.random() - 0.5) * 15))
|
|
}
|
|
|
|
const refreshData = () => {
|
|
console.log('Refreshing monitoring data...')
|
|
simulateActivityUpdate()
|
|
updatePerformanceMetrics()
|
|
}
|
|
|
|
const clearActivityFeed = () => {
|
|
liveActivityFeed.value = []
|
|
}
|
|
|
|
const acknowledgeAlert = (index) => {
|
|
errorAlerts.value.splice(index, 1)
|
|
}
|
|
|
|
const acknowledgeAllAlerts = () => {
|
|
errorAlerts.value = []
|
|
}
|
|
|
|
const exportPerformanceData = () => {
|
|
console.log('Exporting performance data...')
|
|
alert('Exporting performance data. (Implementation pending)')
|
|
}
|
|
|
|
// Lifecycle
|
|
onMounted(() => {
|
|
if (isMonitoring.value) {
|
|
startMonitoring()
|
|
}
|
|
})
|
|
|
|
onUnmounted(() => {
|
|
stopMonitoring()
|
|
})
|
|
</script>
|
|
|
|
<style lang="scss" scoped>
|
|
// Custom styles for FormKit consistency
|
|
:deep(.formkit-outer) {
|
|
margin-bottom: 0;
|
|
}
|
|
|
|
:deep(.formkit-label) {
|
|
font-weight: 500;
|
|
color: rgb(107 114 128);
|
|
font-size: 0.875rem;
|
|
margin-bottom: 0.5rem;
|
|
}
|
|
|
|
:deep(.formkit-input) {
|
|
border-radius: 0.5rem;
|
|
}
|
|
|
|
// Badge component styles (if RsBadge doesn't exist, these can be adjusted)
|
|
.rs-badge {
|
|
@apply inline-flex items-center px-2.5 py-0.5 rounded-full text-xs font-medium;
|
|
}
|
|
|
|
.rs-badge.variant-success {
|
|
@apply bg-green-100 text-green-800;
|
|
}
|
|
|
|
.rs-badge.variant-danger {
|
|
@apply bg-red-100 text-red-800;
|
|
}
|
|
|
|
.rs-badge.variant-warning {
|
|
@apply bg-yellow-100 text-yellow-800;
|
|
}
|
|
|
|
.rs-badge.variant-info {
|
|
@apply bg-blue-100 text-blue-800;
|
|
}
|
|
|
|
.rs-badge.variant-secondary {
|
|
@apply bg-gray-100 text-gray-800;
|
|
}
|
|
</style> |