Python 基于http.server模塊實現(xiàn)簡單http服務的代碼舉例
測試環(huán)境
win11專業(yè)版
python 3.9
代碼實現(xiàn)
# -*- coding:utf-8 -*-
import json
import traceback
import uuid
from http.server import HTTPServer, ThreadingHTTPServer, BaseHTTPRequestHandler
from urllib.parse import urlparse, parse_qs
class MyHTTPRequestHandler(BaseHTTPRequestHandler):
def _send_response(self, status_code, content_type, body):
'''發(fā)送響應信息'''
self.send_response(status_code)
self.send_header('Content-type', content_type)
self.send_header('Cache-Control', 'no-store') # 防止緩存舊編碼響應
self.end_headers()
self.wfile.write(body.encode('utf-8'))
def _handle_home(self):
'''訪問主頁請求處理'''
html = '<meta http-equiv="Content-Type" content="text/html; charset=utf-8"><h1>Home Page</h1>'
self._send_response(200, 'text/html; charset=utf-8', html)
def _handle_404(self):
'''請求不存在資源處理'''
# json_respose = {"success": False, "message": "Not Found"}
# self._send_response(404, 'application/json;charset=utf-8', json.dumps(json_respose))
# Content-Type 指定 charset=utf-8 可避免瀏覽器GET請求,界面中文顯示亂碼問題
self._send_response(404, 'text/html; charset=utf-8', "<h1>404 Not Found</h1>")
def _handle_login(self, login_req_data):
'''處理登錄請求'''
try:
data = json.loads(login_req_data)
username = data.get('username')
password = data.get('password')
ip = data.get('ip')
response = {
'code': 0,
'token': uuid.uuid4().hex,
'description': 'success'
}
self._send_response(200, 'application/json; charset=utf-8', json.dumps(response))
except Exception as e:
error_msg = traceback.format_exc()
print(error_msg)
response = {
'code': 1,
'token': '',
'description': error_msg
}
self._send_response(500, 'application/json; charset=utf-8', json.dumps(response))
def do_GET(self):
'''處理GET請求'''
parsed_path = urlparse(self.path)
path = parsed_path.path
query_params = parse_qs(parsed_path.query) # 獲取URL攜帶的查詢參數(shù)
# print('收到GET請求參數(shù):', query_params)
if path == '/':
self._handle_home()
else:
self._handle_404()
def do_POST(self):
'''處理POST請求'''
content_length = int(self.headers['Content-Length'])
post_data = self.rfile.read(content_length)
parsed_path = urlparse(self.path)
path = parsed_path.path
query_params = parse_qs(parsed_path.query) # 獲取URL 查詢參數(shù)
# print("收到POST數(shù)據(jù):", post_data.decode())
# 路由匹配邏輯
if path == '/':
self._handle_home()
elif path == '/north/login':
self._handle_login(post_data.decode())
else:
self._handle_404()
if __name__ == '__main__':
# server = HTTPServer(('0.0.0.0', 8000), MyHandler) # 阻塞式運行
server = ThreadingHTTPServer(('localhost', 8000), MyHTTPRequestHandler)
print('正在啟動服務,訪問地址:http://localhost:8000')
server.serve_forever()擴展:如果我們希望服務在處理請求的時,調用其它類實例的方法,或者更新其它類實例的屬性,咋處理呢?
答案:將其它類實例初始化為RequestHandler的類屬性,然后在相關請求處理函數(shù)中進行調用
示例
class Subsystem():
def __init__(self, http_server_port):
self.http_server_port = http_server_port
self.server = ThreadingHTTPServer(('0.0.0.0', self.http_server_port), lambda *args: MyHTTPRequestHandler(self, *args))
self.north_server_info = {}
def start_http_server(self):
self.server.serve_forever()
def push_data(self, data):
'''測試函數(shù)'''
logger.info(f'pushing data to server')
def update_north_server_info(self, data):
'''測試函數(shù)'''
self.north_server_info[data.get('ip')] = data
class MyHTTPRequestHandler(BaseHTTPRequestHandler):
def __init__(self, subsystem, *args, **kwargs):
self.subsystem = subsystem # 保存SubSystem實例引用
super().__init__(*args, **kwargs)
# ....略
def _handle_login(self, login_req_data):
'''處理登錄請求'''
try:
data = json.loads(login_req_data)
username = data.get('username')
password = data.get('password')
ip = data.get('ip')
response = {
'code': 0,
'token': uuid.uuid4().hex,
'description': 'success'
}
self.subsystem.push_data('')
self.subsystem.update_north_server_info({'username': username, 'password': password, 'ip': ip})
self._send_response(200, 'application/json; charset=utf-8', json.dumps(response))
except Exception as e:
error_msg = traceback.format_exc()
logger.error(error_msg)
response = {
'code': 1,
'token':'',
'description': error_msg
}
self._send_response(500, 'application/json; charset=utf-8', json.dumps(response))
# 測試
if __name__ == '__main__':
http_port = 8000
for i in range(2):
system = Subsystem(http_port)
thread = threading.Thread(target=system.start_http_server)
thread.start()
http_port += 1相關介紹
模塊簡介
http.server 模塊定義了用于實現(xiàn)HTTP服務器(Web服務器)的類。
其中一個類,HTTPServer是socketserver.TCPServer子類。它創(chuàng)建并監(jiān)聽HTTP套接字,將請求分派給處理程序。創(chuàng)建和運行服務器的代碼示例如下:
def run(server_class=HTTPServer, handler_class=BaseHTTPRequestHandler):
server_address = ('', 8000)
httpd = server_class(server_address, handler_class)
httpd.serve_forever()類及相關函數(shù)簡介
class http.server.HTTPServer(server_address, RequestHandlerClass)- 該類基于基于TCPServer 類構建,通過將將server地址存儲為名為
server_name和server_port的實例變量。通過handler訪問服務,通常是通過handler的server實例變量 class http.server.ThreadingHTTPServer(server_address, RequestHandlerClass)- 同
HTTPServer一樣,除了通過使用 ThreadingMixIn使用線程處理請求。 3.7版本中新增
HTTPServer 和ThreadingHTTPServer 必須在實例化時給它一個RequestHandlerClass,此模塊提供了三種不同的變體:
class http.server.BaseHTTPRequestHandler(request, client_address, server)- 此類用于處理到達服務器的HTTP請求。就其本身而言,它無法響應任何實際的HTTP請求;它必須被子類化以處理每個請求方法(例如
GET或POST)。BaseHTTPRequestHandler提供了許多類和實例變量以及子類使用的方法。 BaseHTTPRequestHandler實例變量:BaseHTTPRequestHandler屬性:BaseHTTPRequestHandler實例方法:client_address- 包含一個
(host, port)形式,表示客戶端地址的元組 server- 包含服務器實例
close_connection- handle_one_request() 返回前需要設置的布爾值。指示是否可能需要另一個請求,或者是否應該關閉連接。
requestline- 包含HTTP請求行的字符串表示,不含終止
CRLF。此屬性應通過[handle_one_request()設置。如果沒有處理有效的請求行,則應將其設置為空字符串。 command- 包含命令(請求類型)。例如,
'GET'. path- 包含請求路徑。 如果URL的查詢參數(shù)部分存在,則
path包括查詢參數(shù)。 request_version- 包含請求版本字符串,形如'HTTP/1.0'`。
headers- 保存由 MessageClass類變量指定的類的實例。此實例解析和管理HTTP請求中的請求頭。parse_headers()函數(shù)來自 http.client,用于解析標頭,它要求http請求提供有效的RFC 2822樣式請求頭。
rfile- 一個io.BufferedIOBase 輸入流,準備從可選輸入數(shù)據(jù)的開頭讀取。
wfile- 包含用于將響應寫回客戶端的輸出流。在寫入此流時,必須正確遵守HTTP協(xié)議,以實現(xiàn)與HTTP客戶端的成功交互操作。在3.6版本中更改:這是一個 io.BufferedIOBase 流。
server_version- 指定服務器軟件版本。你可能希望覆蓋此內容。格式是多個空格分隔的字符串,其中每個字符串的格式為
name[/version]。例如,'BaseHTTP/0.2'。 sys_version- 包含Python系統(tǒng)版本,其形式和
version_string方法及server_version]類變量使用的相同。例如,'Python/1.4'。 error_message_format- 指定
send_error()應使用的格式字符串,用于構建對客戶端的錯誤響應的。默認情況下,基于傳遞給send_error的狀態(tài)代碼,用responds中的變量填充該字符串。 error_content_type- 指定發(fā)送到客戶端的錯誤響應的
Content-TypeHTTP請求頭。默認值為'text/html'。 protocol_version- 這指定了響應中使用的HTTP協(xié)議版本。如果設置為
'HTTP/1.1',服務器將允許HTTP持久連接;但是,服務器在對客戶端的所有響應中必須包含一個準確的Content-Length請求頭(使用send_header())。為了向后兼容,設置默認為“HTTP/1.0”。 MessageClass- 指定一個email.message.Message之類的類,用于解析HTTP頭。通常,這是不可覆蓋的,默認為
http.client.HTTPMessage。 responses- 此屬性包含整數(shù)錯誤代碼到包含短消息和長消息的兩個元素元組的映射。例如,
{code: (shortmessage, longmessage)}。*shortmessage*通常用作錯誤響應中的message鍵,longmessage用作explain鍵。供send_response_only和send_error()方法使用。 handle()- 調用
handle_one_request()實現(xiàn)適當?shù)?code>do_*() 方法。 handle_one_request()- 此方法將解析請求并將其分發(fā)到適當?shù)?code>do_*()方法。你應永遠不需要重寫它。
handle_expect_100()- 當符合HTTP/1.1標準的服務器收到
Expect: 100-continue請求頭時,它會以100 Continue和“200 OK頭作為響應。如果服務器不希望客戶端繼續(xù),則可以重寫此方法以引發(fā)錯誤。例如,服務器可以選擇發(fā)送417 Expectation Failed作為響應頭,并返回False。 - 3.2版本中的新功能
send_error(code, message=None, explain=None)- 向客戶端發(fā)送并記錄完整的錯誤回復。數(shù)字code指定HTTP錯誤代碼,message作為可選的、簡短的、人類可讀的錯誤描述。explain參數(shù)可用于提供有關錯誤的更詳細信息;它將使用 error_message_format 屬性進行格式化,并在一組完整的標頭之后作為響應體發(fā)出。response屬性包含message和explain的默認值,如果沒有提供值,將使用這些值;對于未知代碼,兩者的默認值都是字符串
???。如果方法是HEAD或響應代碼是以下代碼之一: 1xx,204 No Content,205 Reset Content,304 Not Modified,則請求體將為空。- 在版本3.4中變更:錯誤響應包含
Content-Length頭。添加了explain參數(shù)。 send_response(code, message=None)- 添加一個響應頭添加到headers緩沖區(qū),并記錄接受的請求。HTTP響應行被寫入內部緩沖區(qū),后面跟隨Server和Date響應頭。這兩個響應頭的值分別從version_string() 和date_time_string()方法中獲取。如果服務器不打算使用
send_header()發(fā)送任何其他請求頭,則send_response()后面應該跟一個end_headers()調用。 - 3.3版本中變更:header存儲在內部緩沖區(qū),且需要顯示調用
end_headers()。 send_header(keyword, value)- 將HTTP頭添加到內部緩沖區(qū),當
end_headers()或者flush_headers()被調用時,該緩沖區(qū)中的內容將被寫入到輸出流。keyword應指定header,value指定其值。請注意,在send_headers調用完成后,必須調用end_headers()才能完成操作。 - 版本3.2中變更:HTTP頭存儲在內部緩沖區(qū)中
send_response_only(code, message=None)- 僅發(fā)送響應頭,服務器向客戶端發(fā)送
100 Continue響應時使用。響應頭未緩沖,直接發(fā)送到輸出流。如果未指定message,則發(fā)送與響應code對應的HTTP消息。 - 版本3.2中的新功能
end_headers()- 在header緩沖區(qū)中添加一個空行(表示響應中HTTP頭的結束)并調用
flush_headers()。 - 版本3.2中變更:緩沖的HTTP頭被寫入輸出流
flush_headers()- 最后將HTTP頭發(fā)送到輸出流并刷新內部HTTP頭緩沖區(qū)。3.3版本中的新功能。
log_request(code='-', size='-')- 記錄已接受(成功)的請求。code應指定與響應關聯(lián)的數(shù)字HTTP code。如果可獲取響應的大小,則應將其作為size參數(shù)傳遞。
log_error(...)- 當請求無法滿足時記錄錯誤。默認情況下,它將消息傳遞給
log_message(),因此它采用相同的參數(shù)(format和其它參數(shù))。 log_message(format, ...)- 將任意消息記錄到
sys.stderr。這通常會被重寫以創(chuàng)建自定義錯誤日志記錄機制。format參數(shù)是一個標準的類printf風格的格式字符串,其中log_message()的附加參數(shù)作為格式化的輸入。客戶端ip地址和當前日期和時間都會作為記錄的每條消息的前綴。 version_string()- 返回服務器軟件的版本字符串。這是
server_version以及sys_version屬性的混合。 date_time_string(timestamp=None)- 返回timestamp給出的日期和時間(必須為
None或采用time.time()返回的格式),格式化為消息頭。如果省略timestamp,則使用當前日期和時間。結果看起來像'Sun, 06 Nov 1994 08:49:37 GMT'。 log_date_time_string()- 返回當前日期和時間,格式化為日志記錄格式。
address_string()- 返回客戶端地址。
- 3.3版本中變更:以前執(zhí)行了名稱查找。為了避免名稱解析延遲,它現(xiàn)在總是返回IP地址。
class http.server.SimpleHTTPRequestHandler(request, client_address, server, directory=None)- 介紹略
class http.server.``CGIHTTPRequestHandler(*request*, client_address, server)- 介紹略
參考鏈接
https://docs.python.org/3.9/library/http.server.html
到此這篇關于Python 基于http.server模塊實現(xiàn)簡單http服務的文章就介紹到這了,更多相關Python http.server模塊http服務內容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關文章希望大家以后多多支持腳本之家!
相關文章
如何使用pytorch實現(xiàn)LocallyConnected1D
由于LocallyConnected1D是Keras中的函數(shù),為了用pytorch實現(xiàn)LocallyConnected1D并在960×33的數(shù)據(jù)集上進行訓練和驗證,本文分步驟給大家介紹如何使用pytorch實現(xiàn)LocallyConnected1D,感興趣的朋友一起看看吧2023-09-09
Python OpenCV之圖片縮放的實現(xiàn)(cv2.resize)
這篇文章主要介紹了Python OpenCV之圖片縮放的實現(xiàn),文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧2019-06-06
Python使用combinations實現(xiàn)排列組合的方法
今天小編就為大家分享一篇Python使用combinations實現(xiàn)排列組合的方法,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧2018-11-11
Python MySQL如何通過Binlog獲取變更記錄恢復數(shù)據(jù)
本文介紹了如何使用Python和pymysqlreplication庫通過MySQL的二進制日志(Binlog)獲取數(shù)據(jù)庫的變更記錄,并展示了一個簡單的Python腳本,該腳本讀取Binlog事件并打印出插入、更新和刪除操作的SQL語句,此外,還提到可以使用pandas將結果輸出到Excel表格中進行數(shù)據(jù)分析處理2025-01-01

