SpringBoot实现即时通讯

SpringBoot实现即时通讯

功能简述

  • 好友管理
  • 群组管理
  • 聊天模式:私聊、群聊
  • 消息类型:系统消息、文本、语音、图片、视频
  • 会话列表、发送消息、接收消息

核心代码

java 复制代码
package com.qiangesoft.im.core;

import com.alibaba.fastjson2.JSONObject;
import com.qiangesoft.im.constant.ChatTypeEnum;
import com.qiangesoft.im.constant.ImMessageBodyTypeEnum;
import com.qiangesoft.im.service.IImGroupUserService;
import com.qiangesoft.im.util.SpringUtil;
import com.qiangesoft.im.pojo.dto.PingDTO;
import com.qiangesoft.im.pojo.vo.PongVO;
import com.qiangesoft.im.pojo.vo.ImMessageVO;
import com.qiangesoft.im.pojo.dto.ImMessageDTO;
import lombok.extern.slf4j.Slf4j;
import org.springframework.stereotype.Component;

import javax.websocket.*;
import javax.websocket.server.PathParam;
import javax.websocket.server.ServerEndpoint;
import java.io.IOException;
import java.util.List;
import java.util.Map;
import java.util.concurrent.ConcurrentHashMap;

/**
 * 聊天会话
 *
 * @author qiangesoft
 * @date 2023-08-30
 */
@Slf4j
@ServerEndpoint("/ws/im/{userId}")
@Component
public class ImWebSocketServer {

    /**
     * concurrent包的线程安全Set,用来存放每个客户端对应的WebSocket对象。
     */
    private static final ConcurrentHashMap<Long, ImWebSocketServer> WEBSOCKET_MAP = new ConcurrentHashMap<>();
    /**
     * 与某个客户端的连接会话,需要通过它来给客户端发送数据
     */
    private Session session;

    /**
     * 连接建立成功调用的方法:用map存客户端对应的WebSocket对象
     */
    @OnOpen
    public void onOpen(Session session, @PathParam("userId") Long userId) {
        this.session = session;
        if (WEBSOCKET_MAP.containsKey(userId)) {
            WEBSOCKET_MAP.remove(userId);
            WEBSOCKET_MAP.put(userId, this);
        } else {
            WEBSOCKET_MAP.put(userId, this);
        }
        log.info("User [{}] connection opened=====>", userId);

        PongVO pongVO = new PongVO();
        pongVO.setType(ImMessageBodyTypeEnum.SUCCESS.getCode());
        pongVO.setContent("连接成功");
        pongVO.setTimestamp(System.currentTimeMillis());
        doSendMessage(JSONObject.toJSONString(pongVO));
    }

    /**
     * 收到客户端消息后调用的方法
     */
    @OnMessage
    public void onMessage(Session session, @PathParam("userId") Long userId, String message) {
        log.info("User [{}] send a message, content is [{}]", userId, message);

        PingDTO pingDTO = null;
        try {
            pingDTO = JSONObject.parseObject(message, PingDTO.class);
        } catch (Exception e) {
            log.error("消息解析失败");
            e.printStackTrace();
        }
        if (pingDTO == null || !ImMessageBodyTypeEnum.PING.getCode().equals(pingDTO.getType())) {
            sendInValidMessage();
            return;
        }

        PongVO pongVO = new PongVO();
        pongVO.setType(ImMessageBodyTypeEnum.PONG.getCode());
        pongVO.setContent("已收到消息~");
        pongVO.setTimestamp(System.currentTimeMillis());
        doSendMessage(JSONObject.toJSONString(pongVO));
    }

    /**
     * 连接关闭调用的方法
     */
    @OnClose
    public void onClose(Session session, @PathParam("userId") Long userId) {
        close(session, userId);
        log.info("User {} connection is closed<=====", userId);
    }

    /**
     * 报错
     *
     * @param session
     * @param error
     */
    @OnError
    public void onError(Session session, Throwable error) {
        error.printStackTrace();
    }

    /**
     * 指定的userId服务端向客户端发送消息
     */
    public static void sendMessage(ImMessageDTO message) {
        String chatType = message.getChatType();
        if (ChatTypeEnum.GROUP.getCode().equals(chatType)) {
            sendGroupMessage(message);
        }

        if (ChatTypeEnum.PERSON.getCode().equals(chatType)) {
            sendPersonMessage(message);
        }
    }

    /**
     * 指定的userId服务端向客户端发送消息
     */
    public static void offline(Long userId) {
        ImWebSocketServer webSocketServer = WEBSOCKET_MAP.get(userId);
        if (webSocketServer != null) {
            PongVO pongVO = new PongVO();
            pongVO.setType(ImMessageBodyTypeEnum.OFFLINE.getCode());
            pongVO.setContent("设备被挤下线");
            pongVO.setTimestamp(System.currentTimeMillis());
            webSocketServer.doSendMessage(JSONObject.toJSONString(pongVO));

            close(webSocketServer.session, userId);
        }
    }

    /**
     * 自定义关闭
     *
     * @param session
     * @param userId
     */
    public static void close(Session session, Long userId) {
        if (WEBSOCKET_MAP.containsKey(userId)) {
            try {
                session.close();
            } catch (IOException e) {
                e.printStackTrace();
            }
            WEBSOCKET_MAP.remove(userId);
        }
    }

    /**
     * 获取在线用户信息
     *
     * @return
     */
    public static Map<Long, ImWebSocketServer> getOnlineUser() {
        return WEBSOCKET_MAP;
    }

    /**
     * 发送无效消息
     */
    private void sendInValidMessage() {
        PongVO pongVO = new PongVO();
        pongVO.setType(ImMessageBodyTypeEnum.FAIL.getCode());
        pongVO.setContent("无效消息");
        pongVO.setTimestamp(System.currentTimeMillis());
        doSendMessage(JSONObject.toJSONString(pongVO));
    }

    /**
     * 发送群组消息
     *
     * @param message
     */
    private static void sendGroupMessage(ImMessageDTO message) {
        Long receiverId = message.getReceiverId();
        IImGroupUserService groupUserService = SpringUtil.getBean(IImGroupUserService.class);
        List<Long> userIdList = groupUserService.listUserIdByGroupId(receiverId);

        MessageHandlerService messageHandlerService = SpringUtil.getBean(MessageHandlerService.class);
        ImMessageVO messageVO = messageHandlerService.buildVo(message);

        PongVO pongVO = new PongVO();
        pongVO.setType(ImMessageBodyTypeEnum.MESSAGE.getCode());
        pongVO.setContent(messageVO);
        pongVO.setTimestamp(System.currentTimeMillis());
        String messageStr = JSONObject.toJSONString(pongVO);

        for (Long userId : userIdList) {
            ImWebSocketServer webSocketServer = WEBSOCKET_MAP.get(userId);
            if (webSocketServer != null) {
                if (!userId.equals(message.getSenderId())) {
                    webSocketServer.doSendMessage(messageStr);
                }
            }
        }
    }

    /**
     * 发送私聊消息
     *
     * @param message
     */
    private static void sendPersonMessage(ImMessageDTO message) {
        Long receiverId = message.getReceiverId();
        ImWebSocketServer webSocketServer = WEBSOCKET_MAP.get(receiverId);
        if (webSocketServer != null) {
            MessageHandlerService messageHandlerService = SpringUtil.getBean(MessageHandlerService.class);
            ImMessageVO messageVO = messageHandlerService.buildVo(message);

            PongVO pongVO = new PongVO();
            pongVO.setType(ImMessageBodyTypeEnum.MESSAGE.getCode());
            pongVO.setContent(messageVO);
            pongVO.setTimestamp(System.currentTimeMillis());
            webSocketServer.doSendMessage(JSONObject.toJSONString(pongVO));
        }
    }

    /**
     * 实现服务器推送到对应的客户端
     */
    private void doSendMessage(String message) {
        try {
            this.session.getBasicRemote().sendText(message);
        } catch (IOException e) {
            e.printStackTrace();
        }
    }
}
相关推荐
小池先生38 分钟前
springboot启动不了 因一个spring-boot-starter-web底下的tomcat-embed-core依赖丢失
java·spring boot·后端
苹果醋32 小时前
2020重新出发,MySql基础,MySql表数据操作
java·运维·spring boot·mysql·nginx
小蜗牛慢慢爬行2 小时前
如何在 Spring Boot 微服务中设置和管理多个数据库
java·数据库·spring boot·后端·微服务·架构·hibernate
azhou的代码园2 小时前
基于JAVA+SpringBoot+Vue的制造装备物联及生产管理ERP系统
java·spring boot·制造
wm10432 小时前
java web springboot
java·spring boot·后端
龙少95434 小时前
【深入理解@EnableCaching】
java·后端·spring
溟洵6 小时前
Linux下学【MySQL】表中插入和查询的进阶操作(配实操图和SQL语句通俗易懂)
linux·运维·数据库·后端·sql·mysql
SomeB1oody9 小时前
【Rust自学】6.1. 定义枚举
开发语言·后端·rust
SomeB1oody9 小时前
【Rust自学】5.3. struct的方法(Method)
开发语言·后端·rust
路在脚下@10 小时前
spring boot的配置文件属性注入到类的静态属性
java·spring boot·sql