nginx?sticky實(shí)現(xiàn)基于cookie負(fù)載均衡示例詳解
前言
sticky 是一個(gè)nginx的第三方模塊 它不在nginx發(fā)行版中 需要額外編譯這個(gè)模塊的, 它的思想就是不依靠后端生成cookie , 而是sticky在nginx這里生成cookie ,然后下發(fā)到客戶端, 客戶端收到cookie后 以后的請(qǐng)求帶著這個(gè)cookie 就會(huì)通過(guò)這個(gè)cookie 進(jìn)行hash 被一直定位到后端的某一臺(tái)服務(wù)器了
優(yōu)點(diǎn):
- 它比純
ip hash負(fù)載有個(gè)優(yōu)點(diǎn)就是 純 ip hash 像局域網(wǎng)內(nèi)的訪問(wèn)ip 訪問(wèn)會(huì)導(dǎo)致ip傾斜 - 它比 hash
$cookie_jsessionid的優(yōu)點(diǎn)就是 它不依賴后端 不用后端生成 session 從而減少后端的 資源
思考
想想為什么要用這個(gè) sticky 來(lái)把用戶盡量一直定位到一臺(tái)服務(wù)器呢? 在多臺(tái)后臺(tái)服務(wù)器的環(huán)境下,我們?yōu)榱舜_保一個(gè)客戶只和一臺(tái)服務(wù)器通信,我們勢(shì)必使用長(zhǎng)連接。使用什么方式來(lái)實(shí)現(xiàn)這種連接呢,常見(jiàn)的有使用nginx自帶的ip_hash來(lái)做,我想這絕對(duì)不是一個(gè)好的辦法,如果前端是CDN,或者說(shuō)一個(gè)局域網(wǎng)的客戶同時(shí)訪問(wèn)服務(wù)器,導(dǎo)致出現(xiàn)服務(wù)器分配不均衡,以及不能保證每次訪問(wèn)都粘滯在同一臺(tái)服務(wù)器。
如果基于cookie會(huì)是一種什么情形,想想看, 每臺(tái)電腦都會(huì)有不同的cookie,在保持長(zhǎng)連接的同時(shí)還保證了服務(wù)器的壓力均衡,nginx sticky值得推薦。
如果瀏覽器不支持cookie,那么sticky不生效,畢竟整個(gè)模塊是給予cookie實(shí)現(xiàn)的.
1.cookie_jsessionid 負(fù)載均衡
在說(shuō)sticky 之前先來(lái)看看 nginx 通過(guò) cookie_jessionid 的負(fù)載均衡方式
1.1 后端準(zhǔn)備
@Autowired
lateinit var env: Environment
@GetMapping("/server")
fun server(request:HttpServletRequest):String {
//獲取當(dāng)前服務(wù)的端口
val port = env.getProperty("local.server.port")
println("now port: $port")
//調(diào)用了request.getSession(true) 則會(huì)沒(méi)有session的時(shí)候創(chuàng)建session
val session = request.getSession(true)
val name = session.getAttribute("name")
println("name: $name")
if (name == null){
session.setAttribute("name","johnny")
}
return "success"
}
1.2 hash $cookie_jsessionid;配置
在upstream 里面配置 hash 的方式 使用 cookie_jsessionid 去做hash
#user nobody;
worker_processes 1;
#error_log logs/error.log;
#error_log logs/error.log notice;
#error_log logs/error.log info;
#pid logs/nginx.pid;
events {
worker_connections 1024;
}
http {
include mime.types;
default_type application/octet-stream;
sendfile on;
#tcp_nopush on;
#keepalive_timeout 0;
keepalive_timeout 65;
#gzip on;
upstream backend {
# 指定hash 方式是 cookie_jessionid nginx自帶的方式
hash $cookie_jsessionid;
server 172.16.225.1:8081;
server 172.16.225.1:8080;
}
server {
listen 80;
server_name localhost;
#charset koi8-r;
#access_log logs/host.access.log main;
location / {
# 指定負(fù)載到后端upstream
proxy_pass http://backend;
}
error_page 500 502 503 504 /50x.html;
location = /50x.html {
root html;
}
}
}
可以看到 服務(wù)器下發(fā)了 cookie JSESSIONID 并且多次請(qǐng)求這個(gè) 都不會(huì)改變 因?yàn)閚ginx 根據(jù) JSESSIONID 它進(jìn)行hash 每次都負(fù)載到同一臺(tái)后端服務(wù)器, 因?yàn)檫@個(gè)后端服務(wù)器已經(jīng)存在了 這個(gè)session 所以不會(huì)再次創(chuàng)建

可以看到 多次請(qǐng)求 都打到這個(gè) 8081 的后端服務(wù)了

2.nginx sticky 負(fù)載均衡
2.1 下載 sticky
Bitbucket

2.2 重新編譯升級(jí)nginx
1)下載完成,放入服務(wù)器解壓,記住解壓的位置,后面要用
2)進(jìn)入到nginx的安裝文件
3)配置nginx
tar -xvf nginx-goodies-nginx-sticky-module-ng-c78b7dd79d0d.tar.gz mv nginx-goodies-nginx-sticky-module-ng-c78b7dd79d0d nginx-sticky # 添加sticky 模塊 ./configure \ --prefix=/usr/local/nginx \ --add-module=/opt/nginx-sticky
make 編譯的時(shí)候有可能會(huì)報(bào)錯(cuò)
找到sticky剛剛的解壓目錄,進(jìn)入修改文件 vim ngx_http_sticky_misc.c ,加入下面的頭文件
#include <openssl/sha.h> #include <openssl/md5.h>

再次make , 當(dāng)然后面如果還報(bào)錯(cuò)的話,openssl檢查是否安裝
apt-get install -y openssl
2.3 upstream 配置 sticky
配置好后重啟nginx
upstream backend {
#hash $cookie_jsessionid;
sticky; #指定使用 sticky 進(jìn)行負(fù)載均衡
server 172.16.225.1:8081;
server 172.16.225.1:8080;
}

2.4 修改后端不再創(chuàng)建session
此時(shí)后端不會(huì)創(chuàng)建session 也不會(huì)下發(fā)cookie jsessionid 了
@Autowired
lateinit var env: Environment
@GetMapping("/server")
fun server(request:HttpServletRequest):String {
val port = env.getProperty("local.server.port")
println("now port: $port")
return "success"
}
2.5 再次 多次請(qǐng)求
可以看到stick 幫我們下發(fā)了 route 這個(gè)cookie , 并且這個(gè)不會(huì)變 默認(rèn)關(guān)閉瀏覽器就會(huì)失效

可以看到請(qǐng)求還是只會(huì)落在一臺(tái)服務(wù)器上

3.sticky 其他用法
sticky [name=route] [domain=.foo.bar] [path=/] [expires=1h] [hash=index|md5|sha1] [no_fallback];
name: 可以為任何的string字符,默認(rèn)是route
domain:哪些域名下可以使用這個(gè)cookie
path:哪些路徑對(duì)啟用sticky,例如path/test,那么只有test這個(gè)目錄才會(huì)使用sticky做負(fù)載均衡
expires:cookie過(guò)期時(shí)間,默認(rèn)瀏覽器關(guān)閉就過(guò)期,也就是會(huì)話方式。
no_fallbackup:如果設(shè)置了這個(gè),cookie對(duì)應(yīng)的服務(wù)器宕機(jī)了,那么將會(huì)返回502(bad gateway 或者 proxy error),建議不啟用
總結(jié)
本篇主要介紹了 nginx sticky 負(fù)載均衡,它不需要后端去生成session 下發(fā)jsessionid 而是nginx的sticky模塊幫我們?nèi)ハ掳l(fā)一個(gè) route 的 cookie , nginx 使用這個(gè)cookie 進(jìn)行hash 負(fù)載, 從而實(shí)現(xiàn)了 客戶每次訪問(wèn)都粘滯在同一臺(tái)服務(wù)器
以上就是nginx sticky實(shí)現(xiàn)基于cookie負(fù)載均衡示例詳解的詳細(xì)內(nèi)容,更多關(guān)于nginx sticky cookie負(fù)載均衡的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
nginx負(fù)載均衡下的webshell上傳的實(shí)現(xiàn)
本文主要介紹了nginx負(fù)載均衡下的webshell上傳的實(shí)現(xiàn),文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2023-03-03
Nginx服務(wù)器下使用rewrite重寫url以實(shí)現(xiàn)偽靜態(tài)的示例
這篇文章主要介紹了Nginx服務(wù)器下使用rewrite重寫url以實(shí)現(xiàn)偽靜態(tài)的示例,這里舉了Discuz!和WordPress這兩個(gè)常用的PHP程序,需要的朋友可以參考下2015-12-12
nginx tcp負(fù)載均衡的具體實(shí)現(xiàn)
Nginx是比較不錯(cuò)的開(kāi)源Web服務(wù)器之一,它也可以用作TCP和UDP負(fù)載均衡器,本文主要介紹了nginx tcp負(fù)載均衡的具體實(shí)現(xiàn),具有一定的參考價(jià)值,感興趣的可以了解一下2024-05-05
Nginx反向代理實(shí)現(xiàn)Vue跨域的示例
本文主要介紹了Nginx反向代理實(shí)現(xiàn)Vue跨域的示例,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2023-06-06
Nginx禁止IP訪問(wèn)只允許域名訪問(wèn)及防盜鏈設(shè)置
我們?cè)谑褂玫臅r(shí)候會(huì)遇到很多的惡意IP攻擊,這個(gè)時(shí)候就要用到Nginx 禁止IP訪問(wèn)了,本文主要介紹了Nginx禁止IP訪問(wèn)只允許域名訪問(wèn)及Nginx防盜鏈設(shè)置,具有一定的參考價(jià)值,感興趣的可以了解一下2023-10-10
淺析nginx剛剛發(fā)布的JavaScript能力nginScript
Nginx [engine x]是全球最受歡迎,也是最優(yōu)秀的web服務(wù)器、反向代理服務(wù)器。nginScript是JavaScript/ECMAscript的子集,nginScript不是通過(guò)V8引擎實(shí)現(xiàn)的。本文給大家介紹nginx剛剛發(fā)布的JavaScript能力nginScript,感興趣的朋友跟著小編一起了解了解吧2015-09-09

