make I/O non-blocking and handle dropped connections
This commit is contained in:
parent
275e5eec3e
commit
34b15e9543
48
server.py
48
server.py
@ -1,3 +1,4 @@
|
|||||||
|
from datetime import datetime
|
||||||
from typing import Dict
|
from typing import Dict
|
||||||
|
|
||||||
import cv2
|
import cv2
|
||||||
@ -12,19 +13,40 @@ UDP_PORT = 5005
|
|||||||
|
|
||||||
sock = socket.socket(socket.AF_INET, # Internet
|
sock = socket.socket(socket.AF_INET, # Internet
|
||||||
socket.SOCK_DGRAM) # UDP
|
socket.SOCK_DGRAM) # UDP
|
||||||
|
sock.setblocking(False)
|
||||||
sock.bind((UDP_IP, UDP_PORT))
|
sock.bind((UDP_IP, UDP_PORT))
|
||||||
|
|
||||||
HEIGHT = 480
|
HEIGHT = 480
|
||||||
WIDTH = 640
|
WIDTH = 640
|
||||||
|
|
||||||
frames: Dict[str, np.ndarray] = {}
|
class Client:
|
||||||
|
def __init__(self):
|
||||||
|
self.last_updated = datetime.now()
|
||||||
|
self.frame = np.ndarray((HEIGHT, WIDTH, 3), dtype=np.uint8)
|
||||||
|
|
||||||
|
def update(self, pkt: InterlacedPacket):
|
||||||
|
if pkt.even:
|
||||||
|
self.frame[y + 1:y + 32:2, x:x + 16] = arr
|
||||||
|
else:
|
||||||
|
self.frame[y:y + 32:2, x:x + 16] = arr
|
||||||
|
self.last_updated = datetime.now()
|
||||||
|
|
||||||
|
def latency(self) -> float:
|
||||||
|
return (datetime.now() - self.last_updated).total_seconds()
|
||||||
|
|
||||||
|
def read(self) -> np.ndarray:
|
||||||
|
return self.frame
|
||||||
|
|
||||||
|
|
||||||
|
frames: Dict[str, Client] = {}
|
||||||
|
|
||||||
while True:
|
while True:
|
||||||
# break the array down into 16-bit chunks, then transmit them as UDP packets
|
# break the array down into 16-bit chunks, then transmit them as UDP packets
|
||||||
for i in range(0, HEIGHT, 16):
|
for repeats in range(10000):
|
||||||
for j in range(0, WIDTH, 16):
|
try:
|
||||||
data, addr = sock.recvfrom(InterlacedPacket.size) # buffer size is 768 bytes
|
data, addr = sock.recvfrom(InterlacedPacket.size) # buffer size is 768 bytes
|
||||||
# print("received packet from", addr)
|
# print("received packet from", addr)
|
||||||
|
if data:
|
||||||
pkt = from_bytes_int(data)
|
pkt = from_bytes_int(data)
|
||||||
|
|
||||||
uuid = str(pkt.uuid)
|
uuid = str(pkt.uuid)
|
||||||
@ -33,16 +55,22 @@ while True:
|
|||||||
arr = pkt.array
|
arr = pkt.array
|
||||||
|
|
||||||
if uuid not in frames.keys():
|
if uuid not in frames.keys():
|
||||||
frames[uuid] = frame = np.ndarray((HEIGHT, WIDTH, 3), dtype=np.uint8)
|
print("New client acquired, naming %s", uuid)
|
||||||
|
frames[uuid] = Client()
|
||||||
|
|
||||||
if pkt.even:
|
frames[uuid].update(pkt)
|
||||||
frames[uuid][y+1:y+32:2, x:x+16] = arr
|
|
||||||
else:
|
except BlockingIOError:
|
||||||
frames[uuid][y:y + 32:2, x:x + 16] = arr
|
pass
|
||||||
|
|
||||||
# Display the resulting frame
|
# Display the resulting frame
|
||||||
for id in frames.keys():
|
for id in list(frames.keys()):
|
||||||
cv2.imshow(id, frames[id])
|
if frames[id].latency() >= 5:
|
||||||
|
print("Client likely lost connection, dropping %s", id)
|
||||||
|
cv2.destroyWindow(id)
|
||||||
|
frames.pop(id)
|
||||||
|
else:
|
||||||
|
cv2.imshow(id, frames[id].read())
|
||||||
|
|
||||||
# Break the loop if 'q' key is pressed
|
# Break the loop if 'q' key is pressed
|
||||||
if cv2.waitKey(1) == ord('q'):
|
if cv2.waitKey(1) == ord('q'):
|
||||||
|
Loading…
x
Reference in New Issue
Block a user