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