2026-04-21 12:35:48 +08:00
|
|
|
import { logger } from './logger'
|
|
|
|
|
|
2026-05-05 13:03:14 +08:00
|
|
|
export function bindShutdown(server: any, groupChatServer?: any, chatRunServer?: any): void {
|
2026-04-20 20:37:32 +08:00
|
|
|
let isShuttingDown = false
|
|
|
|
|
|
|
|
|
|
const shutdown = async (signal: string) => {
|
|
|
|
|
if (isShuttingDown) return
|
|
|
|
|
isShuttingDown = true
|
|
|
|
|
|
2026-04-21 12:35:48 +08:00
|
|
|
logger.info('Shutting down (%s)...', signal)
|
2026-04-20 20:37:32 +08:00
|
|
|
|
|
|
|
|
try {
|
2026-05-05 13:03:14 +08:00
|
|
|
// Close ChatRunSocket first to abort all active runs and close EventSource connections
|
|
|
|
|
if (chatRunServer) {
|
|
|
|
|
chatRunServer.close()
|
|
|
|
|
logger.info('ChatRunSocket closed')
|
|
|
|
|
}
|
|
|
|
|
|
2026-04-24 20:41:14 +08:00
|
|
|
// Disconnect Socket.IO before HTTP server to prevent hanging
|
|
|
|
|
if (groupChatServer) {
|
|
|
|
|
groupChatServer.agentClients.disconnectAll()
|
|
|
|
|
groupChatServer.getIO().close()
|
|
|
|
|
logger.info('Socket.IO closed')
|
|
|
|
|
}
|
|
|
|
|
|
2026-04-20 20:37:32 +08:00
|
|
|
if (server) {
|
|
|
|
|
await new Promise<void>((resolve) => {
|
|
|
|
|
server.close(() => {
|
2026-04-21 12:35:48 +08:00
|
|
|
logger.info('HTTP server closed')
|
2026-04-20 20:37:32 +08:00
|
|
|
resolve()
|
|
|
|
|
})
|
|
|
|
|
})
|
|
|
|
|
}
|
|
|
|
|
} catch (err) {
|
2026-04-21 12:35:48 +08:00
|
|
|
logger.error(err, 'Shutdown error')
|
2026-04-20 20:37:32 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
process.exit(0)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
process.once('SIGUSR2', shutdown)
|
|
|
|
|
process.on('SIGINT', shutdown)
|
|
|
|
|
process.on('SIGTERM', shutdown)
|
|
|
|
|
}
|