Nginx正向代理實(shí)現(xiàn)局域網(wǎng)電腦訪問外網(wǎng)的過程詳解
引言
在網(wǎng)絡(luò)環(huán)境中,有時(shí)候我們需要讓局域網(wǎng)內(nèi)的電腦訪問外網(wǎng),但是由于網(wǎng)絡(luò)策略或其他原因,直接訪問外網(wǎng)是不可行的。這時(shí)候,可以借助 Nginx 來搭建一個(gè)正向代理服務(wù)器,實(shí)現(xiàn)局域網(wǎng)內(nèi)電腦通過 Nginx 轉(zhuǎn)發(fā)訪問外網(wǎng)的需求。
在工作中我遇到了一個(gè)類似的情況:在公司網(wǎng)絡(luò)中,由于管理要求,局域網(wǎng)內(nèi)的電腦不能直接訪問外網(wǎng)。但是,工作上領(lǐng)導(dǎo)吩咐需要讓局域網(wǎng)內(nèi)的電腦能夠訪問外網(wǎng)上的某個(gè)網(wǎng)站,這時(shí)候就需要用到正向代理。本文將介紹如何配置 Nginx 實(shí)現(xiàn)這一功能。
準(zhǔn)備工作
首先,我們需要一臺可以訪問外網(wǎng)的服務(wù)器,例如一臺服務(wù)器、電腦或者虛擬機(jī)。確保該服務(wù)器已經(jīng)安裝了 Nginx,并且網(wǎng)絡(luò)配置正確。
我這里是有一臺服務(wù)器是可以正常訪問外網(wǎng)的(IP:192.168.0.10),同時(shí)局域網(wǎng)內(nèi)的其他電腦也可以訪問這臺服務(wù)器。
修改nginx.conf
worker_processes 1; events { worker_connections 1024; } http { include mime.types; default_type application/octet-stream; sendfile on; keepalive_timeout 65; # 正向代理配置內(nèi)容 server { listen 9000; # 監(jiān)聽端口 server_name localhost; set $url "proxy_server_doman_or_ip"; # 設(shè)置代理的域名或IP變量,這里替換成自己需要代理的網(wǎng)站 location / { proxy_pass http://$url:8082; # 將請求轉(zhuǎn)發(fā)到由 $url 變量表示的地址。 } } }
重啟nginx,打開局域網(wǎng)電腦測試訪問:192.168.0.10:9000,網(wǎng)頁跳轉(zhuǎn)成功。不就是請求轉(zhuǎn)發(fā)嘛,so easy!
問題解決?如果這么簡單我就不寫這篇文章了??
果然第二天就出問題了,有同事反饋網(wǎng)站進(jìn)不去了。我想不可能吧,一測試果然不行。果斷查閱了一下資料,找到問題原因:nginx在進(jìn)行域名轉(zhuǎn)發(fā)時(shí)會(huì)先把域名解析成IP保存在內(nèi)存中,后面的訪問都是通過IP直接訪問,如果目標(biāo)網(wǎng)站的 IP 地址發(fā)生變化,我們的代理就會(huì)失效。
找到原因就好辦了,添加域名解析功能嘛
域名解析
為了解決 IP 地址變化的問題,我們可以使用域名解析來動(dòng)態(tài)獲取目標(biāo)網(wǎng)站的 IP 地址。直接奉上完整配置,修改 Nginx 配置如下:
# 設(shè)置Nginx啟動(dòng)一個(gè)工作進(jìn)程,這是Nginx的一個(gè)核心設(shè)置,它決定了Nginx會(huì)使用多少個(gè)工作進(jìn)程來處理客戶端請求。 worker_processes 1; # 配置定義了Nginx如何處理事件,例如連接請求,數(shù)據(jù)傳輸?shù)? events { worker_connections 1024; # 定義了每個(gè)工作進(jìn)程允許的最大并發(fā)連接數(shù)。 } # HTTP模塊的配置塊,包含了HTTP服務(wù)器的一般配置 http { include mime.types; default_type application/octet-stream; sendfile on; keepalive_timeout 65; # 正向代理配置內(nèi)容 server { listen 9000; # 監(jiān)聽端口 server_name localhost; # 設(shè)置DNS解析器的地址為8.8.8.8,并且設(shè)置了解析器的緩存時(shí)間為300秒(這樣每隔300s就會(huì)重新解析一次)。ipv6=off 是關(guān)閉IPv6的解析支持。 resolver 8.8.8.8 valid=300 ipv6=off; resolver_timeout 3s; # 設(shè)置解析DNS的超時(shí)時(shí)間為3秒 proxy_read_timeout 60s; proxy_send_timeout 60s; proxy_connect_timeout 60s; set $url "proxy_server_doman_or_ip"; # 設(shè)置代理的域名或IP變量,這里替換成自己需要代理的網(wǎng)站 location / { proxy_pass http://$url:9000; # 將請求轉(zhuǎn)發(fā)到由 $url 變量表示的地址。9000是目標(biāo)網(wǎng)站的端口。 proxy_buffers 256 4K; # 設(shè)置用于緩存后端響應(yīng)的緩沖區(qū)大小為256個(gè),每個(gè)大小為4K。 proxy_max_temp_file_size 0; # 設(shè)置Nginx暫存響應(yīng)數(shù)據(jù)的最大臨時(shí)文件大小為0,即不使用臨時(shí)文件。 proxy_cache_valid 200 302 1m; # 針對狀態(tài)碼為200和302的響應(yīng),設(shè)置緩存有效期為1分鐘。 proxy_cache_valid 301 1h; # 針對狀態(tài)碼為301的響應(yīng),設(shè)置緩存有效期為1小時(shí)。 proxy_cache_valid any 1m; # 對于其他任何響應(yīng)狀態(tài)碼,設(shè)置緩存有效期為1分鐘。 } } }
重啟nginx,問題解決。到這里,正向代理功能就實(shí)現(xiàn)了。如果只想實(shí)現(xiàn)正向代理功能看到這里就可以了。后面是我在遇到問題時(shí)的一些調(diào)試經(jīng)驗(yàn)技巧!
轉(zhuǎn)發(fā)代理調(diào)試
我在實(shí)現(xiàn)代理后,發(fā)現(xiàn)正向代理響應(yīng)的速度非常慢有時(shí)候需要一分鐘才能響應(yīng)結(jié)果。但有時(shí)候又很快。就好奇為啥會(huì)這樣,為了方便調(diào)試和監(jiān)控,我設(shè)置添加了Nginx 的訪問日志。
修改 Nginx 配置如下:
# 設(shè)置日志記錄格式 log_format main '$remote_addr - $remote_user [$time_local] "$request" ' '$status $body_bytes_sent "$http_referer" ' '"$http_user_agent" "$http_x_forwarded_for"' 'upstreamIP: $upstream_addr' 'upgrade: $http_upgrade'; # 正向代理配置內(nèi)容 server { listen 9000; # 監(jiān)聽端口 server_name localhost; # 設(shè)置DNS解析器的地址為8.8.8.8,并且設(shè)置了解析器的緩存時(shí)間為300秒(這樣每隔300s就會(huì)重新解析一次)。ipv6=off 是關(guān)閉IPv6的解析支持。 resolver 8.8.8.8 valid=300 ipv6=off; resolver_timeout 3s; # 設(shè)置解析DNS的超時(shí)時(shí)間為3秒 proxy_read_timeout 60s; proxy_send_timeout 60s; proxy_connect_timeout 60s; set $url "proxy_server_doman_or_ip"; # 設(shè)置代理的域名或IP變量,這里替換成自己需要代理的網(wǎng)站 location / { proxy_pass http://$url:9000; # 將請求轉(zhuǎn)發(fā)到由 $url 變量表示的地址。9000是目標(biāo)網(wǎng)站的端口。 proxy_buffers 256 4K; proxy_max_temp_file_size 0; proxy_cache_valid 200 302 1m; proxy_cache_valid 301 1h; proxy_cache_valid any 1m; access_log logs/proxy/access.log main; # 定義了訪問日志的路徑和格式。 error_log logs/proxy/error.log; # 定義了錯(cuò)誤日志的路徑 } }
這樣,Nginx 將會(huì)在nginx項(xiàng)目下的 logs/proxy/access.log 文件中記錄所有代理轉(zhuǎn)發(fā)日志。
日志信息:
192.168.0.28 - - [27/Feb/2024:17:02:03 +0800] "GET /prod-api/system/todo/listAll?pageNum=1&pageSize=10&active=true HTTP/1.1" 200 64 "http://192.168.20.2:8082/user-task/todo" "Mozilla/5.0 (Windows NT 10.0; Win64; x64; rv:123.0) Gecko/20100101 Firefox/123.0" "-"upstreamIP: 182.148.159.30:8082
192.168.0.28 - - [27/Feb/2024:17:02:03 +0800] "GET /prod-api/process/business/listAll?pageNum=1&pageSize=10&active=true HTTP/1.1" 200 979 "http://192.168.20.2:8082/user-task/todo" "Mozilla/5.0 (Windows NT 10.0; Win64; x64; rv:123.0) Gecko/20100101 Firefox/123.0" "-"upstreamIP: 182.148.159.30:8082
192.168.0.28 - - [27/Feb/2024:17:02:06 +0800] "GET /prod-api/system/user/select HTTP/1.1" 499 0 "http://192.168.20.2:8082/login?redirect=%2Findex" "Mozilla/5.0 (Windows NT 10.0; Win64; x64; rv:123.0) Gecko/20100101 Firefox/123.0" "-"upstreamIP: 172.16.30.6:8082
通過日志信息可以看到,請求轉(zhuǎn)發(fā)后的upstreamIP有兩個(gè)不同的IP:182.148.159.30和172.16.30.6,說明請求被轉(zhuǎn)發(fā)到了兩個(gè)不同的IP,當(dāng)請求IP為182.148.159.30時(shí),響應(yīng)正常;當(dāng)請求IP為172.16.30.6時(shí),響應(yīng)失敗導(dǎo)致等待。
發(fā)現(xiàn)這個(gè)情況后,我也是很疑惑為啥會(huì)將域名解析出兩個(gè)IP,第一反應(yīng)是域名解析錯(cuò)誤,為了驗(yàn)證我將DNS解析的服務(wù)器換成了離我最近的四川電信服務(wù)器:61.139.2.69,測試發(fā)現(xiàn)結(jié)果也是一樣的。雖然感覺解析是沒有問題的,保險(xiǎn)起見我還是自己在網(wǎng)上通過解析網(wǎng)站測試了一下域名解析的結(jié)果:結(jié)果顯示域名確實(shí)綁定了兩個(gè)IP,一個(gè)是正常的一個(gè)是不行的。
然后我就聯(lián)系我們這個(gè)域名對應(yīng)網(wǎng)站的負(fù)責(zé)人,得出的結(jié)果就是他們確實(shí)綁定了兩個(gè)IP,有一個(gè)IP是綁定的內(nèi)網(wǎng)IP,所以才導(dǎo)致外網(wǎng)請求時(shí)失效。
失敗的原因:nginx獲取解析后的IP是通過輪詢往IP轉(zhuǎn)發(fā)請求,如果輪詢到內(nèi)網(wǎng)IP就會(huì)導(dǎo)致請求失敗。
解決辦法:
手動(dòng)選擇IP地址:在Nginx配置中,你可以手動(dòng)指定代理的目標(biāo)IP地址,而不是使用解析出的IP地址。這樣可以避免將請求發(fā)送到內(nèi)網(wǎng)IP。
使用域名解析出的IP地址進(jìn)行請求:你可以通過解析域名獲取到的IP地址列表,然后使用一些方法(比如按照某種規(guī)則選擇IP地址)來保證請求不會(huì)發(fā)送到內(nèi)網(wǎng)IP地址。
使用nginx的upstream模塊進(jìn)行負(fù)載均衡:通過upstream模塊,你可以指定多個(gè)代理服務(wù)器,然后使用一定的負(fù)載均衡策略,比如輪詢、權(quán)重等,來分配請求到不同的服務(wù)器。在這里,你可以手動(dòng)配置upstream模塊,指定外網(wǎng)IP,而不包括內(nèi)網(wǎng)IP。
修改DNS設(shè)置:聯(lián)系網(wǎng)站負(fù)責(zé)人,修改域名解析,只將外網(wǎng)IP綁定到域名上,而不包括內(nèi)網(wǎng)IP。
你們猜我最后咋解決的???
必須是第四條?。。?!
到此這篇關(guān)于利用Nginx正向代理實(shí)現(xiàn)局域網(wǎng)電腦訪問外網(wǎng)的文章就介紹到這了,更多相關(guān)Nginx局域網(wǎng)電腦訪問外網(wǎng)內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Nginx使用的php-fpm的兩種進(jìn)程管理方式及優(yōu)化
這篇文章主要介紹了Nginx使用的php-fpm的兩種進(jìn)程管理方式及優(yōu)化,需要的朋友可以參考下2016-09-09Nginx access.log日志詳解及統(tǒng)計(jì)分析小結(jié)
nginx有一個(gè)非常靈活的日志記錄模式,本文主要介紹了Nginx access.log日志詳解及統(tǒng)計(jì)分析小結(jié),文中通過示例代碼介紹的非常詳細(xì),需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2024-03-03Nginx服務(wù)器中l(wèi)ocation配置的一些基本要點(diǎn)解析
這篇文章主要介紹了Nginx服務(wù)器中l(wèi)ocation配置的一些基本要點(diǎn)解析,特別對管理以及查找匹配作出了詳細(xì)的講解,需要的朋友可以參考下2015-12-12