Python socket如何解析HTTP請求內容

socket解析HTTP請求內容

思路

1. 解析HTTP請求的頭部

HTTP請求頭部的結束符行為"\r\n",可以按行讀取HTTP請求頭的內容,如果讀到一行為"\r\n",說明HTTP請求頭結束。

2. 請求頭裡面含有Content-Length參數

如果HTTP請求裡面有Content-Length參數,說明HTTP請求的內容大小是確定的,請求直接讀取Content-Length的值,然後讀取相應字節的的內容即可。

3. 請求頭裡面含有Transfer-Encoding: chunked 參數

如果HTTP請求裡面有Transfer-Encoding參數,說明HTTP請求的內容大小是不確定的,這種內容的結束符是"0\r\n\r\n",因此可以按行讀取HTTP請求的內容部分,如果連續讀到"0\r\n"和"\r\n"說明內容讀取完畢。

代碼實現

代碼中: self._file 代表的是socket.makefile() 

 def get_http_content(self):
        content_length = 0
        transfer_encoding = False
        while True:
            req_line = self._file.readline()
            req_line = str(req_line, "utf-8")
 
            # 遇到http頭結束符
            # 讀取http內容
            if req_line == "\r\n":
                if content_length != 0:
                    content = self._file.read(content_length)
                    content = str(content, "utf-8")
                    self._content = content
                    return None
 
                if transfer_encoding:
                    content = ""
                    self._file.readline()
                    while True:
                        line = self._file.readline()
                        line = str(line, "utf-8")
                        if line == "0\r\n":
                            sub_line = self._file.readline()
                            sub_line = str(sub_line, "utf-8")
                            if sub_line == "\r\n":
                                self._content = content
                                return None
                        else:
                            content += line
                            continue
                    self._content = False
 
            # 頭文件沒有結束
            # 並且沒有找到關於內容大小的字段
            else:
                if content_length == 0 and transfer_encoding is False:
                    words = req_line.split()
                    if words[0] == "Content-Length:":
                        content_length = int(words[1])
                    if words[0] == "Transfer-Encoding:":
                        transfer_encoding = True
 
            self._content = False

socket 模擬http請求

# coding: utf-8
import socket
from urllib.parse import urlparse
def get_url(url):
    url = urlparse(url)
    host = url.netloc
    path = url.path
    if path == "":
        path = "/"
    # 建立 socket 連接
    client = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
    client.connect((host, 80))
    client.send("GET {} HTTP/1.1\r\nHost:{}\r\nConnection:close\r\n\r\n".format(path, host).encode("utf-8"))
    data = b""
    while True:
        d = client.recv(1024)
        if d:
            data += d
        else:
            break
    data = data.decode("utf-8")
    html_data = data.split("\r\n\r\n")[1]
    print(html_data)
    client.close()
    pass
if __name__ == '__main__':
    get_url("http://www.baidu.com")

以上為個人經驗,希望能給大傢一個參考,也希望大傢多多支持WalkonNet。

推薦閱讀: