python實現靜態web伺服器
HTTP協議簡介
HTTP請求
1:瀏覽器首先向伺服器傳送HTTP請求,請求包括:
方法:GET還是POST,GET僅請求資源,POST會附帶使用者資料;
路徑:/full/url/path;
域名:由Host頭指定:Host: www.sina.com以及其他相關的Header;
如果是POST,那麼請求還包括一個Body,包含使用者資料
2:伺服器向瀏覽器返回HTTP響應,響應包括:
響應程式碼:200表示成功,3xx表示重定向,4xx表示客戶端傳送的請求有錯誤,5xx表示伺服器端處理時發生了錯誤;
響應型別:由Content-Type指定;
以及其他相關的Header;
通常伺服器的HTTP響應會攜帶內容,也就是有一個Body,包含響應的內容,網頁的HTML原始碼就在Body中。
3:如果瀏覽器還需要繼續向伺服器請求其他資源,比如圖片,就再次發出HTTP請求,重複步驟1、2。
Web採用的HTTP協議採用了非常簡單的請求-響應模式,從而大大簡化了開發。當我們編寫一個頁面時,我們只需要在HTTP請求中把HTML傳送出去,不需要考慮如何附帶圖片、視訊等,瀏覽器如果需要請求圖片和視訊,它會發送另一個HTTP請求,因此,一個HTTP請求只處理一個資源(此時就可以理解為TCP協議中的短連線,每個連結只獲取一個資源,如需要多個就需要建立多個連結)
HTTP格式
每個HTTP請求和響應都遵循相同的格式,一個HTTP包含Header和Body兩部分,其中Body是可選的。
HTTP協議是一種文字協議,所以,它的格式也非常簡單。
1 HTTP GET請求的格式:
GET /path HTTP/1.1 Header1: Value1 Header2: Value2 Header3: Value3
每個Header一行一個,換行符是\r\n。
2 HTTP POST請求的格式:
POST /path HTTP/1.1 Header1: Value1 Header2: Value2 Header3: Value3 body data goes here...
3 HTTP響應的格式:
200 OK Header1: Value1 Header2: Value2 Header3: Value3 body data goes here...
HTTP響應如果包含body,也是通過\r\n\r\n來分隔的。
請再次注意,Body的資料型別由Content-Type頭來確定,如果是網頁,Body就是文字,如果是圖片,Body就是圖片的二進位制資料。
當存在Content-Encoding時,Body資料是被壓縮的,最常見的壓縮方式是gzip,所以,看到Content-Encoding: gzip時,需要將Body資料先解壓縮,才能得到真正的資料。壓縮的目的在於減少Body的大小,加快網路傳輸。
demo:靜態web伺服器
import socket import re import time def service_client(new_socket): """為這個客戶端服務""" # 1.接收瀏覽器傳送過來的請求,即http請求 # GET / HTTP/1.1 # -------- request = new_socket.recv(1024).decode('utf-8') # 判斷客戶端意外斷開連結返回空字串 if not request: # 關閉套接字並退出 new_socket.close() print("==="*30) return # 分隔套接字 request_lines = request.splitlines() print() print(">"*20) print(request_lines) file_name = "" ret = re.match(r"[^/]+(/[^ ]*)",request_lines[0]) if ret: file_name = ret.group(1) if file_name == "/": file_name = "/index.html" # 2.返回http格式資料 給瀏覽器 try: f = open("./html" + file_name,"rb") except: response = "HTTP/1.1 404 NOT FOUND\r\n" response += "Content-Type:text/html;charset=utf-8\r\n" response += "\r\n" response += "<h1>404 not found <br> 沒有發現所請求資源</h1>" response += str(time.strftime('%Y-%m-%d %H:%M:%S',time.localtime(time.time()))) new_socket.send(response.encode('utf-8')) else: html_content = f.read() f.close() # 2.1準備傳送給瀏覽器的資料---header response = "HTTP/1.1 200 OK\r\n" response += "\r\n" # 2.2準備傳送給瀏覽器的資料 ---body # 將response header傳送給瀏覽器 new_socket.send(response.encode("utf-8")) # 將response body傳送給瀏覽器 new_socket.send(html_content) # 關閉套接字 new_socket.close() def main(): """用來完成整體的控制""" # 1.建立套接字 tcp_server_socket = socket.socket(socket.AF_INET,socket.SOCK_STREAM) # 埠複用 tcp_server_socket.setsockopt(socket.SOL_SOCKET,socket.SO_REUSEADDR,1) # 2.繫結 tcp_server_socket.bind(("",7890)) # 3.變為套接字 tcp_server_socket.listen(128) while True: # 4.等待客戶端的連結 new_socket,client_addr = tcp_server_socket.accept() print(client_addr) # 5.為這個客戶端服務 service_client(new_socket) # 關閉監聽套接字 tcp_server_socket.close() if __name__ == '__main__': main()
以上就是本文的全部內容,希望對大家的學習有所幫助,也希望大家多多支援我們。