690 lines
22 KiB
Vue
690 lines
22 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";
|
|
|
|
// Use the notification logs composable
|
|
const {
|
|
monitoringData,
|
|
monitoringLoading,
|
|
monitoringError,
|
|
fetchMonitoringData,
|
|
formatTimeAgo,
|
|
} = useNotificationLogs();
|
|
|
|
// 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 - will be updated from API
|
|
const systemStatus = computed(
|
|
() =>
|
|
monitoringData.value?.systemStatus || [
|
|
{
|
|
title: "System Health",
|
|
value: "Healthy",
|
|
icon: "ic:outline-favorite",
|
|
status: "healthy",
|
|
},
|
|
{
|
|
title: "Throughput",
|
|
value: "0/hr",
|
|
icon: "ic:outline-speed",
|
|
status: "healthy",
|
|
},
|
|
{
|
|
title: "Error Rate",
|
|
value: "0.00%",
|
|
icon: "ic:outline-error-outline",
|
|
status: "healthy",
|
|
},
|
|
{
|
|
title: "Response Time",
|
|
value: "0ms",
|
|
icon: "ic:outline-timer",
|
|
status: "healthy",
|
|
},
|
|
]
|
|
);
|
|
|
|
// Performance metrics - will be updated from API
|
|
const performanceMetrics = computed(
|
|
() =>
|
|
monitoringData.value?.performanceMetrics || {
|
|
cpu: 0,
|
|
memory: 0,
|
|
queueLoad: 0,
|
|
}
|
|
);
|
|
|
|
// Live activity feed - will be updated from API
|
|
const liveActivityFeed = computed(() => monitoringData.value?.recentActivity || []);
|
|
|
|
// Error alerts - will be updated from API
|
|
const errorAlerts = computed(() => monitoringData.value?.errorAlerts || []);
|
|
|
|
// Queue status - will be updated from API
|
|
const queueStatus = computed(() => monitoringData.value?.queueStatus || []);
|
|
|
|
// Recent logs - using the same activity feed
|
|
const recentLogs = computed(() => liveActivityFeed.value);
|
|
|
|
// 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;
|
|
|
|
// Fetch immediately
|
|
fetchMonitoringData();
|
|
|
|
// Then set up the interval
|
|
refreshIntervalId.value = setInterval(async () => {
|
|
await fetchMonitoringData();
|
|
}, intervalMs);
|
|
};
|
|
|
|
const stopMonitoring = () => {
|
|
if (refreshIntervalId.value) {
|
|
clearInterval(refreshIntervalId.value);
|
|
refreshIntervalId.value = null;
|
|
}
|
|
};
|
|
|
|
const refreshData = async () => {
|
|
await fetchMonitoringData();
|
|
};
|
|
|
|
const clearActivityFeed = () => {
|
|
// For now, just refresh the data - in a real app, you might have an API endpoint to clear the feed
|
|
fetchMonitoringData();
|
|
};
|
|
|
|
const acknowledgeAlert = (index) => {
|
|
// In a real app, you would call an API to acknowledge the alert
|
|
errorAlerts.value.splice(index, 1);
|
|
};
|
|
|
|
const acknowledgeAllAlerts = () => {
|
|
// In a real app, you would call an API to acknowledge all alerts
|
|
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>
|