客戶端連接失敗原因
連接失敗客戶端一般會有兩種報錯,connection refuse
和 connection timeout
connection refuse(連接拒絕)
一般是以下原因:
1、客戶端連接的端口錯了
2、客戶端連接的域名或者ip錯了
3、如果客戶端使用了域名連接,域名可能指向了錯誤的服務器ip
4、服務器使用了cdn等加速代理,導致連接的實際ip與預期ip不一致
5、服務端沒有啟動或者端口沒有被監(jiān)聽
6、使用了網絡代理軟件
7、服務端監(jiān)聽ip與訪問地址不在一個地址段。例如服務端監(jiān)聽127.0.0.1,則客戶端只能通過127.0.0.1連接,不能通過局域網ip或者外網ip連接。建議監(jiān)聽地址設置為0.0.0.0,這樣本機、內網、外網都可以連接。
connection timeout(連接超時)
一般是以下原因:
1、服務器防火墻阻止了連接,可以臨時關閉防火墻試下
2、如果是云服務器,安全組也可能會阻止連接建立,需要到管理后臺開放對應端口
3、如果用了寶塔等面板,需要在寶塔中開放對應端口
4、服務器不存在或者沒有啟動
5、如果客戶端使用了域名連接,域名可能指向了錯誤的服務器ip
6、客戶端訪問的ip是服務器內網ip,并且客戶端和服務端不在一個局域網
cannot assign requested address (無法分配請求地址)
作為客戶端時,每發(fā)起一個連接需要占用本地一個臨時端口,一臺服務器默認可用臨時端口大概在2-3萬,如果向特定服務器發(fā)起的連接數(shù)超過這個值后將無法分配可用端口,會產生這個錯誤。
可以通過更改內核參數(shù)/etc/sysctl.conf
中的 net.ipv4.ip_local_port_range
來增加本地臨時端口數(shù)量,例如設置成10000 65535
(本地端口范圍設置成10000 65535,也就是本地端口數(shù)增加到55535個),運行sysctl -p
生效。
另外連接斷開后連接變成TIME_WAIT狀態(tài),仍然會占用對應本地端口一段時間,也就是短時間內發(fā)起大量(超過2-3w)短連接也會報Cannot assign requested address
,如果是這種情況可以通過設置內核快速回收TIME_WAIT來解決,參考內核調優(yōu)。
注意
本地端口數(shù)限制僅限于客戶端,服務端沒有本地端口限制,只要資源足夠,服務端維持連接數(shù)量可以看作是無限。
其它報錯
如果發(fā)生的報錯不是connection refuse
和 connection timeout
則一般是以下原因:
1、客戶端使用的通訊協(xié)議與服務端不一致。
例如服務端是http通訊協(xié)議,客戶端使用websocket通訊協(xié)議訪問是無法連接的。如果客戶端用websocket協(xié)議連接,那么服務端必須也是websocket協(xié)議。如果服務端是http協(xié)議的服務,那么客戶端必須用http協(xié)議訪問。
這里的原理類似如果你要和英國人交流,那么要使用英語。如果要和日本人交流,那么要使用日語。這里的語言就類似通訊協(xié)議,雙方(客戶端和服務端)必須使用相同的語言才能交流,否則無法通訊。
通訊協(xié)議不一致導致的常見的報錯有:
WebSocket connection to 'ws://xxx.com:xx/' failed: Error during WebSocket handshake: Unexpected response code: xxx
WebSocket connection to 'ws://xxx.com:xx/' failed: Error during WebSocket handshake: net::ERR_INVALID_HTTP_RESPONSE
解決辦法:
從上面兩條報錯看出,客戶端使用的是ws連接是websocket協(xié)議。服務端也需要是websocket協(xié)議才行,服務端監(jiān)聽部分代碼需要指定websocket協(xié)議才能通訊,例如下面這樣
如果是gatewayWorker,監(jiān)聽部分代碼類似
// websocket協(xié)議,這樣客戶端才能用ws://...來連。xxxx為端口不用改動
$gateway = new Gateway('websocket://0.0.0.0:xxxx');
如果是Workerman則是
// websocket協(xié)議,這樣客戶端才能用ws://...來連。xxxx為端口不用改動
$worker = new Worker('websocket://0.0.0.0:xxxx');