【网关开发】7.Openresty使用cosocket API 发送http与tcp网络请求
1.【网关开发】1.编译Openresty 服务器初始化搭建2.【网关开发】2.Openresty 使用lua扩展 链接etcd数据库3.【网关开发】6.lua绑定委托(delegate)实现多播调用4.【网关开发】5.Openresty 自定义负载均衡与流量转发5.【网关开发】4.Openresty 使用events插件进行事件通知6.【网关开发】3.Openresty lua使用lmdb数据库7.【网关开发】10.Openresty 网关全链路的数据生成与数据上报8.【网关开发】9.Openresty 自定义流量分流策略支持灰度(金丝雀)等发布业务场景9.【网关开发】8.Openresty 网关自定义健康检查的设计与实现
10.【网关开发】7.Openresty使用cosocket API 发送http与tcp网络请求
背景
为网关提供健康检查功能时需要对节点发送http或者tcp探活请求。Openresty 提供cosocket来处理非阻塞IO。
实现
跟工程结合在一起,这里简单拼接数据结构
local function __default_check_alive(status)
return status >= 200 and status <= 299
end
local function debug_ctx()
local ctx = {
peer = {ip = '10.218.22.239', port = '8090'}, --目标机器
ahc = {
type = 'http',
timeout = 3,
check_http_send = "GET /ping HTTP/1.1\r\nHost: service_test.com\r\n\r\n", -- 发送的数据内容
},
status_check = __default_check_alive
}
return ctx
end
发送http请求
local stream_sock = ngx.socket.tcp -- 引入模块
local re_find = ngx.re.find
local function __check_http_peer(ahc, peer, status_check)
local ok
local req = ahc.check_http_send
local sock, err = stream_sock() -- 创建 TCP 的 cosocket 对象
if not sock then
ngx.log(ngx.ERR, "failed to create stream socket: " .. err)
return false, err
end
sock:settimeout(ahc.timeout * 1000) --设置超时时间
ok, err = sock:connect(peer.ip, peer.port) --建立连接
if not ok then
return false
end
local bytes, err = sock:send(req) --发送数据
if not bytes then
return false
end
local status_line, err = sock:receive() -- 接收数据
if not status_line then
if err == "timeout" then
sock:close() -- timeout errors do not close the socket.
end
return false
end
if status_check then
local from, to, err = re_find(status_line,
[[^HTTP/\d+\.\d+\s+(\d+)]], --利用正则获取status code
"joi", nil, 1)
if err then
ngx.log(ngx.ERR, "failed to parse status line: "..err)
end
if not from then
sock:close()
return false
end
local status = tonumber(status_line:sub(from, to))
if not status_check(status) then
-- ngx.log(ngx.ERR, status_line)
sock:close()
return false
end
end
sock:close()
return true
end
发送tcp请求
-- functional, check peer by tcp, returns bool indicate up or down
local function __check_tcp_peer(ahc, peer)
local ok
local sock, err = stream_sock()
if not sock then
ngx.log(ngx.ERR, "failed to create stream socket: " .. err)
return false, err
end
sock:settimeout(ahc.timeout * 1000)
ok, err = sock:connect(peer.ip, peer.port)
if not ok then
return false
end
sock:close()
return true
end
遇到的问题
API disabled in the context of init_worker_by_lua*
这是因为我使用的地方是在init_worker_by_lua阶段,这阶段是不允许使用cosocket的,除了这些阶段还有
set_by_lua、log_by_lua、header_filter_by_lua、body_filter_by_lua、init_by_lua* 都是不允许使用的
-- 修改调用
ngx.timer.at(0, function (p, self)
local ctx = down_peer_checker.debug_ctx()
ngx.log(ngx.INFO,"down_peer_checker check_peer "..tostring(down_peer_checker.check_peer(ctx)))
end)
HTTP/1.1 400 Bad Request
主要是请求字符串格式问题、
GET /ping HTTP/1.1\r\n Host: service_test.com\r\n 错误
GET /ping HTTP/1.1\r\nHost: service_test.com\r\n\r\n 正确
总结与思考
cosocket知识与参考文章:https://zhuanlan.zhihu.com/p/507329735
存在部分封装,源码地址 https://github.com/zhaoshoucheng/openresty/blob/main/pkg/lua_script/upstream/down_peer_checker.lua
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· DeepSeek 开源周回顾「GitHub 热点速览」
· 物流快递公司核心技术能力-地址解析分单基础技术分享
· .NET 10首个预览版发布:重大改进与新特性概览!
· AI与.NET技术实操系列(二):开始使用ML.NET
· 单线程的Redis速度为什么快?