日韩av黄I国产麻豆传媒I国产91av视频在线观看I日韩一区二区三区在线看I美女国产在线I麻豆视频国产在线观看I成人黄色短片

歡迎訪問 生活随笔!

生活随笔

當前位置: 首頁 > 编程语言 > python >内容正文

python

python开发客户端_python用700行代码实现http客户端

發布時間:2023/12/10 python 40 豆豆
生活随笔 收集整理的這篇文章主要介紹了 python开发客户端_python用700行代码实现http客户端 小編覺得挺不錯的,現在分享給大家,幫大家做個參考.

本文用python在TCP的基礎上實現一個HTTP客戶端, 該客戶端能夠復用TCP連接, 使用HTTP1.1協議.

一. 創建HTTP請求

HTTP是基于TCP連接的, 它的請求報文格式如下:

因此, 我們只需要創建一個到服務器的TCP連接, 然后按照上面的格式寫好報文并發給服務器, 就實現了一個HTTP請求.

1. HTTPConnection類

基于以上的分析, 我們首先定義一個HTTPConnection類來管理連接和請求內容:

class HTTPConnection:

default_port = 80

_http_vsn = 11

_http_vsn_str = 'HTTP/1.1'

def __init__(self, host: str, port: int = None) -> None:

self.sock = None

self._buffer = []

self.host = host

self.port = port if port is not None else self.default_port

self._state = _CS_IDLE

self._response = None

self._method = None

self.block_size = 8192

def _output(self, s: Union[str, bytes]) -> None:

if hasattr(s, 'encode'):

s = s.encode('latin-1')

self._buffer.append(s)

def connect(self) -> None:

self.sock = socket.create_connection((self.host, self.port))

對于這個HTTPConnection對象, 我們只需要創建TCP連接, 然后按照HTTP協議的格式把請求數據寫入buffer中, 最后把buffer中的數據發送出去就行了.

2. 編寫請求行

請求行的內容比較簡單, 就是說明請求方法, 請求路徑和HTTP協議. 使用下面的方法來編寫一個請求行:

def put_request(self, method: str, url: str) -> None:

self._method = method

url = url or '/'

request = f'{method} {url} {self._http_vsn_str}'

self._output(request)

3. 添加請求頭

HTTP請求頭和python的字典類似, 每行都是一個字段名與值的映射關系. HTTP協議并不要求設置所有合法的請求頭的值, 我們只需要按照需要, 設置特定的請求頭即可. 使用如下代碼添加請求頭:

def put_header(self, header: Union[bytes, str], value: Union[bytes, str, int]) -> None:

if hasattr(header, 'encode'):

header = header.encode('ascii')

if hasattr(value, 'encode'):

value = value.encode('latin-1')

elif isinstance(value, int):

value = str(value).encode('ascii')

header = header + b': ' + value

self._output(header)

此外, 在HTTP請求中, Host請求頭字段是必須的, 否則網站可能會拒絕響應. 因此, 如果用戶沒有設置這個字段, 這里就應該主動把它加上去:

def _add_host(self, url: str) -> None:

# 所有HTTP / 1.1請求報文中必須包含一個Host頭字段

# 如果用戶沒給,就調用這個函數來生成

netloc = ''

if url.startswith('http'):

nil, netloc, nil, nil, nil = urllib.parse.urlsplit(url)

if netloc:

try:

netloc_enc = netloc.encode('ascii')

except UnicodeEncodeError:

netloc_enc = netloc.encode('idna')

self.put_header('Host', netloc_enc)

else:

host = self.host

port = self.port

try:

host_enc = host.encode('ascii')

except UnicodeEncodeError:

host_enc = host.encode('idna')

# 對IPv6的地址進行額外處理

if host.find(':') >= 0:

host_enc = b'[' + host_enc + b']'

if port == self.default_port:

self.put_header('Host', host_enc)

else:

host_enc = host_enc.decode('ascii')

self.put_header('Host', f'{host_enc}:{port}')

4. 發送請求正文

我們接受兩種形式的body數據: 一個基于io.IOBase的可讀文件對象, 或者是一個能通過迭代得到數據的對象. 在傳輸數據之前, 我們首先要確定數據是否采用分塊傳輸:

def request(self, method: str, url: str, headers: dict = None, body: Union[io.IOBase, Iterable] = None,

encode_chunked: bool = False) -> None:

...

if 'content-length' not in header_names:

if 'transfer-encoding' not in header_names:

encode_chunked = False

content_length = self._get_content_length(body, method)

if content_length is None:

if body is not None:

# 在這種情況下, body一般是個生成器或者可讀文件之類的東西,應該分塊傳輸

encode_chunked = True

self.put_header('Transfer-Encoding', 'chunked')

else:

self.put_header('Content-Length', str(content_length))

else:

# 如果設置了transfer-encoding,則根據用戶給的encode_chunked參數決定是否分塊

pass

else:

# 只要給了content-length,那么一定不是分塊傳輸

encode_chunked = False

...

@staticmethod

def _get_content_length(body: Union[str, bytes, bytearray, Iterable, io.IOBase], method: str) -> Optional[int]:

if body is None:

# PUT,POST,PATCH三個方法默認是有body的

if method.upper() in _METHODS_EXPECTING_BODY:

return 0

else:

return None

if hasattr(body, 'read'):

return None

try:

# 對于bytes或者bytearray格式的數據,通過memoryview獲取它的長度

return memoryview(body).nbytes

except TypeError:

pass

if isinstance(body, str):

return len(body)

return None

在確定了是否分塊之后, 就可以把正文發出去了. 如果body是一個可讀文件的話, 就調用_read_readable方法把它封裝為一個生成器:

def _send_body(self, message_body: Union[str, bytes, bytearray, Iterable, io.IOBase], encode_chunked: bool) -> None:

if hasattr(message_body, 'read'):

chunks = self._read_readable(message_body)

else:

try:

memoryview(message_body)

except TypeError:

try:

chunks = iter(message_body)

except TypeError:

raise TypeError(

f'message_body should be a bytes-like object or an iterable, got {repr(type(message_body))}')

else:

# 如果是字節類型的,通過一次迭代把它發出去

chunks = (message_body,)

for chunk in chunks:

if not chunk:

continue

if encode_chunked:

chunk = f'{len(chunk):X}\r\n'.encode('ascii') + chunk + b'\r\n'

self.send(chunk)

if encode_chunked:

self.send(b'0\r\n\r\n')

def _read_readable(self, readable: io.IOBase) -> Generator[bytes, None, None]:

need_encode = False

if isinstance(readable, io.TextIOBase):

need_encode = True

while True:

data_block = readable.read(self.block_size)

if not data_block:

break

if need_encode:

data_block = data_block.encode('utf-8')

yield data_block

二. 獲取響應數據

HTTP響應報文的格式與請求報文大同小異, 它大致是這樣的:

因此, 我們只要用HTTPConnection的socket對象讀取服務器發送的數據, 然后按照上面的格式對數據進行解析就行了.

1. HTTPResponse類

我們首先定義一個簡單的HTTPResponse類. 它的屬性大致上就是socket的文件對象以及一些請求的信息等等, 調用它的begin方法來解析響應行和響應頭的數據, 然后調用read方法讀取響應正文:

class HTTPResponse:

def __init__(self, sock: socket.socket, method: str = None) -> None:

self.fp = sock.makefile('rb')

self._method = method

self.headers = None

self.version = _UNKNOWN

self.status = _UNKNOWN

self.reason = _UNKNOWN

self.chunked = _UNKNOWN

self.chunk_left = _UNKNOWN

self.length = _UNKNOWN

self.will_close = _UNKNOWN

def begin(self) -> None:

...

def read(self, amount: int = None) -> bytes:

...

2. 解析狀態行

狀態行的解析比較簡單, 我們只需要讀取響應的第一行數據, 然后把它解析為HTTP協議版本,狀態碼和原因短語三部分就行了:

def _read_status(self) -> Tuple[str, int, str]:

line = str(self._read_line(), 'latin-1')

if not line:

raise RemoteDisconnected('Remote end closed connection without response')

try:

version, status, reason = line.split(None, 2)

except ValueError:

# reason只是給人看的, 一般和status對應, 所以它有可能不存在

try:

version, status = line.split(None, 1)

reason = ''

except ValueError:

version, status, reason = '', '', ''

if not version.startswith('HTTP/'):

self._close_conn()

raise BadStatusLine(line)

try:

status = int(status)

if status < 100 or status > 999:

raise BadStatusLine(line)

except ValueError:

raise BadStatusLine(line)

return version, status, reason.strip()

如果狀態碼為100, 則客戶端需要解析多個響應狀態行. 它的原理是這樣的: 在請求數據過大的時候, 有的客戶端會先不發送請求數據, 而是先在header中添加一個Expect: 100-continue, 如果服務器愿意接收數據, 會返回100的狀態碼, 這時候客戶端再把數據發過去. 因此, 如果讀取到100的狀態碼, 那么后面往往還會收到一個正式的響應數據, 應該繼續讀取響應頭. 這部分的代碼如下:

def begin(self) -> None:

while True:

version, status, reason = self._read_status()

if status != HTTPStatus.CONTINUE:

break

# 跳過100狀態碼部分的響應頭

while True:

skip = self._read_line().strip()

if not skip:

breakself.status = status

self.reason = reason

if version in ('HTTP/1.0', 'HTTP/0.9'):

self.version = 10

elif version.startswith('HTTP/1.'):

self.version = 11

else:

# HTTP2還沒研究, 這里就不寫了

raise UnknownProtocol(version)

...

3. 解析響應頭

解析響應頭比響應行還要簡單. 因為每個header字段占一行, 我們只需要一直調用read_line方法讀取字段, 直到讀完header為止就行了.

def _parse_header(self) -> None:

headers = {}

while True:

line = self._read_line()

if len(headers) > _MAX_HEADERS:

raise HTTPException('got more than %d headers' % _MAX_HEADERS)

if line in _EMPTY_LINE:

break

line = line.decode('latin-1')

i = line.find(':')

if i == -1:

raise BadHeaderLine(line)

# 這里默認沒有重名的情況

key, value = line[:i].lower(), line[i + 1:].strip()

headers[key] = value

self.headers = headers

4. 接收響應正文

在接收響應正文之前, 首先要確定它的傳輸方式和長度:

def _set_chunk(self) -> None:

transfer_encoding = self.get_header('transfer-encoding')

if transfer_encoding and transfer_encoding.lower() == 'chunked':

self.chunked = True

self.chunk_left = None

else:

self.chunked = False

def _set_length(self) -> None:

# 首先要知道數據是否是分塊傳輸的

if self.chunked == _UNKNOWN:

self._set_chunk()

# 如果狀態碼是1xx或者204(無響應內容)或者304(使用上次緩存的內容),則沒有響應正文

# 如果這是個HEAD請求,那么也不能有響應正文

if (self.status == HTTPStatus.NO_CONTENT or

self.status == HTTPStatus.NOT_MODIFIED or

100 <= self.status < 200 or

self._method == 'HEAD'):

self.length = 0

return

length = self.get_header('content-length')

if length and not self.chunked:

try:

self.length = int(length)

except ValueError:

self.length = None

else:

if self.length < 0:

self.length = None

else:

self.length = None

然后, 我們實現一個read方法, 從body中讀取指定大小的數據:

def read(self, amount: int = None) -> bytes:

if self.is_closed():

return b''

if self._method == 'HEAD':

self.close()

return b''

if amount is None:

return self._read_all()

return self._read_amount(amount)

如果沒有指定需要的數據大小, 就默認讀取所有數據:

def _read_all(self) -> bytes:

if self.chunked:

return self._read_all_chunk()

if self.length is None:

s = self.fp.read()

else:

try:

s = self._read_bytes(self.length)

except IncompleteRead:

self.close()

raise

self.length = 0

self.close()

return s

def _read_all_chunk(self) -> bytes:

assert self.chunked != _UNKNOWN

value = []

try:

while True:

chunk = self._read_chunk()

if chunk is None:

break

value.append(chunk)

return b''.join(value)

except IncompleteRead:

raise IncompleteRead(b''.join(value))

def _read_chunk(self) -> Optional[bytes]:

try:

chunk_size = self._read_chunk_size()

except ValueError:

raise IncompleteRead(b'')

if chunk_size == 0:

self._read_and_discard_trailer()

self.close()

return None

chunk = self._read_bytes(chunk_size)

# 每塊的結尾會有一個\r\n,這里把它讀掉

self._read_bytes(2)

return chunk

def _read_chunk_size(self) -> int:

line = self._read_line(error_message='chunk size')

i = line.find(b';')

if i >= 0:

line = line[:i]

try:

return int(line, 16)

except ValueError:

self.close()

raise

def _read_and_discard_trailer(self) -> None:

# chunk的尾部可能會掛一些額外的信息,比如MD5值,過期時間等等,一般會在header中用trailer字段說明

# 當chunk讀完之后調用這個函數, 這些信息就先舍棄掉得了

while True:

line = self._read_line(error_message='chunk size')

if line in _EMPTY_LINE:

break

否則的話, 就讀取部分數據, 如果正好是分塊數據的話, 就比較復雜了. 簡單來說, 就是用bytearray制造一個所需大小的數組, 然后依次讀取chunk把數據往里面填, 直到填滿或者沒數據為止.? 然后用chunk_left記錄下當前塊剩余的量, 以便下次讀取.

def _read_amount(self, amount: int) -> bytes:

if self.chunked:

return self._read_amount_chunk(amount)

if isinstance(self.length, int) and amount > self.length:

amount = self.length

container = bytearray(amount)

n = self.fp.readinto(container)

if not n and container:

# 如果讀不到字節了,也就可以關了

self.close()

elif self.length is not None:

self.length -= n

if not self.length:

self.close()

return memoryview(container)[:n].tobytes()

def _read_amount_chunk(self, amount: int) -> bytes:

# 調用這個方法,讀取amount大小的chunk類型數據,不足就全部讀取

assert self.chunked != _UNKNOWN

total_bytes = 0

container = bytearray(amount)

mvb = memoryview(container)

try:

while True:

# mvb可以理解為容器的空的那一部分

# 這里一直調用_full_readinto把數據填進去,讓mvb越來越小,同時記錄填入的量

# 等沒數據或者當前數據足夠把mvb填滿之后,跳出循環

chunk_left = self._get_chunk_left()

if chunk_left is None:

break

if len(mvb) <= chunk_left:

n = self._full_readinto(mvb)

self.chunk_left = chunk_left - n

total_bytes += n

break

temp_mvb = mvb[:chunk_left]

n = self._full_readinto(temp_mvb)

mvb = mvb[n:]

total_bytes += n

self.chunk_left = 0

except IncompleteRead:

raise IncompleteRead(bytes(container[:total_bytes]))

return memoryview(container)[:total_bytes].tobytes()

def _full_readinto(self, container: memoryview) -> int:

# 返回讀取的量.如果沒能讀滿,這個方法會報警

amount = len(container)

n = self.fp.readinto(container)

if n < amount:

raise IncompleteRead(bytes(container[:n]), amount - n)

return n

def _get_chunk_left(self) -> Optional[int]:

# 如果當前塊讀了一半,那么直接返回self.chunk_left就行了

# 否則,有三種情況

# 1). chunk_left為None,說明body壓根沒開始讀,于是返回當前這一整塊的長度

# 2). chunk_left為0,說明這塊讀完了,于是返回下一塊的長度

# 3). body數據讀完了,返回None,順便做好善后工作

chunk_left = self.chunk_left

if not chunk_left:

if chunk_left == 0:

# 如果剩余零,說明上一塊已經讀完了,這里把\r\n讀掉

# 如果是None,就說明chunk壓根沒開始讀

self._read_bytes(2)

try:

chunk_left = self._read_chunk_size()

except ValueError:

raise IncompleteRead(b'')

if chunk_left == 0:

self._read_and_discard_trailer()

self.close()

chunk_left = None

self.chunk_left = chunk_left

return chunk_left

三. 復用TCP連接

HTTP通信本質上是基于TCP連接發送和接收HTTP請求和響應, 因此, 只要TCP連接不斷開, 我們就可以繼續用它進行HTTP請求, 這樣就避免了創建和銷毀TCP連接產生的消耗.

1. 判斷連接是否會斷開

在下面幾種情況中, 服務端會自動斷開連接:

HTTP協議小于1.1且沒有在頭部設置了keep-alive

HTTP協議大于等于1.1但是在頭部設置了connection: close

數據沒有分塊傳輸, 也沒有說明數據的長度, 這種情況下, 服務器一般會在發送完成后斷開連接, 讓客戶端知道數據發完了

根據上面列出來的幾種情況, 通過下面的代碼來判斷連接是否會斷開:

def _check_close(self) -> bool:

conn = self.get_header('connection')

if not self.chunked and self.length is None:

return True

if self.version == 11:

if conn and 'close' in conn.lower():

return True

return False

else:

if self.headers.get('keep-alive'):

return False

if conn and 'keep-alive' in conn.lower():

return False

return True

2. 正確地關閉HTTPResponse對象

由于TCP連接的復用, 一個HTTPConnection可以產生多個HTTPResponse對象, 而這些對象在同一個TCP連接上, 會共用這個連接的讀緩沖區. 這就導致, 如果上一個HTTPResponse對象沒有把它的那部分數據讀完, 就會對下一個響應產生影響.

另一方面來看, 我們也需要及時地關閉與這個TCP關聯的文件對象來避免占用資源. 因此, 我們定義如下的close方法關閉一個HTTPResponse對象:

def close(self) -> None:

if self.is_closed():

return

fp = self.fp

self.fp = None

fp.close()

def is_closed(self) -> bool:

return self.fp is None

用戶調用HTTPResponse對象的read方法, 把緩沖區數據讀完之后, 就會自動調用close方法(具體實現見上一章的第四節: 讀取響應數據這部分). 因此, 在獲取下一個響應數據之前, 我們只需要調用這個對象的is_closed方法, 就能判斷讀緩沖區是否已經讀完, 能否繼續接收響應了.

3. HTTP請求的生命周期

不使用管道機制的話, 不同的HTTP請求必須按次序進行, 相互之間不能重疊. 基于這個原因, 我們為HTTPConnection對象設置IDLE, REQ_STARTED和REQ_SENT三種狀態, 一個完整的請求應該經歷這幾種狀態:

根據上面的流程, 對HTTPConnection中對應的方法進行修改:

def get_response(self) -> HTTPResponse:

if self._response and self._response.is_closed():

self._response = None

if self._state != _CS_REQ_SENT or self._response:

raise ResponseNotReady(self._state)

response = HTTPResponse(self.sock, method=self._method)

try:

try:

response.begin()

except ConnectionError:

self.close()

raise

assert response.will_close != _UNKNOWN

self._state = _CS_IDLE

if response.will_close:

self.close()

else:

self._response = response

return response

except Exception as _:

response.close()

raise

def put_request(self, method: str, url: str) -> None:

# 調用這個函數開始新一輪的請求,它負責寫好請求行輸出到緩存里面去

# 調用它的前提是當前處于空閑狀態

# 如果之前的response還在并且已結束,會自動把它消除掉

if self._response and self._response.is_closed():

self._response = None

if self._state == _CS_IDLE:

self._state = _CS_REQ_STARTED

else:

raise CannotSendRequest(self._state)

...

def put_header(self, header: Union[bytes, str], value: Union[bytes, str, int]) -> None:

if self._state != _CS_REQ_STARTED:

raise CannotSendHeader()

...

def end_headers(self, message_body=None, encode_chunked=False) -> None:

if self._state == _CS_REQ_STARTED:

self._state = _CS_REQ_SENT

else:

raise CannotSendHeader()

...

需要注意的是, 如果第二個請求已經進入到獲取響應的階段了, 而上一個請求的響應還沒關閉, 那么就應該直接報錯, 否則讀取到的會是上一個請求剩余的響應部分數據, 導致解析響應出現問題.

事實上, HTTP1.1開始支持管道化技術, 也就是一次提交多個HTTP請求, 然后等待響應, 而不是在接收到上一個請求的響應后, 才發送后面的請求.

基于這種處理模式, 管道化技術理論上可以減少IO時間的損耗, 提升效率, 不過, 需要服務端的支持, 而且會增加程序的復雜程度, 這里就不實現了.

四. 總結

1. 完整代碼

HTTPConnection的完整代碼如下:

class HTTPConnection:

default_port = 80

_http_vsn = 11

_http_vsn_str = 'HTTP/1.1'

def __init__(self, host: str, port: int = None) -> None:

self.sock = None

self._buffer = []

self.host = host

self.port = port if port is not None else self.default_port

self._state = _CS_IDLE

self._response = None

self._method = None

self.block_size = 8192

def request(self, method: str, url: str, headers: dict = None, body: Union[io.IOBase, Iterable] = None,

encode_chunked: bool = False) -> None:

self.put_request(method, url)

headers = headers or {}

header_names = frozenset(k.lower() for k in headers.keys())

if 'host' not in header_names:

self._add_host(url)

if 'content-length' not in header_names:

if 'transfer-encoding' not in header_names:

encode_chunked = False

content_length = self._get_content_length(body, method)

if content_length is None:

if body is not None:

encode_chunked = True

self.put_header('Transfer-Encoding', 'chunked')

else:

self.put_header('Content-Length', str(content_length))

else:

# 如果設置了transfer-encoding,則根據用戶給的encode_chunked參數決定是否分塊

pass

else:

# 只要給了content-length,那么一定不是分塊傳輸

encode_chunked = False

for hdr, value in headers.items():

self.put_header(hdr, value)

if isinstance(body, str):

body = _encode(body)

self.end_headers(body, encode_chunked=encode_chunked)

def send(self, data: bytes) -> None:

if self.sock is None:

self.connect()

self.sock.sendall(data)

def get_response(self) -> HTTPResponse:

if self._response and self._response.is_closed():

self._response = None

if self._state != _CS_REQ_SENT or self._response:

raise ResponseNotReady(self._state)

response = HTTPResponse(self.sock, method=self._method)

try:

try:

response.begin()

except ConnectionError:

self.close()

raise

assert response.will_close != _UNKNOWN

self._state = _CS_IDLE

if response.will_close:

self.close()

else:

self._response = response

return response

except Exception as _:

response.close()

raise

def connect(self) -> None:

self.sock = socket.create_connection((self.host, self.port))

def close(self) -> None:

self._state = _CS_IDLE

try:

sock = self.sock

if sock:

self.sock = None

sock.close()

finally:

response = self._response

if response:

self._response = None

response.close()

def put_request(self, method: str, url: str) -> None:

# 調用這個函數開始新一輪的請求,它負責寫好請求行輸出到緩存里面去

# 調用它的前提是當前處于空閑狀態

# 如果之前的response還在并且已結束,會自動把它消除掉

if self._response and self._response.is_closed():

self._response = None

if self._state == _CS_IDLE:

self._state = _CS_REQ_STARTED

else:

raise CannotSendRequest(self._state)

self._method = method

url = url or '/'

request = f'{method} {url} {self._http_vsn_str}'

self._output(request)

def put_header(self, header: Union[bytes, str], value: Union[bytes, str, int]) -> None:

if self._state != _CS_REQ_STARTED:

raise CannotSendHeader()

if hasattr(header, 'encode'):

header = header.encode('ascii')

if hasattr(value, 'encode'):

value = value.encode('latin-1')

elif isinstance(value, int):

value = str(value).encode('ascii')

header = header + b': ' + value

self._output(header)

def end_headers(self, message_body=None, encode_chunked=False) -> None:

if self._state == _CS_REQ_STARTED:

self._state = _CS_REQ_SENT

else:

raise CannotSendHeader()

self._send_output(message_body, encode_chunked=encode_chunked)

def _add_host(self, url: str) -> None:

# 所有HTTP / 1.1請求報文中必須包含一個Host頭字段

# 如果用戶沒給,就調用這個函數來生成

netloc = ''

if url.startswith('http'):

nil, netloc, nil, nil, nil = urlsplit(url)

if netloc:

try:

netloc_enc = netloc.encode('ascii')

except UnicodeEncodeError:

netloc_enc = netloc.encode('idna')

self.put_header('Host', netloc_enc)

else:

host = self.host

port = self.port

try:

host_enc = host.encode('ascii')

except UnicodeEncodeError:

host_enc = host.encode('idna')

# 對IPv6的地址進行額外處理

if host.find(':') >= 0:

host_enc = b'[' + host_enc + b']'

if port == self.default_port:

self.put_header('Host', host_enc)

else:

host_enc = host_enc.decode('ascii')

self.put_header('Host', f'{host_enc}:{port}')

def _output(self, s: Union[str, bytes]) -> None:

# 將數據添加到緩沖區

if hasattr(s, 'encode'):

s = s.encode('latin-1')

self._buffer.append(s)

def _send_output(self, message_body=None, encode_chunked=False) -> None:

# 發送并清空緩沖數據.然后,如果有請求正文,就也順便發送

self._buffer.extend((b'', b''))

msg = b'\r\n'.join(self._buffer)

self._buffer.clear()

self.send(msg)

if message_body is not None:

self._send_body(message_body, encode_chunked)

def _send_body(self, message_body: Union[bytes, str, bytearray, Iterable, io.IOBase], encode_chunked: bool) -> None:

if hasattr(message_body, 'read'):

chunks = self._read_readable(message_body)

else:

try:

memoryview(message_body)

except TypeError:

try:

chunks = iter(message_body)

except TypeError:

raise TypeError(

f'message_body should be a bytes-like object or an iterable, got {repr(type(message_body))}')

else:

# 如果是字節類型的,通過一次迭代把它發出去

chunks = (message_body,)

for chunk in chunks:

if not chunk:

continue

if encode_chunked:

chunk = f'{len(chunk):X}\r\n'.encode('ascii') + chunk + b'\r\n'

self.send(chunk)

if encode_chunked:

self.send(b'0\r\n\r\n')

def _read_readable(self, readable: io.IOBase) -> Generator[bytes, None, None]:

need_encode = False

if isinstance(readable, io.TextIOBase):

need_encode = True

while True:

data_block = readable.read(self.block_size)

if not data_block:

break

if need_encode:

data_block = data_block.encode('utf-8')

yield data_block

@staticmethod

def _get_content_length(body: Union[str, bytes, bytearray, Iterable, io.IOBase], method: str) -> Optional[int]:

if body is None:

# PUT,POST,PATCH三個方法默認是有body的

if method.upper() in _METHODS_EXPECTING_BODY:

return 0

else:

return None

if hasattr(body, 'read'):

return None

try:

# 對于bytes或者bytearray格式的數據,通過memoryview獲取它的長度

return memoryview(body).nbytes

except TypeError:

pass

if isinstance(body, str):

return len(body)

return None

HTTPResponse的完整代碼如下:

class HTTPResponse:

def __init__(self, sock: socket.socket, method: str = None) -> None:

self.fp = sock.makefile('rb')

self._method = method

self.headers = None

self.version = _UNKNOWN

self.status = _UNKNOWN

self.reason = _UNKNOWN

self.chunked = _UNKNOWN

self.chunk_left = _UNKNOWN

self.length = _UNKNOWN

self.will_close = _UNKNOWN

def begin(self) -> None:

if self.headers is not None:

return

self._parse_status_line()

self._parse_header()

self._set_chunk()

self._set_length()

self.will_close = self._check_close()

def _read_line(self, limit: int = _MAX_LINE + 1, error_message: str = '') -> bytes:

# 注意,這個方法默認不去除line尾部的\r\n

line = self.fp.readline(limit)

if len(line) > _MAX_LINE:

raise LineTooLong(error_message)

return line

def _read_bytes(self, amount: int) -> bytes:

data = self.fp.read(amount)

if len(data) < amount:

raise IncompleteRead(data, amount - len(data))

return data

def _parse_status_line(self) -> None:

while True:

version, status, reason = self._read_status()

if status != HTTPStatus.CONTINUE:

break

while True:

skip = self._read_line(error_message='header line').strip()

if not skip:

break

self.status = status

self.reason = reason

if version in ('HTTP/1.0', 'HTTP/0.9'):

self.version = 10

elif version.startswith('HTTP/1.'):

self.version = 11

else:

raise UnknownProtocol(version)

def _read_status(self) -> Tuple[str, int, str]:

line = str(self._read_line(error_message='status line'), 'latin-1')

if not line:

raise RemoteDisconnected('Remote end closed connection without response')

try:

version, status, reason = line.split(None, 2)

except ValueError:

# reason只是給人看的, 和status對應, 所以它有可能不存在

try:

version, status = line.split(None, 1)

reason = ''

except ValueError:

version, status, reason = '', '', ''

if not version.startswith('HTTP/'):

self.close()

raise BadStatusLine(line)

try:

status = int(status)

if status < 100 or status > 999:

raise BadStatusLine(line)

except ValueError:

raise BadStatusLine(line)

return version, status, reason.strip()

def _parse_header(self) -> None:

headers = {}

while True:

line = self._read_line(error_message='header line')

if len(headers) > _MAX_HEADERS:

raise HTTPException('got more than %d headers' % _MAX_HEADERS)

if line in _EMPTY_LINE:

break

line = line.decode('latin-1')

i = line.find(':')

if i == -1:

raise BadHeaderLine(line)

# 這里默認沒有重名的情況

key, value = line[:i].lower(), line[i + 1:].strip()

headers[key] = value

self.headers = headers

def _set_chunk(self) -> None:

transfer_encoding = self.get_header('transfer-encoding')

if transfer_encoding and transfer_encoding.lower() == 'chunked':

self.chunked = True

self.chunk_left = None

else:

self.chunked = False

def _set_length(self) -> None:

# 首先要知道數據是否是分塊傳輸的

if self.chunked == _UNKNOWN:

self._set_chunk()

# 如果狀態碼是1xx或者204(無響應內容)或者304(使用上次緩存的內容),則沒有響應正文

# 如果這是個HEAD請求,那么也不能有響應正文

assert isinstance(self.status, int)

if (self.status == HTTPStatus.NO_CONTENT or

self.status == HTTPStatus.NOT_MODIFIED or

100 <= self.status < 200 or

self._method == 'HEAD'):

self.length = 0

return

length = self.get_header('content-length')

if length and not self.chunked:

try:

self.length = int(length)

except ValueError:

self.length = None

else:

if self.length < 0:

self.length = None

else:

self.length = None

def _check_close(self) -> bool:

conn = self.get_header('connection')

if not self.chunked and self.length is None:

return True

if self.version == 11:

if conn and 'close' in conn.lower():

return True

return False

else:

if self.headers.get('keep-alive'):

return False

if conn and 'keep-alive' in conn.lower():

return False

return True

def close(self) -> None:

if self.is_closed():

return

fp = self.fp

self.fp = None

fp.close()

def is_closed(self) -> bool:

return self.fp is None

def read(self, amount: int = None) -> bytes:

if self.is_closed():

return b''

if self._method == 'HEAD':

self.close()

return b''

if amount is None:

return self._read_all()

print(amount, amount is None)

return self._read_amount(amount)

def _read_all(self) -> bytes:

if self.chunked:

return self._read_all_chunk()

if self.length is None:

s = self.fp.read()

else:

try:

s = self._read_bytes(self.length)

except IncompleteRead:

self.close()

raise

self.length = 0

self.close()

return s

def _read_all_chunk(self) -> bytes:

assert self.chunked != _UNKNOWN

value = []

try:

while True:

chunk = self._read_chunk()

if chunk is None:

break

value.append(chunk)

return b''.join(value)

except IncompleteRead:

raise IncompleteRead(b''.join(value))

def _read_chunk(self) -> Optional[bytes]:

try:

chunk_size = self._read_chunk_size()

except ValueError:

raise IncompleteRead(b'')

if chunk_size == 0:

self._read_and_discard_trailer()

self.close()

return None

chunk = self._read_bytes(chunk_size)

# 每塊的結尾會有一個\r\n,這里把它讀掉

self._read_bytes(2)

return chunk

def _read_chunk_size(self) -> int:

line = self._read_line(error_message='chunk size')

i = line.find(b';')

if i >= 0:

line = line[:i]

try:

return int(line, 16)

except ValueError:

self.close()

raise

def _read_and_discard_trailer(self) -> None:

# chunk的尾部可能會掛一些額外的信息,比如MD5值,過期時間等等,一般會在header中用trailer字段說明

# 當chunk讀完之后調用這個函數, 這些信息就先舍棄掉得了

while True:

line = self._read_line(error_message='chunk size')

if line in _EMPTY_LINE:

break

def _read_amount(self, amount: int) -> bytes:

if self.chunked:

return self._read_amount_chunk(amount)

if isinstance(self.length, int) and amount > self.length:

amount = self.length

container = bytearray(amount)

n = self.fp.readinto(container)

if not n and container:

# 如果讀不到字節了,也就可以關了

self.close()

elif self.length is not None:

self.length -= n

if not self.length:

self.close()

return memoryview(container)[:n].tobytes()

def _read_amount_chunk(self, amount: int) -> bytes:

# 調用這個方法,讀取amount大小的chunk類型數據,不足就全部讀取

assert self.chunked != _UNKNOWN

total_bytes = 0

container = bytearray(amount)

mvb = memoryview(container)

try:

while True:

# mvb可以理解為容器的空的那一部分

# 這里一直調用_full_readinto把數據填進去,讓mvb越來越小,同時記錄填入的量

# 等沒數據或者當前數據足夠把mvb填滿之后,跳出循環

chunk_left = self._get_chunk_left()

if chunk_left is None:

break

if len(mvb) <= chunk_left:

n = self._full_readinto(mvb)

self.chunk_left = chunk_left - n

total_bytes += n

break

temp_mvb = mvb[:chunk_left]

n = self._full_readinto(temp_mvb)

mvb = mvb[n:]

total_bytes += n

self.chunk_left = 0

except IncompleteRead:

raise IncompleteRead(bytes(container[:total_bytes]))

return memoryview(container)[:total_bytes].tobytes()

def _full_readinto(self, container: memoryview) -> int:

# 返回讀取的量.如果沒能讀滿,這個方法會報警

amount = len(container)

n = self.fp.readinto(container)

if n < amount:

raise IncompleteRead(bytes(container[:n]), amount - n)

return n

def _get_chunk_left(self) -> Optional[int]:

# 如果當前塊讀了一半,那么直接返回self.chunk_left就行了

# 否則,有三種情況

# 1). chunk_left為None,說明body壓根沒開始讀,于是返回當前這一整塊的長度

# 2). chunk_left為0,說明這塊讀完了,于是返回下一塊的長度

# 3). body數據讀完了,返回None,順便做好善后工作

chunk_left = self.chunk_left

if not chunk_left:

if chunk_left == 0:

# 如果剩余零,說明上一塊已經讀完了,這里把\r\n讀掉

# 如果是None,就說明chunk壓根沒開始讀

self._read_bytes(2)

try:

chunk_left = self._read_chunk_size()

except ValueError:

raise IncompleteRead(b'')

if chunk_left == 0:

self._read_and_discard_trailer()

self.close()

chunk_left = None

self.chunk_left = chunk_left

return chunk_left

def get_header(self, name, default: str = None) -> Optional[str]:

if self.headers is None:

raise ResponseNotReady()

return self.headers.get(name, default)

@property

def info(self) -> str:

return repr(self.headers)

這兩個類應該放到同一個py文件中, 同時這個文件內還有其他一些輔助性質的代碼:

import io

import socket

from typing import Generator, Iterable, Optional, Tuple, Union

from urllib.parse import urlsplit

_CS_IDLE = 'Idle'

_CS_REQ_STARTED = 'Request-started'

_CS_REQ_SENT = 'Request-sent'

_METHODS_EXPECTING_BODY = {'PATCH', 'POST', 'PUT'}

_UNKNOWN = 'UNKNOWN'

_MAX_LINE = 65536

_MAX_HEADERS = 100

_EMPTY_LINE = (b'\r\n', b'\n', b'')

class HTTPStatus:

CONTINUE = 100

SWITCHING_PROTOCOLS = 101

PROCESSING = 102

OK = 200

CREATED = 201

ACCEPTED = 202

NON_AUTHORITATIVE_INFORMATION = 203

NO_CONTENT = 204

RESET_CONTENT = 205

PARTIAL_CONTENT = 206

MULTI_STATUS = 207

ALREADY_REPORTED = 208

IM_USED = 226

MULTIPLE_CHOICES = 300

MOVED_PERMANENTLY = 301

FOUND = 302

SEE_OTHER = 303

NOT_MODIFIED = 304

USE_PROXY = 305

TEMPORARY_REDIRECT = 307

PERMANENT_REDIRECT = 308

BAD_REQUEST = 400

UNAUTHORIZED = 401

PAYMENT_REQUIRED = 402

FORBIDDEN = 403

NOT_FOUND = 404

METHOD_NOT_ALLOWED = 405

NOT_ACCEPTABLE = 406

PROXY_AUTHENTICATION_REQUIRED = 407

REQUEST_TIMEOUT = 408

CONFLICT = 409

GONE = 410

LENGTH_REQUIRED = 411

PRECONDITION_FAILED = 412

REQUEST_ENTITY_TOO_LARGE = 413

REQUEST_URI_TOO_LONG = 414

UNSUPPORTED_MEDIA_TYPE = 415

REQUESTED_RANGE_NOT_SATISFIABLE = 416

EXPECTATION_FAILED = 417

MISDIRECTED_REQUEST = 421

UNPROCESSABLE_ENTITY = 422

LOCKED = 423

FAILED_DEPENDENCY = 424

UPGRADE_REQUIRED = 426

PRECONDITION_REQUIRED = 428

TOO_MANY_REQUESTS = 429

REQUEST_HEADER_FIELDS_TOO_LARGE = 431

UNAVAILABLE_FOR_LEGAL_REASONS = 451

INTERNAL_SERVER_ERROR = 500

NOT_IMPLEMENTED = 501

BAD_GATEWAY = 502

SERVICE_UNAVAILABLE = 503

GATEWAY_TIMEOUT = 504

HTTP_VERSION_NOT_SUPPORTED = 505

VARIANT_ALSO_NEGOTIATES = 506

INSUFFICIENT_STORAGE = 507

LOOP_DETECTED = 508

NOT_EXTENDED = 510

NETWORK_AUTHENTICATION_REQUIRED = 511

class HTTPResponse:

...

class HTTPConnection:

...

def _encode(data: str, encoding: str = 'latin-1', name: str = 'data') -> bytes:

# 給請求正文等不知道能怎么轉碼的東西轉碼時用這個,默認使用latin-1編碼

# 它的好處是,轉碼失敗后能拋出詳細的錯誤信息,一目了然

try:

return data.encode(encoding)

except UnicodeEncodeError as err:

raise UnicodeEncodeError(

err.encoding,

err.object,

err.start,

err.end,

"{} ({:.20!r}) is not valid {}. Use {}.encode('utf-8') if you want to send it encoded in UTF-8.".format(

name.title(), data[err.start:err.end], encoding, name)

) from None

class HTTPException(Exception):

pass

class ImproperConnectionState(HTTPException):

pass

class CannotSendRequest(ImproperConnectionState):

pass

class CannotSendHeader(ImproperConnectionState):

pass

class CannotCloseStream(ImproperConnectionState):

pass

class ResponseNotReady(ImproperConnectionState):

pass

class LineTooLong(HTTPException):

def __init__(self, line_type):

HTTPException.__init__(self, 'got more than %d bytes when reading %s'

% (_MAX_LINE, line_type))

class BadStatusLine(HTTPException):

def __init__(self, line):

if not line:

line = repr(line)

self.args = line,

self.line = line

class BadHeaderLine(HTTPException):

def __init__(self, line):

if not line:

line = repr(line)

self.args = line,

self.line = line

class RemoteDisconnected(ConnectionResetError, BadStatusLine):

def __init__(self, *args, **kwargs):

BadStatusLine.__init__(self, '')

ConnectionResetError.__init__(self, *args, **kwargs)

class UnknownProtocol(HTTPException):

def __init__(self, version):

self.args = version,

self.version = version

class UnknownTransferEncoding(HTTPException):

pass

class IncompleteRead(HTTPException):

def __init__(self, partial, expected=None):

self.args = partial,

self.partial = partial

self.expected = expected

def __repr__(self):

if self.expected is not None:

e = f', {self.expected} more expected'

else:

e = ''

return f'{self.__class__.__name__}({len(self.partial)} bytes read{e})'

__str__ = object.__str__

2. 需要注意的點

總的來說, 本文的內容不算復雜, 畢竟HTTP屬于不難理解, 但知識點很多很雜的類型. 這里把本文中一些需要注意的點總結一下:

請求和響應數據的結構大致相同, 都是狀態行+頭部+正文, 狀態行和頭部的每個字段都用一個\r\n分割, 與正文之間用兩個分割;

狀態行是必須的, 請求頭則最少需要host這個字段, 同時為了大家的方便, 你最好也設置一下Accept-encoding和Accept來限制服務器返回給你的數據內容和格式;

正文不是必須的, 特別是對于除了3P(PATCH, POST, PUT)之外的方法來說. 如果你有正文, 你最好在header中使用Content-Length說明正文的長度, 如果是分塊發送, 則使用Transfer-Encoding字段說明;

如果對正文使用分塊傳輸, 每塊的格式是: 16進制的數據長度+\r\n+數據+\r\n, 使用0\r\n\r\n來收尾. 收尾之后, 你還可以放一個trailer, 里面放數據的MD5值或者過期時間什么的, 這時候最好在header中設置trailer字段;

在一個請求的生命周期完成后, TCP連接是否會斷開取決于三點: 響應數據的HTTP版本, 響應頭中的Connection和Keep-Alive字段, 是否知道響應正文的長度;

最最重要的一點, HTTP協議只是一個約定而非限制, 這就和礦泉水的建議零售價差不多, 你可以選擇遵守, 也可以不遵守, 后果自負.

3. 結果測試

首先, 我們用tornado寫一個簡單的服務器, 它會顯示客戶端的地址和接口;

import tornado.web

import tornado.ioloop

class IndexHandler(tornado.web.RequestHandler):

def get(self) -> None:

print(f'new connection from {self.request.connection.context.address}')

self.write('hello world')

app = tornado.web.Application([(r'/', IndexHandler)])

app.listen(8888)

tornado.ioloop.IOLoop.current().start()

然后, 使用我們剛寫好的客戶端進行測試:

from client import HTTPConnection

def fetch(conn: HTTPConnection, url: str = '') -> None:

conn.request('GET', url)

res = conn.get_response()

print(res.read())

connection = HTTPConnection('127.0.0.1', 8888)

for i in range(10):

fetch(connection)

結果如下:

以上就是python用700行代碼實現http客戶端的詳細內容,更多關于python http客戶端的資料請關注WEB開發者其它相關文章!

總結

以上是生活随笔為你收集整理的python开发客户端_python用700行代码实现http客户端的全部內容,希望文章能夠幫你解決所遇到的問題。

如果覺得生活随笔網站內容還不錯,歡迎將生活随笔推薦給好友。

欧美色综合天天久久综合精品 | 免费在线观看亚洲视频 | 九九热视频在线免费观看 | 欧美性生爱| 日韩xxxx视频 | 欧美在线视频一区二区三区 | 国产成人区 | 99在线免费观看视频 | 91九色视频国产 | av电影在线观看完整版一区二区 | 日韩网站视频 | 国产97超碰 | 欧美一级裸体视频 | 国产精品男女 | 一区二区 精品 | 久久天天躁 | 久久久精品在线观看 | 美女在线观看av | 激情欧美日韩一区二区 | 久久99九九99精品 | 中文字幕成人网 | 色婷婷狠狠 | 91视频在线免费观看 | 国产黄色精品视频 | 久久99久久精品国产 | 97超碰资源总站 | 最近的中文字幕大全免费版 | 色综合天天天天做夜夜夜夜做 | 亚洲精品理论 | 黄色免费网站 | 成年人视频在线 | 天天操天天干天天操天天干 | 久热免费在线观看 | 国产精品美女在线 | 欧美 日韩 成人 | 精品一区二三区 | 日韩v欧美v日本v亚洲v国产v | 日韩在线视频免费观看 | 在线精品亚洲一区二区 | 国产精品久久久久久久毛片 | 国内精品久久久久影院优 | 97国产电影 | 国产青草视频在线观看 | 久久综合99 | 99精品欧美一区二区蜜桃免费 | 粉嫩一二三区 | 日日夜精品 | 日韩在线视频一区二区三区 | 久久综合色天天久久综合图片 | 日韩在线中文字幕视频 | 免费看一及片 | 在线中文字幕av观看 | 麻豆传媒一区二区 | 中文字幕在线观看视频一区 | 国产精品欧美久久 | 国产精品h在线观看 | 香蕉视频免费看 | 香蕉影院在线播放 | 在线免费三级 | 成人一级电影在线观看 | 91丨九色丨91啦蝌蚪老版 | 国模一区二区三区四区 | 日韩在线观看视频一区二区三区 | 欧美黑吊大战白妞欧美 | 国产最新视频在线 | 国产一区在线免费观看视频 | 精品久久久久久电影 | 精品二区久久 | av一级久久 | 极品久久久 | 国产视频在线看 | 天天天操天天天干 | 久久久久久久久亚洲精品 | 国产 中文 日韩 欧美 | av色一区| 精品自拍sae8—视频 | 国产欧美三级 | 最近高清中文字幕 | 久久久网| 99热最新 | 视频一区二区在线 | 中文av影院| 成人国产网址 | 国产成人免费在线观看 | 黄色电影在线免费观看 | 免费在线一区二区 | 特级a毛片 | 国产精品人人做人人爽人人添 | 欧美 高跟鞋交 xxxxhd | 中文字幕精品一区久久久久 | 国产麻豆剧果冻传媒视频播放量 | 精品国产免费看 | 日日干美女 | 亚洲国产日韩一区 | 在线免费观看视频一区二区三区 | 国产精品嫩草影视久久久 | 在线精品观看国产 | 国产成年免费视频 | 国产伦理一区二区三区 | 欧美精品在线观看一区 | 超碰在线人人 | 看v片 | 久久免费99精品久久久久久 | 日韩a级黄色片 | 在线视频日韩欧美 | 亚洲春色奇米影视 | 久久乐九色婷婷综合色狠狠182 | 91看片在线看片 | 天天操狠狠操 | 国内一区二区视频 | 日批视频在线 | 色中色资源站 | www色片| 日韩精品免费一区二区三区 | 黄色av一区二区三区 | 久久99国产精品 | 手机成人在线电影 | 亚洲动漫在线观看 | 国产精品欧美日韩在线观看 | 狠狠的操狠狠的干 | 国产亚洲精品女人久久久久久 | 日韩一区二区三区在线看 | 偷拍福利视频一区二区三区 | 五月婷婷中文 | 中文av字幕在线观看 | 国产午夜精品一区二区三区欧美 | 国产精品久久久久国产精品日日 | www亚洲国产 | 天天干婷婷 | 天干啦夜天干天干在线线 | 国产精品一区二区你懂的 | 激情小说久久 | 视频91| 日韩中文字幕a | av在线免费不卡 | 国产不卡在线播放 | 亚洲区色| 久久激情影院 | 久久久久亚洲精品中文字幕 | 国产在线欧美在线 | 嫩草伊人久久精品少妇av | 久草在线免费新视频 | 亚洲欧美综合精品久久成人 | 国产精品久久99精品毛片三a | 成年人视频在线免费观看 | 五月天婷婷丁香花 | 精品久久美女 | 精品一区 精品二区 | 中文字幕av一区二区三区四区 | 日韩中文久久 | 色网免费观看 | 99久久影院 | 国产在线播放一区二区 | 国产 字幕 制服 中文 在线 | 久久伊人免费视频 | 五月天激情婷婷 | 天天操天天操天天操天天操天天操天天操 | 国产精品免费在线播放 | 亚洲激情精品 | 在线观看免费色 | 久久婷婷色综合 | 国产人成一区二区三区影院 | 99久久精品免费看国产一区二区三区 | 开心激情综合网 | 四虎在线观看精品视频 | www.狠狠干| 日本久久久久久 | 精品国偷自产国产一区 | av 在线观看| 最近中文字幕在线中文高清版 | 99视频在线免费播放 | 美女久久一区 | 国产精品久久久久久久久久久久冷 | 国产片网站 | 久久国产高清视频 | 爱av在线网 | 久久久99精品免费观看乱色 | 精品国产伦一区二区三区观看体验 | 亚洲视频在线播放 | 成年免费在线视频 | 婷婷激情站| 天堂在线一区二区 | 韩国av一区二区 | 国产一级二级三级在线观看 | 99热在 | 又紧又大又爽精品一区二区 | 精品夜夜嗨av一区二区三区 | 国产一区二区不卡视频 | 国产在线 一区二区三区 | 国产盗摄精品一区二区 | 九九久| 精品国产91亚洲一区二区三区www | 久久久久久蜜桃一区二区 | 色午夜| 91你懂的| 天无日天天操天天干 | www狠狠| 美国av大片| 日韩在线第一区 | 96国产精品| 成人网页在线免费观看 | av激情五月 | 在线看片91 | 大胆欧美gogo免费视频一二区 | 亚洲精品在线免费 | 日韩av片无码一区二区不卡电影 | 少妇做爰k8经典 | 久久精品a | 久久视频二区 | 成人在线视频一区 | 色夜视频 | 天天操天天爽天天干 | 黄色片网站免费 | 免费精品视频在线观看 | 久久久私人影院 | 中文字幕 国产专区 | 91天堂素人约啪 | 国产精品久久久久国产a级 激情综合中文娱乐网 | 欧美日韩国产在线精品 | 午夜黄色| 日日摸日日 | 亚洲激色 | 亚洲经典在线 | 三上悠亚一区二区在线观看 | 国产69精品久久久久久 | 天天干夜夜爱 | 精品久久影院 | 正在播放一区二区 | 色综合久久悠悠 | 日韩中文字幕免费在线观看 | 欧美国产日韩一区二区三区 | 欧美激情一区不卡 | 国产精品高清免费在线观看 | 国产精品美女久久久久久久 | 91看国产| 中文电影网 | 国产一级h | 91精品色 | 69视频永久免费观看 | 毛片基地黄久久久久久天堂 | 日韩av电影手机在线观看 | 久久久久久久久久毛片 | 亚洲精品玖玖玖av在线看 | av线上看 | 色干综合 | 99情趣网视频 | 欧美日产在线观看 | 97视频成人 | 欧美日bb | 99久久久久免费精品国产 | 亚洲精品成人av在线 | 成人av免费看 | 中文字幕电影在线 | 国产99久久精品一区二区永久免费 | 久久在线播放 | 国产精品嫩草影院123 | 国产精品久久久毛片 | 国产精品淫 | 日日摸日日添夜夜爽97 | 久久96国产精品久久99软件 | 国产一区二区三区 在线 | 国产免费观看高清完整版 | 中文字幕网址 | 国产色综合天天综合网 | 九九在线免费视频 | 奇米四色影狠狠爱7777 | 免费看黄在线观看 | 国产亚洲精品福利 | 国产一级电影在线 | 能在线看的av | 亚洲精品456在线播放第一页 | 国产高清视频在线观看 | 91精品啪在线观看国产线免费 | 国产系列 在线观看 | www视频在线免费观看 | 午夜精品福利一区二区三区蜜桃 | 西西大胆啪啪 | 不卡av免费在线观看 | 91久久精品日日躁夜夜躁国产 | 国产成人一区二区三区免费看 | 精品亚洲成人 | 国产亚洲精品成人 | 91九色精品女同系列 | 国产中文字幕第一页 | 久久人人爽人人 | 精品中文字幕在线播放 | 99精品视频在线播放观看 | 日韩激情在线 | 国产一区二区在线免费 | 日韩视频免费观看高清 | 在线国产视频 | 欧美乱淫视频 | 91看片淫黄大片在线播放 | 久久人人精 | 99视频在线| 中文字幕电影网 | 午夜国产福利在线 | 精品久久久免费 | 色婷婷激情网 | 性色va| 婷婷国产v亚洲v欧美久久 | 久久国产精品视频观看 | 成人一级在线 | 国产精品一区二区久久久久 | 国产在线观看黄 | 二区三区中文字幕 | 精品视频在线视频 | 五月婷色| 成人av地址 | 欧美色图亚洲图片 | 国产在线视频在线观看 | 夜夜躁日日躁狠狠久久av | 久久综合99| 亚洲国产mv| 久久综合干 | 91精品国产欧美一区二区成人 | 中文字幕视频三区 | 天天操 夜夜操 | 国产成人精品一区二 | a天堂一码二码专区 | 免费亚洲黄色 | 成人av电影免费观看 | a v在线观看| 在线观看免费黄色 | 婷婷久久网站 | 中文国产字幕在线观看 | www.色午夜 | 丁香五月亚洲综合在线 | 91成人网在线观看 | 五月婷婷丁香 | 97视频在线观看视频免费视频 | 精品国产伦一区二区三区 | 亚洲国产精品免费 | 欧美aa一级 | 天天干天天做天天操 | 麻豆视频www | 最近更新好看的中文字幕 | 久久伊人91| 日韩欧美一区二区在线播放 | 成人在线观看免费 | 黄色录像av | 国产成人福利在线观看 | 亚洲人天堂 | 国产日韩中文字幕在线 | 国产手机视频在线播放 | 91久久爱热色涩涩 | 国产精品激情偷乱一区二区∴ | 久久久久免费精品国产 | 麻豆视频一区 | 日韩在线电影一区 | www.亚洲精品视频 | 免费中文字幕在线观看 | 91黄色免费网站 | 一区二区三高清 | 婷婷色在线播放 | 久久爱资源网 | 亚洲精品午夜视频 | 国产一区 在线播放 | 草久草久 | 亚洲一级在线观看 | 久久久精品一区二区 | 中文字幕在线免费观看 | 久久不射电影网 | 国产精品久久久久久久99 | 国产精品精品久久久久久 | 欧美aaa大片 | 午夜久久久影院 | 成人在线观看网址 | 激情欧美日韩一区二区 | 99热网站| 久久伊人91| 草久电影 | 日韩激情久久 | 欧美在线观看禁18 | 精品久久久久久久久久国产 | 国产精品情侣视频 | 中文字幕在线资源 | 久热久草在线 | 91久久精品一区二区三区 | 亚洲在线免费视频 | 国产精品99久久久久 | 精品一区二区影视 | 国产视频在线观看一区二区 | 四虎最新域名 | 亚洲四虎在线 | 成人影音av| 日韩专区一区二区 | 国产视频每日更新 | 色婷婷亚洲婷婷 | 国产视频丨精品|在线观看 国产精品久久久久久久久久久久午夜 | a天堂最新版中文在线地址 久久99久久精品国产 | 五月婷婷在线综合 | 色婷婷福利视频 | 麻豆成人小视频 | 在线av资源 | ,午夜性刺激免费看视频 | 亚洲精品短视频 | 国产成人高清av | 日韩中文字幕免费在线观看 | 中文字幕色在线视频 | 欧美日韩在线播放一区 | 在线中文字幕视频 | 久久综合色8888 | 日本xxxx裸体xxxx17 | 公与妇乱理三级xxx 在线观看视频在线观看 | 亚洲精品一区二区久 | 狠狠的干 | 亚洲一区精品人人爽人人躁 | 中国一级特黄毛片大片久久 | 久草在线资源视频 | 天天射天 | 五月婷婷色丁香 | 91精品国产99久久久久久久 | 中文字幕免费在线 | 99久热在线精品视频观看 | 2023av| 久久久亚洲国产精品麻豆综合天堂 | 久久免费视频99 | 天堂久色 | 欧美在线aa | 69精品久久久| 国模一二三区 | 日韩二三区 | 中文字幕精品三级久久久 | 欧美成年人在线观看 | 久久综合亚洲鲁鲁五月久久 | 中文字幕免费久久 | 18av在线视频 | 欧美 日韩 久久 | 欧美高清视频不卡网 | 私人av| 色www精品视频在线观看 | 99热99热 | 四虎5151久久欧美毛片 | 日韩欧美中文 | 在线va网站 | 亚洲成人精品在线 | 激情久久久久久久久久久久久久久久 | 天天综合入口 | 久久精品7| 亚洲一区黄色 | 欧美一级淫片videoshd | 日韩av快播电影网 | 免费午夜视频在线观看 | 久久久久久精 | www好男人| 91久久一区二区 | 婷婷伊人网 | 99这里只有精品视频 | 午夜影院先 | 国产日韩中文字幕在线 | 黄色免费国产 | 综合色狠狠 | 中文字幕一区二区在线播放 | 精品一二三区 | 国产v在线 | 国产精品第72页 | 成人在线播放网站 | 久久av免费| 久久久国产精品网站 | 欧美精品黑人性xxxx | 精品一区欧美 | 亚洲国产中文字幕在线视频综合 | 亚洲成av人片在线观看 | 国产视频久久久 | 五月天婷婷免费视频 | 综合久久综合久久 | 久久久久激情视频 | 久久9视频| 中文字幕在线播放日韩 | 国产大陆亚洲精品国产 | 欧美日韩一级久久久久久免费看 | 爱爱一区| 国内精品小视频 | 韩国一区二区在线观看 | av免费在线观 | 人人插人人澡 | 国产91粉嫩白浆在线观看 | 国产精品高清一区二区三区 | 成人av资源网站 | 亚洲精品美女久久久 | 国产日韩视频在线观看 | 91av成人| 亚洲专区欧美专区 | 97在线免费| 亚洲另类人人澡 | 亚洲在线日韩 | 精品久久久久久久久中文字幕 | 九色精品免费永久在线 | 久久99国产视频 | 国产成人一区二区三区免费看 | 在线国产中文字幕 | 久久综合久久综合这里只有精品 | 少妇18xxxx性xxxx片| av观看久久久 | 天天色成人 | 蜜臀久久99精品久久久无需会员 | 欧洲亚洲激情 | 久久综合色影院 | 欧美亚洲免费在线一区 | 中文字幕一区二区三区在线播放 | 伊人伊成久久人综合网小说 | 国产精品一区二区三区视频免费 | 中文字幕久久精品 | 99久精品| 久久网站最新地址 | av三级在线免费观看 | 国产一级二级在线播放 | www五月天com| 综合色久 | 狠狠干在线 | 欧美伦理一区 | 国产香蕉视频在线观看 | 91免费高清观看 | 亚洲首页 | 在线国产91 | 久精品视频 | 国产成人精品一区二区三区 | 欧美一级视频一区 | 亚洲精品裸体 | 91av在线播放 | 天天摸日日摸人人看 | 免费在线播放av电影 | 国产特级毛片aaaaaa高清 | 日韩精品一区二区三区免费观看 | 国产精品久久久免费看 | 成人国产精品久久久 | 日日天天 | 久久精品欧美日韩精品 | 国产美女被啪进深处喷白浆视频 | 美女精品网站 | 亚洲一区不卡视频 | 99久久婷婷国产 | 亚洲成人精品国产 | 国产精品久久久久久影院 | 欧美一区二区三区在线视频观看 | 中文字幕日韩高清 | 欧美一区二视频在线免费观看 | 波多野结衣一区 | 成人午夜电影久久影院 | 99久久这里只有精品 | 色成人亚洲网 | 久久久久福利视频 | 免费日韩一级片 | 亚洲无吗视频在线 | 麻豆视频免费观看 | 欧美一区二区在线看 | 97精品国自产拍在线观看 | 色99久久| 天天天天天天干 | 日韩精品无 | 天天干,天天干 | 91精品国产91久久久久久三级 | 激情欧美xxxx| 九九久久成人 | 成人黄色小说视频 | 黄视频色网站 | av理论电影| 久久草在线精品 | 99精品视频在线观看 | 亚洲欧美日韩在线一区二区 | 日产乱码一二三区别免费 | 欧美国产日韩一区二区三区 | 在线观看黄色的网站 | 亚洲综合在线播放 | www.亚洲激情.com | 黄色国产高清 | 亚洲无毛专区 | 国产成人在线精品 | 天天干,夜夜操 | 国产成人精品一区一区一区 | 国产色就色| 日韩av中文字幕在线 | 国产精品一区在线观看你懂的 | 国产日韩精品一区二区在线观看播放 | 久久国产精品成人免费浪潮 | 免费看高清毛片 | 日韩电影精品 | 久久人人97超碰国产公开结果 | 日韩精品在线免费播放 | 免费在线播放黄色 | 久久免费国产精品1 | 中文字幕在线视频一区二区三区 | 亚洲视频精选 | 久久精品精品电影网 | 久久久国产精品视频 | 在线免费观看国产视频 | 玖玖玖在线| 特级毛片在线免费观看 | 欧美专区日韩专区 | 中文字幕在线观看视频一区二区三区 | 粉嫩av一区二区三区免费 | 青青河边草免费观看 | 四虎免费在线观看视频 | 国产久草在线 | 日韩欧美大片免费观看 | 午夜精品在线看 | 青青看片| 国产中文欧美日韩在线 | 久久综合色影院 | 亚洲精品毛片一级91精品 | 国产中文视 | 国内99视频 | 天天色天天射天天综合网 | 国产精品女主播一区二区三区 | 精品在线视频一区 | 人人超碰人人 | 久久综合九色欧美综合狠狠 | 国产字幕在线观看 | 国产亚洲欧美日韩高清 | 国产成人精品综合久久久久99 | 91一区啪爱嗯打偷拍欧美 | 美女免费视频观看网站 | 五月婷婷丁香六月 | 日韩免费观看高清 | 亚洲欧美视频一区二区三区 | www国产亚洲 | 欧美日韩一区二区在线观看 | 2019天天干夜夜操 | 在线黄色免费av | 亚洲国产精品一区二区久久,亚洲午夜 | 欧美另类xxxxx | 日韩毛片在线一区二区毛片 | 精品美女在线观看 | 日本字幕网 | 综合久久久久久 | 91看片成人 | 麻豆91在线 | 中文字幕免费 | 国产精品久久久久国产a级 激情综合中文娱乐网 | 美女免费视频网站 | 日韩久久精品一区 | 深夜成人av | 一级免费观看 | 视频福利在线观看 | 亚洲精品福利在线观看 | 亚洲粉嫩av | 国产一级片播放 | 国产免费又粗又猛又爽 | 色天天综合久久久久综合片 | 日本 在线 视频 中文 有码 | 久久免费成人 | 久久婷婷国产色一区二区三区 | 在线看日韩av | 日日摸日日 | 99操视频 | 久久综合九色 | 久久久久久黄色 | 99久久精品免费看国产四区 | av天天草 | 中文字幕a∨在线乱码免费看 | 国产一级黄色免费看 | 日韩精品播放 | 国产一级片一区二区三区 | 中文字幕在线第一页 | 99精品免费 | 在线观看一级 | 亚洲精品理论 | 伊人色综合久久天天 | 日韩欧在线 | 国产成人91 | 亚洲国产精品电影在线观看 | 亚洲精品在线一区二区 | 国产成人精品一区一区一区 | 精品国产一区二区三区不卡 | 麻豆精品视频 | 中文字幕专区高清在线观看 | 亚州av成人 | 日日夜夜天天操 | 又黄又爽又湿又无遮挡的在线视频 | 亚洲精品在线一区二区三区 | 区一区二区三区中文字幕 | 日韩av视屏| 国产高清免费 | 国产高清在线a视频大全 | 欧美大片在线看免费观看 | av中文字幕在线看 | 黄色网中文字幕 | 日本中文字幕在线 | 99久久精品国产亚洲 | 日韩精品中文字幕一区二区 | 午夜av一区二区三区 | 精品嫩模福利一区二区蜜臀 | 午夜10000 | 午夜视频在线观看一区二区 | 亚洲视频在线免费看 | 精品不卡av| 久久精品久久久久电影 | 在线看福利av | 久久99九九99精品 | 91污在线 | 久久成人18免费网站 | 激情av综合 | 午夜丁香视频在线观看 | 欧美做受高潮1 | 99在线热播 | 91成人精品国产刺激国语对白 | 中文免费观看 | 亚洲日本中文字幕在线观看 | 天天综合视频在线观看 | 中日韩三级视频 | 欧美xxxx性xxxxx高清 | 五月开心六月伊人色婷婷 | 91精品福利在线 | 亚洲综合黄色 | 欧美做受xxx | 国产经典 欧美精品 | 亚洲好视频 | 91插插插免费视频 | 香蕉视频在线观看免费 | 麻豆一精品传二传媒短视频 | 狠狠天天 | 欧美日韩在线精品一区二区 | 色综合久久88色综合天天6 | 一区二区中文字幕在线观看 | 丰满少妇高潮在线观看 | 午夜 在线| 五月天激情视频 | 国产一区在线不卡 | 精品一区二区三区久久久 | 亚洲伊人婷婷 | 欧美日韩中文在线视频 | 色全色在线资源网 | 欧美aaaxxxx做受视频 | 国产精品乱码高清在线看 | av网站免费线看精品 | 亚洲 欧美 另类人妖 | 在线国产视频观看 | 啪啪av在线 | 免费日韩一区二区三区 | 亚洲美女精品 | 久久综合影音 | 久久久精品99 | 日韩一区二区在线免费观看 | 东方av免费在线观看 | 日日碰狠狠添天天爽超碰97久久 | 成年美女黄网站色大片免费看 | 日日碰狠狠躁久久躁综合网 | 国产午夜精品免费一区二区三区视频 | 丁香色天天 | 亚洲日本中文字幕在线观看 | 色在线视频 | 国产剧情在线一区 | 久久a v视频 | 毛片激情永久免费 | 日本精a在线观看 | 看国产黄色片 | 91精品视频导航 | 久久久首页 | 国产1区2区3区在线 亚洲自拍偷拍色图 | 黄色影院在线免费观看 | 深夜免费福利 | 日本久热| 欧美一级黄大片 | 国产精品久久久久久久7电影 | 免费色视频| 欧美整片sss | 国产韩国日本高清视频 | 久久66热这里只有精品 | 久久精品—区二区三区 | 免费av电影网站 | 久久久福利视频 | 又黄又刺激的视频 | 欧美三级在线播放 | 人人爽人人爽av | 中文在线字幕免费观看 | 国产精品porn | 99人成在线观看视频 | 美女网站视频一区 | 日韩精品不卡在线观看 | 99精品视频在线观看播放 | 黄色免费网站下载 | 久草在线网址 | 欧美a级在线播放 | 国产亚洲永久域名 | 狠狠色伊人亚洲综合网站野外 | 日韩成人欧美 | 免费视频区 | 久久久久久久久精 | www.久久免费| 日韩视频在线观看免费 | 夜夜骑日日 | 国产成人精品久 | 久久伦理 | 国产视频一区二区在线 | 99福利影院| 国产爽妇网 | 亚洲黄色在线观看 | 精品色综合 | 久久精品1区 | 亚洲视频,欧洲视频 | 人人澡超碰碰 | 久久久久久久久久久综合 | 国产最新在线视频 | 欧美成人高清 | 手机看片1042| 99久久99视频只有精品 | 日本黄色免费网站 | 免费观看mv大片高清 | 片黄色毛片黄色毛片 | 夜夜嗨av色一区二区不卡 | 国产97在线视频 | 国产99精品| 久久综合一本 | 国内成人av | www.国产毛片 | 麻豆91在线 | 伊人资源站 | 中文视频一区二区 | 国产尤物一区二区三区 | 中文字幕一区二区三区四区视频 | 日p在线观看| 久久 亚洲视频 | 奇米网8888 | 亚洲视频一区二区三区在线观看 | 国产精品女教师 | 欧美日韩亚洲一 | 欧美一级大片在线观看 | 国产手机视频在线 | 婷婷成人在线 | 人人狠狠综合久久亚洲 | 国产成人a v电影 | 国产最新视频在线 | 国产亚洲精品久久 | 亚洲婷婷伊人 | 久久99网 | 99精品国自产在线 | 国产在线视频在线观看 | 国产一区二区不卡视频 | 久久精品—区二区三区 | av+在线播放在线播放 | 成人福利在线 | 91在线观| 99久久这里只有精品 | 免费久久99精品国产婷婷六月 | 91看片淫黄大片在线播放 | 美女视频又黄又免费 | 99视频精品在线 | 久精品视频 | 久久成人综合视频 | 天天综合视频在线观看 | 最近能播放的中文字幕 | 欧美精品三级 | 婷婷婷国产在线视频 | 亚洲人成影院在线 | 日韩成人免费在线 | 日韩理论电影在线观看 | www.亚洲黄 | 亚洲综合少妇 | 伊人伊成久久人综合网站 | 福利电影久久 | 久久久久国产精品一区 | 成人毛片网 | 手机av电影在线 | 最近日本韩国中文字幕 | 久久精品视频2 | 五月婷婷.com | 国产在线最新 | 久久99精品久久久久久久久久久久 | 欧美一级特黄aaaaaa大片在线观看 | 免费在线观看av不卡 | 韩国av免费在线观看 | 欧美激情综合五月色丁香小说 | 国产精品一区二区三区四 | 综合在线色 | 手机成人在线 | 日韩亚洲国产中文字幕 | 国产精品成人av久久 | www日韩高清 | 中文字幕亚洲不卡 | av线上免费观看 | 爱爱av网 | 国产精品va | 亚洲伦理一区二区 | av高清在线 | 久久久久夜色 | 91片黄在线观看动漫 | 日韩av电影国产 | 亚洲国产操 | 成人网在线免费视频 | 日韩久久影院 | av中文字幕剧情 | 久久久国产电影 | 99久久精品视频免费 | 精品久久久久久国产 | 色在线观看网站 | 亚洲国产中文在线观看 | 99c视频高清免费观看 | 麻豆视频在线 | 亚洲在线高清 | 亚洲精品国产精品乱码在线观看 | av在线等| 国产精品久免费的黄网站 | 久久免费视频网站 | 99久久精品国产系列 | 欧美性脚交 | 午夜神马福利 | 国产麻豆成人传媒免费观看 | 国产精品美女久久久久久久久久久 | 日本黄色免费看 | 视频在线观看一区 | 又黄又刺激的网站 | 亚洲欧洲一区二区在线观看 | 日韩精品久久久久久久电影竹菊 | 99久久国产免费,99久久国产免费大片 | 天天干天天拍天天操天天拍 | 亚洲精品视频网站在线观看 | 日韩区在线观看 | 国产一二三区av | 欧美一级片 | 波多野结衣一区三区 | 日韩啪视频 | 97在线超碰 | 最近中文字幕在线中文高清版 | 国产中文在线字幕 | 草久中文字幕 | 久久久精品国产免费观看一区二区 | 欧美精品久久久久久久久免 | 黄色成人影院 | 国产99久| 日韩在线欧美在线 | 免费网站黄色 | 久草视频在线新免费 | 波多野结衣网址 | 免费在线激情电影 | 亚洲精品影视在线观看 | 东方av在| 国产高清日韩欧美 | 国产欧美精品在线观看 | 国产精品亚洲a | 在线影院 国内精品 | 久久久久久久久久福利 | 一区二区视频电影在线观看 | 97在线超碰 | 亚洲综合国产精品 | 最新国产精品拍自在线播放 | 久久久久国产精品视频 | 日韩免费b | 免费观看国产精品视频 | 亚洲经典在线 | 91视频麻豆视频 | 99精品视频一区二区 | 免费在线观看黄色网 | 免费在线观看a v | 亚洲精品在线免费看 | 婷婷丁香在线 | 国内丰满少妇猛烈精品播 | 日韩乱色精品一区二区 | 亚洲午夜精品久久久久久久久久久久 | 欧美精品一区在线 | 亚洲精品系列 | 国产亚洲片 | 97香蕉久久国产在线观看 | 国产精品久久久久久影院 | 黄色网在线播放 | 99免费视频 | 蜜桃视频在线视频 | 在线看片中文字幕 | 午夜久久久久久久久久久 | 波多野结衣视频一区二区 | 欧美一区二区三区免费看 | 97在线观看免费观看高清 | 丁香一区二区 | 99精品国产aⅴ | 国产91精品一区二区绿帽 | 国产69精品久久久久9999apgf | 国产 一区二区三区 在线 | 69中文字幕 | 视频一区二区免费 | 国产色妞影院wwwxxx | 国内久久 | av一级片网站 | 国产视频一 | 亚洲男男gaygay无套同网址 | 中文字幕在线免费看 | 成人九九视频 | 黄色毛片网站在线观看 | 国产成人在线观看 | 国产精品久久久精品 | 午夜精品久久久久久久99婷婷 | 超碰人人99| 免费h在线观看 | 一区二区三区电影 | 天天色天天操综合网 | av一级久久 | 伊人狠狠色丁香婷婷综合 | 99久久综合精品五月天 | www.夜夜操.com| 狠狠色丁香久久综合网 | 999视频网 | 天堂网av 在线| 日本中文字幕观看 | 深夜福利视频在线观看 | 天天综合91| 国产精品亚洲精品 | 欧美在线视频a | 一级精品视频在线观看宜春院 | 狠狠色狠狠色 |