104 lines
3.7 KiB
Python
104 lines
3.7 KiB
Python
|
import json
|
||
|
import logging
|
||
|
import time
|
||
|
import urllib.parse
|
||
|
import urllib3
|
||
|
import websocket
|
||
|
|
||
|
from threading import Thread
|
||
|
|
||
|
from odoo.addons.hw_drivers import main
|
||
|
from odoo.addons.hw_drivers.tools import helpers
|
||
|
|
||
|
_logger = logging.getLogger(__name__)
|
||
|
websocket.enableTrace(True, level=logging.getLevelName(_logger.getEffectiveLevel()))
|
||
|
|
||
|
def send_to_controller(print_id, device_identifier):
|
||
|
server = helpers.get_odoo_server_url()
|
||
|
try:
|
||
|
urllib3.disable_warnings()
|
||
|
http = urllib3.PoolManager(cert_reqs='CERT_NONE')
|
||
|
http.request(
|
||
|
'POST',
|
||
|
server + "/iot/printer/status",
|
||
|
body=json.dumps(
|
||
|
{'params': {
|
||
|
'print_id': print_id,
|
||
|
'device_identifier': device_identifier,
|
||
|
}}).encode('utf8'),
|
||
|
headers={
|
||
|
'Content-type': 'application/json',
|
||
|
'Accept': 'text/plain',
|
||
|
},
|
||
|
)
|
||
|
except Exception:
|
||
|
_logger.exception('Could not reach configured server: %s', server)
|
||
|
|
||
|
|
||
|
def on_message(ws, messages):
|
||
|
"""
|
||
|
When a message is receive, this function is triggered
|
||
|
The message is load and if its type is 'print', is sent to the printer
|
||
|
"""
|
||
|
messages = json.loads(messages)
|
||
|
for document in messages:
|
||
|
if (document['message']['type'] == 'print'):
|
||
|
payload = document['message']['payload']
|
||
|
if helpers.get_mac_address() in payload['iotDevice']['iotIdentifiers']:
|
||
|
#send box confirmation
|
||
|
for device in payload['iotDevice']['identifiers']:
|
||
|
if device['identifier'] in main.iot_devices:
|
||
|
main.iot_devices[device["identifier"]]._action_default(payload)
|
||
|
send_to_controller(payload['print_id'], device['identifier'])
|
||
|
|
||
|
|
||
|
def on_error(ws, error):
|
||
|
_logger.error("websocket received an error: %s", error)
|
||
|
|
||
|
|
||
|
def on_close(ws, close_status_code, close_msg):
|
||
|
_logger.debug("websocket closed with status: %s", close_status_code)
|
||
|
|
||
|
|
||
|
class WebsocketClient(Thread):
|
||
|
iot_channel = ""
|
||
|
|
||
|
def on_open(self, ws):
|
||
|
"""
|
||
|
When the client is setup, this function send a message to subscribe to the iot websocket channel
|
||
|
"""
|
||
|
ws.send(
|
||
|
json.dumps({'event_name': 'subscribe', 'data': {'channels': [self.iot_channel], 'last': 0}})
|
||
|
)
|
||
|
|
||
|
def __init__(self, url):
|
||
|
url_parsed = urllib.parse.urlsplit(url)
|
||
|
scheme = url_parsed.scheme.replace("http", "ws", 1)
|
||
|
self.url = urllib.parse.urlunsplit((scheme, url_parsed.netloc, 'websocket', '', ''))
|
||
|
Thread.__init__(self)
|
||
|
|
||
|
def run(self):
|
||
|
self.ws = websocket.WebSocketApp(self.url,
|
||
|
on_open=self.on_open, on_message=on_message,
|
||
|
on_error=on_error, on_close=on_close)
|
||
|
|
||
|
# The IoT synchronised servers can stop in 2 ways that we need to handle:
|
||
|
# A. Gracefully:
|
||
|
# In this case a disconnection signal is sent to the IoT-box
|
||
|
# The websocket is properly closed, but it needs to be established a new connection when
|
||
|
# the server will be back.
|
||
|
#
|
||
|
# B. Forced/killed:
|
||
|
# In this case there is no disconnection signal received
|
||
|
#
|
||
|
# This will also happen with the graceful quit as `reconnect` will trigger if the server
|
||
|
# is offline while attempting the new connection
|
||
|
while True:
|
||
|
try:
|
||
|
run_res = self.ws.run_forever(reconnect=10)
|
||
|
_logger.debug("websocket run_forever return with %s", run_res)
|
||
|
except Exception:
|
||
|
_logger.exception("An unexpected exception happened when running the websocket")
|
||
|
_logger.debug('websocket will try to restart in 10 seconds')
|
||
|
time.sleep(10)
|