doubly interlace video and make server asynchronous
This commit is contained in:
parent
34b15e9543
commit
3e7ba62d3a
17
client.py
17
client.py
@ -5,7 +5,7 @@ import uuid
|
|||||||
|
|
||||||
uuid = uuid.uuid4()
|
uuid = uuid.uuid4()
|
||||||
|
|
||||||
from common import StdPacket, InterlacedPacket
|
from common import StdPacket, InterlacedPacket, DoublyInterlacedPacket
|
||||||
|
|
||||||
UDP_IP = "127.0.0.1"
|
UDP_IP = "127.0.0.1"
|
||||||
UDP_PORT = 5005
|
UDP_PORT = 5005
|
||||||
@ -48,6 +48,19 @@ def breakdown_image_interlaced(frame):
|
|||||||
pkt = InterlacedPacket(uuid, j, i, True, frame[i + 1:i + 32:2, j:j + 16])
|
pkt = InterlacedPacket(uuid, j, i, True, frame[i + 1:i + 32:2, j:j + 16])
|
||||||
send_packet(sock, pkt.to_bytestr())
|
send_packet(sock, pkt.to_bytestr())
|
||||||
|
|
||||||
|
def breakdown_image_dint(frame):
|
||||||
|
(cols, rows, colors) = frame.shape
|
||||||
|
# break the array into 16x32 chunks. we'll split those further into odd and even rows
|
||||||
|
# and send each as UDP packets. this should make packet loss less obvious
|
||||||
|
for l in range(0, 4):
|
||||||
|
for i in range(0, cols, 32):
|
||||||
|
for j in range(0, rows, 32):
|
||||||
|
# print("Sending frame segment (%d, %d)", i, j)
|
||||||
|
i_even = l % 2 == 0
|
||||||
|
j_even = l >= 2
|
||||||
|
pkt = DoublyInterlacedPacket(uuid, j, i, j_even, i_even, frame[i + i_even:i + 32:2, j + j_even:j + 32:2])
|
||||||
|
send_packet(sock, pkt.to_bytestr())
|
||||||
|
|
||||||
while True:
|
while True:
|
||||||
# Capture frame-by-frame
|
# Capture frame-by-frame
|
||||||
ret, frame = cap.read()
|
ret, frame = cap.read()
|
||||||
@ -57,7 +70,7 @@ while True:
|
|||||||
print("Can't receive frame (stream end?). Exiting ...")
|
print("Can't receive frame (stream end?). Exiting ...")
|
||||||
break
|
break
|
||||||
|
|
||||||
breakdown_image_interlaced(frame)
|
breakdown_image_dint(frame)
|
||||||
|
|
||||||
# Release the capture and close all windows
|
# Release the capture and close all windows
|
||||||
cap.release()
|
cap.release()
|
31
common.py
31
common.py
@ -53,3 +53,34 @@ def from_bytes_int(b: bytes) -> InterlacedPacket:
|
|||||||
array = np.frombuffer(b[28:], np.uint8).reshape(16, 16, 3)
|
array = np.frombuffer(b[28:], np.uint8).reshape(16, 16, 3)
|
||||||
|
|
||||||
return InterlacedPacket(uuid, x, y, even, array)
|
return InterlacedPacket(uuid, x, y, even, array)
|
||||||
|
|
||||||
|
class DoublyInterlacedPacket:
|
||||||
|
size = 16 + 4 + 4 + 4 + 768
|
||||||
|
def __init__(self, uuid: UUID, x: int, y: int, even_x: bool, even_y: bool, array: np.ndarray):
|
||||||
|
self.uuid = uuid
|
||||||
|
self.x = x
|
||||||
|
self.y = y
|
||||||
|
self.even_x = even_x
|
||||||
|
self.even_y = even_y
|
||||||
|
self.array = array
|
||||||
|
|
||||||
|
def to_bytestr(self) -> bytes:
|
||||||
|
bytestr = b""
|
||||||
|
bytestr += self.uuid.bytes
|
||||||
|
bytestr += self.x.to_bytes(length=4, signed=False)
|
||||||
|
bytestr += self.y.to_bytes(length=4, signed=False)
|
||||||
|
bytestr += self.even_x.to_bytes(length=2)
|
||||||
|
bytestr += self.even_y.to_bytes(length=2)
|
||||||
|
bytestr += self.array.tobytes()
|
||||||
|
return bytestr
|
||||||
|
|
||||||
|
|
||||||
|
def from_bytes_dint(b: bytes) -> DoublyInterlacedPacket:
|
||||||
|
uuid = UUID(bytes=b[0:16])
|
||||||
|
x = int.from_bytes(b[16:20], signed=False)
|
||||||
|
y = int.from_bytes(b[20:24], signed=False)
|
||||||
|
even_x = bool.from_bytes(b[24:26])
|
||||||
|
even_y = bool.from_bytes(b[26:28])
|
||||||
|
array = np.frombuffer(b[28:], np.uint8).reshape(16, 16, 3)
|
||||||
|
|
||||||
|
return DoublyInterlacedPacket(uuid, x, y, even_x, even_y, array)
|
62
server.py
62
server.py
@ -1,21 +1,17 @@
|
|||||||
from datetime import datetime
|
|
||||||
from typing import Dict
|
from typing import Dict
|
||||||
|
|
||||||
import cv2
|
import cv2
|
||||||
import socket
|
import socket
|
||||||
import numpy as np
|
import numpy as np
|
||||||
|
from datetime import datetime
|
||||||
|
import asyncio
|
||||||
|
|
||||||
from common import InterlacedPacket, from_bytes_int
|
from common import DoublyInterlacedPacket, from_bytes_dint
|
||||||
|
|
||||||
# bind any IP address
|
# bind any IP address
|
||||||
UDP_IP = ""
|
UDP_IP = ""
|
||||||
UDP_PORT = 5005
|
UDP_PORT = 5005
|
||||||
|
|
||||||
sock = socket.socket(socket.AF_INET, # Internet
|
|
||||||
socket.SOCK_DGRAM) # UDP
|
|
||||||
sock.setblocking(False)
|
|
||||||
sock.bind((UDP_IP, UDP_PORT))
|
|
||||||
|
|
||||||
HEIGHT = 480
|
HEIGHT = 480
|
||||||
WIDTH = 640
|
WIDTH = 640
|
||||||
|
|
||||||
@ -24,11 +20,12 @@ class Client:
|
|||||||
self.last_updated = datetime.now()
|
self.last_updated = datetime.now()
|
||||||
self.frame = np.ndarray((HEIGHT, WIDTH, 3), dtype=np.uint8)
|
self.frame = np.ndarray((HEIGHT, WIDTH, 3), dtype=np.uint8)
|
||||||
|
|
||||||
def update(self, pkt: InterlacedPacket):
|
def update(self, pkt: DoublyInterlacedPacket):
|
||||||
if pkt.even:
|
x = pkt.x
|
||||||
self.frame[y + 1:y + 32:2, x:x + 16] = arr
|
y = pkt.y
|
||||||
else:
|
arr = pkt.array
|
||||||
self.frame[y:y + 32:2, x:x + 16] = arr
|
self.frame[y + pkt.even_y:y + 32:2, x + pkt.even_x:x + 32:2] = arr
|
||||||
|
|
||||||
self.last_updated = datetime.now()
|
self.last_updated = datetime.now()
|
||||||
|
|
||||||
def latency(self) -> float:
|
def latency(self) -> float:
|
||||||
@ -40,19 +37,17 @@ class Client:
|
|||||||
|
|
||||||
frames: Dict[str, Client] = {}
|
frames: Dict[str, Client] = {}
|
||||||
|
|
||||||
while True:
|
async def read_packet():
|
||||||
|
while True:
|
||||||
|
for i in range(0, 1200):
|
||||||
# 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 repeats in range(10000):
|
|
||||||
try:
|
try:
|
||||||
data, addr = sock.recvfrom(InterlacedPacket.size) # buffer size is 768 bytes
|
data, addr = sock.recvfrom(DoublyInterlacedPacket.size) # buffer size is 768 bytes
|
||||||
# print("received packet from", addr)
|
# print("received packet from", addr)
|
||||||
if data:
|
if data:
|
||||||
pkt = from_bytes_int(data)
|
pkt = from_bytes_dint(data)
|
||||||
|
|
||||||
uuid = str(pkt.uuid)
|
uuid = str(pkt.uuid)
|
||||||
x = pkt.x
|
|
||||||
y = pkt.y
|
|
||||||
arr = pkt.array
|
|
||||||
|
|
||||||
if uuid not in frames.keys():
|
if uuid not in frames.keys():
|
||||||
print("New client acquired, naming %s", uuid)
|
print("New client acquired, naming %s", uuid)
|
||||||
@ -63,6 +58,10 @@ while True:
|
|||||||
except BlockingIOError:
|
except BlockingIOError:
|
||||||
pass
|
pass
|
||||||
|
|
||||||
|
await asyncio.sleep(0.001)
|
||||||
|
|
||||||
|
async def show_frames():
|
||||||
|
while True:
|
||||||
# Display the resulting frame
|
# Display the resulting frame
|
||||||
for id in list(frames.keys()):
|
for id in list(frames.keys()):
|
||||||
if frames[id].latency() >= 5:
|
if frames[id].latency() >= 5:
|
||||||
@ -72,9 +71,24 @@ while True:
|
|||||||
else:
|
else:
|
||||||
cv2.imshow(id, frames[id].read())
|
cv2.imshow(id, frames[id].read())
|
||||||
|
|
||||||
# Break the loop if 'q' key is pressed
|
cv2.waitKey(1)
|
||||||
if cv2.waitKey(1) == ord('q'):
|
await asyncio.sleep(0.01)
|
||||||
break
|
|
||||||
|
|
||||||
# Release the capture and close all windows
|
if __name__ == "__main__":
|
||||||
cv2.destroyAllWindows()
|
sock = socket.socket(socket.AF_INET, # Internet
|
||||||
|
socket.SOCK_DGRAM) # UDP
|
||||||
|
sock.setblocking(False)
|
||||||
|
sock.bind((UDP_IP, UDP_PORT))
|
||||||
|
|
||||||
|
loop = asyncio.new_event_loop()
|
||||||
|
asyncio.set_event_loop(loop)
|
||||||
|
loop.create_task(read_packet())
|
||||||
|
loop.create_task(show_frames())
|
||||||
|
try:
|
||||||
|
loop.run_forever()
|
||||||
|
finally:
|
||||||
|
loop.run_until_complete(loop.shutdown_asyncgens())
|
||||||
|
loop.close()
|
||||||
|
|
||||||
|
# Release the capture and close all windows
|
||||||
|
cv2.destroyAllWindows()
|
Loading…
x
Reference in New Issue
Block a user