一般服务器上版本更新迭代,需要把老版本目录替换成新版本,让用户全部使用新版本。如果新版本有bug,或者新版本用户使用不习惯导致用户体验差甚至流失。而灰度发布时介于黑白之间的发布,可以让一部分用户使用新版本,其他用户继续使用老版本,这样新版本有问题也可以把损失降到最低。

toc

Lua脚本基础语法

Lua 是一个简洁、轻量、可扩展的脚本语言
直接使用 yum install lua -y 安装,在base库。CentOS7.2以上版本默认安装

## 命令行交互模式运行
[root@Nginx ~]# lua
Lua 5.1.4 Copyright (C) 1994-2008 Lua.org, PUC-Rio
> print("hello world")
hello world
## 脚本方式运行
[root@Nginx ~]# vim test.lua
#!/usr/bin/lua
print("hello world")
[root@Nginx ~]# lua test.lua
hello world

判断语句

if  如果你有房 then
    我就嫁给你
elseif  你有车 or 你有钱 then
    我就嫁给你
else
    再见
end

while循环语句

## 1到100之间所有数字相加
[root@Nginx ~]# vim while.lua
#!/usr/bin/lua
sum=0
num=1
while num <=100 do
        sum=sum+num
        num=num+1
end
print("sum="..sum)

for循环语句

## 1到100之间所有数字相加
[root@Nginx ~]# vim for.lua
#!/usr/bin/lua
sum=0
for i=1,100 do
        sum=sum+i
end
print("sum="..sum)

Nginx+Lua环境搭建

默认情况下 Nginx 不支持 Lua 模块, 需要安装 LuaJIT 解释器, 并且需要重新编译 Nginx

安装依赖包

[root@Nginx ~]# yum -y install gcc gcc-c++ make pcre-devel zlib-devel openssl-devel 

下载并解压最新的包

## 创建目录
[root@Nginx ~]# mkdir /soft/src -p && cd /soft/src
## 下载 luajit 和 ngx_devel_kit 以及 lua-nginx-module
[root@Nginx src]# wget http://luajit.org/download/LuaJIT-2.0.5.tar.gz
[root@Nginx src]# wget https://github.com/simplresty/ngx_devel_kit/archive/v0.3.1rc1.tar.gz
[root@Nginx src]# wget https://github.com/openresty/lua-nginx-module/archive/v0.10.15.tar.gz
## 解压 luajit 和 ngx_devel_kit 以及 lua-nginx-module
[root@Nginx src]# tar xf v0.10.15.tar.gz
[root@Nginx src]# tar xf v0.3.1rc1.tar.gz
[root@Nginx src]# tar xf LuaJIT-2.0.5.tar.gz

安装 Lua 即时编译器 luajit

[root@Nginx src]# cd LuaJIT-2.0.5/
[root@Nginx LuaJIT-2.0.5]# make && make install

编译安装 Nginx

## 返回原来的目录
[root@Nginx LuaJIT-2.0.5]# cd /soft/src
## 下载Nginx源码包
[root@Nginx src]# wget http://nginx.org/download/nginx-1.16.0.tar.gz
## 解压并进入源码包
[root@Nginx src]# tar xf nginx-1.16.0.tar.gz
[root@Nginx src]# cd nginx-1.16.0/
## 安装源码包
[root@Nginx nginx-1.16.0]# ./configure --prefix=/etc/nginx \
 --with-http_ssl_module --with-http_stub_status_module --with-http_dav_module \
--add-module=../ngx_devel_kit-0.3.1rc1/ --add-module=../lua-nginx-module-0.10.15
[root@Nginx nginx-1.16.0]# make && make install
## 建立软链接
[root@Nginx nginx-1.16.0]# ln -s /usr/local/lib/libluajit-5.1.so.2 /lib64/libluajit-5.1.so.2
## 内核加入lua库
[root@Nginx nginx-1.16.0]# echo "/usr/local/LuaJIT/lib" >> /etc/ld.so.conf

Nginx 调用 Lua 模块的语法和变量

  • 语法:
    • set_by_lua --- 设置 Nginx 变量
    • set_by_lua_file --- 指定设置 Nginx 变量的文件
    • access_by_lua --- 请求访问阶段的处理,可以设置访问控制
    • access_by_lua_file --- 指定访问控制文件
    • content_by_lua --- 返回的内容
    • content_by_lua_file --- 指定返回内容文件
  • 变量
    • ngx.var --- Nginx 变量
    • ngx.req.get_headers --- 获取请求头
    • ngx.req.get_url_args --- 获取url请求参数
    • ngx.redirect --- 重定向
    • ngx.print --- 输出响应内容
    • ngx.say --- 输出响应内容,最后加个换行
    • ngx.header --- 输出响应头

测试 NginxLua 是否连接成功

[root@Nginx ~]# vim /etc/nginx/conf/nginx.conf
## 在配置文件server标签里下面内容
        location /test {
                default_type text/html;
                content_by_lua_block {
                        ngx.say("Hello World")
                }

## 启动Nginx
[root@Nginx ~]# /etc/nginx/sbin/nginx

浏览器访问 IP地址/test 是否为 Hello World (注意关闭selinux和firewalld)

Nginx+Lua实现灰度发布

Nginx+Lua实现灰度发布原理:
当用户请求访问前端代理 Nginx 时,内嵌Lua模块会解析 Nginx 配置文件中 Lua 脚本,Lua 脚本会获取客户端IP地址,查看 Memcached 缓存中是否存在该键值,如果存在则会反向代理到新版本的upstream池,不存在则会反向代理到老版本的upstream池。

这只主要是灰度发布,另外两个节点是什么web容器不重要,可以正常访问就行

新老版本服务器配置

## 关闭防火墙和selinux
[root@node ~]# setenforce 0
[root@node ~]# systemctl stop firewalld
## 安装启动httpd
[root@node ~]# yum install httpd -y
[root@node ~]# systemctl start httpd
## 新版本配置
[root@node ~]# echo "这是新版本" > /var/www/html/index.html
## 老版本配置
[root@node ~]# echo "这是老版本" > /var/www/html/index.html

以上就是两个节点配置IP可以正常访问就不用动它们了

配置 Memcached

## 安装Memcached
[root@Nginx src]# yum install memcached -y
## 下载Lua和memcached连接库,让Lua可以连接Memcached
[root@Nginx src]# cd /soft/src
[root@Nginx src]# wget https://github.com/openresty/lua-resty-memcached/archive/v0.14.tar.gz
[root@Nginx src]# tar xf v0.14.tar.gz
[root@Nginx src]# mkdir /etc/nginx/conf/lua
[root@Nginx src]# cp -r lua-resty-memcached-0.14/lib/resty/memcached.lua /etc/nginx/conf/lua/
## 启动Memcached
[root@Nginx src]# systemctl start memcached
[root@Nginx src]# systemctl enable memcached

配置 Nginx 负载均衡

## 把Nginx配置文件server标签和server标签内容都删了换成下面引用其他配置文件
    include conf.d/*.conf
## 创建引用目录
[root@Nginx ~]# mkdir /etc/nginx/conf/conf.d
## 编写Nginx配置文件
[root@Nginx ~]# vim /etc/nginx/conf/conf.d/lua.conf
## 调用刚刚复制的memcached.lua文件
lua_package_path "/etc/nginx/conf/lua/memcached.lua";
upstream new_version {
    server 192.168.1.2:80;
}
upstream old_version {
    server 192.168.1.3:80;
}
server {
    listen 80;
    server_name 192.168.1.1;
    location / {
        default_type 'text/plain';
        content_by_lua_file /etc/nginx/conf/lua/index.lua;   ## 访问网站交给这个lua脚本
    }
    location @new_version {
        proxy_pass http://new_version;
        proxy_set_header Host $http_host;
    }
    location @old_version {
        proxy_pass http://old_version;
        proxy_set_header Host $http_host;
    }
}

编写 Lua 脚本

[root@Nginx ~]# vim /etc/nginx/conf/lua/index.lua
-- 获取客户端头部信息的 x-real-ip
clientIP = ngx.req.get_headers()["X-Real-IP"]
-- 如果clientIP没有获取到客户端IP就获取 x_forwarded_for
if clientIP == nil then
clientIP = ngx.req.get_headers()["x_forwarded_for"]
end
-- 如果clientIP还是没有获取到客户端IP就获取 remote_addr
if clientIP == nil then
clientIP = ngx.var.remote_addr
end
-- 本地加载 memcached
local memcached = require "resty.memcached"
-- 实例化对象
local memc, err = memcached:new()
-- 判断连接是否存在错误
if not memc then
ngx.say("failed to instantiate memc: "..err)
return
 end
-- 建立 memcache 连接
local ok, err = memc:connect("127.0.0.1", 11211)
-- 无法连接往前端抛出错误信息
if not ok then
ngx.say("failed to connect: "..err)
return
end
-- 获取对象中的 ip 存在值赋给 res
local res, flags, err = memc:get(clientIP)
if err then
ngx.say("failed to get clientIP "..err)
return
end
-- 如果 res 值为 1 则调用 local-@new_version
if res == "1" then
ngx.exec("@new_version")
return
end
-- 否则调用 local-@old_version
ngx.exec("@old_version")
return

启动 Nginx 后可以上浏览器测试一下啦

[root@Nginx ~]# /etc/nginx/sbin/nginx

把IP加入 Memcached 让其访问新版本

## 安装telnet(没有Memcached客户端用telnet暂时代替一下)
[root@Nginx ~]# yum install telnet -y
## 连接Memcached
[root@Nginx ~]# telnet 127.0.0.1 11211
## set 客户端访问的IP(后面的分别是flags、exptime、bytes对应的值,不用管,这不重要)
set 192.168.1.254 0 0 1
## 填写IP键的值(出现STORED就可以了)
1
12-28 15:14