socketserver和WSGI服务端实现教程

Python socketserver 和 WSGI 服务端实现教程

在本文中,我们将详细解析一个使用 socketserver 模块实现的简单 WSGI 服务器。该服务器能够处理 HTTP 请求,支持 WSGI 应用,并正确处理响应头和错误。

代码概述

这段代码定义了一个 run_wsgi 方法,用于处理 HTTP 请求并运行 WSGI 应用。它实现了以下功能:

  1. 处理 Expect: 100-continue 请求头。
  2. 创建 WSGI 环境字典。
  3. 定义 writestart_response 函数来处理 WSGI 响应。
  4. 执行 WSGI 应用,并将响应发送回客户端。
  5. 处理连接错误和超时。
  6. 在发生异常时记录错误,并返回 InternalServerError 响应。
代码详细解析
def run_wsgi(self):# 处理 'Expect: 100-continue' 请求头if self.headers.get("Expect", "").lower().strip() == "100-continue":self.wfile.write(b"HTTP/1.1 100 Continue\r\n\r\n")# 创建 WSGI 环境字典self.environ = environ = self.make_environ()headers_set = []headers_sent = []# 定义 write 函数,用于发送响应数据def write(data):assert headers_set, "write() before start_response"if not headers_sent:status, response_headers = headers_sent[:] = headers_settry:code, msg = status.split(None, 1)except ValueError:code, msg = status, ""code = int(code)self.send_response(code, msg)header_keys = set()for key, value in response_headers:self.send_header(key, value)key = key.lower()header_keys.add(key)if not ("content-length" in header_keysor environ["REQUEST_METHOD"] == "HEAD"or code < 200or code in (204, 304)):self.close_connection = Trueself.send_header("Connection", "close")if "server" not in header_keys:self.send_header("Server", self.version_string())if "date" not in header_keys:self.send_header("Date", self.date_time_string())self.end_headers()assert isinstance(data, bytes), "applications must write bytes"self.wfile.write(data)self.wfile.flush()# 定义 start_response 函数,用于设置响应头def start_response(status, response_headers, exc_info=None):if exc_info:try:if headers_sent:reraise(*exc_info)finally:exc_info = Noneelif headers_set:raise AssertionError("Headers already set")headers_set[:] = [status, response_headers]return write# 执行 WSGI 应用def execute(app):application_iter = app(environ, start_response)try:for data in application_iter:write(data)if not headers_sent:write(b"")finally:if hasattr(application_iter, "close"):application_iter.close()application_iter = Nonetry:execute(self.server.app)except (_ConnectionError, socket.timeout) as e:self.connection_dropped(e, environ)except Exception:if self.server.passthrough_errors:raisefrom .debug.tbtools import get_current_tracebacktraceback = get_current_traceback(ignore_system_exceptions=True)try:if not headers_sent:del headers_set[:]execute(InternalServerError())except Exception:passself.server.log("error", "Error on request:\n%s", traceback.plaintext)

分步骤讲解

  1. 处理 Expect: 100-continue 请求头
if self.headers.get("Expect", "").lower().strip() == "100-continue":self.wfile.write(b"HTTP/1.1 100 Continue\r\n\r\n")

当客户端发送带有 Expect: 100-continue 请求头的请求时,服务器应先响应 100 Continue,然后再继续处理请求。这段代码处理了这一逻辑。

  1. 创建 WSGI 环境字典
self.environ = environ = self.make_environ()

这行代码调用 make_environ 方法,创建一个 WSGI 环境字典 environ,包含了请求的所有必要信息,如请求方法、路径、头信息等。

  1. 定义 write 函数
def write(data):assert headers_set, "write() before start_response"if not headers_sent:status, response_headers = headers_sent[:] = headers_settry:code, msg = status.split(None, 1)except ValueError:code, msg = status, ""code = int(code)self.send_response(code, msg)header_keys = set()for key, value in response_headers:self.send_header(key, value)key = key.lower()header_keys.add(key)if not ("content-length" in header_keysor environ["REQUEST_METHOD"] == "HEAD"or code < 200or code in (204, 304)):self.close_connection = Trueself.send_header("Connection", "close")if "server" not in header_keys:self.send_header("Server", self.version_string())if "date" not in header_keys:self.send_header("Date", self.date_time_string())self.end_headers()assert isinstance(data, bytes), "applications must write bytes"self.wfile.write(data)self.wfile.flush()

write 函数负责发送响应数据。在首次调用时,它还会发送响应头。它确保在 start_response 被调用后才允许发送数据,并确保数据是字节类型。

  1. 定义 start_response 函数
def start_response(status, response_headers, exc_info=None):if exc_info:try:if headers_sent:reraise(*exc_info)finally:exc_info = Noneelif headers_set:raise AssertionError("Headers already set")headers_set[:] = [status, response_headers]return write

start_response 函数用于设置响应头。它会在 WSGI 应用中被调用,接受状态码和响应头列表。如果发生异常且响应头已经发送,它会重新引发异常。

  1. 执行 WSGI 应用
def execute(app):application_iter = app(environ, start_response)try:for data in application_iter:write(data)if not headers_sent:write(b"")finally:if hasattr(application_iter, "close"):application_iter.close()application_iter = None

execute 函数执行 WSGI 应用,并迭代应用的输出。它调用 write 函数发送响应数据,并在最后关闭应用迭代器。

  1. 错误处理
try:execute(self.server.app)
except (_ConnectionError, socket.timeout) as e:self.connection_dropped(e, environ)
except Exception:if self.server.passthrough_errors:raisefrom .debug.tbtools import get_current_tracebacktraceback = get_current_traceback(ignore_system_exceptions=True)try:if not headers_sent:del headers_set[:]execute(InternalServerError())except Exception:passself.server.log("error", "Error on request:\n%s", traceback.plaintext)

这部分代码处理各种异常。如果发生连接错误或超时,会调用 connection_dropped 方法处理。如果发生其他异常且 passthrough_errors 未启用,会记录错误并返回 InternalServerError 响应。

总结

通过使用 socketserver 模块和 WSGI 规范,可以实现一个简单但功能强大的网络服务器。本文中的代码展示了如何处理请求头、创建 WSGI 环境、发送响应数据、处理错误等。希望通过这篇教程,你能够更好地理解和使用 socketserver 模块来开发自己的网络服务器应用。更多详细信息和示例请参考官方文档。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/web/43249.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

【深入理解JVM】关于Object o = new Object()

1. 解释一下对象的创建过程 “半初始化”状态通常指的是对象在内存分配后、但在完全初始化之前的一种状态。在Java中&#xff0c;虽然JVM的规范和设计努力避免对象处于这种不稳定的状态&#xff0c;但在多线程环境下&#xff0c;由于指令重排序等并发问题&#xff0c;仍有可能…

Apache Spark详解

目录 性能优化 银行业务案例&#xff1a; 步骤1&#xff1a;环境准备和数据加载 步骤2&#xff1a;数据探索和预处理 步骤3&#xff1a;特征工程 步骤4&#xff1a;数据转换 步骤5&#xff1a;构建机器学习模型 步骤6&#xff1a;模型评估 步骤7&#xff1a;部署和监控…

Spring JdbcTemplate使用

maven引入Spring JDBC <dependency><groupId>org.springframework</groupId><artifactId>spring-jdbc</artifactId><version>5.3.19</version></dependency> Spring配置中配置 <!-- DataSource配置 --><bean id"…

java代理简单理解

一、什么是代理 举例说明&#xff1a;当我想买一台电脑&#xff0c;国内太贵了。委托好友A在国外帮忙买。 这个情节中我要实现的动作和好友实现的动作一样&#xff0c;都是买电脑。好友帮我完成了这个动作&#xff0c;这就是代理。 类A和类B都实现一个interface接口C&#x…

【LeetCode刷题笔记】LeetCode.24.两两交换链表中的节点

创作不易&#xff0c;本篇文章如果帮助到了你&#xff0c;还请点赞 关注支持一下♡>&#x16966;<)!! 主页专栏有更多知识&#xff0c;如有疑问欢迎大家指正讨论&#xff0c;共同进步&#xff01; 更多算法知识专栏&#xff1a;算法分析&#x1f525; 给大家跳段街舞感谢…

新手小白的pytorch学习第一弹-------张量

1 导入pytorch包 import torch2 创建张量&#xff08;tensor&#xff09; scalar标量 scalar torch.tensor(7) scalartensor(7)scalar.ndim查看scalar的维度&#xff0c;因为scalar是标量&#xff0c;所以维度为0 0scalar.shapetorch.Size([])torch.item()7vector&#xf…

Apache功能配置:访问控制、日志分割; 部署AWStats日志分析工具

目录 保持连接 访问控制 只允许指定ip访问 拒绝指定主机其他正常访问 用户授权 日志格式 日志分割 操作步骤 使用第三方工具cronolog分割日志 AWStats日志分析 操作步骤 访问AwStats分析系统 保持连接 Apache通过设置配置文件httpd-default.conf中相关的连接保持参…

基于Java的科大讯飞大模型API调用实现

写在前面&#xff1a;因为现在自己实习的公司新拓展的一个业务是结合AI的低代码平台&#xff0c;我负责后端的开发&#xff0c;之前一直都是直接使用gpt或者文心一言等ui界面来直接使用大模型&#xff0c;从来没有自己调接口过&#xff0c;所以本文记录一下自己第一次使用大模型…

源代码防泄漏的正确方法

为了保护公司的源代码不被泄露&#xff0c;IT企业可以采取一系列严格的安全措施。这些措施涵盖技术手段、管理策略和操作流程&#xff0c;形成多层次的防护体系做到源代码防泄漏工作。 技术手段 1、源代码加密&#xff1a; 采用高级加密标准&#xff08;AES&#xff09;或其他…

【QT】QComboBox允许输入查询,且不区分大小写

目录 0.简介 1.环境 2.详细代码 3.参考 0.简介 项目需求&#xff0c;原本有一个下拉框&#xff0c;但是条目太多&#xff0c;不好搜索&#xff0c;所以用户要求可以输入查找 修改前 &#xff1a; 修改后&#xff1a; 1.环境 windows11 vs-code qt5.12 2.详细代码 QComboB…

中小企业和数智化的距离,只差一块华为IdeaHub

每次谈及中小企业数智化的话题&#xff0c;被提到最多的总是“三不”难题&#xff0c;即不想转、不敢转、不会转。 为了破解这一困局&#xff0c;政府多次在工作报告中提到“深入开展中小企业数字化赋能专项行动”&#xff0c;并在各地为中小企业创新提供政策支持。此外&#…

Android --- Kotlin学习之路:基础语法学习笔记

------>可读可写变量 var name: String "Hello World";------>只读变量 val name: String "Hello World"------>类型推断 val name: String "Hello World" 可以写成 val name "Hello World"------>基本数据类型 1…

MD5加密和注册页面的编写

MD5加密 1.导入包 npm install --save ts-md5 2.使用方式 import { Md5 } from ts-md5; //md5加密后的密码 const md5PwdMd5.hashStr("123456").toUpperCase(); 遇见的问题及用到的技术 注册页面 register.vue代码 <template><div class"wappe…

从零开始学习嵌入式----Linux 命令行,常用命令速记指南

目录 一、文件操作 二、文本操作 三、系统管理 四、网络操作 五、其他常用命令 六、学习建议 在 Linux 世界里&#xff0c;命令行就像一把瑞士军刀&#xff0c;掌握了它&#xff0c;你就能游刃有余地操控整个系统。但面对茫茫多的命令&#xff0c;新手往往会感到无所适从…

关于Python中的字典你所不知道的七个技巧

01 引言 Python是我最喜欢的编程语言之一&#xff0c;它向来以其简单性、多功能性和可读性而闻名。 字典作为Python中最常使用的数据类型&#xff0c;大家几乎每个人都或多或少在项目中使用过字典&#xff0c;但是字典里有一些潜在的技巧可能并不是每个同学都会用到。 在本文…

相同含义但不同类型字段作为join条件时注意事项

假设表A和表B中都有表示学号的stu_id字段&#xff0c;但该字段在表A和表B中类型分别为bigint和string。当直接通过该字段进行join时&#xff0c;一般情况下可以得到我们预期的结果。 select a.stu_id from a as r join b as l on r.stu_id l.stu_id 但是如果学号长度较长的…

【UE5.1 角色练习】16-枪械射击——瞄准

目录 效果 步骤 一、瞄准时拉近摄像机位置 二、瞄准偏移 三、向指定方向射击 四、连发 效果 步骤 一、瞄准时拉近摄像机位置 打开角色蓝图&#xff0c;在事件图表中添加如下节点&#xff0c;当进入射击状态时设置目标臂长度为300&#xff0c;从而拉近视角。 但是这样切…

勇攀新高峰|暴雨信息召开2024年中述职工作会议

7月8日至9日&#xff0c;暴雨信息召开2024年中述职工作会议&#xff0c;总结回顾了上半年的成绩和不足&#xff0c;本次会议采用线上线下的方式举行&#xff0c;公司各部门管理人员、前台市场营销人员参加述职&#xff0c;公司领导班子出席会议。 本次述职采取了现场汇报点评的…

关于宏v4l2_subdev_call的拆解

struct v4l2_subdev *sd结构体 struct v4l2_subdev { #if defined(CONFIG_MEDIA_CONTROLLER)struct media_entity entity; #endifstruct list_head list;struct module *owner;bool owner_v4l2_dev;u32 flags;struct v4l2_device *v4l2_dev;const struct v4l2_subdev_ops *op…

数字滚动动画~

前言 数字从0.00滚动到某个数值的动画 实现&#xff08;React版本&#xff09; Dom <div className"number" ref{numberRef}>0.00</div> JS const _initNumber () > {const targetNumber 15454547.69;const duration 1500;const numberElement…