# Python rough implementation of a C# TCP client import socket import time from threading import Lock from errors import ReadCancelledError from utils import BinaryWriter class TcpClient: def __init__(self): self.connected = False self.socket = socket.socket(socket.AF_INET, socket.SOCK_STREAM) # Support for multi-threading advantages and safety self.cancelled = False # Has the read operation been cancelled? self.delay = 0.1 # Read delay when there was no data available self.lock = Lock() def connect(self, ip, port): """Connects to the specified IP and port number""" self.socket.connect((ip, port)) self.connected = True def close(self): """Closes the connection""" self.socket.close() self.connected = False self.socket.setblocking(True) def write(self, data): """Writes (sends) the specified bytes to the connected peer""" # Ensure that only one thread can send data at once with self.lock: # Set blocking so it doesn't error self.socket.setblocking(True) self.socket.sendall(data) def read(self, buffer_size): """Reads (receives) the specified bytes from the connected peer""" # Ensure that only one thread can receive data at once with self.lock: # Ensure it is not cancelled at first, so we can enter the loop self.cancelled = False # Set non-blocking so it can be cancelled self.socket.setblocking(False) with BinaryWriter() as writer: while writer.written_count < buffer_size: # Only do cancel if no data was read yet # Otherwise, carry on reading and finish if self.cancelled and writer.written_count == 0: raise ReadCancelledError() try: # When receiving from the socket, we may not receive all the data at once # This is why we need to keep checking to make sure that we receive it all left_count = buffer_size - writer.written_count partial = self.socket.recv(left_count) writer.write(partial) except BlockingIOError: # There was no data available for us to read. Sleep a bit time.sleep(self.delay) # If everything went fine, return the read bytes return writer.get_bytes() def cancel_read(self): """Cancels the read operation IF it hasn't yet started, raising a ReadCancelledError""" self.cancelled = True