Почему apachebench постоянно повторяет свои запросы в сеансе HTTP 1.0 через одно и то же соединение? - PullRequest
1 голос
/ 31 января 2012

Я сделал простой скрипт на Python с видом HTTP-сервера:

import SocketServer

response  = b'HTTP/1.0 200 OK\r\nDate: Mon, 1 Jan 1996 01:01:01 GMT\r\n'
response += b'Content-Type: text/plain\r\nContent-Length: 14\r\n\r\n'
response += b'Hello, world!\n'

class MyTCPHandler(SocketServer.StreamRequestHandler):

    def handle(self):
        while True:
            line = self.rfile.readline().strip()
            print "{} {}:{} wrote: {}".format(self.connection, self.client_address[0], self.client_address[1], line)
            if not line:
                self.wfile.write(response)        
                print 'Sent hello world'
                #break

if __name__ == "__main__":
    HOST, PORT = "localhost", 9999

    server = SocketServer.TCPServer((HOST, PORT), MyTCPHandler)
    server.serve_forever()

Теперь я запускаю ab для выполнения 1 запроса через 1 соединение: ab -n1 -c1 http://127.0.0.1:9999/

Когда я не закрываю соединение после ответа, ожидая, что клиент закроет сокет на своем конце, как только он получит мой ответ, происходит что-то странное.

Ab повторяет все время, отправляя один и тот же запрос через одно и то же соединение:

~$ python2.7 ./socket_server.py
<socket._socketobject object at 0x1068ca600> 127.0.0.1:64626 wrote: GET / HTTP/1.0
<socket._socketobject object at 0x1068ca600> 127.0.0.1:64626 wrote: Host: 127.0.0.1:9999
<socket._socketobject object at 0x1068ca600> 127.0.0.1:64626 wrote: User-Agent: ApacheBench/2.3
<socket._socketobject object at 0x1068ca600> 127.0.0.1:64626 wrote: Accept: */*
<socket._socketobject object at 0x1068ca600> 127.0.0.1:64626 wrote:
Sent hello world
<socket._socketobject object at 0x1068ca600> 127.0.0.1:64626 wrote: GET / HTTP/1.0
<socket._socketobject object at 0x1068ca600> 127.0.0.1:64626 wrote: Host: 127.0.0.1:9999
<socket._socketobject object at 0x1068ca600> 127.0.0.1:64626 wrote: User-Agent: ApacheBench/2.3
<socket._socketobject object at 0x1068ca600> 127.0.0.1:64626 wrote: Accept: */*
<socket._socketobject object at 0x1068ca600> 127.0.0.1:64626 wrote:
Sent hello world
<socket._socketobject object at 0x1068ca600> 127.0.0.1:64626 wrote: GET / HTTP/1.0
<socket._socketobject object at 0x1068ca600> 127.0.0.1:64626 wrote: Host: 127.0.0.1:9999
<socket._socketobject object at 0x1068ca600> 127.0.0.1:64626 wrote: User-Agent: ApacheBench/2.3
<socket._socketobject object at 0x1068ca600> 127.0.0.1:64626 wrote: Accept: */*
<socket._socketobject object at 0x1068ca600> 127.0.0.1:64626 wrote:
Sent hello world

... и он бесконечно повторяется через одно и то же гнездо.

Я попытался прочитать необработанные данные из сокета:

import SocketServer

class MyTCPHandler2(SocketServer.BaseRequestHandler):

    def handle(self):
        while True:
            self.data = self.request.recv(1024).strip()
            print "{} wrote:".format(self.client_address[0])
            print self.data

if __name__ == "__main__":
    HOST, PORT = "localhost", 9999

    server = SocketServer.TCPServer((HOST, PORT), MyTCPHandler2)
    server.serve_forever()

Ab по-прежнему даже не пытается дождаться ответа, он устремляется с тоннами запросов в одном соединении.

Почему? Это какое-то соединение, повторно используемое в HTTP 1.0, которое мне не хватает? Должен ли я всегда закрывать соединение независимо от данных, следующих после первой пары \ r \ n \ r \ n?

Какими будут правильные параметры httperf, чтобы попытаться воспроизвести то же поведение?

...