鍍金池/ 教程/ Java/ 與其他 location 配合
定時(shí)任務(wù)
函數(shù)的參數(shù)
超時(shí)
一個(gè) openresty 內(nèi)存“泄漏”問題
獲取 uri 參數(shù)
局部變量
sleep
灰度發(fā)布
TIME_WAIT
代碼覆蓋率
連接池
CentOS 平臺(tái)安裝
稀疏數(shù)組
如何只啟動(dòng)一個(gè) timer 工作?
變量的共享范圍
break,return 關(guān)鍵字
Nginx
SQL 注入
如何引用第三方 resty 庫
不同階段共享變量
獲取請(qǐng)求 body
動(dòng)態(tài)生成的 lua-resty-redis 模塊方法
動(dòng)態(tài)加載證書和 OCSP stapling
repeat 控制結(jié)構(gòu)
編碼為 array 還是 object
Nginx 靜態(tài)文件服務(wù)
執(zhí)行階段概念
Lua 函數(shù)
日期時(shí)間函數(shù)
健康監(jiān)測(cè)
與其他 location 配合
for 控制結(jié)構(gòu)
函數(shù)定義
HTTPS 時(shí)代
點(diǎn)號(hào)與冒號(hào)操作符的區(qū)別
String 庫
文件操作
OpenResty 最佳實(shí)踐
<code>ngx.shared.DICT</code> 非隊(duì)列性質(zhì)
使用動(dòng)態(tài) DNS 來完成 HTTP 請(qǐng)求
代碼規(guī)范
什么是 JIT?
Windows 平臺(tái)安裝
正確的記錄日志
LuaNginxModule
不用標(biāo)準(zhǔn)庫
C10K 編程
控制結(jié)構(gòu)
請(qǐng)求中斷后的處理
Lua 環(huán)境搭建
Test::Nginx 能指定現(xiàn)成的 nginx.conf,而不是自動(dòng)生成一個(gè)嗎
Lua 基礎(chǔ)數(shù)據(jù)類型
動(dòng)態(tài)限速
PostgresNginxModule
簡(jiǎn)單API Server框架
API 測(cè)試
location 匹配規(guī)則
虛變量
單元測(cè)試
防止 SQL 注入
select + set_keepalive 組合操作引起的數(shù)據(jù)讀寫錯(cuò)誤
阻塞操作
全動(dòng)態(tài)函數(shù)調(diào)用
Web 服務(wù)
典型應(yīng)用場(chǎng)景
Nginx 新手起步
TLS session resumption
輸出響應(yīng)體
調(diào)用代碼前先定義函數(shù)
module 是邪惡的
怎樣理解 cosocket
模塊
Socket 編程發(fā)展
如何對(duì) Nginx Lua module 添加新 api
如何在后臺(tái)開啟輕量級(jí)線程完成定時(shí)任務(wù)?
如何定位問題
table 庫
json 解析的異常捕獲
如何安裝火焰圖生成工具
lua 中如何 continue
if 是邪惡的
為什么我們的域名不能被解析
抵制使用 module() 定義模塊
測(cè)試
body 在 location 中的傳遞
Lua 入門
子查詢
pipeline 壓縮請(qǐng)求數(shù)量
如何發(fā)起新 HTTP 請(qǐng)求
Lua 簡(jiǎn)介
緩存失效風(fēng)暴
Ubuntu 平臺(tái)安裝
日志輸出
緩存
Lua 面向?qū)ο缶幊?/span>
Nginx 陷阱和常見錯(cuò)誤
Redis 接口的二次封裝(發(fā)布訂閱)
日志
訪問有授權(quán)驗(yàn)證的 Redis
正則表達(dá)式
lock
熱裝載代碼
調(diào)用 FFI 出現(xiàn) &quot;table overflow&quot;
數(shù)據(jù)合法性檢測(cè)
禁止某些終端訪問
控制結(jié)構(gòu) if-else
調(diào)試
與 Docker 使用的網(wǎng)絡(luò)瓶頸
PostgresNginxModule 模塊的調(diào)用方式
用 do-end 整理你的代碼
FFI
什么時(shí)候使用
簡(jiǎn)介
環(huán)境搭建
Mac OS X 平臺(tái)安裝
火焰圖
負(fù)載均衡
while 型控制結(jié)構(gòu)
如何定位 openresty 崩潰 bug
使用 Nginx 內(nèi)置綁定變量
判斷數(shù)組大小
請(qǐng)求返回后繼續(xù)執(zhí)行
Redis 接口的二次封裝
KeepAlive
反向代理
協(xié)議無痛升級(jí)
數(shù)學(xué)庫
元表
Vanilla 介紹
HelloWorld
LuaCjsonLibrary
持續(xù)集成
代碼靜態(tài)分析
網(wǎng)上有大量對(duì) Lua 調(diào)優(yōu)的推薦,我們應(yīng)該如何看待?
script 壓縮復(fù)雜請(qǐng)求
非空判斷
性能測(cè)試
函數(shù)返回值
API 的設(shè)計(jì)
kong 介紹
表達(dá)式
不支持事務(wù)
LuaRestyDNSLibrary 簡(jiǎn)介

與其他 location 配合

nginx 世界的 location 是異常強(qiáng)大的,畢竟 nginx 的主要應(yīng)用場(chǎng)景是在負(fù)載均衡、API server,在不同 server、location 之間跳轉(zhuǎn)更是家常便飯。利用不同 location 的功能組合,我們可以完成內(nèi)部調(diào)用、流水線方式跳轉(zhuǎn)、外部重定向等幾大不同方式,下面將給大家介紹幾個(gè)主要應(yīng)用,就當(dāng)拋磚引玉。

內(nèi)部調(diào)用

例如對(duì)數(shù)據(jù)庫、內(nèi)部公共函數(shù)的統(tǒng)一接口,可以把它們放到統(tǒng)一的 location 中。通常情況下,為了保護(hù)這些內(nèi)部接口,都會(huì)把這些接口設(shè)置為 internal 。這么做的最主要好處就是可以讓這個(gè)內(nèi)部接口相對(duì)獨(dú)立,不受外界干擾。

示例代碼:

location = /sum {
    # 只允許內(nèi)部調(diào)用
    internal;

    # 這里做了一個(gè)求和運(yùn)算只是一個(gè)例子,可以在這里完成一些數(shù)據(jù)庫、
    # 緩存服務(wù)器的操作,達(dá)到基礎(chǔ)模塊和業(yè)務(wù)邏輯分離目的
    content_by_lua_block {
        local args = ngx.req.get_uri_args()
        ngx.say(tonumber(args.a) + tonumber(args.b))
    }
}

location = /app/test {
    content_by_lua_block {
        local res = ngx.location.capture(
                        "/sum", {args={a=3, b=8}}
                        )
        ngx.say("status:", res.status, " response:", res.body)
    }
}

緊接著,稍微擴(kuò)充一下,并行請(qǐng)求的效果,示例如下:

location = /sum {
    internal;
    content_by_lua_block {
        ngx.sleep(0.1)
        local args = ngx.req.get_uri_args()
        ngx.print(tonumber(args.a) + tonumber(args.b))
    }
}

location = /subduction {
    internal;
    content_by_lua_block {
        ngx.sleep(0.1)
        local args = ngx.req.get_uri_args()
        ngx.print(tonumber(args.a) - tonumber(args.b))
    }
}

location = /app/test_parallels {
    content_by_lua_block {
        local start_time = ngx.now()
        local res1, res2 = ngx.location.capture_multi( {
                        {"/sum", {args={a=3, b=8}}},
                        {"/subduction", {args={a=3, b=8}}}
                    })
        ngx.say("status:", res1.status, " response:", res1.body)
        ngx.say("status:", res2.status, " response:", res2.body)
        ngx.say("time used:", ngx.now() - start_time)
    }
}

location = /app/test_queue {
    content_by_lua_block {
        local start_time = ngx.now()
        local res1 = ngx.location.capture_multi( {
                        {"/sum", {args={a=3, b=8}}}
                    })
        local res2 = ngx.location.capture_multi( {
                        {"/subduction", {args={a=3, b=8}}}
                    })
        ngx.say("status:", res1.status, " response:", res1.body)
        ngx.say("status:", res2.status, " response:", res2.body)
        ngx.say("time used:", ngx.now() - start_time)
    }
}

測(cè)試結(jié)果:

?  ~ curl 127.0.0.1/app/test_parallels
status:200 response:11
status:200 response:-5
time used:0.10099983215332
?  ~ curl 127.0.0.1/app/test_queue
status:200 response:11
status:200 response:-5
time used:0.20199990272522

利用 ngx.location.capture_multi 函數(shù),直接完成了兩個(gè)子請(qǐng)求并行執(zhí)行。當(dāng)兩個(gè)請(qǐng)求沒有相互依賴,這種方法可以極大提高查詢效率。兩個(gè)無依賴請(qǐng)求,各自是 100ms,順序執(zhí)行需要 200ms,但通過并行執(zhí)行可以在 100ms 完成兩個(gè)請(qǐng)求。實(shí)際生產(chǎn)中查詢時(shí)間可能沒這么規(guī)整,但思想大同小異,這個(gè)特性是很有用的。

http://wiki.jikexueyuan.com/project/openresty/images/work_location_flow_1.png" alt="圖例" />

該方法,可以被廣泛應(yīng)用于廣告系統(tǒng)(1:N模型,一個(gè)請(qǐng)求,后端從N家供應(yīng)商中獲取條件最優(yōu)廣告)、高并發(fā)前端頁面展示(并行無依賴界面、降級(jí)開關(guān)等)。

流水線方式跳轉(zhuǎn)

現(xiàn)在的網(wǎng)絡(luò)請(qǐng)求,已經(jīng)變得越來越擁擠。各種不同 API 、下載請(qǐng)求混雜在一起,就要求不同廠商對(duì)下載的動(dòng)態(tài)調(diào)整有各種不同的定制策略,而這些策略在一天的不同時(shí)間段,規(guī)則可能還不一樣。這時(shí)候我們還可以效仿工廠的流水線模式,逐層過濾、處理。

示例代碼:

location ~ ^/static/([-_a-zA-Z0-9/]+).jpg {
    set $image_name $1;
    content_by_lua_block {
        ngx.exec("/download_internal/images/"
                .. ngx.var.image_name .. ".jpg");
    };
}

location /download_internal {
    internal;
    # 這里還可以有其他統(tǒng)一的 download 下載設(shè)置,例如限速等
    alias ../download;
}

注意,ngx.exec 方法與 ngx.redirect 是完全不同的,前者是個(gè)純粹的內(nèi)部跳轉(zhuǎn)并且沒有引入任何額外 HTTP 信號(hào)。 這里的兩個(gè) location 更像是流水線上工人之間的協(xié)作關(guān)系。第一環(huán)節(jié)的工人對(duì)完成自己處理部分后,直接交給第二環(huán)節(jié)處理人(實(shí)際上可以有更多環(huán)節(jié)),它們之間的數(shù)據(jù)流是定向的。

http://wiki.jikexueyuan.com/project/openresty/images/work_location_flow_2.png" alt="圖例" />

外部重定向

不知道大家什么時(shí)候開始注意的,百度的首頁已經(jīng)不再是 HTTP 協(xié)議,它已經(jīng)全面修改到了 HTTPS 協(xié)議上。但是對(duì)于大家的輸入習(xí)慣,估計(jì)還是在地址欄里面輸入 baidu.com ,回車后發(fā)現(xiàn)它會(huì)自動(dòng)跳轉(zhuǎn)到 https://www.baidu.com ,這時(shí)候就需要的外部重定向了。

location = /foo {
    content_by_lua_block {
        ngx.say([[I am foo]])
    }
}

location = / {
    rewrite_by_lua_block {
        return ngx.redirect('/foo');
    }
}

執(zhí)行測(cè)試,結(jié)果如下:

?  ~  curl 127.0.0.1 -i
HTTP/1.1 302 Moved Temporarily
Server: openresty/1.9.3.2rc3
Date: Sun, 22 Nov 2015 11:04:03 GMT
Content-Type: text/html
Content-Length: 169
Connection: keep-alive
Location: /foo

<html>
<head><title>302 Found</title></head>
<body bgcolor="white">
<center><h1>302 Found</h1></center>
<hr><center>openresty/1.9.3.2rc3</center>
</body>
</html>

?  ~  curl 127.0.0.1/foo -i
HTTP/1.1 200 OK
Server: openresty/1.9.3.2rc3
Date: Sun, 22 Nov 2015 10:43:51 GMT
Content-Type: text/html
Transfer-Encoding: chunked
Connection: keep-alive

I am foo

當(dāng)我們使用瀏覽器訪問頁面 http://127.0.0.1 就可以發(fā)現(xiàn)瀏覽器會(huì)自動(dòng)跳轉(zhuǎn)到 http://127.0.0.1/foo 。

與之前兩個(gè)應(yīng)用實(shí)例不同的,外部重定向是可以跨域名的。例如從 A 網(wǎng)站跳轉(zhuǎn)到 B 網(wǎng)站是絕對(duì)允許的。在 CDN 場(chǎng)景的大量下載應(yīng)用中,一般分為調(diào)度、存儲(chǔ)兩個(gè)重要環(huán)節(jié)。調(diào)度就是通過根據(jù)請(qǐng)求方 IP 、下載文件等信息尋找最近、最快節(jié)點(diǎn),應(yīng)答跳轉(zhuǎn)給請(qǐng)求方完成下載。

上一篇:table 庫下一篇:Windows 平臺(tái)安裝