mirror of
https://gitlab.com/MoonTestUse1/AdministrationItDepartmens.git
synced 2025-08-14 00:25:46 +02:00
add websockets support22
This commit is contained in:
@@ -45,6 +45,9 @@ async def create_request(
|
|||||||
# Отправляем уведомление в Telegram
|
# Отправляем уведомление в Telegram
|
||||||
await notify_new_request(db_request.id)
|
await notify_new_request(db_request.id)
|
||||||
|
|
||||||
|
# Получаем актуальную статистику
|
||||||
|
stats = requests.get_statistics(db)
|
||||||
|
|
||||||
# Получаем полные данные о заявке для отправки через WebSocket
|
# Получаем полные данные о заявке для отправки через WebSocket
|
||||||
request_data = {
|
request_data = {
|
||||||
"id": db_request.id,
|
"id": db_request.id,
|
||||||
@@ -61,7 +64,11 @@ async def create_request(
|
|||||||
# Отправляем уведомление через WebSocket всем админам
|
# Отправляем уведомление через WebSocket всем админам
|
||||||
await notification_manager.broadcast_to_admins({
|
await notification_manager.broadcast_to_admins({
|
||||||
"type": "new_request",
|
"type": "new_request",
|
||||||
"data": request_data
|
"data": request_data,
|
||||||
|
"statistics": {
|
||||||
|
"total": stats["total"],
|
||||||
|
"by_status": stats["by_status"]
|
||||||
|
}
|
||||||
})
|
})
|
||||||
return db_request
|
return db_request
|
||||||
|
|
||||||
@@ -96,12 +103,19 @@ async def update_request_status(
|
|||||||
if db_request is None:
|
if db_request is None:
|
||||||
raise HTTPException(status_code=404, detail="Request not found")
|
raise HTTPException(status_code=404, detail="Request not found")
|
||||||
|
|
||||||
|
# Получаем актуальную статистику
|
||||||
|
stats = requests.get_statistics(db)
|
||||||
|
|
||||||
# Отправляем уведомление через WebSocket
|
# Отправляем уведомление через WebSocket
|
||||||
await notification_manager.broadcast_to_admins({
|
await notification_manager.broadcast_to_admins({
|
||||||
"type": "status_update",
|
"type": "status_update",
|
||||||
"data": {
|
"data": {
|
||||||
"id": request_id,
|
"id": request_id,
|
||||||
"status": db_request.status
|
"status": db_request.status
|
||||||
|
},
|
||||||
|
"statistics": {
|
||||||
|
"total": stats["total"],
|
||||||
|
"by_status": stats["by_status"]
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
return db_request
|
return db_request
|
||||||
|
@@ -1,6 +1,9 @@
|
|||||||
from fastapi import WebSocket
|
from fastapi import WebSocket
|
||||||
from typing import Dict, List
|
from typing import Dict, List
|
||||||
import json
|
import json
|
||||||
|
import logging
|
||||||
|
|
||||||
|
logger = logging.getLogger(__name__)
|
||||||
|
|
||||||
class NotificationManager:
|
class NotificationManager:
|
||||||
def __init__(self):
|
def __init__(self):
|
||||||
@@ -12,25 +15,47 @@ class NotificationManager:
|
|||||||
async def connect(self, websocket: WebSocket, client_type: str):
|
async def connect(self, websocket: WebSocket, client_type: str):
|
||||||
await websocket.accept()
|
await websocket.accept()
|
||||||
self.active_connections[client_type].append(websocket)
|
self.active_connections[client_type].append(websocket)
|
||||||
|
logger.info(f"New {client_type} connection. Total connections: {len(self.active_connections[client_type])}")
|
||||||
|
|
||||||
def disconnect(self, websocket: WebSocket, client_type: str):
|
def disconnect(self, websocket: WebSocket, client_type: str):
|
||||||
self.active_connections[client_type].remove(websocket)
|
if websocket in self.active_connections[client_type]:
|
||||||
|
self.active_connections[client_type].remove(websocket)
|
||||||
|
logger.info(f"{client_type} disconnected. Remaining connections: {len(self.active_connections[client_type])}")
|
||||||
|
|
||||||
async def broadcast_to_admins(self, message: dict):
|
async def broadcast_to_admins(self, message: dict):
|
||||||
"""Отправка сообщения всем подключенным админам"""
|
"""Отправка сообщения всем подключенным админам"""
|
||||||
|
logger.info(f"Broadcasting to admins: {message}")
|
||||||
|
disconnected = []
|
||||||
|
|
||||||
for connection in self.active_connections["admin"]:
|
for connection in self.active_connections["admin"]:
|
||||||
try:
|
try:
|
||||||
await connection.send_json(message)
|
await connection.send_json(message)
|
||||||
except:
|
logger.info("Message sent successfully")
|
||||||
# Если не удалось отправить сообщение, пропускаем
|
except Exception as e:
|
||||||
|
logger.error(f"Error sending message: {e}")
|
||||||
|
disconnected.append(connection)
|
||||||
continue
|
continue
|
||||||
|
|
||||||
|
# Удаляем отключенные соединения
|
||||||
|
for connection in disconnected:
|
||||||
|
self.disconnect(connection, "admin")
|
||||||
|
|
||||||
async def broadcast_to_employees(self, employee_id: int, message: dict):
|
async def broadcast_to_employees(self, employee_id: int, message: dict):
|
||||||
"""Отправка сообщения конкретному сотруднику"""
|
"""Отправка сообщения конкретному сотруднику"""
|
||||||
|
logger.info(f"Broadcasting to employee {employee_id}: {message}")
|
||||||
|
disconnected = []
|
||||||
|
|
||||||
for connection in self.active_connections["employee"]:
|
for connection in self.active_connections["employee"]:
|
||||||
try:
|
try:
|
||||||
await connection.send_json(message)
|
await connection.send_json(message)
|
||||||
except:
|
logger.info("Message sent successfully")
|
||||||
|
except Exception as e:
|
||||||
|
logger.error(f"Error sending message: {e}")
|
||||||
|
disconnected.append(connection)
|
||||||
continue
|
continue
|
||||||
|
|
||||||
|
# Удаляем отключенные соединения
|
||||||
|
for connection in disconnected:
|
||||||
|
self.disconnect(connection, "employee")
|
||||||
|
|
||||||
notification_manager = NotificationManager()
|
notification_manager = NotificationManager()
|
@@ -72,7 +72,7 @@
|
|||||||
</template>
|
</template>
|
||||||
|
|
||||||
<script setup lang="ts">
|
<script setup lang="ts">
|
||||||
import { ref, onMounted, onUnmounted } from 'vue'
|
import { ref, onMounted, onUnmounted, watch } from 'vue'
|
||||||
import axios from '@/plugins/axios'
|
import axios from '@/plugins/axios'
|
||||||
import { wsClient } from '@/plugins/websocket'
|
import { wsClient } from '@/plugins/websocket'
|
||||||
import { formatDate } from '@/utils/date'
|
import { formatDate } from '@/utils/date'
|
||||||
@@ -101,32 +101,21 @@ const fetchData = async () => {
|
|||||||
const handleWebSocketMessage = async (data: any) => {
|
const handleWebSocketMessage = async (data: any) => {
|
||||||
console.log('Received WebSocket message:', data)
|
console.log('Received WebSocket message:', data)
|
||||||
|
|
||||||
|
// Обновляем статистику, если она пришла в сообщении
|
||||||
|
if (data.statistics) {
|
||||||
|
statistics.value = data.statistics
|
||||||
|
}
|
||||||
|
|
||||||
if (data.type === 'new_request') {
|
if (data.type === 'new_request') {
|
||||||
try {
|
// Добавляем новую заявку в начало списка
|
||||||
// Получаем актуальную статистику
|
if (data.data) {
|
||||||
const statsResponse = await axios.get('/api/requests/statistics')
|
requests.value = [data.data, ...requests.value]
|
||||||
statistics.value = statsResponse.data
|
|
||||||
|
|
||||||
// Добавляем новую заявку в начало списка
|
|
||||||
if (data.data) {
|
|
||||||
requests.value = [data.data, ...requests.value]
|
|
||||||
}
|
|
||||||
} catch (error) {
|
|
||||||
console.error('Error updating data:', error)
|
|
||||||
}
|
}
|
||||||
} else if (data.type === 'status_update') {
|
} else if (data.type === 'status_update') {
|
||||||
try {
|
// Обновляем статус заявки в списке
|
||||||
// Получаем актуальную статистику
|
const request = requests.value.find(r => r.id === data.data.id)
|
||||||
const statsResponse = await axios.get('/api/requests/statistics')
|
if (request) {
|
||||||
statistics.value = statsResponse.data
|
request.status = data.data.status
|
||||||
|
|
||||||
// Обновляем статус заявки в списке
|
|
||||||
const request = requests.value.find(r => r.id === data.data.id)
|
|
||||||
if (request) {
|
|
||||||
request.status = data.data.status
|
|
||||||
}
|
|
||||||
} catch (error) {
|
|
||||||
console.error('Error updating status:', error)
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -141,6 +130,15 @@ onMounted(() => {
|
|||||||
}, 1000)
|
}, 1000)
|
||||||
})
|
})
|
||||||
|
|
||||||
|
// Переподключение WebSocket при потере соединения
|
||||||
|
watch(() => wsClient.isConnected, (isConnected) => {
|
||||||
|
if (!isConnected) {
|
||||||
|
setTimeout(() => {
|
||||||
|
wsClient.connect('admin')
|
||||||
|
}, 3000)
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
// Отключение от WebSocket при размонтировании компонента
|
// Отключение от WebSocket при размонтировании компонента
|
||||||
onUnmounted(() => {
|
onUnmounted(() => {
|
||||||
wsClient.removeMessageHandler(handleWebSocketMessage)
|
wsClient.removeMessageHandler(handleWebSocketMessage)
|
||||||
|
Reference in New Issue
Block a user