深入理解ngx_http_proxy_connect_module模块(下)
目录
- 5. 源码分析
- 5.1 模块的初始化代码
- 5.2 请求入口点函数分析
- 5.2.1 ngx_http_proxy_connect_post_read_handler
- 5.2.2 ngx_http_proxy_connect_handler
- 5.3 域名解析回调
- 5.4 向上游服务器发起连接
- 5.4.1 ngx_http_proxy_connect_process_connect
- 5.4.2 ngx_http_proxy_connect_write_upstream
- 5.5 连接建立后向客户端发送CONNECT请求的响应
- 5.6 数据隧道透传
- 5.6.1 上下游连接的读写回调函数
- 5.6.1 ngx_http_proxy_connect_tunnel
- 5.7 关闭会话
- 6. 总结
对于模块的配置指令和内置变量的相关信息可以查看上半部分:深入理解ngx_http_proxy_connect_module模块(上)
5. 源码分析
废话不多说,直接进入源码环节。
5.1 模块的初始化代码
static ngx_http_module_t ngx_http_proxy_connect_module_ctx = {ngx_http_proxy_connect_add_variables, /* preconfiguration */ngx_http_proxy_connect_init, /* postconfiguration */NULL, /* create main configuration */NULL, /* init main configuration */NULL, /* create server configuration */NULL, /* merge server configuration */ngx_http_proxy_connect_create_loc_conf, /* create location configuration */ngx_http_proxy_connect_merge_loc_conf /* merge location configuration */
};
本模块设置了preconfiguration回调,用来在nginx框架中添加第4节列出的变量;本模块又设置了postconfiguration回调,用来设置回调钩子函数。ngx_http_proxy_connect_init代码如下:
static ngx_int_t
ngx_http_proxy_connect_init(ngx_conf_t *cf)
{ngx_http_core_main_conf_t *cmcf;ngx_http_handler_pt *h;cmcf = ngx_http_conf_get_module_main_conf(cf, ngx_http_core_module);h = ngx_array_push(&cmcf->phases[NGX_HTTP_POST_READ_PHASE].handlers);if (h == NULL) {return NGX_ERROR;}*h = ngx_http_proxy_connect_post_read_handler;return NGX_OK;
}
ngx_http_proxy_connect_init代码非常简单,就是在NGX_HTTP_POST_READ_PHASE阶段设置一个回调函数ngx_http_proxy_connect_post_read_handler, NGX_HTTP_POST_READ_PHASE阶段是nginx 异步http处理框架收到客户端的http请求包后的第一个处理阶段。
然后再看一下proxy_connect配置指令的代码,如下:
static char *
ngx_http_proxy_connect(ngx_conf_t *cf, ngx_command_t *cmd, void *conf)
{ngx_http_core_loc_conf_t *clcf;ngx_http_proxy_connect_loc_conf_t *pclcf;clcf = ngx_http_conf_get_module_loc_conf(cf, ngx_http_core_module);clcf->handler = ngx_http_proxy_connect_handler;pclcf = ngx_http_conf_get_module_loc_conf(cf, ngx_http_proxy_connect_module);pclcf->accept_connect = 1;return NGX_CONF_OK;
}
也是非常简单,就是设置一个NGX_HTTP_CONTENT_PHASE阶段的回调函数,以便在这个阶段来接管connect请求的处理逻辑。
5.2 请求入口点函数分析
5.2.1 ngx_http_proxy_connect_post_read_handler
ngx_http_proxy_connect_post_read_handler函数在NGX_HTTP_POST_READ_PHASE阶段被回调,如果发现当前的是CONNECT请求,则判断是否开启了proxy_connect,如果没有开启,则返回NGX_HTTP_NOT_ALLOWED,反之,则对当前的请求设置一个ngx_http_proxy_connect_ctx_t上下文,源码如下:
static ngx_int_t
ngx_http_proxy_connect_post_read_handler(ngx_http_request_t *r)
{ngx_http_proxy_connect_ctx_t *ctx;ngx_http_proxy_connect_loc_conf_t *pclcf;if (r->method == NGX_HTTP_CONNECT) {pclcf = ngx_http_get_module_loc_conf(r, ngx_http_proxy_connect_module);if (!pclcf->accept_connect) {ngx_log_error(NGX_LOG_INFO, r->connection->log, 0,"proxy_connect: client sent connect method");return NGX_HTTP_NOT_ALLOWED;}/* init ctx */ctx = ngx_pcalloc(r->pool, sizeof(ngx_http_proxy_connect_ctx_t));if (ctx == NULL) {return NGX_ERROR;}ctx->buf.pos = (u_char *) NGX_HTTP_PROXY_CONNECT_ESTABLISTHED;ctx->buf.last = ctx->buf.pos +sizeof(NGX_HTTP_PROXY_CONNECT_ESTABLISTHED) - 1;ctx->buf.memory = 1;ctx->connect_timeout = pclcf->connect_timeout;ctx->send_timeout = pclcf->send_timeout;ctx->data_timeout = pclcf->data_timeout;ngx_http_set_ctx(r, ctx, ngx_http_proxy_connect_module);}/* 返回NGX_DECLINED表示如果本阶段有其他的模块,就继续执行这些模块的回调函数。return NGX_DECLINED;
}
5.2.2 ngx_http_proxy_connect_handler
nginx在经过以上ngx_http_proxy_connect_post_read_handler处理后,正常情况下都会会进入到NGX_HTTP_CONTENT_PHASE阶段,在这时就会回调ngx_http_proxy_connect_handler进行处理,源码如下:
static ngx_int_t
ngx_http_proxy_connect_handler(ngx_http_request_t *r)
{ngx_url_t url;ngx_int_t rc;ngx_resolver_ctx_t *rctx, temp;ngx_http_core_loc_conf_t *clcf;ngx_http_proxy_connect_ctx_t *ctx;ngx_http_proxy_connect_upstream_t *u;ngx_http_proxy_connect_loc_conf_t *plcf;plcf = ngx_http_get_module_loc_conf(r, ngx_http_proxy_connect_module);/* 如果不是CONNECT请求或者配置中没有开启proxy_connect, 则本模块直接放弃处理 */if (r->method != NGX_HTTP_CONNECT || !plcf->accept_connect) {return NGX_DECLINED;}/* 判断客户端请求的端口是否在允许的范围内,如果不在范围内,则本模块直接放弃处理 */rc = ngx_http_proxy_connect_allow_handler(r, plcf);if (rc != NGX_OK) {return rc;}/* 获取在ngx_http_proxy_connect_post_read_handler设置的上下文信息 */ctx = ngx_http_get_module_ctx(r, ngx_http_proxy_connect_module);;if (ngx_http_proxy_connect_upstream_create(r, ctx) != NGX_OK) {return NGX_HTTP_INTERNAL_SERVER_ERROR;}u = ctx->u;u->conf = plcf;ngx_memzero(&url, sizeof(ngx_url_t));/* 如果在配置文件中设置了proxy_connect_address,则根据设置的值作为连接上游服务器的地址 */if (plcf->address) {if (ngx_http_complex_value(r, plcf->address, &url.url) != NGX_OK) {return NGX_HTTP_INTERNAL_SERVER_ERROR;}if (url.url.len == 0 || url.url.data == NULL) {url.url.len = r->connect_host.len;url.url.data = r->connect_host.data;}} else {/* 没有设置proxy_connect_address,则用CONNECT请求头中的url地址中的host部分作为上游服务器的地址 */url.url.len = r->connect_host.len;url.url.data = r->connect_host.data;}/* 设置待连接上游服务器的端口 */url.default_port = r->connect_port_n;url.no_resolve = 1;ngx_log_debug1(NGX_LOG_DEBUG_HTTP, r->connection->log, 0,"proxy_connect: connect handler: parse url: %V" , &url.url);if (ngx_parse_url(r->pool, &url) != NGX_OK) {if (url.err) {ngx_log_error(NGX_LOG_ERR, r->connection->log, 0,"proxy_connect: %s in connect host \"%V\"",url.err, &url.url);return NGX_HTTP_FORBIDDEN;}return NGX_HTTP_INTERNAL_SERVER_ERROR;}/* 将当前请求的read和write的i/o事件处理回调函数进行设置,因为当前暂时还不需要处理读写操作,只是用来进行连接是否中断的检测 */r->read_event_handler = ngx_http_proxy_connect_rd_check_broken_connection;r->write_event_handler = ngx_http_proxy_connect_wr_check_broken_connection;/* NOTE:* We use only one address in u->resolved,* and u->resolved.host is "<address:port>" format.* u->resolved用来存放最终向上游服务器连接的ip+端口的地址。*/u->resolved = ngx_pcalloc(r->pool, sizeof(ngx_http_upstream_resolved_t));if (u->resolved == NULL) {return NGX_HTTP_INTERNAL_SERVER_ERROR;}/* rc = NGX_DECLINED */if (url.addrs) {/* 如果url.addrs中已经有目标地址,则用第一个地址来设置u->resolved */ngx_log_debug0(NGX_LOG_DEBUG_HTTP, r->connection->log, 0,"proxy_connect: upstream address given directly");u->resolved->sockaddr = url.addrs[0].sockaddr;u->resolved->socklen = url.addrs[0].socklen;
#if defined(nginx_version) && nginx_version >= 1011007u->resolved->name = url.addrs[0].name;
#endifu->resolved->naddrs = 1;}u->resolved->host = url.host;u->resolved->port = (in_port_t) (url.no_port ? r->connect_port_n : url.port);u->resolved->no_port = url.no_port;if (u->resolved->sockaddr) {/* 目标地址已经设置好了,接下去就不需要进行域名解析直接进行连接了 */rc = ngx_http_proxy_connect_sock_ntop(r, u);if (rc != NGX_OK) {return rc;}/* 当前的ngx_http_request_t的引用计数+1 */r->main->count++;/* 向上游服务器发起TCP连接请求 */ngx_http_proxy_connect_process_connect(r, u);return NGX_DONE;}/* 因为将连接的上游服务器是域名形式提供的,因此需要先通过域名解析后方可以发起连接 */ngx_str_t *host = &url.host;clcf = ngx_http_get_module_loc_conf(r, ngx_http_core_module);temp.name = *host;/* 设置请求上游服务器的开始时间 */u->start_time = ngx_current_msec;u->state.resolve_time = (ngx_msec_t) -1相关文章:
深入理解ngx_http_proxy_connect_module模块(下)
目录 5. 源码分析5.1 模块的初始化代码5.2 请求入口点函数分析5.2.1 ngx_http_proxy_connect_post_read_handler5.2.2 ngx_http_proxy_connect_handler5.3 域名解析回调5.4 向上游服务器发起连接5.4.1 ngx_http_proxy_connect_process_connect5.4.2 ngx_http_proxy_connect_wri…...
HTTP详解(HTTP的特点,状态码,工作原理,GET和POST的区别,如何解决无状态通信)!!!
文章目录 一、HTTP协议简介二、HTTP的主要特点三、HTTP之URL四、Request和Respons五、HTTP的状态码六、HTTP工作原理七、GET和POST请求的区别八、解决HTTP无状态通信——Cookie和Session 一、HTTP协议简介 HTTP协议是Hyper Text Transfer Protocol(超文本传输协议&…...
【QT+QGIS跨平台编译】之五十七:【QGIS_CORE跨平台编译】—【VECTOR_TILE生成】
文章目录 一、protoc二、生成来源三、构建过程一、protoc Protocol Buffers(简称 protobuf)是一种轻量级、高效的数据序列化框架,它可以将结构化数据序列化为二进制格式,同时还可以进行反序列化和数据压缩。相比于 XML 和 JSON 等传统的文本序列化格式,protobuf 采用二进制…...
2024年腾讯云优惠政策_腾讯云TOP10优惠活动
腾讯云服务器多少钱一年?62元一年起,2核2G3M配置,腾讯云2核4G5M轻量应用服务器218元一年、756元3年,4核16G12M服务器32元1个月、312元一年,8核32G22M服务器115元1个月、345元3个月,腾讯云服务器网txyfwq.co…...
SpringMVC 学习(二)之第一个 SpringMVC 案例
目录 1 通过 Maven 创建一个 JavaWeb 工程 2 配置 web.xml 文件 3 创建 SpringMVC 配置文件 spring-mvc.xml 4 创建控制器 HelloController 5 创建视图 index.jsp 和 success.jsp 6 运行过程 7 参考文档 1 通过 Maven 创建一个 JavaWeb 工程 可以参考以下博文&#x…...
qt5与qt6的cmake区别
文章目录 使用cmake构建qt项目,坑很多。一是本身就麻烦,二是,确实坑,因为不同的qtcreator版本,选了不同的kits(套件) 生成的CMakeList.txt文件也不一样。 如果可以的话都选择Qt6的相关选项&…...
【计算机网络】一些乱七八糟内容
MAC Media Access Control 用于在局域网(LAN)或广域网(WAN)中实现设备自动接入网络 "载波侦听多路访问"(Carrier Sense Multiple Access) CSMA/CD 是CSMA的升级版本,加入了序列号检测机制。 CSMA/CA 是CSM…...
基于ESP32的MicroPython项目量产烧写指南
背景 前段时间用MicroPython开发了一个项目,硬件是ESP32-C3,目前准备量产,我需要提供固件以供加工厂批量烧录,需要把我有程序的板子里的程序读出来,然后下到别的板子上,以下做这件事情的过程记录。 1.固件…...
线性规划的标准型转换
对于任意给定的线性规划的问题,其实其本身可能是不符合线性规划标准型的需求的,但是如果通过一系列的等价变化的话,是可以将该问题转换为标准型的线性规划问题,例如如下的线性规划问题: 添加图片注释,不超过 140 字&am…...
机器学习:探寻智能化时代的科技奇迹
在数字化浪潮席卷全球的今天,机器学习已然成为科技领域的一颗璀璨明星,引领着人工智能不断向前发展。那么,机器学习究竟是什么?它为何能在众多科技中脱颖而出,成为改变世界的力量?本文将带您一探究竟&#…...
《Flask入门教程》学习笔记
《Flask入门教程》官网:https://tutorial.helloflask.com/ 目录 第一章:准备工作第二章:Hello, Flask!第三章:模板第四章:静态文件第五章:数据库第六章:模板优化第七章:表单第八章&a…...
go语言基础 -- map的定义与使用
map的定义与使用 map声明基础语法map的基本使用map的遍历map切片map排序 map声明基础语法 // map的声明 var xxx_map map[key_type]value_typemap的key可以是基本数据类型,channel,接口,结构体,数组,但不能是slice&am…...
讯方·智汇云校第五期名师班火热报名中!
第三期名师班回顾 授课情况 课堂上,同学们热情高涨,积极参与互动。他们紧跟名师的步伐,深入探索云服务的奥秘。张梁老师在为同学们讲述完知识点后,会根据所讲知识给同学们布置对应的实验,由同学们分组讨论练习。 每…...
为什么企业需要使用云电子邮箱?
作为一家机构的负责人,您比大多数人都清楚,您的工作日不会在下午5点就结束。很可能,当您的员工已经打卡下班回家很久之后,您还在以这样或那样的方式继续工作。作为一名企业主,埋头苦干对您来说并不是什么新鲜事&#x…...
[DEBUG] spring boot-如何处理链接中的空格等特殊字符
问题: get或者post中提交的内容可能有空格、#等特殊字符,不做处理的话可能解析错误。 解决: html中: <a th:href"{/listSgrna(id${item.getGeneId()},geneName${item.getGeneName()},genome${genome},sgrnaNum${sgrnaN…...
通过配置数据库事件(Event)来实现定时导出 MySQL 数据库
首先,确保 MySQL 服务器已启用事件调度器功能。你可以通过以下 SQL 语句查询: SHOW VARIABLES LIKE event_scheduler; 如果 event_scheduler 的值为 ON,则表示事件调度器已启用;如果为 OFF,则可以使用以下语句启用&…...
基于x86架构的OpenHarmony应用生态挑战赛等你来战!
为了更快速推进OpenHarmony在PC领域的进一步落地,加快x86架构下基于OpenHarmony的应用生态的繁荣,为北向应用开发者提供一个更加便捷的开发环境,推动OpenHarmony北向应用开发者的增加,助力OpenHarmony在PC领域实现新的突破&#x…...
LeetCode每日一题2673. Make Costs of Paths Equal in a Binary Tree
文章目录 一、题目二、题解 一、题目 You are given an integer n representing the number of nodes in a perfect binary tree consisting of nodes numbered from 1 to n. The root of the tree is node 1 and each node i in the tree has two children where the left ch…...
贝叶斯分类器
贝叶斯分类器 1. 引言 贝叶斯分类器是一种基于贝叶斯定理的分类算法,它利用特征之间的关系和类别的先验概率来进行分类。贝叶斯分类器在文本分类、垃圾邮件过滤、医学诊断等领域有着广泛的应用。 贝叶斯分类算法是统计学的一种分类方法,是一类利用概率…...
游戏服务之会话管理
会话的概念与作用 游戏服务器 Session(会话)是指在游戏服务器和客户端之间建立的一个临时的连接。它可以用于存储和管理用户的游戏状态和信息。 当用户登录游戏时,服务器会为该用户创建一个 Session,可用于记录用户的登录状态、角色信息等个人信息。服务器会为每个会话分…...
前端倒计时误差!
提示:记录工作中遇到的需求及解决办法 文章目录 前言一、误差从何而来?二、五大解决方案1. 动态校准法(基础版)2. Web Worker 计时3. 服务器时间同步4. Performance API 高精度计时5. 页面可见性API优化三、生产环境最佳实践四、终极解决方案架构前言 前几天听说公司某个项…...
mongodb源码分析session执行handleRequest命令find过程
mongo/transport/service_state_machine.cpp已经分析startSession创建ASIOSession过程,并且验证connection是否超过限制ASIOSession和connection是循环接受客户端命令,把数据流转换成Message,状态转变流程是:State::Created 》 St…...
页面渲染流程与性能优化
页面渲染流程与性能优化详解(完整版) 一、现代浏览器渲染流程(详细说明) 1. 构建DOM树 浏览器接收到HTML文档后,会逐步解析并构建DOM(Document Object Model)树。具体过程如下: (…...
数据链路层的主要功能是什么
数据链路层(OSI模型第2层)的核心功能是在相邻网络节点(如交换机、主机)间提供可靠的数据帧传输服务,主要职责包括: 🔑 核心功能详解: 帧封装与解封装 封装: 将网络层下发…...
04-初识css
一、css样式引入 1.1.内部样式 <div style"width: 100px;"></div>1.2.外部样式 1.2.1.外部样式1 <style>.aa {width: 100px;} </style> <div class"aa"></div>1.2.2.外部样式2 <!-- rel内表面引入的是style样…...
大模型多显卡多服务器并行计算方法与实践指南
一、分布式训练概述 大规模语言模型的训练通常需要分布式计算技术,以解决单机资源不足的问题。分布式训练主要分为两种模式: 数据并行:将数据分片到不同设备,每个设备拥有完整的模型副本 模型并行:将模型分割到不同设备,每个设备处理部分模型计算 现代大模型训练通常结合…...
12.找到字符串中所有字母异位词
🧠 题目解析 题目描述: 给定两个字符串 s 和 p,找出 s 中所有 p 的字母异位词的起始索引。 返回的答案以数组形式表示。 字母异位词定义: 若两个字符串包含的字符种类和出现次数完全相同,顺序无所谓,则互为…...
华为云Flexus+DeepSeek征文|DeepSeek-V3/R1 商用服务开通全流程与本地部署搭建
华为云FlexusDeepSeek征文|DeepSeek-V3/R1 商用服务开通全流程与本地部署搭建 前言 如今大模型其性能出色,华为云 ModelArts Studio_MaaS大模型即服务平台华为云内置了大模型,能助力我们轻松驾驭 DeepSeek-V3/R1,本文中将分享如何…...
在web-view 加载的本地及远程HTML中调用uniapp的API及网页和vue页面是如何通讯的?
uni-app 中 Web-view 与 Vue 页面的通讯机制详解 一、Web-view 简介 Web-view 是 uni-app 提供的一个重要组件,用于在原生应用中加载 HTML 页面: 支持加载本地 HTML 文件支持加载远程 HTML 页面实现 Web 与原生的双向通讯可用于嵌入第三方网页或 H5 应…...
Python+ZeroMQ实战:智能车辆状态监控与模拟模式自动切换
目录 关键点 技术实现1 技术实现2 摘要: 本文将介绍如何利用Python和ZeroMQ消息队列构建一个智能车辆状态监控系统。系统能够根据时间策略自动切换驾驶模式(自动驾驶、人工驾驶、远程驾驶、主动安全),并通过实时消息推送更新车…...
