客戶端連接失敗原因
連接失敗客戶端一般會有兩種報錯,connection refuse 和 connection timeout
connection refuse(連接拒絕)
一般是以下原因:
1、客戶端連接的端口錯了
2、客戶端連接的域名或者ip錯了
3、如果客戶端使用了域名連接,域名可能指向了錯誤的伺服器ip
4、伺服器使用了cdn等加速代理,導致連接的實際ip與預期ip不一致
5、服務端沒有啟動或者端口沒有被監聽
6、使用了網路代理軟體
7、服務端監聽ip與訪問地址不在一個地址段。例如服務端監聽127.0.0.1,則客戶端只能透過127.0.0.1連接,不能透過區域網ip或者外網ip連接。建議監聽地址設置為0.0.0.0,這樣本機、內網、外網都可以連接。
connection timeout(連接超時)
一般是以下原因:
1、伺服器防火牆阻止了連接,可以暫時關閉防火牆試下
2、如果是雲伺服器,安全組也可能會阻止連接建立,需要到管理後台開放對應端口
3、如果用了寶塔等面板,需要在寶塔中開放對應端口
4、伺服器不存在或者沒有啟動
5、如果客戶端使用了域名連接,域名可能指向了錯誤的伺服器ip
6、客戶端訪問的ip是伺服器內網ip,並且客戶端和服務端不在一個區域網
cannot assign requested address (無法分配請求地址)
作為客戶端時,每發起一個連接需要佔用本地一個臨時端口,一台伺服器預設可用臨時端口大概在2-3萬,如果向特定伺服器發起的連接數超過這個值後將無法分配可用端口,會產生這個錯誤。
可以透過更改內核參數/etc/sysctl.conf 中的 net.ipv4.ip_local_port_range 來增加本地臨時端口數量,例如設置成10000 65535(本地端口範圍設置成10000 65535,也就是本地端口數增加到55535個),運行sysctl -p生效。
另外連接斷開後連接變成TIME_WAIT狀態,仍然會佔用對應本地端口一段時間,也就是短時間內發起大量(超過2-3w)短連接也會報Cannot assign requested address,如果是這種情況可以透過設置內核快速回收TIME_WAIT來解決,參考內核調優。
注意
本地端口數限制僅限於客戶端,服務端沒有本地端口限制,只要資源足夠,服務端維持連接數量可以看作是無限。
其它報錯
如果發生的報錯不是connection refuse 和 connection timeout則一般是以下原因:
1、客戶端使用的通訊協議與服務端不一致。
例如服務端是http通訊協議,客戶端使用websocket通訊協議訪問是無法連接的。如果客戶端用websocket協議連接,那麼服務端必須也是websocket協議。如果服務端是http協議的服務,那麼客戶端必須用http協議訪問。
這裡的原理類似如果你要和英國人交流,那麼要使用英文。如果要和日本人交流,那麼要使用日語。這裡的語言就類似通訊協議,雙方(客戶端和服務端)必須使用相同的語言才能交流,否則無法通訊。
通訊協議不一致導致的常見的報錯有:
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協議。服務端也需要是websocket協議才行,服務端監聽部分代碼需要指定websocket協議才能通訊,例如下面這樣
如果是gatewayWorker,監聽部分代碼類似
// websocket協議,這樣客戶端才能用ws://...來連。xxxx為端口不用改動
$gateway = new Gateway('websocket://0.0.0.0:xxxx');
如果是Workerman則是
// websocket協議,這樣客戶端才能用ws://...來連。xxxx為端口不用改動
$worker = new Worker('websocket://0.0.0.0:xxxx');