diff options
author | Robert Zorko <robertz@gurucue.com> | 2015-10-13 15:28:18 +0200 |
---|---|---|
committer | Robert Zorko <robertz@gurucue.com> | 2015-10-13 15:28:18 +0200 |
commit | ba122f03ee8f591859c8c276e580755615c4c299 (patch) | |
tree | 5bc52f4b4ec0089aa1bb198922c5f24ba45bb038 /robot | |
parent | c1171038a0d8b41c9e10fcdd66c25532890df7fb (diff) | |
parent | 21f7f2bfadbf49240693fa4d592869461055ea06 (diff) |
Merge branch 'master' of ssh://212.235.189.51:22122/codeq-server
Diffstat (limited to 'robot')
-rwxr-xr-x | robot/main.py | 127 |
1 files changed, 127 insertions, 0 deletions
diff --git a/robot/main.py b/robot/main.py new file mode 100755 index 0000000..06f10a0 --- /dev/null +++ b/robot/main.py @@ -0,0 +1,127 @@ +#!/usr/bin/python3 + +# This file must be run on the EV3 brick. It serves websocket connections, +# sending sensor updates and running user programs. + +PORT = 8000 + +import json +import multiprocessing +import os +import os.path +import signal + +import engineio +import eventlet + +import ev3dev + +ev3_names = { + 'lego-ev3-gyro': 'gyroscope', + 'lego-ev3-touch': 'touch', + 'lego-ev3-us': 'ultrasonic' +} + +ev3_motors = [ + ev3dev.motor(ev3dev.OUTPUT_A), + ev3dev.motor(ev3dev.OUTPUT_B), + ev3dev.motor(ev3dev.OUTPUT_C), + ev3dev.motor(ev3dev.OUTPUT_D) +] + +connections = set() +process = None + +def _run_exec(program): + import ev3dev + from mindstorms_widgets import mindstorms_widgets + exec(program) + +def run(program): + global process + stop() + process = multiprocessing.Process(target=_run_exec, args=[program]) + process.start() + +def stop(): + global process + if process is not None: + try: + os.kill(process.pid, signal.SIGKILL) + except: + pass + process = None + + for motor in ev3_motors: + if motor.connected: + motor.stop(stop_command='brake') + +def notifier(): + def read_sensor(path): + with open(os.path.join(path, 'driver_name'), 'r') as f: + name = f.read().strip() + with open(os.path.join(path, 'value0'), 'r') as f: + value = int(f.read().strip()) + with open(os.path.join(path, 'decimals'), 'r') as f: + decimals = int(f.read().strip()) + if decimals > 0: + value /= 10**decimals + with open(os.path.join(path, 'units'), 'r') as f: + units = f.read().strip() + if units: + value = '{} {}'.format(value, units) + return (ev3_names.get(name, name), value) + + sensors_path = '/sys/class/lego-sensor' + while True: + eventlet.sleep(0.2) + try: + message = {'event': 'update', 'sensors': {}} + for sensor in os.listdir(sensors_path): + name, value = read_sensor(os.path.join(sensors_path, sensor)) + message['sensors'][name] = value + + text = json.dumps(message) + for sid in connections: + eio.send(sid, text) + except: + pass + +eio = engineio.Server(async_mode='eventlet', cookie=None) +app = engineio.Middleware(eio) + +@eio.on('connect') +def connect(sid, environ): + print('New connection: {}'.format(sid)) + connections.add(sid) + +@eio.on('disconnect') +def disconnect(sid): + print('Dropped connection: {}'.format(sid)) + try: + connections.remove(sid) + except: + pass + +@eio.on('message') +def message(sid, msg): + reply = None + action = msg.get('action') + + if action is None: + reply = {'code': 1, 'message': 'Request does not contain an action'} + elif action == 'run': + program = msg.get('program') + if program is None: + reply = {'code': 2, 'message': 'Program not specified'} + else: + run(program) + reply = {'code': 0} + elif action == 'stop': + stop() + reply = {'code': 0} + + eio.send(sid, reply) + +eventlet.spawn_n(notifier) +eventlet.wsgi.server(eventlet.listen(('', PORT)), app) |