frp
是一個用Go
語言開發的,可用於內網穿透
的高性能的反向代理應用,支持 tcp, udp 、 http 和 https。可將一個部署在本機的web服務映射到外網。html
本文主要講如何基於frp
+ nginx
配置http 內網穿透服務,承載多人同時使用,從而支持微信公衆號,微信小程序的本地開發調試linux
所需資源:nginx
msh.com
爲例)本文涉及的環境git
(請仔細閱讀原理,在不理解原理的狀況下上手配置容易出錯,且很難定位緣由。這都是本人所經歷的慘痛教訓)github
以本人搭建的frp
內網穿透服務爲例:web
第一步: 配置無誤的狀況下,frp服務端和frp客戶端前後啓動,創建通訊隧道
,其中:shell
7071
端口(此端口可自定義),接收此端口下全部外網用戶請求8585
, 8686
端口爲例第二步: 經過配置nginx
反向代理,將指向本臺公網服務器的dev.msh.com
下的子域名,映射到服務器的7071
端口,也就是frp監聽的那個端口。 外網用戶訪問dev.msh.com
下的子域名,例如 :小程序
a.dev.msh.com
b.dev.msh.com
等同於訪問msh.com:7071
,會 觸發 frp服務端和客戶端的互動,從而http請求由frp服務端傳遞到frp客戶端windows
第三步: frp客戶端收到http請求後,基於自定義配置,則作以下處理:微信小程序
a.dev.msh.com
,則將請求轉發到我本地的8585
web服務端口b.dev.msh.com
,則將請求轉發到我本地的8686
web服務端口第四步: 本地的web服務收到http請求後,對請求作處理,並完成響應
第五步: frp客戶端將響應結果回傳給frp的服務端。服務端最終將響應回傳給外網用戶
第六步: 最終的實測效果爲:
a.dev.msh.com
,等同於訪問我本地的localhost:8585
b.dev.msh.com
,等同於訪問我本地的localhost:8686
本文以msh.com
爲例:
登陸域名的解析後臺,在msh.com
下增長兩條A記錄: dev
, *.dev
,記錄值爲部署frp服務端的公網服務器的ip。
表明dev.msh.com
下的全部的子域名,會所有指向此臺公網服務器。
由於本文采用的是綠色安裝,因此不須要配置go語言環境。多謝 Tylerrrkd 指正
下載解壓
# 下載
wget https://github.com/fatedier/frp/releases/download/v0.22.0/frp_0.22.0_linux_amd64.tar.gz
# 解壓
tar -zxvf frp_0.22.0_linux_amd64.tar.gz
複製代碼
修改配置文件
解壓後進入解壓目錄,找到frps.ini
文件, 作以下配置 。配置說明請參見各項對應的註釋
[common]
# frp監聽的端口,用做服務端和客戶端通訊
bind_port = 7000
# 服務端經過此端口接監聽和接收公網用戶的http請求
vhost_http_port = 7071
# frp提供了一個控制檯,能夠經過這個端口訪問到控制檯。可查看frp當前有多少代理鏈接以及對應的狀態
dashboard_port = 7500
# 服務端的subdomain_host須要和客戶端配置文件中的subdomain、local_port配合使用,
# 可經過{subdomain}.{subdomain_host} 的域名格式來訪問本身本地的 web 服務。
# 假如服務端的subdomain_host爲dev.msh.com,客戶端某個配置組中的
# subdomain爲a,local_port爲8585,
# 則:
# 訪問 a.dev.msh.com ,等同於訪問本地的localhost:8585
subdomain_host = dev.msh.com
複製代碼
啓動frp服務端
./frps -c frps.ini
複製代碼
(關於如何安裝nginx
就不過多贅述,詳細安裝教程請參考 Nginx Linux詳細安裝部署教程)
修改 nginx.conf
文件
# frp的接收http請求的反向代理
server {
listen 80;
server_name *.dev.msh.com dev.msh.com;
location / {
# 7071端口即爲frp監聽的http端口
proxy_pass http://127.0.0.1:7071;
proxy_set_header Host $host:80;
proxy_set_header X-Real-IP $remote_addr;
proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for;
proxy_set_header Upgrade $http_upgrade;
proxy_set_header Connection "upgrade";
proxy_connect_timeout 7d;
proxy_send_timeout 7d;
proxy_read_timeout 7d;
}
# 防止爬蟲抓取
if ($http_user_agent ~* "360Spider|JikeSpider|Spider|spider|bot|Bot|2345Explorer|curl|wget|webZIP|qihoobot|Baiduspider|Googlebot|Googlebot-Mobile|Googlebot-Image|Mediapartners-Google|Adsbot-Google|Feedfetcher-Google|Yahoo! Slurp|Yahoo! Slurp China|YoudaoBot|Sosospider|Sogou spider|Sogou web spider|MSNBot|ia_archiver|Tomato Bot|NSPlayer|bingbot")
{
return 403;
}
}
複製代碼
讓nginx從新加載配置文件
/usr/local/nginx/sbin/nginx -s reload
複製代碼
# 開啓防火牆端口 7000端口和7071端口即爲上面配置的bind_port和vhost_http_port端口
firewall-cmd --zone=public --add-port=7000/tcp --permanent
firewall-cmd --zone=public --add-port=7071/tcp --permanent
# 開啓後重啓防火牆,使得剛剛的修改生效
firewall-cmd --reload
複製代碼
下載客戶端
去github上面下載最新版的 windows客戶端 github.com/fatedier/fr… ,找到 frp_0.23.1_windows_amd64.zip
,點擊下載便可
(Mac用戶請下載Mac版本的客戶端)
解壓後,編輯 frpc.ini 文件
[common]
# 部署frp服務端的公網服務器的ip
server_addr = 132.232.64.79
# 和服務端的bind_port保持一致
server_port = 7000
# 代理服務一 ,[]內的代理服務名稱在全局範圍內確保惟一,每一個人的每一個代理服務不能重名,
# 不然會影響正常使用。
[http-a]
type = http
# local_port表明你想要暴露給外網的本地web服務端口
local_port = 8585
# subdomain 在全局範圍內要確保惟一,每一個代理服務的subdomain不能重名,不然會影響正常使用。
# 客戶端的subdomain需和服務端的subdomain_host配合使用
subdomain = a
# 代理服務二 ,各項配置說明請參考配置組一
[http-b]
type = http
local_port = 8686
subdomain = b
複製代碼
啓動 客戶端
在frp解壓目錄下右鍵打開 powershell
或者cmd
,執行以下命令
./frpc.exe -c .\frpc.ini
複製代碼
若是窗口提示 『start proxy success』 ,則表明frp服務端和frp客戶端的通訊隧道
創建成功
測試訪問
在瀏覽器裏面訪問 http://a.dev.msh.com
、測試本地的web服務是否已經暴露給外網
疑問: 爲何要搭建本身的內網穿透服務,而不是購買相似花生殼這種收費的內網穿透服務?
解疑: 花生殼收費太高,一個旗艦版的一年都須要868元,並且只有4個端口映射,意味着只有4名開發人員同時用均可能不夠。一箇中型的互聯網公司都有四五十人或者上百人,若是用花生殼的話,每一年都須要三四萬花費,顯然不是個小數目。而對於一家互聯網公司,都是有本身的服務器資源和域名資源的,既然如此,何不搭建一個自有的內網穿透服務?
疑問:微信小程序只支持https協議,而剛搭建的是http內網穿透,不適用怎麼辦?
解疑: 能夠在微信Web開發者工具裏面找到項目設置,把 『不校驗合法域名、業務域名、TLS版本以及HTTPS證書』 項勾選便可。這樣就能夠在生產環境下走https
協議,本地開發環境下走http
協議
關於生產環境下怎麼部署https
,請參考本人在掘金上的這篇文章全站HTTPS升級系列
另外關於本地開發環境下怎麼部署https,曾經嘗試過mkcert
、jdk的keystore
,然而最終沒有找打一個切實可行的方案
疑問:我搭建的內網穿透服務,怎麼限定只有內部成員可用,防止外人隨意『搭便車』?
解疑: 能夠基於token參數來完成身份驗證。服務端和客戶端的 common 配置中的 token 參數一致則身份驗證經過。