gateway + websocket 实现权限校验

2024-09-04 05:44

本文主要是介绍gateway + websocket 实现权限校验,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

添加websocket的依赖

<!-- SpringBoot Websocket -->
<dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-websocket</artifactId>
</dependency>

修改默认的数据传输大小及会话超时

@Configuration
public class WebSocketConfig {@Beanpublic ServerEndpointExporter serverEndpointExporter() {return new ServerEndpointExporter();}@Beanpublic ServletServerContainerFactoryBean servletServerContainerFactoryBean() {ServletServerContainerFactoryBean factoryBean = new ServletServerContainerFactoryBean();// 1MfactoryBean.setMaxTextMessageBufferSize(1024 * 1024);factoryBean.setMaxBinaryMessageBufferSize(1024 * 1024);// 30 分钟factoryBean.setMaxSessionIdleTimeout(1000 * 60 * 30L);return factoryBean;}
}

websocket服务端

@Slf4j
@Component
@ServerEndpoint(value = "/websocket/{clientId}", encoders = {WebSocketObjectEncoder.class})
public class WebSocketHandler {private RedisService redisService = SpringUtils.getBean(RedisService.class);private static final ConcurrentMap<String, Session> SESSION_MAP = new ConcurrentHashMap<>();private String clientId;@OnOpenpublic void onOpen(Session session, @PathParam("clientId") String clientId) throws IOException {String sessionId = session.getId();log.info("onOpen sessionId: {}, clientId: {}", sessionId, clientId);this.clientId = clientId;// 增加权限校验boolean res = this.validateToken(session.getRequestParameterMap());if (!res) {// 未校验通过直接断开session.close(new CloseReason(CloseReason.CloseCodes.PROTOCOL_ERROR, "非法请求"));return;}SESSION_MAP.put(clientId, session);this.sendText(session, String.format("客户端【%s】已连接", clientId));}@OnMessagepublic void onMessage(Session session, @PathParam("clientId") String clientId, String message) {String sessionId = session.getId();log.info("onMessage sessionId:{}, clientId:{}, message:{}", sessionId, clientId, message);this.sendText(session, String.format("客户端【%s】消息已收到", clientId));}@OnClosepublic void onClose(Session session) {log.info("sessionId-{} onClose ...", session.getId());SESSION_MAP.remove(this.clientId);}@OnErrorpublic void onError(Session session, Throwable throwable) throws IOException {log.error("Error for session " + session.getId() + " : " + throwable.getMessage());if (session.isOpen()) {session.close(new CloseReason(CloseReason.CloseCodes.UNEXPECTED_CONDITION, throwable.getMessage()));}SESSION_MAP.remove(this.clientId);}/*** 检验客户端身份** @param params* @return*/private boolean validateToken(Map<String, List<String>> params) {try {boolean res = true;String token = params.getOrDefault("token", new ArrayList<>()).get(0);if (StringUtils.isBlank(token)) {res = false;}Claims claims = JwtUtils.parseToken(token);String jwt_claims_key = "user_id";if (claims == null || !claims.containsKey(jwt_claims_key)) {res = false;}// RedisKey.USER_LOGIN_TOKEN = "user_login_token:%s"String redisKey = String.format(RedisKey.USER_LOGIN_TOKEN, JwtUtils.getUserId(claims));if(!redisService.hasKey(redisKey)){res = false;}return res;} catch (Exception e) {log.info("WebSocket token validate error: {}", e.getMessage());}return false;}/*** 发送消息,需要对Object进行序列化,所以WebSocketObjectEncoder是必须的* 或者直接在这里使用JSONObject.toJSONString(obj) 也是可以的** @param clientId* @param data*/public void sendMessage(String clientId, Object data) {try {if (SESSION_MAP.containsKey(clientId)) {SESSION_MAP.get(clientId).getBasicRemote().sendObject(data);}} catch (Exception e) {log.error("sendMessage error:{}", e.getMessage());}}/*** 发送文本消息** @param session* @param message*/private void sendText(Session session, String message) {try {session.getBasicRemote().sendText(message);} catch (IOException e) {e.printStackTrace();}}
}

序列化工具 WebSocketObjectEncoder

public class WebSocketObjectEncoder implements Encoder.Text<Object> {@Overridepublic String encode(Object obj) throws EncodeException {return JSONObject.toJSONString(obj);}@Overridepublic void init(EndpointConfig endpointConfig) {}@Overridepublic void destroy() {}
}

在gateway中配置路由

spring:cloud:gateway:discovery:locator:lowerCaseServiceId: trueenabled: falseroutes:# WEBSOCKET服务- id: websocketuri: lb:ws://websocketpredicates: - Path=/websocket/**# 安全配置
security:# 不校验白名单ignore:whites:- /websocket/**

测试
使用Postman,点击左上角的Menu - File - New...
在这里插入图片描述
找到WebSocket
在这里插入图片描述
url中填写ws://localhost/websocket/{clientId}?token={token},如果是使用ssl证书的域名,则填写wss://www.xxx.com/websocket/{clientId}?token={token}
在这里插入图片描述
如果能正常接收到服务端返回的消息说明连接成功
在这里插入图片描述
如果token校验错误则会立即断开连接,点击右边的箭头可以查看具体异常信息
在这里插入图片描述

这篇关于gateway + websocket 实现权限校验的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1135155

相关文章

如何使用Java实现请求deepseek

《如何使用Java实现请求deepseek》这篇文章主要为大家详细介绍了如何使用Java实现请求deepseek功能,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 目录1.deepseek的api创建2.Java实现请求deepseek2.1 pom文件2.2 json转化文件2.2

python使用fastapi实现多语言国际化的操作指南

《python使用fastapi实现多语言国际化的操作指南》本文介绍了使用Python和FastAPI实现多语言国际化的操作指南,包括多语言架构技术栈、翻译管理、前端本地化、语言切换机制以及常见陷阱和... 目录多语言国际化实现指南项目多语言架构技术栈目录结构翻译工作流1. 翻译数据存储2. 翻译生成脚本

Android 悬浮窗开发示例((动态权限请求 | 前台服务和通知 | 悬浮窗创建 )

《Android悬浮窗开发示例((动态权限请求|前台服务和通知|悬浮窗创建)》本文介绍了Android悬浮窗的实现效果,包括动态权限请求、前台服务和通知的使用,悬浮窗权限需要动态申请并引导... 目录一、悬浮窗 动态权限请求1、动态请求权限2、悬浮窗权限说明3、检查动态权限4、申请动态权限5、权限设置完毕后

如何通过Python实现一个消息队列

《如何通过Python实现一个消息队列》这篇文章主要为大家详细介绍了如何通过Python实现一个简单的消息队列,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 目录如何通过 python 实现消息队列如何把 http 请求放在队列中执行1. 使用 queue.Queue 和 reque

Python如何实现PDF隐私信息检测

《Python如何实现PDF隐私信息检测》随着越来越多的个人信息以电子形式存储和传输,确保这些信息的安全至关重要,本文将介绍如何使用Python检测PDF文件中的隐私信息,需要的可以参考下... 目录项目背景技术栈代码解析功能说明运行结php果在当今,数据隐私保护变得尤为重要。随着越来越多的个人信息以电子形

使用 sql-research-assistant进行 SQL 数据库研究的实战指南(代码实现演示)

《使用sql-research-assistant进行SQL数据库研究的实战指南(代码实现演示)》本文介绍了sql-research-assistant工具,该工具基于LangChain框架,集... 目录技术背景介绍核心原理解析代码实现演示安装和配置项目集成LangSmith 配置(可选)启动服务应用场景

使用Python快速实现链接转word文档

《使用Python快速实现链接转word文档》这篇文章主要为大家详细介绍了如何使用Python快速实现链接转word文档功能,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 演示代码展示from newspaper import Articlefrom docx import

前端原生js实现拖拽排课效果实例

《前端原生js实现拖拽排课效果实例》:本文主要介绍如何实现一个简单的课程表拖拽功能,通过HTML、CSS和JavaScript的配合,我们实现了课程项的拖拽、放置和显示功能,文中通过实例代码介绍的... 目录1. 效果展示2. 效果分析2.1 关键点2.2 实现方法3. 代码实现3.1 html部分3.2

Java深度学习库DJL实现Python的NumPy方式

《Java深度学习库DJL实现Python的NumPy方式》本文介绍了DJL库的背景和基本功能,包括NDArray的创建、数学运算、数据获取和设置等,同时,还展示了如何使用NDArray进行数据预处理... 目录1 NDArray 的背景介绍1.1 架构2 JavaDJL使用2.1 安装DJL2.2 基本操

最长公共子序列问题的深度分析与Java实现方式

《最长公共子序列问题的深度分析与Java实现方式》本文详细介绍了最长公共子序列(LCS)问题,包括其概念、暴力解法、动态规划解法,并提供了Java代码实现,暴力解法虽然简单,但在大数据处理中效率较低,... 目录最长公共子序列问题概述问题理解与示例分析暴力解法思路与示例代码动态规划解法DP 表的构建与意义动