Python本地搭建静态Web服务器的实现
ZSYL 人气:0静态Web服务器-返回固定页面数据
学习目标
能够写出组装固定页面数据的响应报文
1. 开发自己的静态Web服务器
实现步骤:
- 编写一个TCP服务端程序
- 获取浏览器发送的http请求报文数据
- 读取固定页面数据,把页面数据组装成HTTP响应报文数据发送给浏览器。
- HTTP响应报文数据发送完成以后,关闭服务于客户端的套接字。
2. 静态Web服务器-返回固定页面数据的示例代码
import socket # 判断是否是主模块的代码 if __name__ == '__main__': # 创建tcp服务端套接字 tcp_server_socket = socket.socket(socket.AF_INET, socket.SOCK_STREAM) # 设置端口号复用,程序退出端口号立即释放 tcp_server_socket.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, True) # 绑定端口号 tcp_server_socket.bind(("", 8000)) # 设置监听 tcp_server_socket.listen(128) # 循环等待接受客户端的连接请求 while True: # 等待接受客户端的连接请求 new_socket, ip_port = tcp_server_socket.accept() # 代码执行到此,说明连接建立成功 # 接收客户端的请求信息 recv_data = new_socket.recv(4096) print(recv_data) # 打开文件读取文件中的数据 with open("static/index.html", "r") as file: # 这里的file表示打开文件的对象 file_data = file.read() # 提示: with open 关闭文件这步操作不用程序员来完成,系统帮我们来完成 # 响应行 response_line = "HTTP/1.1 200 OK\r\n" # 响应头 response_header = "Server: PWS/1.0\r\n" # 响应体 response_body = file_data # 把数据封装成http 响应报文格式的数据 response = response_line + response_header + "\r\n" + response_body # 把字符串编码成二进制 response_data = response.encode("utf-8") # 发送给浏览器的响应报文数据 new_socket.send(response_data) # 关闭服务于客户端的套接字 new_socket.close()
3. 小结
编写一个TCP服务端程序
tcp_server_socket = socket.socket(socket.AF_INET, socket.SOCK_STREAM) # 循环接受客户端的连接请求 while True: conn_socket, ip_port = tcp_server_socket.accept()
获取浏览器发送的http请求报文数据
client_request_data = conn_socket.recv(4096)
读取固定页面数据,把页面数据组装成HTTP响应报文数据发送给浏览器。
response_data = (response_line + response_header + "\r\n").encode("utf-8") + response_body conn_socket.send(response_data)
HTTP响应报文数据发送完成以后,关闭服务于客户端的套接字。
conn_socket.close()
静态Web服务器-返回指定页面数据
学习目标
能够写出组装指定页面数据的响应报文
1. 静态Web服务器的问题
目前的Web服务器,不管用户访问什么页面,返回的都是固定页面的数据,接下来需要根据用户的请求返回指定页面的数据
返回指定页面数据的实现步骤:
- 获取用户请求资源的路径
- 根据请求资源的路径,读取指定文件的数据
- 组装指定文件数据的响应报文,发送给浏览器
- 判断请求的文件在服务端不存在,组装404状态的响应报文,发送给浏览器
2. 静态Web服务器-返回指定页面数据的示例代码
import socket import os def main(): # 创建tcp服务端套接字 tcp_server_socket = socket.socket(socket.AF_INET, socket.SOCK_STREAM) # 设置端口号复用,程序退出端口号立即释放 tcp_server_socket.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, True) # 绑定端口号 tcp_server_socket.bind(("", 8000)) # 设置监听 tcp_server_socket.listen(128) # 循环等待接受客户端的连接请求 while True: # 等待接受客户端的连接请求 new_socket, ip_port = tcp_server_socket.accept() # 代码执行到此,说明连接建立成功 # 接收客户端的请求信息 recv_data = new_socket.recv(4096) # 判断接收的数据长度是否为0 if len(recv_data) == 0: new_socket.close() return # 对二进制数据进行解码 recv_content = recv_data.decode("utf-8") print(recv_content) # 对数据按照空格进行分割 request_list = recv_content.split(" ", maxsplit=2) # 获取请求的资源路径 request_path = request_list[1] print(request_path) # 判断请求的是否是根目录,如果是根目录设置返回的信息 if request_path == "/": request_path = "/index.html" # 1. os.path.exits # os.path.exists("static/" + request_path) # 2. try-except # 打开文件读取文件中的数据, 提示:这里使用rb模式,兼容打开图片文件 with open("static" + request_path, "rb") as file: # 这里的file表示打开文件的对象 file_data = file.read() # 提示: with open 关闭文件这步操作不用程序员来完成,系统帮我们来完成 # 代码执行到此,说明文件存在,返回200状态信息 # 响应行 response_line = "HTTP/1.1 200 OK\r\n" # 响应头 response_header = "Server: PWS/1.0\r\n" # 响应体 response_body = file_data # 把数据封装成http 响应报文格式的数据 response = (response_line + response_header + "\r\n").encode("utf-8") + response_body # 发送给浏览器的响应报文数据 new_socket.send(response) # 关闭服务于客户端的套接字 new_socket.close() # 判断是否是主模块的代码 if __name__ == '__main__': main()
3. 小结
获取用户请求资源的路径
request_list = client_request_conent.split(" ", maxsplit=2) request_path = request_list[1]
根据请求资源的路径,读取请求指定文件的数据
with open("static" + request_path, "rb") as file: file_data = file.read()
组装指定文件数据的响应报文,发送给浏览器
response_data = (response_line + response_header + "\r\n").encode("utf-8") + response_body conn_socket.send(response_data)
判断请求的文件在服务端不存在,组装404状态的响应报文,发送给浏览器
try: # 打开指定文件,代码省略... except Exception as e: conn_socket.send(404响应报文数据)
静态Web服务器-返回404页面数据
import socket import os def main(): # 创建tcp服务端套接字 tcp_server_socket = socket.socket(socket.AF_INET, socket.SOCK_STREAM) # 设置端口号复用,程序退出端口号立即释放 tcp_server_socket.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, True) # 绑定端口号 tcp_server_socket.bind(("", 8000)) # 设置监听 tcp_server_socket.listen(128) # 循环等待接受客户端的连接请求 while True: # 等待接受客户端的连接请求 new_socket, ip_port = tcp_server_socket.accept() # 代码执行到此,说明连接建立成功 # 接收客户端的请求信息 recv_data = new_socket.recv(4096) # 判断接收的数据长度是否为0 if len(recv_data) == 0: new_socket.close() return # 对二进制数据进行解码 recv_content = recv_data.decode("utf-8") print(recv_content) # 对数据按照空格进行分割 request_list = recv_content.split(" ", maxsplit=2) # 获取请求的资源路径 request_path = request_list[1] print(request_path) # 判断请求的是否是根目录,如果是根目录设置返回的信息 if request_path == "/": request_path = "/index.html" # 1. os.path.exits # os.path.exists("static/" + request_path) # 2. try-except try: # 打开文件读取文件中的数据, 提示:这里使用rb模式,兼容打开图片文件 with open("static" + request_path, "rb") as file: # 这里的file表示打开文件的对象 file_data = file.read() # 提示: with open 关闭文件这步操作不用程序员来完成,系统帮我们来完成 except Exception as e: # 代码执行到此,说明没有请求的该文件,返回404状态信息 # 响应行 response_line = "HTTP/1.1 404 Not Found\r\n" # 响应头 response_header = "Server: PWS/1.0\r\n" # 读取404页面数据 with open("static/error.html", "rb") as file: file_data = file.read() # 响应体 response_body = file_data # 把数据封装成http 响应报文格式的数据 response = (response_line + response_header + "\r\n").encode("utf-8") + response_body # 发送给浏览器的响应报文数据 new_socket.send(response) else: # 代码执行到此,说明文件存在,返回200状态信息 # 响应行 response_line = "HTTP/1.1 200 OK\r\n" # 响应头 response_header = "Server: PWS/1.0\r\n" # 响应体 response_body = file_data # 把数据封装成http 响应报文格式的数据 response = (response_line + response_header + "\r\n").encode("utf-8") + response_body # 发送给浏览器的响应报文数据 new_socket.send(response) finally: # 关闭服务于客户端的套接字 new_socket.close() # 判断是否是主模块的代码 if __name__ == '__main__': main()
加载全部内容