home *** CD-ROM | disk | FTP | other *** search
/ GameStar 2006 January / Gamestar_80_2006-01_dvd.iso / Dema / Civilization4 / data1.cab / Civ4DemoComponent / Assets / Python / System / httplib.pyc (.txt) < prev    next >
Encoding:
Python Compiled Bytecode  |  2005-11-09  |  35.0 KB  |  1,318 lines

  1. # Source Generated with Decompyle++
  2. # File: in.pyc (Python 2.4)
  3.  
  4. '''HTTP/1.1 client library
  5.  
  6. <intro stuff goes here>
  7. <other stuff, too>
  8.  
  9. HTTPConnection go through a number of "states", which defines when a client
  10. may legally make another request or fetch the response for a particular
  11. request. This diagram details these state transitions:
  12.  
  13.     (null)
  14.       |
  15.       | HTTPConnection()
  16.       v
  17.     Idle
  18.       |
  19.       | putrequest()
  20.       v
  21.     Request-started
  22.       |
  23.       | ( putheader() )*  endheaders()
  24.       v
  25.     Request-sent
  26.       |
  27.       | response = getresponse()
  28.       v
  29.     Unread-response   [Response-headers-read]
  30.       |\\____________________
  31.       |                     |
  32.       | response.read()     | putrequest()
  33.       v                     v
  34.     Idle                  Req-started-unread-response
  35.                      ______/|
  36.                    /        |
  37.    response.read() |        | ( putheader() )*  endheaders()
  38.                    v        v
  39.        Request-started    Req-sent-unread-response
  40.                             |
  41.                             | response.read()
  42.                             v
  43.                           Request-sent
  44.  
  45. This diagram presents the following rules:
  46.   -- a second request may not be started until {response-headers-read}
  47.   -- a response [object] cannot be retrieved until {request-sent}
  48.   -- there is no differentiation between an unread response body and a
  49.      partially read response body
  50.  
  51. Note: this enforcement is applied by the HTTPConnection class. The
  52.       HTTPResponse class does not enforce this state machine, which
  53.       implies sophisticated clients may accelerate the request/response
  54.       pipeline. Caution should be taken, though: accelerating the states
  55.       beyond the above pattern may imply knowledge of the server\'s
  56.       connection-close behavior for certain requests. For example, it
  57.       is impossible to tell whether the server will close the connection
  58.       UNTIL the response headers have been read; this means that further
  59.       requests cannot be placed into the pipeline until it is known that
  60.       the server will NOT be closing the connection.
  61.  
  62. Logical State                  __state            __response
  63. -------------                  -------            ----------
  64. Idle                           _CS_IDLE           None
  65. Request-started                _CS_REQ_STARTED    None
  66. Request-sent                   _CS_REQ_SENT       None
  67. Unread-response                _CS_IDLE           <response_class>
  68. Req-started-unread-response    _CS_REQ_STARTED    <response_class>
  69. Req-sent-unread-response       _CS_REQ_SENT       <response_class>
  70. '''
  71. import errno
  72. import mimetools
  73. import socket
  74. from urlparse import urlsplit
  75.  
  76. try:
  77.     from cStringIO import StringIO
  78. except ImportError:
  79.     from StringIO import StringIO
  80.  
  81. __all__ = [
  82.     'HTTP',
  83.     'HTTPResponse',
  84.     'HTTPConnection',
  85.     'HTTPSConnection',
  86.     'HTTPException',
  87.     'NotConnected',
  88.     'UnknownProtocol',
  89.     'UnknownTransferEncoding',
  90.     'UnimplementedFileMode',
  91.     'IncompleteRead',
  92.     'InvalidURL',
  93.     'ImproperConnectionState',
  94.     'CannotSendRequest',
  95.     'CannotSendHeader',
  96.     'ResponseNotReady',
  97.     'BadStatusLine',
  98.     'error']
  99. HTTP_PORT = 80
  100. HTTPS_PORT = 443
  101. _UNKNOWN = 'UNKNOWN'
  102. _CS_IDLE = 'Idle'
  103. _CS_REQ_STARTED = 'Request-started'
  104. _CS_REQ_SENT = 'Request-sent'
  105. CONTINUE = 100
  106. SWITCHING_PROTOCOLS = 101
  107. PROCESSING = 102
  108. OK = 200
  109. CREATED = 201
  110. ACCEPTED = 202
  111. NON_AUTHORITATIVE_INFORMATION = 203
  112. NO_CONTENT = 204
  113. RESET_CONTENT = 205
  114. PARTIAL_CONTENT = 206
  115. MULTI_STATUS = 207
  116. IM_USED = 226
  117. MULTIPLE_CHOICES = 300
  118. MOVED_PERMANENTLY = 301
  119. FOUND = 302
  120. SEE_OTHER = 303
  121. NOT_MODIFIED = 304
  122. USE_PROXY = 305
  123. TEMPORARY_REDIRECT = 307
  124. BAD_REQUEST = 400
  125. UNAUTHORIZED = 401
  126. PAYMENT_REQUIRED = 402
  127. FORBIDDEN = 403
  128. NOT_FOUND = 404
  129. METHOD_NOT_ALLOWED = 405
  130. NOT_ACCEPTABLE = 406
  131. PROXY_AUTHENTICATION_REQUIRED = 407
  132. REQUEST_TIMEOUT = 408
  133. CONFLICT = 409
  134. GONE = 410
  135. LENGTH_REQUIRED = 411
  136. PRECONDITION_FAILED = 412
  137. REQUEST_ENTITY_TOO_LARGE = 413
  138. REQUEST_URI_TOO_LONG = 414
  139. UNSUPPORTED_MEDIA_TYPE = 415
  140. REQUESTED_RANGE_NOT_SATISFIABLE = 416
  141. EXPECTATION_FAILED = 417
  142. UNPROCESSABLE_ENTITY = 422
  143. LOCKED = 423
  144. FAILED_DEPENDENCY = 424
  145. UPGRADE_REQUIRED = 426
  146. INTERNAL_SERVER_ERROR = 500
  147. NOT_IMPLEMENTED = 501
  148. BAD_GATEWAY = 502
  149. SERVICE_UNAVAILABLE = 503
  150. GATEWAY_TIMEOUT = 504
  151. HTTP_VERSION_NOT_SUPPORTED = 505
  152. INSUFFICIENT_STORAGE = 507
  153. NOT_EXTENDED = 510
  154.  
  155. class HTTPMessage(mimetools.Message):
  156.     
  157.     def addheader(self, key, value):
  158.         '''Add header for field key handling repeats.'''
  159.         prev = self.dict.get(key)
  160.         if prev is None:
  161.             self.dict[key] = value
  162.         else:
  163.             combined = ', '.join((prev, value))
  164.             self.dict[key] = combined
  165.  
  166.     
  167.     def addcontinue(self, key, more):
  168.         '''Add more field data from a continuation line.'''
  169.         prev = self.dict[key]
  170.         self.dict[key] = prev + '\n ' + more
  171.  
  172.     
  173.     def readheaders(self):
  174.         '''Read header lines.
  175.  
  176.         Read header lines up to the entirely blank line that terminates them.
  177.         The (normally blank) line that ends the headers is skipped, but not
  178.         included in the returned list.  If a non-header line ends the headers,
  179.         (which is an error), an attempt is made to backspace over it; it is
  180.         never included in the returned list.
  181.  
  182.         The variable self.status is set to the empty string if all went well,
  183.         otherwise it is an error message.  The variable self.headers is a
  184.         completely uninterpreted list of lines contained in the header (so
  185.         printing them will reproduce the header exactly as it appears in the
  186.         file).
  187.  
  188.         If multiple header fields with the same name occur, they are combined
  189.         according to the rules in RFC 2616 sec 4.2:
  190.  
  191.         Appending each subsequent field-value to the first, each separated
  192.         by a comma. The order in which header fields with the same field-name
  193.         are received is significant to the interpretation of the combined
  194.         field value.
  195.         '''
  196.         self.dict = { }
  197.         self.unixfrom = ''
  198.         self.headers = hlist = []
  199.         self.status = ''
  200.         headerseen = ''
  201.         firstline = 1
  202.         startofline = None
  203.         unread = None
  204.         tell = None
  205.         if hasattr(self.fp, 'unread'):
  206.             unread = self.fp.unread
  207.         elif self.seekable:
  208.             tell = self.fp.tell
  209.         
  210.         while True:
  211.             if tell:
  212.                 
  213.                 try:
  214.                     startofline = tell()
  215.                 except IOError:
  216.                     startofline = None
  217.                     tell = None
  218.                     self.seekable = 0
  219.                 except:
  220.                     None<EXCEPTION MATCH>IOError
  221.                 
  222.  
  223.             None<EXCEPTION MATCH>IOError
  224.             line = self.fp.readline()
  225.             if not line:
  226.                 self.status = 'EOF in headers'
  227.                 break
  228.             
  229.             if firstline and line.startswith('From '):
  230.                 self.unixfrom = self.unixfrom + line
  231.                 continue
  232.             
  233.             firstline = 0
  234.             if headerseen and line[0] in ' \t':
  235.                 hlist.append(line)
  236.                 self.addcontinue(headerseen, line.strip())
  237.                 continue
  238.             elif self.iscomment(line):
  239.                 continue
  240.             elif self.islast(line):
  241.                 break
  242.             
  243.             headerseen = self.isheader(line)
  244.             if headerseen:
  245.                 hlist.append(line)
  246.                 self.addheader(headerseen, line[len(headerseen) + 1:].strip())
  247.                 continue
  248.                 continue
  249.             if not self.dict:
  250.                 self.status = 'No headers'
  251.             else:
  252.                 self.status = 'Non-header line where header expected'
  253.             if unread:
  254.                 unread(line)
  255.             elif tell:
  256.                 self.fp.seek(startofline)
  257.             else:
  258.                 self.status = self.status + '; bad seek'
  259.             break
  260.  
  261.  
  262.  
  263. class HTTPResponse:
  264.     
  265.     def __init__(self, sock, debuglevel = 0, strict = 0, method = None):
  266.         self.fp = sock.makefile('rb', 0)
  267.         self.debuglevel = debuglevel
  268.         self.strict = strict
  269.         self._method = method
  270.         self.msg = None
  271.         self.version = _UNKNOWN
  272.         self.status = _UNKNOWN
  273.         self.reason = _UNKNOWN
  274.         self.chunked = _UNKNOWN
  275.         self.chunk_left = _UNKNOWN
  276.         self.length = _UNKNOWN
  277.         self.will_close = _UNKNOWN
  278.  
  279.     
  280.     def _read_status(self):
  281.         line = self.fp.readline()
  282.         if self.debuglevel > 0:
  283.             print 'reply:', repr(line)
  284.         
  285.         if not line:
  286.             raise BadStatusLine(line)
  287.         
  288.         
  289.         try:
  290.             (version, status, reason) = line.split(None, 2)
  291.         except ValueError:
  292.             
  293.             try:
  294.                 (version, status) = line.split(None, 1)
  295.                 reason = ''
  296.             except ValueError:
  297.                 version = ''
  298.             except:
  299.                 None<EXCEPTION MATCH>ValueError
  300.             
  301.  
  302.             None<EXCEPTION MATCH>ValueError
  303.  
  304.         if not version.startswith('HTTP/'):
  305.             if self.strict:
  306.                 self.close()
  307.                 raise BadStatusLine(line)
  308.             else:
  309.                 self.fp = LineAndFileWrapper(line, self.fp)
  310.                 return ('HTTP/0.9', 200, '')
  311.         
  312.         
  313.         try:
  314.             status = int(status)
  315.             if status < 100 or status > 999:
  316.                 raise BadStatusLine(line)
  317.         except ValueError:
  318.             raise BadStatusLine(line)
  319.  
  320.         return (version, status, reason)
  321.  
  322.     
  323.     def begin(self):
  324.         if self.msg is not None:
  325.             return None
  326.         
  327.         while True:
  328.             (version, status, reason) = self._read_status()
  329.             if status != CONTINUE:
  330.                 break
  331.             
  332.             while True:
  333.                 skip = self.fp.readline().strip()
  334.                 if not skip:
  335.                     break
  336.                 
  337.                 if self.debuglevel > 0:
  338.                     print 'header:', skip
  339.                     continue
  340.         self.status = status
  341.         self.reason = reason.strip()
  342.         if version == 'HTTP/1.0':
  343.             self.version = 10
  344.         elif version.startswith('HTTP/1.'):
  345.             self.version = 11
  346.         elif version == 'HTTP/0.9':
  347.             self.version = 9
  348.         else:
  349.             raise UnknownProtocol(version)
  350.         if self.version == 9:
  351.             self.chunked = 0
  352.             self.will_close = 1
  353.             self.msg = HTTPMessage(StringIO())
  354.             return None
  355.         
  356.         self.msg = HTTPMessage(self.fp, 0)
  357.         if self.debuglevel > 0:
  358.             for hdr in self.msg.headers:
  359.                 print 'header:', hdr,
  360.             
  361.         
  362.         self.msg.fp = None
  363.         tr_enc = self.msg.getheader('transfer-encoding')
  364.         if tr_enc and tr_enc.lower() == 'chunked':
  365.             self.chunked = 1
  366.             self.chunk_left = None
  367.         else:
  368.             self.chunked = 0
  369.         self.will_close = self._check_close()
  370.         length = self.msg.getheader('content-length')
  371.         if length and not (self.chunked):
  372.             
  373.             try:
  374.                 self.length = int(length)
  375.             except ValueError:
  376.                 self.length = None
  377.             except:
  378.                 None<EXCEPTION MATCH>ValueError
  379.             
  380.  
  381.         None<EXCEPTION MATCH>ValueError
  382.         self.length = None
  383.         if not status == NO_CONTENT and status == NOT_MODIFIED:
  384.             if status <= status:
  385.                 pass
  386.             elif status < 200 or self._method == 'HEAD':
  387.                 self.length = 0
  388.             
  389.         if not (self.will_close) and not (self.chunked) and self.length is None:
  390.             self.will_close = 1
  391.         
  392.  
  393.     
  394.     def _check_close(self):
  395.         conn = self.msg.getheader('connection')
  396.         if self.version == 11:
  397.             conn = self.msg.getheader('connection')
  398.             if conn and 'close' in conn.lower():
  399.                 return True
  400.             
  401.             return False
  402.         
  403.         if self.msg.getheader('keep-alive'):
  404.             return False
  405.         
  406.         if conn and 'keep-alive' in conn.lower():
  407.             return False
  408.         
  409.         pconn = self.msg.getheader('proxy-connection')
  410.         if pconn and 'keep-alive' in pconn.lower():
  411.             return False
  412.         
  413.         return True
  414.  
  415.     
  416.     def close(self):
  417.         if self.fp:
  418.             self.fp.close()
  419.             self.fp = None
  420.         
  421.  
  422.     
  423.     def isclosed(self):
  424.         return self.fp is None
  425.  
  426.     
  427.     def read(self, amt = None):
  428.         if self.fp is None:
  429.             return ''
  430.         
  431.         if self.chunked:
  432.             return self._read_chunked(amt)
  433.         
  434.         if amt is None:
  435.             if self.length is None:
  436.                 s = self.fp.read()
  437.             else:
  438.                 s = self._safe_read(self.length)
  439.                 self.length = 0
  440.             self.close()
  441.             return s
  442.         
  443.         if self.length is not None:
  444.             if amt > self.length:
  445.                 amt = self.length
  446.             
  447.         
  448.         s = self.fp.read(amt)
  449.         if self.length is not None:
  450.             self.length -= len(s)
  451.         
  452.         return s
  453.  
  454.     
  455.     def _read_chunked(self, amt):
  456.         if not self.chunked != _UNKNOWN:
  457.             raise AssertionError
  458.         chunk_left = self.chunk_left
  459.         value = ''
  460.         while True:
  461.             if chunk_left is None:
  462.                 line = self.fp.readline()
  463.                 i = line.find(';')
  464.                 if i >= 0:
  465.                     line = line[:i]
  466.                 
  467.                 chunk_left = int(line, 16)
  468.                 if chunk_left == 0:
  469.                     break
  470.                 
  471.             
  472.             if amt is None:
  473.                 value += self._safe_read(chunk_left)
  474.             elif amt < chunk_left:
  475.                 value += self._safe_read(amt)
  476.                 self.chunk_left = chunk_left - amt
  477.                 return value
  478.             elif amt == chunk_left:
  479.                 value += self._safe_read(amt)
  480.                 self._safe_read(2)
  481.                 self.chunk_left = None
  482.                 return value
  483.             else:
  484.                 value += self._safe_read(chunk_left)
  485.                 amt -= chunk_left
  486.             self._safe_read(2)
  487.             chunk_left = None
  488.         while True:
  489.             line = self.fp.readline()
  490.             if line == '\r\n':
  491.                 break
  492.                 continue
  493.         self.close()
  494.         return value
  495.  
  496.     
  497.     def _safe_read(self, amt):
  498.         '''Read the number of bytes requested, compensating for partial reads.
  499.  
  500.         Normally, we have a blocking socket, but a read() can be interrupted
  501.         by a signal (resulting in a partial read).
  502.  
  503.         Note that we cannot distinguish between EOF and an interrupt when zero
  504.         bytes have been read. IncompleteRead() will be raised in this
  505.         situation.
  506.  
  507.         This function should be used when <amt> bytes "should" be present for
  508.         reading. If the bytes are truly not available (due to EOF), then the
  509.         IncompleteRead exception can be used to detect the problem.
  510.         '''
  511.         s = ''
  512.         while amt > 0:
  513.             chunk = self.fp.read(amt)
  514.             if not chunk:
  515.                 raise IncompleteRead(s)
  516.             
  517.             s += chunk
  518.             amt -= len(chunk)
  519.         return s
  520.  
  521.     
  522.     def getheader(self, name, default = None):
  523.         if self.msg is None:
  524.             raise ResponseNotReady()
  525.         
  526.         return self.msg.getheader(name, default)
  527.  
  528.     
  529.     def getheaders(self):
  530.         '''Return list of (header, value) tuples.'''
  531.         if self.msg is None:
  532.             raise ResponseNotReady()
  533.         
  534.         return self.msg.items()
  535.  
  536.  
  537.  
  538. class HTTPConnection:
  539.     _http_vsn = 11
  540.     _http_vsn_str = 'HTTP/1.1'
  541.     response_class = HTTPResponse
  542.     default_port = HTTP_PORT
  543.     auto_open = 1
  544.     debuglevel = 0
  545.     strict = 0
  546.     
  547.     def __init__(self, host, port = None, strict = None):
  548.         self.sock = None
  549.         self._buffer = []
  550.         self._HTTPConnection__response = None
  551.         self._HTTPConnection__state = _CS_IDLE
  552.         self._method = None
  553.         self._set_hostport(host, port)
  554.         if strict is not None:
  555.             self.strict = strict
  556.         
  557.  
  558.     
  559.     def _set_hostport(self, host, port):
  560.         if port is None:
  561.             i = host.rfind(':')
  562.             j = host.rfind(']')
  563.             if i > j:
  564.                 
  565.                 try:
  566.                     port = int(host[i + 1:])
  567.                 except ValueError:
  568.                     raise InvalidURL("nonnumeric port: '%s'" % host[i + 1:])
  569.  
  570.                 host = host[:i]
  571.             else:
  572.                 port = self.default_port
  573.             if host and host[0] == '[' and host[-1] == ']':
  574.                 host = host[1:-1]
  575.             
  576.         
  577.         self.host = host
  578.         self.port = port
  579.  
  580.     
  581.     def set_debuglevel(self, level):
  582.         self.debuglevel = level
  583.  
  584.     
  585.     def connect(self):
  586.         '''Connect to the host and port specified in __init__.'''
  587.         msg = 'getaddrinfo returns an empty list'
  588.         for res in socket.getaddrinfo(self.host, self.port, 0, socket.SOCK_STREAM):
  589.             (af, socktype, proto, canonname, sa) = res
  590.             
  591.             try:
  592.                 self.sock = socket.socket(af, socktype, proto)
  593.                 if self.debuglevel > 0:
  594.                     print 'connect: (%s, %s)' % (self.host, self.port)
  595.                 
  596.                 self.sock.connect(sa)
  597.             except socket.error:
  598.                 msg = None
  599.                 if self.debuglevel > 0:
  600.                     print 'connect fail:', (self.host, self.port)
  601.                 
  602.                 if self.sock:
  603.                     self.sock.close()
  604.                 
  605.                 self.sock = None
  606.                 continue
  607.  
  608.         
  609.         if not self.sock:
  610.             raise socket.error, msg
  611.         
  612.  
  613.     
  614.     def close(self):
  615.         '''Close the connection to the HTTP server.'''
  616.         if self.sock:
  617.             self.sock.close()
  618.             self.sock = None
  619.         
  620.         if self._HTTPConnection__response:
  621.             self._HTTPConnection__response.close()
  622.             self._HTTPConnection__response = None
  623.         
  624.         self._HTTPConnection__state = _CS_IDLE
  625.  
  626.     
  627.     def send(self, str):
  628.         """Send `str' to the server."""
  629.         if self.sock is None:
  630.             if self.auto_open:
  631.                 self.connect()
  632.             else:
  633.                 raise NotConnected()
  634.         
  635.         if self.debuglevel > 0:
  636.             print 'send:', repr(str)
  637.         
  638.         
  639.         try:
  640.             self.sock.sendall(str)
  641.         except socket.error:
  642.             v = None
  643.             if v[0] == 32:
  644.                 self.close()
  645.             
  646.             raise 
  647.  
  648.  
  649.     
  650.     def _output(self, s):
  651.         '''Add a line of output to the current request buffer.
  652.  
  653.         Assumes that the line does *not* end with \\r\\n.
  654.         '''
  655.         self._buffer.append(s)
  656.  
  657.     
  658.     def _send_output(self):
  659.         '''Send the currently buffered request and clear the buffer.
  660.  
  661.         Appends an extra \\r\\n to the buffer.
  662.         '''
  663.         self._buffer.extend(('', ''))
  664.         msg = '\r\n'.join(self._buffer)
  665.         del self._buffer[:]
  666.         self.send(msg)
  667.  
  668.     
  669.     def putrequest(self, method, url, skip_host = 0, skip_accept_encoding = 0):
  670.         """Send a request to the server.
  671.  
  672.         `method' specifies an HTTP request method, e.g. 'GET'.
  673.         `url' specifies the object being requested, e.g. '/index.html'.
  674.         `skip_host' if True does not add automatically a 'Host:' header
  675.         `skip_accept_encoding' if True does not add automatically an
  676.            'Accept-Encoding:' header
  677.         """
  678.         if self._HTTPConnection__response and self._HTTPConnection__response.isclosed():
  679.             self._HTTPConnection__response = None
  680.         
  681.         if self._HTTPConnection__state == _CS_IDLE:
  682.             self._HTTPConnection__state = _CS_REQ_STARTED
  683.         else:
  684.             raise CannotSendRequest()
  685.         self._method = method
  686.         if not url:
  687.             url = '/'
  688.         
  689.         str = '%s %s %s' % (method, url, self._http_vsn_str)
  690.         self._output(str)
  691.         if self._http_vsn == 11:
  692.             if not skip_host:
  693.                 netloc = ''
  694.                 if url.startswith('http'):
  695.                     (nil, netloc, nil, nil, nil) = urlsplit(url)
  696.                 
  697.                 if netloc:
  698.                     self.putheader('Host', netloc.encode('idna'))
  699.                 elif self.port == HTTP_PORT:
  700.                     self.putheader('Host', self.host.encode('idna'))
  701.                 else:
  702.                     self.putheader('Host', '%s:%s' % (self.host.encode('idna'), self.port))
  703.             
  704.             if not skip_accept_encoding:
  705.                 self.putheader('Accept-Encoding', 'identity')
  706.             
  707.         
  708.  
  709.     
  710.     def putheader(self, header, value):
  711.         """Send a request header line to the server.
  712.  
  713.         For example: h.putheader('Accept', 'text/html')
  714.         """
  715.         if self._HTTPConnection__state != _CS_REQ_STARTED:
  716.             raise CannotSendHeader()
  717.         
  718.         str = '%s: %s' % (header, value)
  719.         self._output(str)
  720.  
  721.     
  722.     def endheaders(self):
  723.         '''Indicate that the last header line has been sent to the server.'''
  724.         if self._HTTPConnection__state == _CS_REQ_STARTED:
  725.             self._HTTPConnection__state = _CS_REQ_SENT
  726.         else:
  727.             raise CannotSendHeader()
  728.         self._send_output()
  729.  
  730.     
  731.     def request(self, method, url, body = None, headers = { }):
  732.         '''Send a complete request to the server.'''
  733.         
  734.         try:
  735.             self._send_request(method, url, body, headers)
  736.         except socket.error:
  737.             v = None
  738.             if v[0] != 32 or not (self.auto_open):
  739.                 raise 
  740.             
  741.             self._send_request(method, url, body, headers)
  742.  
  743.  
  744.     
  745.     def _send_request(self, method, url, body, headers):
  746.         header_names = []([ k.lower() for k in headers ])
  747.         skips = { }
  748.         if 'accept-encoding' in header_names:
  749.             skips['skip_accept_encoding'] = 1
  750.         
  751.         self.putrequest(method, url, **skips)
  752.         if body and 'content-length' not in header_names:
  753.             self.putheader('Content-Length', str(len(body)))
  754.         
  755.         for hdr, value in headers.iteritems():
  756.             self.putheader(hdr, value)
  757.         
  758.         self.endheaders()
  759.         if body:
  760.             self.send(body)
  761.         
  762.  
  763.     
  764.     def getresponse(self):
  765.         '''Get the response from the server.'''
  766.         if self._HTTPConnection__response and self._HTTPConnection__response.isclosed():
  767.             self._HTTPConnection__response = None
  768.         
  769.         if self._HTTPConnection__state != _CS_REQ_SENT or self._HTTPConnection__response:
  770.             raise ResponseNotReady()
  771.         
  772.         if self.debuglevel > 0:
  773.             response = self.response_class(self.sock, self.debuglevel, strict = self.strict, method = self._method)
  774.         else:
  775.             response = self.response_class(self.sock, strict = self.strict, method = self._method)
  776.         response.begin()
  777.         if not response.will_close != _UNKNOWN:
  778.             raise AssertionError
  779.         self._HTTPConnection__state = _CS_IDLE
  780.         if response.will_close:
  781.             self.close()
  782.         else:
  783.             self._HTTPConnection__response = response
  784.         return response
  785.  
  786.  
  787.  
  788. class SharedSocket:
  789.     
  790.     def __init__(self, sock):
  791.         self.sock = sock
  792.         self._refcnt = 0
  793.  
  794.     
  795.     def incref(self):
  796.         self._refcnt += 1
  797.  
  798.     
  799.     def decref(self):
  800.         self._refcnt -= 1
  801.         if not self._refcnt >= 0:
  802.             raise AssertionError
  803.         self
  804.         if self._refcnt == 0:
  805.             self.sock.close()
  806.         
  807.  
  808.     
  809.     def __del__(self):
  810.         self.sock.close()
  811.  
  812.  
  813.  
  814. class SharedSocketClient:
  815.     
  816.     def __init__(self, shared):
  817.         self._closed = 0
  818.         self._shared = shared
  819.         self._shared.incref()
  820.         self._sock = shared.sock
  821.  
  822.     
  823.     def close(self):
  824.         if not self._closed:
  825.             self._shared.decref()
  826.             self._closed = 1
  827.             self._shared = None
  828.         
  829.  
  830.  
  831.  
  832. class SSLFile(SharedSocketClient):
  833.     '''File-like object wrapping an SSL socket.'''
  834.     BUFSIZE = 8192
  835.     
  836.     def __init__(self, sock, ssl, bufsize = None):
  837.         SharedSocketClient.__init__(self, sock)
  838.         self._ssl = ssl
  839.         self._buf = ''
  840.         if not bufsize:
  841.             pass
  842.         self._bufsize = self.__class__.BUFSIZE
  843.  
  844.     
  845.     def _read(self):
  846.         buf = ''
  847.         while True:
  848.             
  849.             try:
  850.                 buf = self._ssl.read(self._bufsize)
  851.             except socket.sslerror:
  852.                 err = None
  853.                 if err[0] == socket.SSL_ERROR_WANT_READ or err[0] == socket.SSL_ERROR_WANT_WRITE:
  854.                     continue
  855.                 
  856.                 if err[0] == socket.SSL_ERROR_ZERO_RETURN or err[0] == socket.SSL_ERROR_EOF:
  857.                     break
  858.                 
  859.                 raise 
  860.                 continue
  861.                 except socket.error:
  862.                     err = None
  863.                     if err[0] == errno.EINTR:
  864.                         continue
  865.                     
  866.                     if err[0] == errno.EBADF:
  867.                         break
  868.                     
  869.                     raise 
  870.                     continue
  871.                 else:
  872.                     break
  873.                 None<EXCEPTION MATCH>socket.error
  874.             return buf
  875.  
  876.  
  877.     
  878.     def read(self, size = None):
  879.         L = [
  880.             self._buf]
  881.         avail = len(self._buf)
  882.         while size is None or avail < size:
  883.             s = self._read()
  884.             if s == '':
  885.                 break
  886.             
  887.             L.append(s)
  888.             avail += len(s)
  889.         all = ''.join(L)
  890.         if size is None:
  891.             self._buf = ''
  892.             return all
  893.         else:
  894.             self._buf = all[size:]
  895.             return all[:size]
  896.  
  897.     
  898.     def readline(self):
  899.         L = [
  900.             self._buf]
  901.         self._buf = ''
  902.         while None:
  903.             i = L[-1].find('\n')
  904.             if i >= 0:
  905.                 break
  906.             
  907.             s = self._read()
  908.             if s == '':
  909.                 break
  910.             
  911.         if i == -1:
  912.             return ''.join(L)
  913.         else:
  914.             all = ''.join(L)
  915.             i = all.find('\n') + 1
  916.             line = all[:i]
  917.             self._buf = all[i:]
  918.             return line
  919.  
  920.     
  921.     def readlines(self, sizehint = 0):
  922.         total = 0
  923.         list = []
  924.         while True:
  925.             line = self.readline()
  926.             if not line:
  927.                 break
  928.             
  929.             list.append(line)
  930.             total += len(line)
  931.             if sizehint and total >= sizehint:
  932.                 break
  933.                 continue
  934.         return list
  935.  
  936.     
  937.     def fileno(self):
  938.         return self._sock.fileno()
  939.  
  940.     
  941.     def __iter__(self):
  942.         return self
  943.  
  944.     
  945.     def next(self):
  946.         line = self.readline()
  947.         if not line:
  948.             raise StopIteration
  949.         
  950.         return line
  951.  
  952.  
  953.  
  954. class FakeSocket(SharedSocketClient):
  955.     
  956.     class _closedsocket:
  957.         
  958.         def __getattr__(self, name):
  959.             raise error(9, 'Bad file descriptor')
  960.  
  961.  
  962.     
  963.     def __init__(self, sock, ssl):
  964.         sock = SharedSocket(sock)
  965.         SharedSocketClient.__init__(self, sock)
  966.         self._ssl = ssl
  967.  
  968.     
  969.     def close(self):
  970.         SharedSocketClient.close(self)
  971.         self._sock = self.__class__._closedsocket()
  972.  
  973.     
  974.     def makefile(self, mode, bufsize = None):
  975.         if mode != 'r' and mode != 'rb':
  976.             raise UnimplementedFileMode()
  977.         
  978.         return SSLFile(self._shared, self._ssl, bufsize)
  979.  
  980.     
  981.     def send(self, stuff, flags = 0):
  982.         return self._ssl.write(stuff)
  983.  
  984.     sendall = send
  985.     
  986.     def recv(self, len = 1024, flags = 0):
  987.         return self._ssl.read(len)
  988.  
  989.     
  990.     def __getattr__(self, attr):
  991.         return getattr(self._sock, attr)
  992.  
  993.  
  994.  
  995. class HTTPSConnection(HTTPConnection):
  996.     '''This class allows communication via SSL.'''
  997.     default_port = HTTPS_PORT
  998.     
  999.     def __init__(self, host, port = None, key_file = None, cert_file = None, strict = None):
  1000.         HTTPConnection.__init__(self, host, port, strict)
  1001.         self.key_file = key_file
  1002.         self.cert_file = cert_file
  1003.  
  1004.     
  1005.     def connect(self):
  1006.         '''Connect to a host on a given (SSL) port.'''
  1007.         sock = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
  1008.         sock.connect((self.host, self.port))
  1009.         ssl = socket.ssl(sock, self.key_file, self.cert_file)
  1010.         self.sock = FakeSocket(sock, ssl)
  1011.  
  1012.  
  1013.  
  1014. class HTTP:
  1015.     '''Compatibility class with httplib.py from 1.5.'''
  1016.     _http_vsn = 10
  1017.     _http_vsn_str = 'HTTP/1.0'
  1018.     debuglevel = 0
  1019.     _connection_class = HTTPConnection
  1020.     
  1021.     def __init__(self, host = '', port = None, strict = None):
  1022.         '''Provide a default host, since the superclass requires one.'''
  1023.         if port == 0:
  1024.             port = None
  1025.         
  1026.         self._setup(self._connection_class(host, port, strict))
  1027.  
  1028.     
  1029.     def _setup(self, conn):
  1030.         self._conn = conn
  1031.         self.send = conn.send
  1032.         self.putrequest = conn.putrequest
  1033.         self.endheaders = conn.endheaders
  1034.         self.set_debuglevel = conn.set_debuglevel
  1035.         conn._http_vsn = self._http_vsn
  1036.         conn._http_vsn_str = self._http_vsn_str
  1037.         self.file = None
  1038.  
  1039.     
  1040.     def connect(self, host = None, port = None):
  1041.         """Accept arguments to set the host/port, since the superclass doesn't."""
  1042.         if host is not None:
  1043.             self._conn._set_hostport(host, port)
  1044.         
  1045.         self._conn.connect()
  1046.  
  1047.     
  1048.     def getfile(self):
  1049.         """Provide a getfile, since the superclass' does not use this concept."""
  1050.         return self.file
  1051.  
  1052.     
  1053.     def putheader(self, header, *values):
  1054.         '''The superclass allows only one value argument.'''
  1055.         self._conn.putheader(header, '\r\n\t'.join(values))
  1056.  
  1057.     
  1058.     def getreply(self):
  1059.         '''Compat definition since superclass does not define it.
  1060.  
  1061.         Returns a tuple consisting of:
  1062.         - server status code (e.g. \'200\' if all goes well)
  1063.         - server "reason" corresponding to status code
  1064.         - any RFC822 headers in the response from the server
  1065.         '''
  1066.         
  1067.         try:
  1068.             response = self._conn.getresponse()
  1069.         except BadStatusLine:
  1070.             e = None
  1071.             self.file = self._conn.sock.makefile('rb', 0)
  1072.             self.close()
  1073.             self.headers = None
  1074.             return (-1, e.line, None)
  1075.  
  1076.         self.headers = response.msg
  1077.         self.file = response.fp
  1078.         return (response.status, response.reason, response.msg)
  1079.  
  1080.     
  1081.     def close(self):
  1082.         self._conn.close()
  1083.         self.file = None
  1084.  
  1085.  
  1086. if hasattr(socket, 'ssl'):
  1087.     
  1088.     class HTTPS(HTTP):
  1089.         '''Compatibility with 1.5 httplib interface
  1090.  
  1091.         Python 1.5.2 did not have an HTTPS class, but it defined an
  1092.         interface for sending http requests that is also useful for
  1093.         https.
  1094.         '''
  1095.         _connection_class = HTTPSConnection
  1096.         
  1097.         def __init__(self, host = '', port = None, key_file = None, cert_file = None, strict = None):
  1098.             if port == 0:
  1099.                 port = None
  1100.             
  1101.             self._setup(self._connection_class(host, port, key_file, cert_file, strict))
  1102.             self.key_file = key_file
  1103.             self.cert_file = cert_file
  1104.  
  1105.  
  1106.  
  1107.  
  1108. class HTTPException(Exception):
  1109.     pass
  1110.  
  1111.  
  1112. class NotConnected(HTTPException):
  1113.     pass
  1114.  
  1115.  
  1116. class InvalidURL(HTTPException):
  1117.     pass
  1118.  
  1119.  
  1120. class UnknownProtocol(HTTPException):
  1121.     
  1122.     def __init__(self, version):
  1123.         self.args = (version,)
  1124.         self.version = version
  1125.  
  1126.  
  1127.  
  1128. class UnknownTransferEncoding(HTTPException):
  1129.     pass
  1130.  
  1131.  
  1132. class UnimplementedFileMode(HTTPException):
  1133.     pass
  1134.  
  1135.  
  1136. class IncompleteRead(HTTPException):
  1137.     
  1138.     def __init__(self, partial):
  1139.         self.args = (partial,)
  1140.         self.partial = partial
  1141.  
  1142.  
  1143.  
  1144. class ImproperConnectionState(HTTPException):
  1145.     pass
  1146.  
  1147.  
  1148. class CannotSendRequest(ImproperConnectionState):
  1149.     pass
  1150.  
  1151.  
  1152. class CannotSendHeader(ImproperConnectionState):
  1153.     pass
  1154.  
  1155.  
  1156. class ResponseNotReady(ImproperConnectionState):
  1157.     pass
  1158.  
  1159.  
  1160. class BadStatusLine(HTTPException):
  1161.     
  1162.     def __init__(self, line):
  1163.         self.args = (line,)
  1164.         self.line = line
  1165.  
  1166.  
  1167. error = HTTPException
  1168.  
  1169. class LineAndFileWrapper:
  1170.     '''A limited file-like object for HTTP/0.9 responses.'''
  1171.     
  1172.     def __init__(self, line, file):
  1173.         self._line = line
  1174.         self._file = file
  1175.         self._line_consumed = 0
  1176.         self._line_offset = 0
  1177.         self._line_left = len(line)
  1178.  
  1179.     
  1180.     def __getattr__(self, attr):
  1181.         return getattr(self._file, attr)
  1182.  
  1183.     
  1184.     def _done(self):
  1185.         self._line_consumed = 1
  1186.         self.read = self._file.read
  1187.         self.readline = self._file.readline
  1188.         self.readlines = self._file.readlines
  1189.  
  1190.     
  1191.     def read(self, amt = None):
  1192.         if self._line_consumed:
  1193.             return self._file.read(amt)
  1194.         
  1195.         if not self._line_left:
  1196.             raise AssertionError
  1197.         if amt is None or amt > self._line_left:
  1198.             s = self._line[self._line_offset:]
  1199.             self._done()
  1200.             if amt is None:
  1201.                 return s + self._file.read()
  1202.             else:
  1203.                 return s + self._file.read(amt - len(s))
  1204.         elif not amt <= self._line_left:
  1205.             raise AssertionError
  1206.         i = self._line_offset
  1207.         j = i + amt
  1208.         s = self._line[i:j]
  1209.         self._line_offset = j
  1210.         self._line_left -= amt
  1211.         if self._line_left == 0:
  1212.             self._done()
  1213.         
  1214.         return s
  1215.  
  1216.     
  1217.     def readline(self):
  1218.         if self._line_consumed:
  1219.             return self._file.readline()
  1220.         
  1221.         if not self._line_left:
  1222.             raise AssertionError
  1223.         s = self._line[self._line_offset:]
  1224.         self._done()
  1225.         return s
  1226.  
  1227.     
  1228.     def readlines(self, size = None):
  1229.         if self._line_consumed:
  1230.             return self._file.readlines(size)
  1231.         
  1232.         if not self._line_left:
  1233.             raise AssertionError
  1234.         L = [
  1235.             self._line[self._line_offset:]]
  1236.         self._done()
  1237.         if size is None:
  1238.             return L + self._file.readlines()
  1239.         else:
  1240.             return L + self._file.readlines(size)
  1241.  
  1242.  
  1243.  
  1244. def test():
  1245.     '''Test this module.
  1246.  
  1247.     A hodge podge of tests collected here, because they have too many
  1248.     external dependencies for the regular test suite.
  1249.     '''
  1250.     import sys
  1251.     import getopt
  1252.     (opts, args) = getopt.getopt(sys.argv[1:], 'd')
  1253.     dl = 0
  1254.     for o, a in opts:
  1255.         if o == '-d':
  1256.             dl = dl + 1
  1257.             continue
  1258.     
  1259.     host = 'www.python.org'
  1260.     selector = '/'
  1261.     if args[0:]:
  1262.         host = args[0]
  1263.     
  1264.     if args[1:]:
  1265.         selector = args[1]
  1266.     
  1267.     h = HTTP()
  1268.     h.set_debuglevel(dl)
  1269.     h.connect(host)
  1270.     h.putrequest('GET', selector)
  1271.     h.endheaders()
  1272.     (status, reason, headers) = h.getreply()
  1273.     print 'status =', status
  1274.     print 'reason =', reason
  1275.     print 'read', len(h.getfile().read())
  1276.     print 
  1277.     if headers:
  1278.         for header in headers.headers:
  1279.             print header.strip()
  1280.         
  1281.     
  1282.     print 
  1283.     
  1284.     class HTTP11(HTTP):
  1285.         _http_vsn = 11
  1286.         _http_vsn_str = 'HTTP/1.1'
  1287.  
  1288.     h = HTTP11('www.python.org')
  1289.     h.putrequest('GET', 'http://www.python.org/~jeremy/')
  1290.     h.endheaders()
  1291.     h.getreply()
  1292.     h.close()
  1293.     if hasattr(socket, 'ssl'):
  1294.         for host, selector in (('sourceforge.net', '/projects/python'),):
  1295.             print 'https://%s%s' % (host, selector)
  1296.             hs = HTTPS()
  1297.             hs.set_debuglevel(dl)
  1298.             hs.connect(host)
  1299.             hs.putrequest('GET', selector)
  1300.             hs.endheaders()
  1301.             (status, reason, headers) = hs.getreply()
  1302.             print 'status =', status
  1303.             print 'reason =', reason
  1304.             print 'read', len(hs.getfile().read())
  1305.             print 
  1306.             if headers:
  1307.                 for header in headers.headers:
  1308.                     print header.strip()
  1309.                 
  1310.             
  1311.             print 
  1312.         
  1313.     
  1314.  
  1315. if __name__ == '__main__':
  1316.     test()
  1317.  
  1318.