OpenResty動態(tài)更新后端地址

OpenResty
需求:

OpenResty運行在Docker中岂座,后端代理的server存在IP不固定的情況,需要OpenResty動態(tài)更新后端地址

實現(xiàn)思路:
  1. 在OpenResty中定義共享Dict洞斯,將后端IP保存在其中
  2. 設(shè)置/add/delete的location,通過Web調(diào)用增刪后端IP
  3. 設(shè)置proxy_pass代理坑赡,代理的地址從共享Dict中獲取實現(xiàn)請求
過程驗證(這里所有服務(wù)在一個Centos 7 Docker中啟動):
  1. 啟動兩個后端服務(wù)
  • 啟動三個終端烙如,在第一個終端中啟動web服務(wù)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服務(wù)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并配置
  • 在第三個終端中,執(zhí)行下面命令安裝Openresty(此處為CentOS系統(tǒng)毅否,其余系統(tǒng)的安裝命令請自行搜索)
[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. 增加后端服務(wù)并驗證(/get通過no查詢后端服務(wù)的IP和端口)
  • 在add的時候指定后端服務(wù)的no(序列號)、IP和端口信息
  • no用來區(qū)分不同后端服務(wù)螟加,執(zhí)行相同no的提交會覆蓋原來的no信息
  • proxy_pass拿到的后端服務(wù)隨機產(chǎn)生(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. 刪除后端服務(wù)并驗證
    /delete通過no刪除后端服務(wù)的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來維護后端服務(wù)信息徘溢,需要保證與序號no的對應
  2. proxy_pass中的后端服務(wù)地址和端口對通過隨機拿到吞琐,存在負載不均衡的情況,可以根據(jù)自己的實際需求修改然爆,比如修改為ip_hash站粟,但需要考慮保存在共享內(nèi)存Dict中后端服務(wù)的數(shù)量
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市曾雕,隨后出現(xiàn)的幾起案子奴烙,更是在濱河造成了極大的恐慌,老刑警劉巖剖张,帶你破解...
    沈念sama閱讀 216,402評論 6 499
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件切诀,死亡現(xiàn)場離奇詭異,居然都是意外死亡搔弄,警方通過查閱死者的電腦和手機幅虑,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,377評論 3 392
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來顾犹,“玉大人倒庵,你說我怎么就攤上這事§潘ⅲ” “怎么了哄芜?”我有些...
    開封第一講書人閱讀 162,483評論 0 353
  • 文/不壞的土叔 我叫張陵,是天一觀的道長柬唯。 經(jīng)常有香客問我认臊,道長,這世上最難降的妖魔是什么锄奢? 我笑而不...
    開封第一講書人閱讀 58,165評論 1 292
  • 正文 為了忘掉前任失晴,我火速辦了婚禮,結(jié)果婚禮上拘央,老公的妹妹穿的比我還像新娘涂屁。我一直安慰自己,他們只是感情好灰伟,可當我...
    茶點故事閱讀 67,176評論 6 388
  • 文/花漫 我一把揭開白布拆又。 她就那樣靜靜地躺著,像睡著了一般栏账。 火紅的嫁衣襯著肌膚如雪帖族。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 51,146評論 1 297
  • 那天挡爵,我揣著相機與錄音竖般,去河邊找鬼。 笑死茶鹃,一個胖子當著我的面吹牛涣雕,可吹牛的內(nèi)容都是我干的艰亮。 我是一名探鬼主播,決...
    沈念sama閱讀 40,032評論 3 417
  • 文/蒼蘭香墨 我猛地睜開眼挣郭,長吁一口氣:“原來是場噩夢啊……” “哼迄埃!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起兑障,我...
    開封第一講書人閱讀 38,896評論 0 274
  • 序言:老撾萬榮一對情侶失蹤侄非,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后旺垒,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體彩库,經(jīng)...
    沈念sama閱讀 45,311評論 1 310
  • 正文 獨居荒郊野嶺守林人離奇死亡肤无,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 37,536評論 2 332
  • 正文 我和宋清朗相戀三年先蒋,在試婚紗的時候發(fā)現(xiàn)自己被綠了。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片宛渐。...
    茶點故事閱讀 39,696評論 1 348
  • 序言:一個原本活蹦亂跳的男人離奇死亡竞漾,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出窥翩,到底是詐尸還是另有隱情业岁,我是刑警寧澤,帶...
    沈念sama閱讀 35,413評論 5 343
  • 正文 年R本政府宣布寇蚊,位于F島的核電站笔时,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏仗岸。R本人自食惡果不足惜允耿,卻給世界環(huán)境...
    茶點故事閱讀 41,008評論 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望扒怖。 院中可真熱鬧较锡,春花似錦、人聲如沸盗痒。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,659評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽俯邓。三九已至骡楼,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間稽鞭,已是汗流浹背君编。 一陣腳步聲響...
    開封第一講書人閱讀 32,815評論 1 269
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留川慌,地道東北人吃嘿。 一個月前我還...
    沈念sama閱讀 47,698評論 2 368
  • 正文 我出身青樓祠乃,卻偏偏與公主長得像,于是被迫代替她去往敵國和親兑燥。 傳聞我的和親對象是個殘疾皇子亮瓷,可洞房花燭夜當晚...
    茶點故事閱讀 44,592評論 2 353

推薦閱讀更多精彩內(nèi)容