Было несколько вещей, которые я должен был сделать, чтобы сделать эту работу.
Во-первых, Сломанный канал может возникнуть в двух разных местах в коде сервера. Первый - когда подключается новый клиент, а сервер пытается отправить сообщение «Новый клиент подключен» всем клиентам, а второй - когда существующий клиент отправляет сообщение. Итак, нам нужно обработать исключение в обоих местах.
Итак, мы помещаем try / кроме обоих блоков. где он говорит, что если c! = conn.
Теперь о том, как обрабатывать исключение.
Как я сначала подумал, простое удаление клиента c из списка клиентов приведет к работают, но для l oop, для c в клиентах будет выдаваться ошибка времени выполнения, когда мы пытаемся изменить набор клиентов во время итерации.
Я пробовал разные методы, чтобы обойти эту проблему, но это это самый эффективный метод работы, который я получил.
Я изменил клиентов из set () в пустой список []
Затем я изменил clients.add на clients.append
Затем я изменил для l oop на c в диапазоне (len (клиенты)) и использовал клиентов [c] для доступа к клиентам.
Но когда я попробовал это, я увидел, что если утверждение, что клиенты [c]! = conn могут выбросить индекс списка из-за ошибки ошибки, если программа попытается go поверх несуществующего клиента после удаления. Поэтому я тоже поместил его в блок try / исключением и разрешил программе продолжить работу при исключении.
for c in range(len(clients)):
try:
if clients[c] != conn:
try:
message = connection_message.encode(FORMAT)
msg_length = len(message)
send_length = str(msg_length).encode(FORMAT)
send_length += b' ' * (HEADER - len(send_length))
clients[c].sendall(send_length)
clients[c].sendall(message)
except:
clients.remove(clients[c])
except:
continue
Последняя проблема заключалась в том, что даже после удаления клиента поток все еще жив, поэтому активный поток count возвращает больше, чем количество подключенных клиентов. Таким образом, вместо вывода числа активных соединений в виде числа активных потоков - 1, я печатаю len (клиентов) + 1, + 1, поскольку при подключении нового клиента эта строка печатается перед добавлением клиента в список.
print(f"[ACTIVE CONNECTIONS] {len(clients) + 1}")
Итак, вся программа теперь выглядит так:
import socket
import threading
HEADER = 64
PORT = 5050
SERVER = socket.gethostbyname(socket.gethostname())
ADDR = (SERVER, PORT)
FORMAT = 'utf-8'
DISCONNECT_MESSAGE = "!DISCONNECT"
server = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
server.bind(ADDR)
clients = []
clients_lock = threading.Lock()
def handle_client(conn, addr):
name = conn.recv(HEADER).decode(FORMAT)
if name:
name = int(name)
msg_name = conn.recv(name).decode(FORMAT)
print(f"[NEW CONNECTION] {msg_name} connected.")
connection_message = f"{msg_name} connected."
with clients_lock:
for c in range(len(clients)):
try:
if clients[c] != conn:
try:
message = connection_message.encode(FORMAT)
msg_length = len(message)
send_length = str(msg_length).encode(FORMAT)
send_length += b' ' * (HEADER - len(send_length))
clients[c].sendall(send_length)
clients[c].sendall(message)
except:
clients.remove(clients[c])
except:
continue
with clients_lock:
clients.append(conn)
connected = True
try:
while connected:
msg_length = conn.recv(HEADER).decode(FORMAT)
if msg_length:
msg_length = int(msg_length)
msg1 = conn.recv(msg_length).decode(FORMAT)
msg = f"{msg_name}: {msg1}"
if msg1 == DISCONNECT_MESSAGE:
connected = False
print(f"{msg}")
with clients_lock:
for c in range(len(clients)):
try:
if clients[c] != conn:
try:
message = msg.encode(FORMAT)
msg_length = len(message)
send_length = str(msg_length).encode(FORMAT)
send_length += b' ' * (HEADER - len(send_length))
clients[c].sendall(send_length)
clients[c].sendall(message)
except:
clients.remove(clients[c])
except:
continue
msg = f"You: {msg1}"
message = msg.encode(FORMAT)
msg_length = len(message)
send_length = str(msg_length).encode(FORMAT)
send_length += b' ' * (HEADER - len(send_length))
conn.send(send_length)
conn.send(message)
finally:
with clients_lock:
clients.remove(conn)
conn.close()
def start():
server.listen()
print(f"[LISTENING] Server is listening on {SERVER}")
while True:
conn, addr = server.accept()
thread = threading.Thread(target=handle_client, args=(conn, addr))
thread.daemon = True
thread.start()
print(f"[ACTIVE CONNECTIONS] {len(clients) + 1}")
print("[STARTING] server is starting...")
start()