如何限制套接字连接数并在客户端触发超时(Python)

b_p*_*kes 5 python sockets

如何设置服务器套接字一次可以接受的连接数限制?我希望能够设置最大连接数,然后一旦达到该限制,客户端的任何进一步连接尝试都将导致超时。到目前为止,我已经为服务器尝试过类似的操作:

sock = socket.socket()
sock.setblocking(0)
sock.bind(address)
sock.listen(0)

connections = []

while True:
    readable, writable, exceptional = select.select([sock], [], [])

    if readable and len(connections) < MAX_CONNECTIONS:
        connection, client_address = s.accept()
        connections.append(connection)
        # Process connection asynchronously
Run Code Online (Sandbox Code Playgroud)

对于客户:

try:
    sock = socket.create_connection(self.address, timeout=TIMEOUT)
    sock.settimeout(None)
    print "Established connection."
except socket.error as err:
    print >> sys.stderr, "Socket connection error: " + str(err)
    sys.exit(1)

# If connection successful, do stuff
Run Code Online (Sandbox Code Playgroud)

由于程序其余部分的结构,我选择在服务器上使用非阻塞套接字,并且我不想更改它。现在,即使达到限制并且服务器停止接受客户端,客户端也能够连接到服务器。我该如何解决这个问题?谢谢。

Leo*_*nes 2

我相信这里可能存在一些轻微的误解select()。根据联机帮助页,select()返回“为某种 IO 操作做好准备”的文件描述符,其中就绪意味着“可以在不阻塞的情况下执行相应的 IO 操作”。侦听套接字上相应的 IO 操作是accept(),只有当操作系统已经为您完成完整的 TCP 握手时,该操作才能在不阻塞的情况下执行,否则您可能会阻塞等待客户端的最终 ACK。

这意味着只要侦听套接字打开,即使应用程序不处理连接,操作系统也会接受连接。如果您想在设定数量后拒绝连接,基本上有两种选择:

  • 只需接受并在接受后直接关闭。
  • 达到限制后关闭监听套接字,完成后重新打开。

第二个选项更复杂,需要使用该SO_REUSEADDR选项,这可能不适合您的情况。它也可能无法在所有操作系统上工作,尽管它似乎在 Linux 上可靠地工作。

这是第二个解决方案的快速草图(因为第一个解决方案非常简单)。

def get_listening_socket():
    sock = socket.socket()
    sock.setblocking(0)
    sock.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, 1)
    sock.bind(('0.0.0.0', 5555))
    sock.listen(0)
    return sock

sock = get_listening_socket()

LIMIT = 1
conns = {sock}
while True:
    readable, writable, exceptional = select.select(conns, [], [])
    if sock in readable: # new connection on the listening socket
        conn, caddr = sock.accept()
        conns.add(conn)
        if len(conns) > LIMIT: # ">" because "sock" is also in there
            conns.remove(sock)
            sock.close()
    else: # reading from an established connection
        for c in readable:
            buf = c.recv(4096)
            if not buf:
                conns.remove(c)
                sock = get_listening_socket()
                conns.add(sock)
            else:
                print("received: %s" % buf)
Run Code Online (Sandbox Code Playgroud)

然而,您可能需要重新考虑为什么要这样做。如果只是为了节省服务器上的一些内存,那么您可能会过度优化,应该考虑使用 syn-cookies。