如何通过Web套接字向连接的用户发送消息?

时间:2017-03-15 08:50:01

标签: java websocket java-ee-7

我想通过Web套接字向特定用户发送消息。到目前为止,我可以打开一个Web套接字并从客户端读取消息:

@ServerEndpoint(value = "/wsep")
public class WebSocketEndpoint {

    private static final Logger LOGGER = LoggerFactory.getLogger(WebSocketEndpoint.class);

    private Session session;

    @OnOpen
    public void onOpen(Session session) {
        this.session = session;
        try {
            session.getBasicRemote().sendText("You are connected. Your ID is " + session.getId());
        } catch (Exception e) {
            LOGGER.error("Error on open web socket", e);
        }
    }

    @OnMessage
    public void onClientMessage(String message, Session session) {      
        LOGGER.info("Message from {} is: {}", session.getId(), message);
    }

    @OnClose
    public void onClose(Session session) {
        this.session = null;
        LOGGER.info("{} disconnected", session.getId());
    }
}

我有一个独立的服务,可以在目的地为用户创建消息。我的Message课程是一个简单的POJO:

public class Message {
    private String fromUserName;
    private String toUserName;
    private String content;
    ...
}

当在我的MessageService中创建新消息时,我想通知接收者他是否已连接。我想我必须添加方法WebSocketEndpoint.onServerMessage

public void onServerMessage(Session session, Message message) {
    session.getBasicRemote().sendText(message.getContent());
}

但我不知道怎么做那样有用的东西。

1 个答案:

答案 0 :(得分:3)

所有用户都会有一个ServerEndpoint实例。因此,它应该存储所有客户端会话。正如Vu.N建议的那样,一种方法是使用地图:

Map<String, Session> sessions = new ConcurrentHashMap<>();

public void onOpen(Session session) {
    String username = [...]
    sessions.put(username, session);
}

然后,将消息发送给用户将很容易:

public void onServerMessage(Session session, Message message) {
    sessions.get(message.getToUserName())
            .getBasicRemote() // see also getAsyncRemote()
            .sendText(message.getContent());
}

现在最困难的部分是获得username

在我以前的作品中,我已经用3种方式完成了它:

  1. 客户端连接到具有某些&#34;键&#34;的URL。这个&#34;关键&#34;将用于找到正确的用户名。 WebSocket服务器端点如下所示:

    @ServerEndpoint(value="/wsep/{key}") // the URL will have an extra "key"
    public class WebSocketEndpoint {
    [...]
    @OnOpen
    public void onOpen(Session session, @PathParam("key") String key) {
        String username = getUserNameWithKey(key);
        sessions.add(username, session);
    }
    
  2. 客户端在第一条消息中发送一些信息。你只需忽略@OnOpen部分:

    @OnOpen
    public void onOpen(Session session) {
        LOGGER.info("Session open. ID: {}", session.getId());
    }
    
    @OnMessage
    public void onMessage(Session session, String message) {
        String username = getUserNameFromMessage(message);
        sessions.add(username, session);
    }
    
  3. 可以从cookie,JWT或其他东西获取一些用户信息。您需要ServerEndpointConfig.Configurator才能从请求中获取该信息。例如:

    public class CookieServerConfigurator extends ServerEndpointConfig.Configurator {
    
        @Override
        public void modifyHandshake(ServerEndpointConfig sec, HandshakeRequest request, HandshakeResponse response) {
            Map<String,List<String>> headers = request.getHeaders();
            sec.getUserProperties().put("cookie", headers.get("cookie"));
        }
    }
    

    然后,服务器端点将指向该配置程序:

    @ServerEndpoint(value = "/example2/", configurator = CookieServerConfigurator.class)
    public class WebSocketEndpoint {
    

    你可以得到这样的信息:

    @OnOpen
    public void onOpen(Session session, EndpointConfig endpointConfig) {
        String username = getUsername((List<String>)endpointConfig.getUserProperties().get("cookie"));
    
  4. 您可以在此处查看一些有用的示例:https://github.com/matruskan/websocket-example

    对于更复杂的系统,您还可以使用&#34;标记系统&#34;而不是使用Map。然后,每个会话都可以接收发送到任何标签的消息,发送到标签的消息可以定向到许多会话。