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();
        }
    }
}
相关推荐
Victor3561 小时前
MongoDB(51)什么是分片?
后端
Victor3561 小时前
MongoDB(50)副本集中的角色有哪些?
后端
IT_陈寒2 小时前
JavaScript开发者必看:5个让你的代码性能翻倍的隐藏技巧
前端·人工智能·后端
shengjk12 小时前
大数据工程师必看:为什么你的 IN 查询在 Flink/Spark 上慢到离谱?
后端
武子康2 小时前
大数据-252 离线数仓 - Airflow + Crontab 入门实战:定时调度、DAG 编排与常见报错排查
大数据·后端·apache hive
程序员Terry3 小时前
RocketMQ 使用指南
后端·rocketmq
AI茶水间管理员3 小时前
OpenClaw 的 Token 消耗怎么计算?(附实操优化方案)
后端
星浩AI3 小时前
现在最需要被 PUA 的,其实是 AI
人工智能·后端·github
程序员老赵3 小时前
超全 Docker 镜像源配置指南|Windows/Mac/Linux一键搞定,拉镜像再也不卡顿
linux·后端·容器
弹简特3 小时前
【JavaEE18-后端部分】 MyBatis 入门第二篇:使用注解完成增删改查(含有参数传递底层原理)
spring boot·mybatis