I want to send a PyGame Image over socket with Python 3.5 and I always got a bug.
I get the image, I pickle it, and then I send it.
In the client, I receive it, I unpickle it, and I show it.
But I've got an error:
fenetre.blit(img, (20, 30))
pygame.error: display Surface quit
Here is my code for the server (the one which sends the image):
pygame.init()
pygame.camera.init()
cam = pygame.camera.Camera("/dev/video0", (680, 480))
cam.start()
class Streaming(Thread):
def __init__(self):
Thread.__init__(self)
def run(self):
s = socket.socket()
s.bind(('192.168.1.158', 12801))
s.listen(1)
while True:
sc, info = s.accept()
print("Video client connected : "+str(info))
try:
while True:
image = cam.get_image()
str_img = pickle.dumps(image)
sc.send(str_img)
print(str_img)
print("Sending Image")
time.sleep(0.005)
except Exception as e:
print(str(e))
And the code for the client:
fenetre = pygame.display.set_mode((900, 900))
class Receiving(Thread):
def __init__(self):
Thread.__init__(self)
self.clock = pygame.time.Clock()
def run(self):
global fenetre
si = socket.socket()
si.connect(("192.168.1.158", 12801))
while True:
img = si.recv(4096)
img = pickle.loads(img)
fenetre.blit(img, (20, 30))
pygame.display.flip()
self.clock.tick(60)
Thanks in advance!
I made example which sends surface with current time.
(Pygame can't work with my camera).
It uses struc.pack()
to send image size always as 4 bytes before image is send. So client first receives 4 bytes and have image size. Then it can use this information to receive image.
Both use loops to send/receive all the time.
server.py
#!/usr/bin/env python
import pygame
from threading import Thread
import socket
import struct # to send `int` as `4 bytes`
import time # for test
# --- constants ---
ADDRESS = ("localhost", 12801)
SURFACE_SIZE = (640, 480)
WHITE = (255, 255, 255)
BLACK = ( 0, 0, 0)
GREEN = ( 0, 255, 0)
# --- classes ---
class Streaming(Thread):
def __init__(self):
Thread.__init__(self)
pygame.init()
#pygame.camera.init()
#self.cam = pygame.camera.Camera("/dev/video0", SURFACE_SIZE)
#self.cam.start()
# create surface to imitate camera image
self.image = pygame.Surface(SURFACE_SIZE)
self.image_rect = self.image.get_rect()
# create font to display text on surface
self.font = pygame.font.Font(None, 50)
def get_image(self):
# emulate cam.get_image()
# get current time as string
current_time = time.strftime('%H:%M:%S.%s')
# render surface with text (and center it)
text = self.font.render(current_time, True, BLACK, GREEN)
text_rect = text.get_rect(center=self.image_rect.center)
# clear image and put new text
self.image.fill(WHITE)
self.image.blit(text, text_rect)
return self.image
def run(self):
s = socket.socket()
# solution for: "socket.error: [Errno 98] Address already in use"
s.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, 1)
s.bind(ADDRESS)
s.listen(1)
print("Wait for connection")
try:
sc, info = s.accept()
print("Video client connected:", info)
while True:
# get image surface
#image = self.cam.get_image()
image = self.get_image()
# convert surface to string
img_str = pygame.image.tostring(image, 'RGB')
print('len:', len(img_str))
# send string size
len_str = struct.pack('!i', len(img_str))
sc.send(len_str)
# send string image
sc.send(img_str)
# wait
time.sleep(0.5)
except Exception as e:
print(e)
finally:
# exit
print("Closing socket and exit")
sc.close()
s.close()
pygame.quit()
# --- main ---
Streaming().run()
client.py
#!/usr/bin/env python
import pygame
from threading import Thread
import socket
import struct
# --- constants ---
ADDRESS = ("localhost", 12801)
SURFACE_SIZE = (640, 480)
# --- classes ---
class Receiving(Thread):
def __init__(self):
Thread.__init__(self)
pygame.init()
self.screen = pygame.display.set_mode((800, 600))
self.screen_rect = self.screen.get_rect()
self.clock = pygame.time.Clock()
def run(self):
s = socket.socket()
s.connect(ADDRESS)
try:
running = True
while running:
# receive size
len_str = s.recv(4)
size = struct.unpack('!i', len_str)[0]
print('size:', size)
# receive string
img_str = b''
while size > 0:
if size >= 4096:
data = s.recv(4096)
else:
data = s.recv(size)
if not data:
break
size -= len(data)
img_str += data
print('len:', len(img_str))
# convert string to surface
image = pygame.image.fromstring(img_str, SURFACE_SIZE, 'RGB')
image_rect = image.get_rect(center=self.screen_rect.center)
# blit
self.screen.blit(image, image_rect)
pygame.display.flip()
#self.clock.tick(30)
# wait for ESC or close window
for event in pygame.event.get():
if event.type == pygame.QUIT:
running = False
elif event.type == pygame.KEYDOWN:
if event.key == pygame.K_ESCAPE:
running = False
except Exception as e:
print(e)
finally:
# exit
print("Closing socket and exit")
s.close()
pygame.quit()
# --- main ---
Receiving().run()