import json
import logging
import asyncio
import sys
from urllib.parse import parse_qsl

from app.classes.shared.authentication import authentication
from app.classes.shared.helpers import helper
from app.classes.shared.console import console
from app.classes.web.websocket_helper import websocket_helper

logger = logging.getLogger(__name__)

try:
    import tornado.websocket

except ModuleNotFoundError as e:
    logger.critical(f"Import Error: Unable to load {e.name} module", exc_info=True)
    console.critical(f"Import Error: Unable to load {e.name} module")
    sys.exit(1)


class SocketHandler(tornado.websocket.WebSocketHandler):
    page = None
    page_query_params = None
    controller = None
    tasks_manager = None
    translator = None
    io_loop = None

    def initialize(self, controller=None, tasks_manager=None, translator=None):
        self.controller = controller
        self.tasks_manager = tasks_manager
        self.translator = translator
        self.io_loop = tornado.ioloop.IOLoop.current()

    def get_remote_ip(self):
        remote_ip = self.request.headers.get("X-Real-IP") or \
                    self.request.headers.get("X-Forwarded-For") or \
                    self.request.remote_ip
        return remote_ip

    def get_user_id(self):
        _, _, user = authentication.check(self.get_cookie('token'))
        return user['user_id']

    def check_auth(self):
        return authentication.check_bool(self.get_cookie('token'))

    # pylint: disable=arguments-differ
    def open(self):
        logger.debug('Checking WebSocket authentication')
        if self.check_auth():
            self.handle()
        else:
            websocket_helper.send_message(self, 'notification', 'Not authenticated for WebSocket connection')
            self.close()
            self.controller.management.add_to_audit_log_raw('unknown',
                                                            0, 0,
                                                            'Someone tried to connect via WebSocket without proper authentication',
                                                            self.get_remote_ip())
            websocket_helper.broadcast('notification', 'Someone tried to connect via WebSocket without proper authentication')
            logger.warning('Someone tried to connect via WebSocket without proper authentication')

    def handle(self):
        self.page = self.get_query_argument('page')
        self.page_query_params = dict(parse_qsl(helper.remove_prefix(
            self.get_query_argument('page_query_params'),
            '?'
        )))
        websocket_helper.add_client(self)
        logger.debug('Opened WebSocket connection')

    # pylint: disable=arguments-renamed
    @staticmethod
    def on_message(raw_message):

        logger.debug(f'Got message from WebSocket connection {raw_message}')
        message = json.loads(raw_message)
        logger.debug(f"Event Type: {message['event']}, Data: {message['data']}")

    def on_close(self):
        websocket_helper.remove_client(self)
        logger.debug('Closed WebSocket connection')

    async def write_message_int(self, message):
        self.write_message(message)

    def write_message_helper(self, message):
        asyncio.run_coroutine_threadsafe(self.write_message_int(message), self.io_loop.asyncio_loop)