2011-12-15 97 views
4

簡單的異步http客戶端,由於沒有可用的站點而長時間掛起。asyncore timeout

例如在網站www.evtur.ru等待很長時間,十分鐘或更長時間。

我無法找到如何最小化超時的方法,例如有可能在5秒內完成超時嗎?

# coding=utf-8 
import asyncore 
import string, socket 
import StringIO 
import mimetools, urlparse 

class AsyncHTTP(asyncore.dispatcher): 
    # HTTP requestor 

    def __init__(self, uri): 
     asyncore.dispatcher.__init__(self) 

     self.uri = uri 


     # turn the uri into a valid request 
     scheme, host, path, params, query, fragment = urlparse.urlparse(uri) 
     assert scheme == "http", "only supports HTTP requests" 
     try: 
      host, port = string.split(host, ":", 1) 
      port = int(port) 
     except (TypeError, ValueError): 
      port = 80 # default port 
     if not path: 
      path = "/" 
     if params: 
      path = path + ";" + params 
     if query: 
      path = path + "?" + query 

     self.request = "GET %s HTTP/1.0\r\nHost: %s\r\n\r\n" % (path, host) 

     self.host = host 
     self.port = port 

     self.status = None 
     self.header = None 
     self.http_code = None 
     self.data = "" 

     # get things going! 
     self.create_socket(socket.AF_INET, socket.SOCK_STREAM) 
     #self.connect((host, port)) 
     #return 
     try: 
      self.connect((host, port)) 
     except Exception,e: 
      self.close() 
      self.handle_connect_expt(e) 

    def handle_connect(self): 
     self.send(self.request) 

    def handle_expt(self): 
     print "handle_expt error!" 
     self.close() 

    def handle_error(self): 
     print "handle_error error!" 
     self.close() 

    def handle_connect_expt(self,expt): 
     print "connection error:",expt 

    def handle_code(self):   
     print self.host," : ","recv http code: ",self.http_code 


    def handle_read(self): 
     data = self.recv(2048) 
     #print data 
     if not self.header: 
      self.data = self.data + data 
      try: 
       i = string.index(self.data, "\r\n\r\n") 
      except ValueError: 
       return # continue 
      else: 
       # parse header 
       fp = StringIO.StringIO(self.data[:i+4]) 
       # status line is "HTTP/version status message" 
       status = fp.readline() 
       self.status = string.split(status, " ", 2) 
       self.http_code = self.status[1] 
       self.handle_code()  

       # followed by a rfc822-style message header 
       self.header = mimetools.Message(fp) 
       # followed by a newline, and the payload (if any) 
       data = self.data[i+4:] 
       self.data = "" 
       #header recived 
       #self.close() 


    def handle_close(self): 
     self.close() 



c = AsyncHTTP('http://www.python.org') 
c = AsyncHTTP('http://www.evtur.ru') 
asyncore.loop(timeout=0.05) 
+0

我認爲你的代碼需要一些格式。你在混合製表符和空格嗎?如果是這樣,這是一個糟糕的想法。 http://www.python.org/dev/peps/pep-0008/ – Wilduck

+0

正如在[這個答案](http://stackoverflow.com/a/8513221/984421)中解釋你對這個問題的其他問題,問題不在於你的代碼,而在於你的系統設置。 – ekhumoro

+0

[asyncore python hangs]可能的重複(http://stackoverflow.com/questions/8504518/asyncore-python-hangs) – ekhumoro

回答

0

您必須在調度程序套接字上設置超時。 調用

self.create_socket(socket.AF_INET, socket.SOCK_STREAM) 

後,您可以撥打

self.socket.settimeout(nn) 

設置超時的插座上。您也可以像設置其他套接字一樣設置其他套接字選項。

1
self.socket.settimeout(nn) 

這似乎不適合我。

但是,asyncore.loop()具有稱爲count的參數。這裏是僞代碼什麼asyncore.loop()這樣做:

for i in range(count): 
    ... 
    select(... , timeout) 
    ... 

所以,如果你想5秒,你需要做的事:

asyncore.loop(timeout=1, count=5) 

但不建議在這個確切的方式工作。請注意,如果有「事件」,則可能有超過5個「計數」。我使用以下代碼:

start = int(time.time()) 

while True: 
    asyncore.loop(timeout=1, count=5) 
    print "LOOP : %d enqueued, waiting to finish" % len(asyncore.socket_map) 

    if len(asyncore.socket_map) == 0 : 
     break 

    if int(time.time()) - start > timeout : 
     print "LOOP : Timeout" 
     break