From a896346c77427ccdbf0b078aad977a398b9f7fe6 Mon Sep 17 00:00:00 2001 From: grossmj Date: Sat, 12 Jan 2019 16:02:36 +0700 Subject: [PATCH] Fix issue with notification queue that prevented to properly close projects. Fix #1493 --- gns3server/controller/notification.py | 13 ++++++---- .../api/compute/notification_handler.py | 14 ++++++----- .../api/controller/notification_handler.py | 12 +++++----- .../api/controller/project_handler.py | 24 +++++++++---------- 4 files changed, 34 insertions(+), 29 deletions(-) diff --git a/gns3server/controller/notification.py b/gns3server/controller/notification.py index 1e0ef4a6..97e0883c 100644 --- a/gns3server/controller/notification.py +++ b/gns3server/controller/notification.py @@ -17,7 +17,6 @@ import os import aiohttp -import asyncio from contextlib import contextmanager from ..notification_queue import NotificationQueue @@ -43,8 +42,10 @@ class Notification: queue = NotificationQueue() self._project_listeners.setdefault(project.id, set()) self._project_listeners[project.id].add(queue) - yield queue - self._project_listeners[project.id].remove(queue) + try: + yield queue + finally: + self._project_listeners[project.id].remove(queue) @contextmanager def controller_queue(self): @@ -55,8 +56,10 @@ class Notification: """ queue = NotificationQueue() self._controller_listeners.append(queue) - yield queue - self._controller_listeners.remove(queue) + try: + yield queue + finally: + self._controller_listeners.remove(queue) def controller_emit(self, action, event): """ diff --git a/gns3server/handlers/api/compute/notification_handler.py b/gns3server/handlers/api/compute/notification_handler.py index 81cf9d0e..75b6f8b1 100644 --- a/gns3server/handlers/api/compute/notification_handler.py +++ b/gns3server/handlers/api/compute/notification_handler.py @@ -44,15 +44,17 @@ class NotificationHandler: request.app['websockets'].add(ws) asyncio.ensure_future(process_websocket(ws)) - with notifications.queue() as queue: - try: + + try: + with notifications.queue() as queue: while True: notification = await queue.get_json(1) if ws.closed: break await ws.send_str(notification) - finally: - if not ws.closed: - await ws.close() - request.app['websockets'].discard(ws) + finally: + if not ws.closed: + await ws.close() + request.app['websockets'].discard(ws) + return ws diff --git a/gns3server/handlers/api/controller/notification_handler.py b/gns3server/handlers/api/controller/notification_handler.py index 5e26fb27..04c118e0 100644 --- a/gns3server/handlers/api/controller/notification_handler.py +++ b/gns3server/handlers/api/controller/notification_handler.py @@ -67,16 +67,16 @@ class NotificationHandler: request.app['websockets'].add(ws) asyncio.ensure_future(process_websocket(ws)) - with controller.notification.controller_queue() as queue: - try: + try: + with controller.notification.controller_queue() as queue: while True: notification = await queue.get_json(5) if ws.closed: break await ws.send_str(notification) - finally: - if not ws.closed: - await ws.close() - request.app['websockets'].discard(ws) + finally: + if not ws.closed: + await ws.close() + request.app['websockets'].discard(ws) return ws diff --git a/gns3server/handlers/api/controller/project_handler.py b/gns3server/handlers/api/controller/project_handler.py index 265fb96f..66ff6294 100644 --- a/gns3server/handlers/api/controller/project_handler.py +++ b/gns3server/handlers/api/controller/project_handler.py @@ -259,24 +259,24 @@ class ProjectHandler: request.app['websockets'].add(ws) asyncio.ensure_future(process_websocket(ws)) - with controller.notification.project_queue(project) as queue: - try: + try: + with controller.notification.project_queue(project) as queue: while True: notification = await queue.get_json(5) if ws.closed: break await ws.send_str(notification) - finally: - if not ws.closed: - await ws.close() - request.app['websockets'].discard(ws) + finally: + if not ws.closed: + await ws.close() + request.app['websockets'].discard(ws) - if project.auto_close: - # To avoid trouble with client connecting disconnecting we sleep few seconds before checking - # if someone else is not connected - await asyncio.sleep(5) - if not controller.notification.project_has_listeners(project): - await project.close() + if project.auto_close: + # To avoid trouble with client connecting disconnecting we sleep few seconds before checking + # if someone else is not connected + await asyncio.sleep(5) + if not controller.notification.project_has_listeners(project): + await project.close() return ws