OpenResty动态更新后端地址

OpenResty
需求:

OpenResty运行在Docker中,后端代理的server存在IP不固定的情况,需要OpenResty动态更新后端地址

实现思路:
  1. 在OpenResty中定义共享Dict,将后端IP保存在其中
  2. 设置/add/delete的location,通过Web调用增删后端IP
  3. 设置proxy_pass代理,代理的地址从共享Dict中获取实现请求
过程验证(这里所有服务在一个Centos 7 Docker中启动):
  1. 启动两个后端服务
  • 启动三个终端,在第一个终端中启动web服务test1
[root@f2d9f997edac /]# mkdir -p test1 && cd test1 && echo test1 > test
[root@f2d9f997edac test1]# python -m SimpleHTTPServer 8081
Serving HTTP on 0.0.0.0 port 8081 ...
  • 在第二个终端中启动web服务test2
[root@f2d9f997edac /]# mkdir -p test2 && cd test2 && echo test2 > test
[root@f2d9f997edac test2]# python -m SimpleHTTPServer 8082
Serving HTTP on 0.0.0.0 port 8082 ...
  • 在第三个终端中验证test1和test2
[root@f2d9f997edac /]# curl 127.0.0.1:8081/test
test1
[root@f2d9f997edac /]# curl 127.0.0.1:8082/test
test2
  1. 安装OpenResty并配置
  • 在第三个终端中,执行下面命令安装Openresty(此处为CentOS系统,其余系统的安装命令请自行搜索)
[root@f2d9f997edac /]# yum install -y openresty
  • 编辑配置文件,验证配置文件并启动Openrestry
[root@f2d9f997edac /]# cat /usr/local/openresty/nginx/conf/nginx.conf
worker_processes  4;

events {
    worker_connections  1024;
}

http {
    include       mime.types;
    default_type  application/octet-stream;

    log_format  main  '$remote_addr - $remote_user [$time_local] "$request" '
                      '$status $body_bytes_sent "$http_referer" '
                      '"$http_user_agent" "$http_x_forwarded_for"';

    #access_log  logs/access.log  main;

    sendfile        on;
    #tcp_nopush     on;

    keepalive_timeout  65;

    #gzip  on;

    lua_shared_dict ups 10m;

    server {
        listen 80;
        location /add {
            content_by_lua_block {
            local ups = ngx.shared.ups
                local ip = ngx.req.get_uri_args()["ip"]
                local port = ngx.req.get_uri_args()["port"]
                local no = ngx.req.get_uri_args()["no"]
                if ip == nil or port == nil or no == nil then
                    ngx.say("usage: /add?ip=x.x.x.x&port=xx&no=xx")
                    return
                end
                ups:set("host-"..no, ip)
                ups:set("port-"..no, port)
                -- ups["host-"..no] = ip
                -- ups["port-"..no] = port
                local count = 0
                for k in pairs(ups:get_keys(0)) do count = count + 1  end
                ups:set("servers", count)
                ngx.say("OK")
            }
        }
        location /get {
            content_by_lua_block {
                local ups = ngx.shared.ups
                local no = ngx.req.get_uri_args()["no"]
                if no == nil then
                    ngx.say("usage: /get?no=xx")
                    return
                end
                if ups:get("host-" .. no) == nil then
                    ngx.say("the no server is vaild: " .. no)
                    return
                end
                local count = 0
                for k in pairs(ups:get_keys(0)) do count = count + 1  end
                ups:set("servers", count)
                ngx.say("ip: " .. ups:get("host-" .. no))
                ngx.say("port: " .. ups:get("port-" .. no))
                ngx.say("servers: " .. math.floor(ups:get("servers")/2))
            }
        }
        location /delete {
            content_by_lua_block {
                local ups = ngx.shared.ups
                local no = ngx.req.get_uri_args()["no"]
                if no == nil then
                    ngx.say("usage: /get?no=xx")
                    return
                end
                if ups:get("host-" .. no) == nil then
                    ngx.say("the no server is vaild: " .. no)
                    return
                end
                ups:delete("host-" .. no)
                ups:delete("port-" .. no)
                ngx.say("OK")
            }
        }

        location / {
            set_by_lua_block $cur_ups {
                local ups = ngx.shared.ups
                local items = ups:get("servers")
                local n = math.random(math.floor(items/2))
                local ss = string.gsub(ups:get_keys(0)[n], "%a+-", "")
                s = ups:get("host-"..ss) .. ":" .. ups:get("port-"..ss)
                return s
            }
            proxy_next_upstream off;
            proxy_set_header    Host $host;
            proxy_http_version  1.1;
            proxy_set_header    Connection  "";
            proxy_pass $scheme://$cur_ups;
        }
    }
}
[root@f2d9f997edac /]# openresty -t
nginx: the configuration file /usr/local/openresty/nginx/conf/nginx.conf syntax is ok
nginx: configuration file /usr/local/openresty/nginx/conf/nginx.conf test is successful
[root@f2d9f997edac /]# openresty
  1. 增加后端服务并验证(/get通过no查询后端服务的IP和端口)
  • 在add的时候指定后端服务的no(序列号)、IP和端口信息
  • no用来区分不同后端服务,执行相同no的提交会覆盖原来的no信息
  • proxy_pass拿到的后端服务随机产生(2中的math.random(math.floor(items/2))处)
[root@f2d9f997edac /]# curl '127.0.0.1:80/add?no=1&ip=127.0.0.1&port=8081'
OK
[root@f2d9f997edac /]# curl '127.0.0.1:80/get?no=1'
ip: 127.0.0.1
port: 8081
servers: 1
[root@f2d9f997edac /]# curl '127.0.0.1:80/add?no=2&ip=127.0.0.1&port=8082'
OK
[root@f2d9f997edac /]# curl '127.0.0.1:80/get?no=2'
ip: 127.0.0.1
port: 8082
servers: 2
[root@f2d9f997edac /]# curl '127.0.0.1:80/test'
test1
[root@f2d9f997edac /]# curl '127.0.0.1:80/test'
test2
[root@f2d9f997edac /]#
  1. 删除后端服务并验证
    /delete通过no删除后端服务的IP和端口信息
[root@f2d9f997edac /]# curl '127.0.0.1:80/delete?no=1'
OK
[root@f2d9f997edac /]# curl '127.0.0.1:80/get?no=1'
the no server is vaild: 1
[root@f2d9f997edac /]# curl '127.0.0.1:80/get?no=2'
ip: 127.0.0.1
port: 8082
servers: 1
[root@f2d9f997edac /]# curl '127.0.0.1:80/test'
test2
[root@f2d9f997edac /]# curl '127.0.0.1:80/test'
test2
[root@f2d9f997edac /]#

存在不足:

  1. 如果由程序控制OpenResty的/add/delete来维护后端服务信息,需要保证与序号no的对应
  2. proxy_pass中的后端服务地址和端口对通过随机拿到,存在负载不均衡的情况,可以根据自己的实际需求修改,比如修改为ip_hash,但需要考虑保存在共享内存Dict中后端服务的数量
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 159,117评论 4 362
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 67,328评论 1 293
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 108,839评论 0 243
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 44,007评论 0 206
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 52,384评论 3 287
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 40,629评论 1 219
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 31,880评论 2 313
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 30,593评论 0 198
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 34,313评论 1 243
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 30,575评论 2 246
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 32,066评论 1 260
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 28,392评论 2 253
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 33,052评论 3 236
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 26,082评论 0 8
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 26,844评论 0 195
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 35,662评论 2 274
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 35,575评论 2 270

推荐阅读更多精彩内容