PHP用fsockopen连WebSocket卡住的根本原因是未发送HTTP Upgrade请求,而非超时设置问题;必须手动构造含Upgrade: websocket等头的GET请求,否则服务端不响应导致阻塞。
fsockopen 连 WebSocket 时卡住,根本不是超时没设,是没发 Upgrade 请求PHP 原生不支持 WebSocket 协议,fsockopen 只能建 TCP 连接,但 WebSocket 握手必须发 HTTP Upgrade 请求。如果只连上就等响应,服务端不回、客户端不发,就会一直阻塞——看着像“超时”,其实是卡在协议层。
实操建议:
GET / HTTP/1.1 请求头,含 Upgrade: websocket、Connection: Upgrade、Sec-WebSocket-Key(需 base64_encode(random_bytes(16)))fsockopen 调用时传入第 4 个参数 $errno 和第 5 个参数 $errstr,检查连接是否真失败,而非靠后续读取判断fwrite($fp, $handshake_request),再 fread($fp, 2048) 读响应,否则会挂死stream_socket_client 设置连接超时和读写超时stream_socket_client 比 fsockopen 更可控,支持流上下文(context),可分别设连接阶段和 I/O 阶段的超时。
常见错误:只设了 timeout,但 WebSocket 握手后还有帧解析,读响应时仍可能卡住。
实操建议:
timeout(单位秒,如 5),控制 stream_socket_client 建连耗时stream_set_timeout($fp, 5) 设置读操作超时,否则 fgets 或 fread 可能永远等stream_get_contents 读响应体——它不识别 WebSocket 帧结构,容易读不完或读错长度ext-websocket 或 reactphp/socket
原生 PHP 处理 WebSocket 是硬扛协议细节,出错率高、维护成本大。生产环境别自己拼 Sec-WebSocket-Accept 校验逻辑。
实操建议:
ext-websocket(非官方但较成熟),用 websocket_connect 封装握手和 ping/pongreactphp/socket + evenement/emitter 可异步建连,配合 clue/reactphp-buzz 发 Upgrade 请求更稳select 或 poll,Linux 上要确认 ulimit -n 足够,否则并发连多个 ws 会报 Too many open files
别猜。把每一步拆开测:DNS 解析 → TCP 连接 → SSL 握手(

实操建议:
dns_get_record($host) 单独测 DNS,排除域名解析慢stream_socket_client 的 $errstr 返回值里,Connection refused 是端口不通,Operation timed out 是连不上(网络或防火墙),SSL handshake failed 是证书或 TLS 版本问题tcpdump -i any port 8080 -w ws.pcap,Wireshark 打开查有没有 HTTP/1.1 101 响应