openresty lua用Redis的Stream解决消息订阅问题

使用 Redis Streams 解决消息订阅和消费的问题,可以避免在订阅模式下的连接管理问题。下面是如何使用 OpenResty 和 Redis Streams 实现类似的功能。

配置 nginx.conf

确保你的 nginx.conf 文件中配置了 Lua 模块和 Redis 集群的连接信息:

nginx 复制代码
http {
    lua_shared_dict redis_cluster_slot_locks 10m;
    lua_shared_dict redis_cluster_slot_cache 10m;

    init_worker_by_lua_file /path/to/init_worker.lua;

    server {
        listen 8080;

        location /publish {
            content_by_lua_block {
                local redis_cluster = require "resty.rediscluster"
                local config = {
                    name = "testCluster",
                    serv_list = {
                        { ip = "127.0.0.1", port = 7000 },
                        { ip = "127.0.0.1", port = 7001 },
                        { ip = "127.0.0.1", port = 7002 },
                        { ip = "127.0.0.1", port = 7003 },
                        { ip = "127.0.0.1", port = 7004 },
                        { ip = "127.0.0.1", port = 7005 }
                    },
                    keepalive_timeout = 60000,
                    keepalive_cons = 1000,
                    connection_timout = 1000,
                    max_redirection = 5
                }

                local red = redis_cluster:new(config)

                local res, err = red:xadd("mystream", "*", "message", "Hello, World!")
                if not res then
                    ngx.say("failed to publish: ", err)
                    return
                end

                ngx.say("message published to stream mystream")

                local ok, err = red:set_keepalive(10000, 100)
                if not ok then
                    ngx.say("failed to set keepalive: ", err)
                    return
                end
            }
        }
    }
}

init_worker.lua

init_worker.lua 中编写消费逻辑,并确保在消费模式下正确管理连接:

lua 复制代码
local redis_cluster = require "resty.rediscluster"
local config = {
    name = "testCluster",
    serv_list = {
        { ip = "127.0.0.1", port = 7000 },
        { ip = "127.0.0.1", port = 7001 },
        { ip = "127.0.0.1", port = 7002 },
        { ip = "127.0.0.1", port = 7003 },
        { ip = "127.0.0.1", port = 7004 },
        { ip = "127.0.0.1", port = 7005 }
    },
    keepalive_timeout = 60000,
    keepalive_cons = 1000,
    connection_timout = 1000,
    max_redirection = 5
}

local function consume_stream(premature)
    if premature then
        return
    end

    local red = redis_cluster:new(config)
    
    local last_id = "0"  -- 开始读取的起始 ID

    while true do
        local res, err = red:xread("COUNT", 10, "BLOCK", 1000, "STREAMS", "mystream", last_id)
        if not res then
            ngx.log(ngx.ERR, "failed to read stream: ", err)
            break
        end

        if res and res[2] then
            for _, stream in ipairs(res[2]) do
                for _, message in ipairs(stream[2]) do
                    local id = message[1]
                    local fields = message[2]
                    ngx.log(ngx.INFO, "received message: ", fields[2])
                    last_id = id
                end
            end
        end
    end

    red:close()

    -- Schedule another stream consumption attempt
    local ok, err = ngx.timer.at(1, consume_stream)
    if not ok then
        ngx.log(ngx.ERR, "failed to create timer: ", err)
    end
end

-- Schedule the consume_stream function
local ok, err = ngx.timer.at(0, consume_stream)
if not ok then
    ngx.log(ngx.ERR, "failed to create timer: ", err)
end

关键点解释

  1. 避免在消费模式下调用 set_keepalive

    • 在消费模式下,我们不会尝试将连接放入连接池,而是直接读取消息并处理。
  2. 连接管理

    • 消费操作在 consume_stream 函数内进行。
    • 在读取失败时,记录错误并退出循环,然后释放连接。
  3. 使用 ngx.timer.at 调度消费函数

    • 使用 ngx.timer.at(0, consume_stream) 调度消费函数,以便在 worker 初始化时立即开始消费。
  4. 处理 Redis Streams

    • 使用 xread 命令读取流中的消息。
    • 通过循环读取消息并处理。

测试

  1. 启动 OpenResty 并配置上述 nginx.confinit_worker.lua

  2. 使用发布接口发布消息,查看 OpenResty 日志确认消息已接收:

    bash 复制代码
    curl "http://localhost:8080/publish"

    在 OpenResty 日志中应该看到类似如下输出:

    复制代码
    2024/07/09 16:04:00 [info] 12345#0: *1 [lua] init_worker.lua:23: received message: Hello, World!

通过这些配置和代码,你可以在 OpenResty 中使用 Redis Streams 实现对消息的长期消费,并正确处理连接的生命周期和错误。

相关推荐
forestsea6 小时前
深入理解Redisson RLocalCachedMap:本地缓存过期策略全解析
redis·缓存·redisson
佛祖让我来巡山6 小时前
Redis 为什么这么快?——「极速快递站」的故事
redis·redis为什么快?
啦啦啦_99998 小时前
Redis-0-业务逻辑
数据库·redis·缓存
自不量力的A同学8 小时前
Redisson 4.2.0 发布,官方推荐的 Redis 客户端
数据库·redis·缓存
fengxin_rou8 小时前
[Redis从零到精通|第四篇]:缓存穿透、雪崩、击穿
java·redis·缓存·mybatis·idea·多线程
是阿楷啊9 小时前
Java大厂面试场景:音视频场景中的Spring Boot与微服务实战
spring boot·redis·spring cloud·微服务·grafana·prometheus·java面试
笨蛋不要掉眼泪10 小时前
Redis哨兵机制全解析:原理、配置与实战故障转移演示
java·数据库·redis·缓存·bootstrap
ALex_zry1 天前
Redis Cluster 分布式缓存架构设计与实践
redis·分布式·缓存
乔江seven1 天前
【Flask 进阶】3 从同步到异步:基于 Redis 任务队列解决 API 高并发与长耗时任务阻塞
redis·python·flask
这周也會开心1 天前
Redis与MySQL回写中的数据类型存储设计
数据库·redis·mysql