Openresty 配合 redis 实现无感知灰度发布系统
2020/03 作者:ihunter 0 次 0
Openresty 是基于 nginx 的一个成熟的网络平台,集成 nginx 和加强的 luajit,许多实用的 lua 库和一些高品质高性能的第三方 nginx 模块。
本文将使用 Openresty 来搭建一个简单的灰度发布系统。环境为 Cenos7。
安装 Openresty
...
配置 Nginx
在用户目录下创建工作目录,创建 nginx.conf 配置文件,目录结构如下图:
nginx配置文件:
worker_processes 1; error_log logs/error.log; events { worker_connections 1024; } # 通过 redis 获取灰度用户区间,执行灰度操作 # 灰度测试完毕后,所有用户切换到灰度服务器,更新正式服务器 # 清空灰度服务器列表 http { # 灰度服务器 upstream grayscale.server{ server 127.0.0.1:9090; } # 正式服务器 upstream prod.server{ server 127.0.0.1:8080; } server { listen 1111; location / { expires 5s; # 默认为生产环境 set $target 'prod.server'; default_type text/html; access_by_lua_file /Users/wangjie/data/nginx/lua/grayscale.lua; proxy_pass $scheme://$target$request_uri; } } }
grayscale.lua 代码:
--[[ # 解决方案 # redis 中存储用户token 对应用户id # redis 中存储灰度用户id区间 # 通过 token 获取用户 id 判断是否在灰度用户区间内 ]] local cjson = require('cjson') local redis = require('resty.redis') local red = redis:new() red:connect("127.0.0.1",6379) -- 灰度区间 local Grayscale_Range = red:get("Grayscale_Range") local Grayscale_Range_Obj = cjson.decode(Grayscale_Range) local IdStart = tonumber(Grayscale_Range_Obj.idStart) local IdEnd = tonumber(Grayscale_Range_Obj.idEnd) -- 用户信息 local UserToken = ngx.req.get_headers()["TOKEN"] local UserId = tonumber(red:get("TOKEN_"..UserToken)) if (UserToken != nil) then if ((UserId > IdStart) and (UserId < IdEnd)) then ngx.var.target = "grayscale.server" end end
创建两个web服务
创建两个 web服务,这里我使用 springboot 创建了两个简单的接口用于模拟灰度和生产环境分别返回如下内容
启动 redis 服务
redis 服务,代码写的是 6379 无密码,如果有密码加一句 在red:auth("密码") 即可。在 DB0 中新增两个值如下
// 原样拷贝即可 Grayscale_Range = { "idStart":1, "idEnd":5000 }
TOKEN_HGdsakGDKSFGkgcshalGF = 1234
测试
使用 postman 请求 http://localhost:1111/api/test1 并携带 TOKEN 请求头(HGdsakGDKSFGkgcshalGF)时,返回灰度环境内容
修改 redis 键 Grayscale_Range 的值为
{ "idStart":1, "idEnd":100 }
再次请求,发现已经切换回生产环境
至此,大功告成
上篇:
环境准备: SSH互信免密码登陆,时间同步,修改主机名,JDK安装,新用户添加,安装maven
下篇:
Prometheus+Grafana+Alertmanager