当前位置:   article > 正文

SpringCloud集成websocket出现Error: Unexpected server response: 200,服务器将WebSocket连接错误地识别为HTTP请求的问题

error: unexpected server response: 200

最近项目里需要使用到websocket,主要用于前后端实时通信,项目中用到的场景是用户扫码乘车之后司机的设备需要语音提醒,思路是司机在打开乘车二维码时前端根据司机的用户ID发送websocket请求,后端在扫码乘车成功后发送消息,在本地测试一切正常,因为在本地与前端联调时没走网关,直接通过websocket所在服务端口来连接,但是上服务器由于服务端口不能对外暴漏,只能走网关或者通过nginx转发来实现,所以就出现了nginx转发成http请求这种问题。

这是websocket实现类代码:

WebSocketServerConvenientlife
  1. import com.alibaba.fastjson.JSON;
  2. import com.alibaba.fastjson.JSONObject;
  3. import org.apache.commons.lang3.StringUtils;
  4. import org.slf4j.Logger;
  5. import org.slf4j.LoggerFactory;
  6. import org.springframework.stereotype.Component;
  7. import javax.websocket.*;
  8. import javax.websocket.server.PathParam;
  9. import javax.websocket.server.ServerEndpoint;
  10. import java.io.IOException;
  11. import java.util.concurrent.CopyOnWriteArraySet;
  12. //乘客扫码后司机角色登录的设备语音提示
  13. @ServerEndpoint (value = "/convenientlife/websocket/{sessionId}")
  14. @Component
  15. public class WebSocketServerConvenientlife
  16. {
  17. private static Logger LOGGER = LoggerFactory.getLogger(WebSocketServerConvenientlife.class);
  18. //静态变量,用来记录当前在线连接数。应该把它设计成线程安全的。
  19. private static int onlineCount = 0;
  20. //concurrent包的线程安全Set,用来存放每个客户端对应的MyWebSocket对象。
  21. private static CopyOnWriteArraySet<WebSocketServerConvenientlife> webSocketSet = new CopyOnWriteArraySet<>();
  22. //与某个客户端的连接会话,需要通过它来给客户端发送数据
  23. private Session session;
  24. //接收sessionId(默认就是司机账户ID)
  25. private String sessionId;
  26. //接收message
  27. private JSONObject message;
  28. /**
  29. * 连接建立成功调用的方法
  30. */
  31. @OnOpen
  32. public void onOpen(Session session, @PathParam ("sessionId") String sessionId)
  33. {
  34. LOGGER.info("---+++++ convenientlife模块成功连接,sessionId :{}", sessionId);
  35. this.session = session;
  36. this.sessionId = sessionId;
  37. webSocketSet.add(this);
  38. addOnlineCount();
  39. LOGGER.info("当前在线连接数为:"+getOnlineCount());
  40. try
  41. {
  42. sendMessage("{}");
  43. }
  44. catch (IOException e)
  45. {
  46. LOGGER.error("websocket IO异常");
  47. }
  48. }
  49. /**
  50. * 连接关闭调用的方法
  51. */
  52. @OnClose
  53. public void onClose()
  54. {
  55. webSocketSet.remove(this);
  56. subOnlineCount();
  57. LOGGER.info("convenientlife模块有一连接关闭!当前在线连接数为:"+getOnlineCount());
  58. }
  59. /**
  60. * 收到客户端消息后调用的方法
  61. *
  62. * @param message 客户端发送过来的消息*/
  63. @OnMessage
  64. public void onMessage(String message, Session session)
  65. {
  66. JSONObject jsonObject = JSON.parseObject(message);
  67. if(jsonObject != null && StringUtils.isNotBlank(jsonObject.getString("sid")))
  68. {
  69. String sid = jsonObject.getString("sid");
  70. for(WebSocketServerConvenientlife item : webSocketSet)
  71. {
  72. if(sid.equals(item.sessionId))
  73. {
  74. item.message = jsonObject;
  75. }
  76. }
  77. }
  78. }
  79. /**
  80. *
  81. * @param session 参数
  82. * @param error 参数
  83. */
  84. @OnError
  85. public void onError(Session session, Throwable error)
  86. {
  87. LOGGER.error("convenientlife模块websocke 发生错误:"+error.getMessage());
  88. }
  89. /**
  90. * 实现服务器主动推送
  91. */
  92. public void sendMessage(String message) throws IOException
  93. {
  94. this.session.getBasicRemote().sendText(message);
  95. }
  96. /**
  97. * 群发自定义消息
  98. * */
  99. public static synchronized void sendInfo(String message, @PathParam ("sessionId") String sessionId)
  100. {
  101. for(WebSocketServerConvenientlife item : webSocketSet)
  102. {
  103. try
  104. {
  105. //只推送给指定的这个sessionId
  106. if(sessionId != null && item.sessionId.equals(sessionId))
  107. {
  108. LOGGER.info("推送sessionId: "+sessionId+";推送内容:"+message);
  109. item.sendMessage(message);
  110. }
  111. }
  112. catch (IOException e)
  113. {
  114. LOGGER.error("sendInfo error: ", e);
  115. }
  116. }
  117. }
  118. /**
  119. * session是否存在
  120. * @param sessionId 参数
  121. * @return boolean
  122. */
  123. public static synchronized boolean existsSesson(@PathParam ("sessionId") String sessionId)
  124. {
  125. if(StringUtils.isBlank(sessionId))
  126. {
  127. return false;
  128. }
  129. for(WebSocketServerConvenientlife item : webSocketSet)
  130. {
  131. if(item.sessionId.equals(sessionId))
  132. {
  133. return true;
  134. }
  135. }
  136. return false;
  137. }
  138. public static synchronized JSONObject getMessage(@PathParam ("sessionId") String sessionId)
  139. {
  140. if(StringUtils.isBlank(sessionId))
  141. {
  142. return null;
  143. }
  144. for(WebSocketServerConvenientlife item : webSocketSet)
  145. {
  146. if(item.sessionId.equals(sessionId))
  147. {
  148. return item.message;
  149. }
  150. }
  151. return null;
  152. }
  153. public static synchronized int getOnlineCount()
  154. {
  155. return onlineCount;
  156. }
  157. public static synchronized void addOnlineCount()
  158. {
  159. WebSocketServerConvenientlife.onlineCount++;
  160. }
  161. public static synchronized void subOnlineCount()
  162. {
  163. WebSocketServerConvenientlife.onlineCount--;
  164. }
  165. }

 使用 @ServerEndpoint 注解定义 WebSocket 服务器端的终端点,WebSocket 终端点可以处理来自客户端的连接和消息,并可以向客户端发送消息。

WebSocketConfig类:

  1. import org.springframework.context.annotation.Bean;
  2. import org.springframework.context.annotation.Configuration;
  3. import org.springframework.web.socket.server.standard.ServerEndpointExporter;
  4. @Configuration
  5. public class WebSocketConfig {
  6. /** 部署时注释 start**/
  7. @Bean
  8. public ServerEndpointExporter serverEndpointExporter() {
  9. return new ServerEndpointExporter();
  10. }
  11. /** 部署时注释 end**/
  12. }

这个类是必须要的,配置和初始化 WebSocket 相关的设置和组件,以便在应用程序中启用 WebSocket 功能。

服务器nginx配置如下:

  1. location /ws {
  2. proxy_http_version 1.1;
  3. proxy_set_header Upgrade $http_upgrade;
  4. proxy_set_header Connection "upgrade";
  5. proxy_pass http://ip:6100;
  6. }

前缀为/ws开头的请求转发到6100端口去,6100是WebSocket所在服务的端口,于是我用postman测试了一下

就出现了WebSocket连接识别成http请求的错误了,看了一下nginx日志,也没发现报错信息,但确实将WebSocket连接识别成http请求转发到服务中去了                                                               

那这就很奇怪了,我明明nginx配置了WebSocket连接转发请求,然后框框一顿查,以为是服务不支持接收WebSocket请求,检查pom文件、配置文件,最后发现需要在nginx配置端口后面加一个接口前缀,改成这样

  1. location /ws {
  2. proxy_http_version 1.1;
  3. proxy_set_header Upgrade $http_upgrade;
  4. proxy_set_header Connection "upgrade";
  5. proxy_pass http://ip:6100/convenientlife;
  6. }

然后再用postman请求:

连接成功!!!查了一下大概原因:

服务器端应用程序的路径:在更改Nginx配置后,将WebSocket连接的请求代理到http://ip:6100/convenientlife这个路径上。如果你的服务器端应用程序正好在/convenientlife路径下,那么添加了接口前缀后,Nginx将正确地将WebSocket请求转发到正确的路径,从而实现了连接成功。

声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/很楠不爱3/article/detail/179213
推荐阅读
相关标签
  

闽ICP备14008679号