Big refactor (#20)

* First step of server redesign

Only thing left is to re-add disconnect handling

* Convert client to use img

Removing the tiling feature allowed me to switch the client to use img
instead of canvas.

* Rename "canvas" to "img"

"canvas" isn't a good variable name for an <img> element

* Handle disconnects (again)

I had removed disconnect handling when I started the partial rewrite.
This commit re-adds it.

* Remove obsolete code

* Change dependency to websocket-server in requirements.txt

* Fix Chrome click bug

This should fix the Chrome click bug.
This commit is contained in:
Samuel Sloniker 2021-07-19 12:09:42 -07:00 committed by GitHub
parent ebc9831c93
commit 8ea09771e8
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
6 changed files with 127 additions and 226 deletions

View File

@ -18,7 +18,7 @@
</div> </div>
<div id=holder style="display:none"> <div id=holder style="display:none">
<div id=errorbox style="width: 800px; height: 480px">Loading...</div> <div id=errorbox style="width: 800px; height: 480px">Loading...</div>
<canvas id=canvas width=800 height=480 style="width: 800px; height: 480px; display: none"></canvas> <img id=img width=800 height=480 style="width: 800px; height: 480px; display: none"></canvas>
</div> </div>
<script src=js/main.js></script> <script src=js/main.js></script>
</body> </body>

View File

@ -11,8 +11,8 @@ function resize() {
displayWidth = Math.round(maxWidth) displayWidth = Math.round(maxWidth)
displayHeight = Math.round(maxWidth * 0.6) displayHeight = Math.round(maxWidth * 0.6)
} }
canvas.style.width = displayWidth + 'px' img.style.width = displayWidth + 'px'
canvas.style.height = displayHeight + 'px' img.style.height = displayHeight + 'px'
errorbox.style.width = displayWidth + 'px' errorbox.style.width = displayWidth + 'px'
errorbox.style.height = displayHeight + 'px' errorbox.style.height = displayHeight + 'px'
} }
@ -22,29 +22,22 @@ function setup() {
ws = new WebSocket(localStorage.getItem('url')) ws = new WebSocket(localStorage.getItem('url'))
ws.onmessage = function(e) { ws.onmessage = function(e) {
window.m = e window.m = e
let type, pos, x, y, data, img, code, msg let type, data, code, msg
type = e.data.split('%')[0] type = e.data.split('%')[0]
if (type == 'ver') { if (type == 'ver') {
ws.send('pass ' + localStorage.getItem('password')) ws.send('pass ' + localStorage.getItem('password'))
setInterval(function(){ws.send('ack')}, 3000) setInterval(function(){ws.send('ack')}, 3000)
} else if ( type == 'pic' ) { } else if ( type == 'pic' ) {
pos = e.data.split('%')[1]
x = Number(pos.split('x')[0])
y = Number(pos.split('x')[1])
data = e.data.split('%')[2] data = e.data.split('%')[2]
img = new Image()
img.src = data img.src = data
img.addEventListener("load", function(){
ctx.drawImage(img,x,y)
})
errorbox.style.display = 'none' errorbox.style.display = 'none'
canvas.style.display = 'block' img.style.display = 'block'
} else if ( type == 'err' ) { } else if ( type == 'err' ) {
code = e.data.split('%')[1] code = e.data.split('%')[1]
msg = e.data.split('%')[2] msg = e.data.split('%')[2]
errorbox.textContent = msg errorbox.textContent = msg
errorbox.style.display = 'block' errorbox.style.display = 'block'
canvas.style.display = 'none' img.style.display = 'none'
console.error('Server reported error: ' + code + ': ' + msg) console.error('Server reported error: ' + code + ': ' + msg)
if (code[0] == '*') { if (code[0] == '*') {
errorbox.innerHTML = msg + '<br>Please refresh page' errorbox.innerHTML = msg + '<br>Please refresh page'
@ -58,12 +51,12 @@ function setup() {
ws.onerror = function(e){ ws.onerror = function(e){
errorbox.innerHTML = 'Connection lost<br>Please refresh page' errorbox.innerHTML = 'Connection lost<br>Please refresh page'
errorbox.style.display = 'block' errorbox.style.display = 'block'
canvas.style.display = 'none' img.style.display = 'none'
} }
ws.onclose = function(e){ ws.onclose = function(e){
errorbox.innerHTML = 'Connection lost<br>Please refresh page' errorbox.innerHTML = 'Connection lost<br>Please refresh page'
errorbox.style.display = 'block' errorbox.style.display = 'block'
canvas.style.display = 'none' img.style.display = 'none'
} }
ws.onopen = function(e) { ws.onopen = function(e) {
ws.send('maxver 1') ws.send('maxver 1')
@ -80,8 +73,8 @@ function touch(e) {
function release(e) { function release(e) {
x = e.layerX x = e.offsetX || e.layerX
y = e.layerY y = e.offsetY || e.layerY
w = displayWidth w = displayWidth
length = is_short?false:true length = is_short?false:true
ws.send('touch ' + x + ' ' + y + ' ' + w + ' ' + length) ws.send('touch ' + x + ' ' + y + ' ' + w + ' ' + length)
@ -100,7 +93,7 @@ function connect(e) {
} }
let canvas = document.querySelector('canvas') let img = document.querySelector('#img')
let errorbox = document.querySelector('#errorbox') let errorbox = document.querySelector('#errorbox')
let login = document.querySelector('#login') let login = document.querySelector('#login')
@ -113,8 +106,6 @@ let start_button = document.querySelector('#start')
let is_short = true let is_short = true
let click_timeout let click_timeout
let ctx = canvas.getContext("2d")
let displayWidth, displayHeight let displayWidth, displayHeight
let ws let ws
@ -123,7 +114,7 @@ let ws
resize() resize()
window.onresize = resize window.onresize = resize
canvas.onmousedown = touch img.onmousedown = touch
canvas.onmouseup = release img.onmouseup = release
start.onclick = connect start.onclick = connect

View File

@ -1 +0,0 @@
["232x53+0+0", "232x11+0+53", "232x84+0+64", "163x148+232+0", "181x148+395+0", "170x148+576+0", "54x148+746+0", "139x148+0+148", "139x184+0+296", "110x166+139+148", "110x166+249+148", "110x166+359+148", "110x166+469+148", "110x166+579+148", "111x166+689+148", "110x166+139+314", "110x166+249+314", "110x166+359+314", "110x166+469+314", "110x166+579+314", "111x166+689+314"]

View File

@ -3,48 +3,14 @@ import threading
import os import os
import subprocess import subprocess
def run(bmp_path, crop): def get_img():
subprocess.run(['convert', bmp_path, '-crop', crop, f'pieces/{crop.split("+", 1)[1].replace("+", "x")}.jpg',])
def get_split_imgs():
bmp_path = backend.get_img() bmp_path = backend.get_img()
with open('crops.json') as f: subprocess.run(['convert', bmp_path, '-define', 'webp:lossless=true', 'img.webp',])
crops = json.load(f)
threads = []
for crop in crops:
threads.append(threading.Thread(target=run, args=([bmp_path, crop])))
threads[-1].start()
for thread in threads:
thread.join()
os.unlink(bmp_path) os.unlink(bmp_path)
with open('newlist', 'w+b') as f: return 'img.webp'
md5sum = subprocess.Popen(['md5sum', '-c', 'oldlist'], stdout=subprocess.PIPE, stderr=subprocess.DEVNULL)
grep = subprocess.Popen(['grep', 'FAILED'], stdin=md5sum.stdout, stdout=f)
grep.wait()
changed = []
with open('newlist') as f:
lines = f.readlines()
for line in lines:
changed.append(line.split(": ")[0].split('.')[0].split('/')[1])
with open('oldlist', 'w+b') as f:
subprocess.Popen(['md5sum'] + [f'pieces/{i}' for i in os.listdir('pieces')], stdout=f)
return changed
def get_full_img():
bmp_path = backend.get_img()
subprocess.run(['convert', bmp_path, 'img.jpg',])
os.unlink(bmp_path)
return 'img.jpg'
def touch(x, y, w, is_long): def touch(x, y, w, is_long):
x = round(800 * x / w) x = round(800 * x / w)

View File

@ -1,3 +1,3 @@
argon2-cffi argon2-cffi
requests requests
tornado websocket-server

View File

@ -1,19 +1,14 @@
import argon2 import argon2
import asyncio
import base64 import base64
import imgproc import imgproc
import importlib import importlib
import logging
import os import os
import parse_config import parse_config
import random
import requests
import secrets
import shutil
import sys import sys
import tempfile
import threading import threading
import time import time
import tornado.web, tornado.websocket, tornado.ioloop from websocket_server import WebsocketServer
try: try:
conf = sys.argv[1] conf = sys.argv[1]
@ -23,198 +18,148 @@ except IndexError:
with open(conf) as f: with open(conf) as f:
config_data = parse_config.load(f) config_data = parse_config.load(f)
imgproc.backend = importlib.import_module(f'backends.{config_data["backend"]}') backend = importlib.import_module(f'backends.{config_data["backend"]}')
imgproc.backend.config = config_data backend.config = config_data
imgproc.config = config_data imgproc.config = config_data
imgproc.backend = backend
ph = argon2.PasswordHasher() ph = argon2.PasswordHasher()
client = None client = None
token = secrets.token_urlsafe(64)
class DisconnectError(BaseException): class DisconnectError(BaseException):
pass pass
class Client: class Client:
def __init__(self, conn): def __init__(self, server, client_):
global client global client
if client is not None: if client is not None:
raise DisconnectError('err%*inuse%Server already in use') raise DisconnectError('err%*inuse%Server already in use')
client = self client = self
self.conn = conn self.server = server
self.items = {} self.client = client_
self.lock = threading.Lock()
self.good = True
def send(self, item, name):
with self.lock:
self.items[name] = item
def ack(self):
with self.lock:
self.good = False
def get_item_to_send(self):
with self.lock:
if self.items:
name, content = list(self.items.items())[0]
del self.items[name]
return name, content
elif not self.good:
return 'ack', 'ack'
else:
return None, None
class HCRAServer(tornado.websocket.WebSocketHandler):
def open(self):
self.has_auth = False self.has_auth = False
self.ready_for_msgs = False
self.acked = True
self.version = None self.version = None
self.next_msg = None
def on_close(self): def close(self):
global client global client
if client is self.client: self.ready_for_msgs = False
self.client.good = None
client = None client = None
self.is_open = False
imgproc.backend.disconnect() imgproc.backend.disconnect()
def on_message(self, message): def do_send(self):
action = message.split(' ', 1)[0] if self.next_msg is not None and self.ready_for_msgs and self.acked:
if self.version is None: msg = self.next_msg
if action == 'maxver': self.next_msg = None
self.version = min(int(message.split(' ', 1)[1]), 1) self.server.send_message(self.client, msg)
self.write_message(f'ver%{self.version}') self.server.send_message(self.client, 'ack')
else: self.acked = False
self.write_message('err%*mustmaxver%Client must send version')
self.close() def got_ack(self):
self.acked = True
self.do_send()
def on_left(client_, server):
global client
if client_ != client.client:
return return
elif not self.has_auth:
client.close()
def on_message(client_, server, message):
global client
if client_ != client.client:
return
action = message.split(' ', 1)[0]
if client.version is None:
if action == 'maxver':
client.version = min(int(message.split(' ', 1)[1]), 1)
server.send_message(client_, f'ver%{client.version}')
else:
server.send_message(client_, 'err%*mustmaxver%Client must send version')
client_['handler'].send_text("", opcode=0x8)
return
elif not client.has_auth:
if action != 'pass': if action != 'pass':
self.write_message('err%*mustauth%Authentication required') server.send_message(client_, 'err%*mustauth%Authentication required')
self.close() client_['handler'].send_text("", opcode=0x8)
return return
try: try:
ph.verify(config_data['password_argon2'], message.split(' ', 1)[1]) ph.verify(config_data['password_argon2'], message.split(' ', 1)[1])
except argon2.exceptions.VerifyMismatchError: except argon2.exceptions.VerifyMismatchError:
self.write_message(f'err%*badpass%Incorrect password') server.send_message(client_, f'err%*badpass%Incorrect password')
self.close() client_['handler'].send_text("", opcode=0x8)
return return
self.has_auth = True client.has_auth = True
try:
self.client = Client(self)
except DisconnectError as e:
self.write_message(str(e))
return
imgproc.backend.connect() imgproc.backend.connect()
try: client.ready_for_msgs = True
imgname = imgproc.get_full_img() client.do_send()
except Exception as e:
self.write_message('err%noconn%Server failed to capture screenshot')
return
with open(imgname, 'rb') as f:
img = f.read()
os.unlink(imgname)
self.write_message(f'pic%0x0%data:image/jpeg;base64,{base64.b64encode(img).decode("utf-8")}')
self.is_open = True
self.client.ack()
else: else:
if action == 'ack': if action == 'ack':
self.client.good = True client.got_ack()
else: else:
_, x, y, w, is_long = message.split(' ') _, x, y, w, is_long = message.split(' ')
x, y, w, is_long = int(x), int(y), int(w), is_long == 'true' x, y, w, is_long = int(x), int(y), int(w), is_long == 'true'
imgproc.touch(x, y, w, is_long) imgproc.touch(x, y, w, is_long)
def check_origin(self, origin):
return True
def cycle(): def on_connect(client_, server):
try: try:
changed = imgproc.get_split_imgs() client = Client(server, client_)
except Exception as e: except DisconnectError as e:
if client is not None: server.send_message(client_, str(e))
client.send('err%noconn%Server failed to capture screenshot', 'ERR') client_['handler'].send_text("", opcode=0x8)
time.sleep(3)
return return
threads = []
for i in changed:
thread = threading.Thread(target=do_img, args=(i,))
threads.append(thread)
thread.start()
def get_img_msg():
try:
img = imgproc.get_img()
except Exception as e:
return 'err%noconn%Server failed to capture screenshot'
with open(img, 'rb') as f:
img_data = f.read()
os.unlink(img)
return f'pic%0x0%data:image/webp;base64,{base64.b64encode(img_data).decode("utf-8")}'
def cycle():
if client is not None: if client is not None:
client.ack() msg = get_img_msg()
try:
client.next_msg = msg
def do_img(imgname): client.do_send()
if client is not None: except AttributeError as e:
client.send(img(imgname), imgname) if "'NoneType' object has no attribute" in str(e):
return
else:
def img(imgname): raise e
with open(f'pieces/{imgname}.jpg', 'rb') as f:
img = f.read()
response = f'pic%{imgname}%data:image/jpeg;base64,{base64.b64encode(img).decode("utf-8")}'
return response
def do_cycles(): def do_cycles():
while True: while True:
if client is not None:
cycle() cycle()
time.sleep(0.4) time.sleep(0.8)
else:
time.sleep(1)
threading.Thread(target=do_cycles).start()
server = WebsocketServer(int(config_data['port']), host='0.0.0.0', loglevel=logging.INFO)
class Cycler(tornado.web.RequestHandler): server.set_fn_new_client(on_connect)
def get(self): server.set_fn_message_received(on_message)
if client is not None: server.set_fn_client_left(on_left)
name, item = client.get_item_to_send() server.run_forever()
if item is not None:
client.conn.write_message(item)
self.write('OK')
def get_token():
while True:
requests.get('http://localhost:1234/' + token)
tmp = tempfile.mkdtemp(prefix="HCRA-")
try:
shutil.copy('crops.json', tmp)
os.chdir(tmp)
os.mkdir('pieces')
threading.Thread(target=do_cycles).start()
application = tornado.web.Application([
(r"/", HCRAServer),
('/' + token, Cycler),
])
application.listen(int(config['port']))
threading.Thread(target=get_token).start()
tornado.ioloop.IOLoop.current().start()
finally:
os.chdir('/')
shutil.rmtree(tmp)