SpringBoot整合Netty整合WebSocket-带参认证

文章目录

    • 一. VectorNettyApplication启动类配置
    • 二.WebSocketServerBoot初始化服务端Netty
    • 三. WebsocketServerChannelInitializer初始化服务端Netty读写处理器
    • 四.initParamHandler处理器-去参websocket识别
    • 五.MessageHandler核心业务处理类-采用工厂策略模式
      • 5.1 策略上下文
    • 六.统一响应
    • 七.统一输出处理器


一. VectorNettyApplication启动类配置

初始化SpringBoot线程同时初始化Netty线程

/*** @description: 通知启动类* @Title: VectorNotification* @Package com.vector.notification* @Author YuanJie* @Date 2023/3/2 12:57*/
@EnableDiscoveryClient // 开启服务注册与发现
@SpringBootApplication(scanBasePackages = {"com.vector"},exclude = {DataSourceAutoConfiguration.class}) // 开启组件扫描和自动配置
public class VectorNettyApplication implements CommandLineRunner {@Value("${netty.host}")private String host;@Value("${netty.port}")private Integer port;@Resourceprivate WebSocketServerBoot webSocketServerBoot;public static void main(String[] args) {SpringApplication.run(VectorNettyApplication.class, args);}// springboot启动后执行netty服务端启动@Overridepublic void run(String... args) throws Exception {ChannelFuture channelFuture = webSocketServerBoot.bind(host, port);// 优雅关闭, jvm关闭时将netty服务端关闭Runtime.getRuntime().addShutdownHook(new Thread(() -> webSocketServerBoot.destroy()));// 阻塞 直到channel关闭channelFuture.channel().closeFuture().syncUninterruptibly();}
}

二.WebSocketServerBoot初始化服务端Netty

主要进行netty的基本配置

/*** @author YuanJie* @projectName vector-server* @package com.vector.netty.accept* @className com.vector.netty.accept.ServerBootstrap* @copyright Copyright 2020 vector, Inc All rights reserved.* @date 2023/6/9 18:34*/
@Component
@Slf4j
public class WebSocketServerBoot {private final EventLoopGroup parentGroup = new NioEventLoopGroup();private final EventLoopGroup childGroup = new NioEventLoopGroup(2);private Channel channel;@Resourceprivate WebsocketServerChannelInitializer websocketServerChannelInitializer;/*** 初始化服务端* sync():等待Future直到其完成,如果这个Future失败,则抛出失败原因;* syncUninterruptibly():不会被中断的sync();*/public ChannelFuture bind(String host, Integer port) {ChannelFuture channelFuture = null;try {channelFuture = new ServerBootstrap().group(parentGroup, childGroup) // 指定线程模型 一个用于接收客户端连接,一个用于处理客户端读写操作.channel(NioServerSocketChannel.class) // 指定服务端的IO模型.option(ChannelOption.SO_BACKLOG, 1024) // 设置TCP缓冲区.childOption(ChannelOption.SO_KEEPALIVE, true) // 保持连接 tcp底层心跳机制.childHandler(websocketServerChannelInitializer) // 指定处理新连接数据的读写处理逻辑.bind(host, port).addListener(new GenericFutureListener<Future<? super Void>>() {@Overridepublic void operationComplete(Future<? super Void> future) throws Exception {if (future.isSuccess()) {log.info("服务端启动成功,监听端口:{}", port);} else {log.error("服务端启动失败,监听端口:{}", port);bind(host, port + 1);}}}).syncUninterruptibly();// 绑定端口channel = channelFuture.channel(); // 获取channel} finally {if (null == channelFuture) {channel.close();parentGroup.shutdownGracefully();childGroup.shutdownGracefully();}}return channelFuture;}/*** 销毁*/public void destroy() {if (null == channel) return;channel.close();parentGroup.shutdownGracefully();childGroup.shutdownGracefully();}/*** 获取通道** @return*/public Channel getChannel() {return channel;}
}

三. WebsocketServerChannelInitializer初始化服务端Netty读写处理器

主要规划netty的读写处理器

/*** @author YuanJie* @projectName vector-server* @package com.vector.netty.config* @className com.vector.netty.server.ServerChannelInitializer* @copyright Copyright 2020 vector, Inc All rights reserved.* @date 2023/6/9 19:13*/
@Component
public class WebsocketServerChannelInitializer extends ChannelInitializer<SocketChannel> {// @Sharableprivate final LoggingHandler loggingHandler = new LoggingHandler(LogLevel.INFO);public final static String WEBSOCKET_PATH = "/ws";@Resourceprivate InitParamHandler initParamHandler;@Resourceprivate MessageHandler messageHandler;@Resourceprivate OutBoundHandler outBoundHandler;@Overrideprotected void initChannel(SocketChannel socketChannel) throws Exception {ChannelPipeline pipeline = socketChannel.pipeline();// 日志打印pipeline.addLast(loggingHandler);// http报文解析器 线程不安全不能被共享pipeline.addLast(new HttpServerCodec());
//        // 添加对大数据流的支持pipeline.addLast(new ChunkedWriteHandler());
//        // 消息聚合器 8192 8Mpipeline.addLast(new HttpObjectAggregator(1 << 13));// 进行设置心跳检测pipeline.addLast(new IdleStateHandler(60, 30, 60 * 30, TimeUnit.SECONDS));// ================= 上述是用于支持http协议的 ==============//websocket 服务器处理的协议,用于给指定的客户端进行连接访问的路由地址// 处理uri参数 WebSocketServerProtocolHandler不允许带参数 顺序不可调换pipeline.addLast(initParamHandler);pipeline.addLast(new WebSocketServerProtocolHandler(WEBSOCKET_PATH,null, true, 1<<16,true,true,5000));pipeline.addLast(messageHandler);// 自定义出栈处理器pipeline.addLast(outBoundHandler);}
}

四.initParamHandler处理器-去参websocket识别

主要为了去参,WebSocketServerProtocolHandler不允许带参数,同时初始化一些信道用户数据

/*** URL参数处理程序,这时候连接还是个http请求,没有升级成webSocket协议,此处SimpleChannelInboundHandler泛型使用FullHttpRequest** @author YuanJie* @date 2023/5/7 15:07*/
@Slf4j
@ChannelHandler.Sharable
@Component
public class InitParamHandler extends SimpleChannelInboundHandler<FullHttpRequest> {/*** 存储已经登录用户的channel对象*/public static ChannelGroup channelGroup = new DefaultChannelGroup(GlobalEventExecutor.INSTANCE);/*** 存储用户id和用户的channelId绑定*/public static final Map<Long, ChannelId> userMap = new ConcurrentHashMap<>();/*** 用于存储群聊房间号和群聊成员的channel信息*/public static final Map<Long, ChannelGroup> groupMap = new ConcurrentHashMap<>();@DubboReferenceprivate MemberRemote memberRemote;/*** 此处进行url参数提取,重定向URL,访问webSocket的url不支持带参数的,带参数会抛异常,这里先提取参数,将参数放入通道中传递下去,重新设置一个不带参数的url** @param ctx     the {@link ChannelHandlerContext} which this {@link SimpleChannelInboundHandler}*                belongs to* @param request the message to handle* @throws Exception*/@Overrideprotected void channelRead0(ChannelHandlerContext ctx, FullHttpRequest request) throws Exception {if (!this.acceptInboundMessage(request)) {ctx.fireChannelRead(request.retain());}String uri = request.uri();log.info("NettyWebSocketParamHandler.channelRead0 --> : 格式化URL... {}", uri);Map<CharSequence, CharSequence> queryMap = UrlBuilder.ofHttp(uri).getQuery().getQueryMap();//将参数放入通道中传递下去String senderId = "senderId";if (StringUtils.isBlank(queryMap.get(senderId))) {log.info("NettyWebSocketParamHandler.channelRead0 --> : 参数缺失 senderId");ctx.close();}// 验证token
//        verifyToken(ctx,senderId);// 初始化数据
//        initData(ctx, Long.valueOf(queryMap.get(senderId).toString()));// 获取?之前的路径request.setUri(WebsocketServerChannelInitializer.WEBSOCKET_PATH);ctx.fireChannelRead(request.retain());}@Overridepublic void channelActive(ChannelHandlerContext ctx) {//添加到channelGroup通道组channelGroup.add(ctx.channel());ctx.channel().id();}@Overridepublic void channelInactive(ChannelHandlerContext ctx) {// 移除channelGroup 通道组channelGroup.remove(ctx.channel());}@Overridepublic void exceptionCaught(ChannelHandlerContext ctx, Throwable cause) throws Exception {cause.printStackTrace();log.error("NettyWebSocketParamHandler.exceptionCaught --> cause: ", cause);ctx.close();}private void verifyToken(ChannelHandlerContext ctx, Long senderId) {String userKey = CacheConstants.LOGIN_TOKEN_KEY + senderId;RedissonCache redissonCache = SpringContextUtil.getBean(RedissonCache.class);Boolean hasKey = redissonCache.hasKey(userKey);if (!hasKey) {log.info("NettyWebSocketParamHandler.channelRead0 --> : 用户未登录... {}", senderId);ctx.close();}// token续期redissonCache.expire(userKey, SystemConstants.TOKEN_EXPIRE_TIME, TimeUnit.MILLISECONDS);}/*** 加入聊天室** @param ctx* @param senderId* @throws ExecutionException* @throws InterruptedException*/private void joinGroup(ChannelHandlerContext ctx, Long senderId) {R r = null;try {CompletableFuture<R> result = memberRemote.getGroupListById(senderId);r = result.get(3, TimeUnit.SECONDS);} catch (Exception e) {log.error("messageHandler.joinGroup查询群聊列表失败 ===> {}", e.getMessage());ctx.channel().write(WSMessageDTO.error("查询群聊列表失败"));return;}if (r == null || r.getCode() != 200) {log.error("查询群聊列表失败 ====> {}", r.getMsg());ctx.channel().write(WSMessageDTO.error("查询群聊列表失败"));return;}//查询成功//获取群聊列表String json = JacksonInstance.toJson(r.getData());List<Long> groupIds = JacksonInstance.toObjectList(json, new TypeReference<List<Long>>() {});ChannelGroup group;for (Long groupId : groupIds) {group = groupMap.get(groupId);if (group == null) {//如果群聊信道不存在,则创建一个群聊信道group = new DefaultChannelGroup(GlobalEventExecutor.INSTANCE);groupMap.put(groupId, group);}//将当前用户加入到群聊信道中group.add(ctx.channel());}}/*** 加入聊天信道*/private void joinChat(ChannelHandlerContext ctx, Long senderId) {//将当前用户的channelId放入map中userMap.put(senderId, ctx.channel().id());}private void initData(ChannelHandlerContext ctx, Long senderId) {joinChat(ctx, senderId);joinGroup(ctx, senderId);}
}

五.MessageHandler核心业务处理类-采用工厂策略模式

使得业务和通信协议无关,无感知。具体业务可以增加策略

/*** @author YuanJie* @projectName vector-server* @package com.vector.netty.handler* @className com.vector.netty.handler.MessageTypeHandler* @copyright Copyright 2020 vector, Inc All rights reserved.* @date 2023/6/15 16:23*/
@Slf4j
@Component
@ChannelHandler.Sharable
public class MessageHandler extends SimpleChannelInboundHandler<TextWebSocketFrame> {@Resourceprivate MessageStrategyContext messageStrategyContext;@Overridepublic void channelRead0(ChannelHandlerContext ctx, TextWebSocketFrame msg) {// 获取客户端发送的数据WSMessageDTO wsMessageDTO = JacksonInstance.toObject(msg.text(), new TypeReference<WSMessageDTO>() {});wsMessageDTO.setMessageId(SnowFlakeUtil.getNextId());log.info("客户端收到服务器数据:{}", wsMessageDTO.getMessage());verifyParams(ctx, wsMessageDTO);// 根据消息类型获取对应的处理器 核心处理方法messageStrategyContext.messageType(ctx, wsMessageDTO);}private void verifyParams(ChannelHandlerContext ctx, WSMessageDTO wsMessageDTO) {StringBuilder sb = new StringBuilder();if (wsMessageDTO.getSenderId() == null) {sb.append("senderId不能为空");}if (!EnumBusiness.containsBusiness(wsMessageDTO.getBusinessType())) {sb.append("businessType不能为空");}if (!EnumMessage.containsMessage(wsMessageDTO.getMessageType())) {sb.append("messageType不能为空");}if (wsMessageDTO.getMessage() == null) {sb.append("message不能为空");}if (sb.length() > 0) {log.error("参数校验失败:{}", sb.toString());ctx.channel().write(WSMessageDTO.error("参数校验失败:" + sb.toString()));ctx.close();}}@Overridepublic void channelReadComplete(ChannelHandlerContext ctx) throws Exception {ctx.flush();}
}

5.1 策略上下文

具体工厂策略可以详看我的策略模式文章
枚举维护前端参数和bean对象名

/*** @author YuanJie* @projectName vector-server* @package com.vector.netty.enums* @className com.vector.netty.enums.BusinessEnums* @copyright Copyright 2020 vector, Inc All rights reserved.* @date 2023/6/14 16:13*/
public enum EnumBusiness {/*** 单聊*/chatMessage("chat", ChatMessageStrategy.class.getSimpleName()),/*** 群聊*/groupMessage("group", GroupMessageStrategy.class.getSimpleName()),/*** 在线人数*/onlineCount("onlineCount", OnlineCountStrategy.class.getSimpleName()),TEST("test",TestStrategy.class.getSimpleName());private final String businessType;private final String beanName;EnumBusiness(String businessType, String beanName) {this.businessType = businessType;this.beanName = StringUtils.isNotEmpty(beanName)?beanName.toLowerCase():null;}/*** 根据code获取对应的枚举对象*/public static EnumBusiness getEnum(String businessType) {EnumBusiness[] values = EnumBusiness.values(); // 获取枚举列表if (null != businessType && values.length > 0) {for (EnumBusiness value : values) {if (value.businessType.equals(businessType)) {return value;  // 返回枚举对象}}}return null;}/*** 该code在枚举列表code属性是否存在*/public static boolean containsBusiness(String businessType) {EnumBusiness anEnum = getEnum(businessType); // 获取枚举对象return anEnum != null;}/*** 判断code与枚举中的code是否相同*/public static boolean equals(String businessType, EnumBusiness calendarSourceEnum) {return calendarSourceEnum.businessType.equals(businessType);}public String getBusinessType() {return businessType;}public String getBeanName() {return beanName;}
}

策略根据bean名获取实例对象

/*** @author YuanJie* @projectName vector-server* @package com.vector.netty.service* @className com.vector.netty.service.MessageContext* @copyright Copyright 2020 vector, Inc All rights reserved.* @date 2023/6/14 17:02*/
@Component
@Slf4j
public class MessageStrategyContext {/** 策略实例集合 */private final ConcurrentHashMap<String, MessageStrategy> strategyConcurrentHashMap =new ConcurrentHashMap<>(20);/*** 注入策略实例* 如果使用的是构造器注入,可能会有多个参数注入进来。** 如果使用的是field反射注入** 如果使用的是setter方法注入,那么你将不能将属性设置为final。** @param strategyMap*         注意注入类型要是Map基础类型*         注入接口,spring会自动注入他的所有被spring托管的实现类*/@Autowiredpublic MessageStrategyContext(Map<String, MessageStrategy> strategyMap) {//清空集合数据this.strategyConcurrentHashMap.clear();if (!CollectionUtils.isEmpty(strategyMap)) {strategyMap.forEach((beanName, messageStrategy) -> {if (StringUtils.isEmpty(beanName) || messageStrategy == null) {return;}this.strategyConcurrentHashMap.put(beanName.toLowerCase(), messageStrategy);});}}/*** 选择业务方式* 单聊,群聊,统计在线人数...** @param msg 信息*/public void messageType(ChannelHandlerContext ctx, WSMessageDTO msg){EnumBusiness enumerateInstances = EnumBusiness.getEnum(msg.getBusinessType());if (CollectionUtils.isEmpty(strategyConcurrentHashMap)) {log.info("策略实例集合初始化失败,请检查是否正确注入!");}MessageStrategy messageStrategy = strategyConcurrentHashMap.get(enumerateInstances.getBeanName());messageStrategy.messageType(ctx, msg);}}

六.统一响应

注意使用该统一响应对象,所有入栈处理器必须使用即调用方必须是SimpleChannelInboundHandler,详细原因在下文 七.统一输出处理器中

/*** @author YuanJie* @projectName vector-server* @package com.vector.netty.entity* @className com.vector.netty.entity.SocketMessage* @copyright Copyright 2020 vector, Inc All rights reserved.* @date 2023/6/14 19:35*/
@Data
public class WSMessageDTO {/*** 消息发送者*/private Long senderId;/*** 消息接收者/群聊id*/private Long chatId;/*** 消息类型 0文本 1图片 2文件 3视频 4语音 5位置 6名片 7链接 8系统消息* @see com.vector.netty.enums.EnumMessage*/private byte messageType;/*** 业务类型 chat单聊 group群聊 onlineCount在线人数* @see com.vector.netty.enums.EnumBusiness*/private String businessType;/*** 记录每条消息的id*/private Long messageId;/*** 消息内容*/private String message;/*** 消息发送时间*/private LocalDateTime sendTime;/*** 消息接收时间*/private LocalDateTime receiveTime;/*** 最后一条消息内容*/private String lastMessage;/*** 消息状态 0失败 1成功*/private byte code;/*** 封装统一返回格式* @return*/public static TextWebSocketFrame ok(){WSMessageDTO data = new WSMessageDTO();data.setCode((byte) 1);return new TextWebSocketFrame(JacksonInstance.toJson(data)).retain();}public static TextWebSocketFrame ok(WSMessageDTO data){data.setCode((byte) 1);return new TextWebSocketFrame(JacksonInstance.toJson(data)).retain();}public static TextWebSocketFrame error(String message){WSMessageDTO data = new WSMessageDTO();data.setCode((byte) 0);data.setMessage(message);return new TextWebSocketFrame(JacksonInstance.toJson(data)).retain();}
}

七.统一输出处理器

  • 若调用WSMessageDTO方法,必须注意内存泄露
  • 即调用方必须是SimpleChannelInboundHandler<>
  • 严禁使用ChannelInboundHandlerAdapter, 否则将造成严重内存泄露
  • 相应地,必须使用此处的写出@param msg ,释放@param msg 引用
/*** @author YuanJie* @projectName vector-server* @package com.vector.netty.handler* @className com.vector.netty.handler.OutBoundHandler* @copyright Copyright 2020 vector, Inc All rights reserved.* @date 2023/7/24 22:38*/
@Slf4j
@Component
@ChannelHandler.Sharable
public class OutBoundHandler extends ChannelOutboundHandlerAdapter {/*** 若调用WSMessageDTO方法,必须注意内存泄露* 即调用方必须是SimpleChannelInboundHandler<>* 严禁使用ChannelInboundHandlerAdapter, 否则将造成严重内存泄露* 相应地,必须使用此处的写出@param msg ,释放@param msg 引用* @param ctx               the {@link ChannelHandlerContext} for which the write operation is made* @param msg               the message to write* @param promise           the {@link ChannelPromise} to notify once the operation completes* @throws Exception*/@Overridepublic void write(ChannelHandlerContext ctx, Object msg, ChannelPromise promise) throws Exception {if (msg instanceof FullHttpMessage){log.info("webSocket协议升级成功");// 出栈必须得这样写,不能自定义通信消息,可能把websocket反馈的消息覆盖了。  也不能在最后处理器调ctx.fireChannelRead()ctx.writeAndFlush(msg,promise);return;} else if (msg instanceof TextWebSocketFrame) {log.info("我要给客户端发送消息了。。。。");ctx.writeAndFlush(msg, promise);return;}log.error("OutBoundHandler.write: 消息类型错误");ctx.writeAndFlush(WSMessageDTO.error("服务器内部错误: OutBoundHandler.write()"),promise);ctx.close();}
}

在这里插入图片描述
在这里插入图片描述

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/586306.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

服务器设置了端口映射之后外网还是访问不了服务器

目录 排查思路参考&#xff1a; 1、确认服务是否在运行 2、确认端口映射设置是否正确 3、使用防火墙测试到服务器的连通性 4、检查服务内部的配置 5、解决办法 6、学习小分享 我们在一个完整的网络数据存储服务系统设备中都会存有业务服务器、防火墙、交换机、路由器&a…

Allavsoft for Mac v3.27.0.8852注册激活版 优秀的视频下载工具

Allavsoft for Mac是一款功能强大的多媒体下载和转换工具&#xff0c;支持从各种在线视频网站和流媒体服务下载视频、音频和图片。它具备批量下载和转换功能&#xff0c;可将文件转换为多种格式&#xff0c;以适应不同设备的播放需求。此外&#xff0c;Allavsoft还提供视频编辑…

Java | Leetcode Java题解之第3题无重复字符的最长子串

题目&#xff1a; 题解&#xff1a; class Solution {public int lengthOfLongestSubstring(String s) {// 哈希集合&#xff0c;记录每个字符是否出现过Set<Character> occ new HashSet<Character>();int n s.length();// 右指针&#xff0c;初始值为 -1&#…

Linux环境基础和工具的使用

目录 1、Linux软件包管理器---yum 2、Linux开发工具 2.1、vim基本概念 2.2 vim基本操作 2.3 vim正常模式命令集 2.4 vim末行模式命令集 2.5 简单vim配置 2.5.1 配置文件的位置 3 Linux编译器--gcc/g的使用 3.1 背景知识 3.2 gcc完成 4 Linux调试器--gdb使用 4.1 背…

二叉树的深度和高度问题-算法通关村

二叉树的深度和高度问题-算法通关村 1 最大深度问题 LeetCode104: 给定一个二叉树&#xff0c;找出其最大深度。 二叉树的深度为根节点到最远叶子节点的最长路径上的节点数。 说明&#xff1a;叶子节点是指没有子节点的节点。 对于node&#xff08;3&#xff09;&#xff0…

《吴恩达:AI 智能体工作流引领人工智能新趋势》

近期值得看的 AI 视频之一&#xff1a;《吴恩达&#xff1a;AI 智能体工作流引领人工智能新趋势》这是吴恩达老师分享的他在 AI 智能体方面的发现。如果说智人区分于其他物种的能力是我们善用工具&#xff0c;那么对于 AI 来说&#xff0c;智能体就是它的工具。根据吴老师分享的…

服务器远程桌面连接不上怎么办?

随着互联网的发展和远程办公的兴起&#xff0c;服务器远程桌面连接成为了许多企业和个人不可或缺的工具。偶尔我们可能会碰到服务器远程桌面连接不上的情况&#xff0c;这时候我们需要找到解决办法&#xff0c;确保高效地远程访问服务器。 天联组网——突破远程连接障碍 在我们…

【机器学习】机器学习创建算法第3篇:K-近邻算法,学习目标【附代码文档】

机器学习&#xff08;算法篇&#xff09;完整教程&#xff08;附代码资料&#xff09;主要内容讲述&#xff1a;机器学习算法课程定位、目标&#xff0c;K-近邻算法定位,目标,学习目标,1 什么是K-近邻算法,1 Scikit-learn工具介绍,2 K-近邻算法API。K-近邻算法&#xff0c;1.4 …

pygwalker+streamlit python看板库使用体验

算作前言 在 B 站看到 pygwalker 的介绍&#xff0c;很感兴趣。 是一个类似于简化版的 tableau 工具。 原版 docs PyGWalker 文档 – Kanaries 搭建看板 直接结合 streamlit 使用&#xff0c;streamlit 真的神器。 import pygwalker as pyg import pandas as pd import str…

2024年网络安全运营体系建设方案

以下是部分WORD内容&#xff0c;请您参阅。如需下载完整WORD文件&#xff0c;请前往星球获取&#xff1a; 网络安全运营监控工作整体构想 工作目标及原则 工作目标 为进一步落实强化公司网络安全保障&#xff0c;有效支撑公司数字化转型战略&#xff0c;建立健全公司网省两级协…

Echarts实现高亮某一个点

背景 接口会返回所有点的数据&#xff0c;以及最优点的数据。产品要求在绘制图形后&#xff0c;高亮最优点&#xff0c;添加一个红色的样式&#xff0c;如图。点击select选择器时&#xff0c;可选择不同指标和花费对应的关系。 以下介绍实现思路 1、自定义配置选择器的数据源…

【Web】记录Polar靶场<中等>难度题一遍过(全)

目录 到底给不给flag呢 写shell 注入 某函数的复仇 xxe SSTI unpickle BlackMagic 反序列化 找找shell 再来ping一波啊 wu 代码审计1 你的马呢&#xff1f; ezphp 随机值 phpurl search file PlayGame csdn 反正持续一个月&#xff0c;感觉XYCTF…