Torchlight3/Torchlight/AsyncClient.py
2017-08-02 23:41:02 +02:00

98 lines
2.1 KiB
Python

#!/usr/bin/python3
# -*- coding: utf-8 -*-
import asyncio
import logging
import json
class ClientProtocol(asyncio.Protocol):
def __init__(self, loop, master):
self.Loop = loop
self.Master = master
self.Transport = None
self.Buffer = bytearray()
def connection_made(self, transport):
self.Transport = transport
def data_received(self, data):
self.Buffer += data
chunks = self.Buffer.split(b'\0')
if data[-1] == b'\0':
chunks = chunks[:-1]
self.Buffer = bytearray()
else:
self.Buffer = bytearray(chunks[-1])
chunks = chunks[:-1]
for chunk in chunks:
self.Master.OnReceive(chunk)
def connection_lost(self, exc):
self.Transport.close()
self.Transport = None
self.Master.OnDisconnect(exc)
def Send(self, data):
if self.Transport:
self.Transport.write(data)
class AsyncClient():
def __init__(self, loop, host, port, master):
self.Logger = logging.getLogger(__class__.__name__)
self.Loop = loop
self.Host = host
self.Port = port
self.Master = master
self.Protocol = None
self.SendLock = asyncio.Lock()
self.RecvFuture = None
async def Connect(self):
while True:
self.Logger.warn("Connecting...")
try:
_, self.Protocol = await self.Loop.create_connection(
lambda: ClientProtocol(self.Loop, self), host = self.Host, port = self.Port)
break
except:
await asyncio.sleep(1.0)
def OnReceive(self, data):
Obj = json.loads(data)
if "method" in Obj and Obj["method"] == "publish":
self.Master.OnPublish(Obj)
else:
if self.RecvFuture:
self.RecvFuture.set_result(Obj)
def OnDisconnect(self, exc):
self.Protocol = None
if self.RecvFuture:
self.RecvFuture.cancel()
self.Master.OnDisconnect(exc)
async def Send(self, obj):
if not self.Protocol:
return None
Data = json.dumps(obj, ensure_ascii = False, separators = (',', ':')).encode("UTF-8")
with (await self.SendLock):
if not self.Protocol:
return None
self.RecvFuture = asyncio.Future()
self.Protocol.Send(Data)
await self.RecvFuture
if self.RecvFuture.done():
Obj = self.RecvFuture.result()
else:
Obj = None
self.RecvFuture = None
return Obj