[Spring Cloud] (9)XSS拦截器

文章目录

  • 简述
    • 本文涉及代码已开源
    • Fir Cloud 完整项目
    • 防XSS攻击
      • 必要性:
      • 作用:
  • 整体效果
  • 后端
    • 增加拦截器开关配置
    • pom中增加jsoup依赖
    • 添加JSON处理工具类
    • 添加xss拦截工具类
    • 防XSS-请求拦截器
  • 前端

简述

本文涉及代码已开源

本文网关gateway,微服务,vue已开源到gitee
杉极简 / gateway网关阶段学习
https://gitee.com/dong-puen/gateway-stages

Fir Cloud 完整项目

该内容完整项目如下
Fir Cloud v1.0.0
https://gitee.com/dong-puen/fir-cloud
https://github.com/firLucky/fir-cloud

防XSS攻击

XSS攻击是一种常见的网络攻击手段,它允许攻击者将恶意脚本注入到其他用户会浏览的页面中。谁也不想,被注入一段代码,然后在客户的浏览器上被执行,然后造成重大损失,然后被领导叫去喝茶吧。。。

必要性:

  1. 保护用户隐私:XSS攻击可以盗取用户的敏感信息,如登录凭据、个人数据和Cookies。
  2. 维护数据完整性:攻击者可能会利用XSS攻击来修改网页内容,破坏数据的准确性和完整性。
  3. 防止会话劫持:XSS攻击常常被用来窃取用户的会话Cookies,从而允许攻击者冒充用户进行操作。

作用:

  1. 防止恶意脚本执行:通过输入清理、输出编码等措施,防止恶意脚本在用户浏览器中执行。
  2. 防止敏感信息泄露:减少攻击者盗取用户敏感信息的机会,保护用户隐私。
  3. 保护Web应用安全:加强Web应用的整体安全性,使其不易受到XSS攻击。
  4. 防止网页篡改:防止攻击者通过XSS攻击修改网页内容。
  5. 减少安全漏洞:通过实施XSS防御措施,减少Web应用的安全漏洞。

整体效果

前端传递一个攻击代码字符串给后端时,后端将其过滤掉,效果如下:
image.png
image.png

后端

增加拦截器开关配置

  # xss拦截xss: true

image.png
GlobalConfig增加

    /*** 防xss*/private boolean xss;

image.png

pom中增加jsoup依赖

jsoup 是一款基于 Java 的HTML解析器

        <!-- https://mvnrepository.com/artifact/org.jsoup/jsoup --><dependency><groupId>org.jsoup</groupId><artifactId>jsoup</artifactId><version>1.17.2</version></dependency>

添加JSON处理工具类

package com.fir.gateway.utils;import com.alibaba.fastjson.JSONObject;
import org.springframework.util.StringUtils;/*** JSON处理工具类** @author fir*/
public enum JsonUtils {/*** 实例*/INSTANCE;/*** json对象字符串开始标记*/private final static String JSON_OBJECT_START = "{";/*** json对象字符串结束标记*/private final static String JSON_OBJECT_END = "}";/*** json数组字符串开始标记*/private final static String JSON_ARRAY_START = "[";/*** json数组字符串结束标记*/private final static String JSON_ARRAY_END = "]";/*** 判断字符串是否json对象字符串** @param val 字符串* @return true/false*/public boolean isJsonObj(String val) {if (StringUtils.hasLength(val)) {return false;}val = val.trim();if (val.startsWith(JSON_OBJECT_START) && val.endsWith(JSON_OBJECT_END)) {try {JSONObject.parseObject(val);return true;} catch (Exception e) {return false;}}return false;}/*** 判断字符串是否json数组字符串** @param val 字符串* @return true/false*/public boolean isJsonArr(String val) {if (StringUtils.hasLength(val)) {return false;}val = val.trim();if (StringUtils.hasLength(val)) {return false;}val = val.trim();if (val.startsWith(JSON_ARRAY_START) && val.endsWith(JSON_ARRAY_END)) {try {JSONObject.parseArray(val);return true;} catch (Exception e) {return false;}}return false;}/*** 判断对象是否是json对象** @param obj 待判断对象* @return true/false*/public boolean isJsonObj(Object obj) {String str = JSONObject.toJSONString(obj);return this.isJsonObj(str);}/*** 判断字符串是否json字符串** @param str 字符串* @return true/false*/public boolean isJson(String str) {if (StringUtils.hasLength(str)) {return false;}return this.isJsonObj(str) || this.isJsonArr(str);}
}

添加xss拦截工具类

package com.fir.gateway.utils;import com.alibaba.fastjson.JSONObject;
import lombok.SneakyThrows;
import org.jsoup.Jsoup;
import org.jsoup.nodes.Document;
import org.jsoup.safety.Safelist;import java.util.ArrayList;
import java.util.List;
import java.util.Map;
import java.util.Objects;/*** xss拦截工具类** @author lieber*/
public enum XssUtils {/*** 实例*/INSTANCE;private final static String RICH_TEXT = "</";/*** 自定义白名单*/private final static Safelist CUSTOM_WHITELIST = Safelist.relaxed()
//            .addAttributes("video", "width", "height", "controls", "alt", "src")
//            .addAttributes(":all", "style", "class");/*** jsoup不格式化代码*/private final static Document.OutputSettings OUTPUT_SETTINGS = new Document.OutputSettings().prettyPrint(false);/*** 清除json对象中的xss攻击字符** @param val json对象字符串* @return 清除后的json对象字符串*/private String cleanObj(String val) {JSONObject jsonObject = JSONObject.parseObject(val);for (Map.Entry<String, Object> entry : jsonObject.entrySet()) {if (entry.getValue() != null && entry.getValue() instanceof String) {String str = (String) entry.getValue();str = this.cleanXss(str);entry.setValue(str);}}return jsonObject.toJSONString();}/*** 清除json数组中的xss攻击字符** @param val json数组字符串* @return 清除后的json数组字符串*/private String cleanArr(String val) {List<String> list = JSONObject.parseArray(val, String.class);List<String> result = new ArrayList<>(list.size());for (String str : list) {str = this.cleanXss(str);result.add(str);}return JSONObject.toJSONString(result);}/*** 清除xss攻击字符串,此处优化空间较大** @param str 字符串* @return 清除后无害的字符串*/@SneakyThrowspublic String cleanXss(String str) {if (JsonUtils.INSTANCE.isJsonObj(str)) {str = this.cleanObj(str);} else if (JsonUtils.INSTANCE.isJsonArr(str)) {str = this.cleanArr(str);} else {boolean richText = this.richText(str);if (!richText) {str = str.trim();str = str.replaceAll(" +", " ");}String afterClean = Jsoup.clean(str, "", CUSTOM_WHITELIST, OUTPUT_SETTINGS);if (paramError(richText, afterClean, str)) {
//                throw new BizRunTimeException(ApiCode.PARAM_ERROR, "参数包含特殊字符");
//                throw new Exception("参数包含特殊字符");System.out.println("参数包含特殊字符");}str = richText ? afterClean : this.backSpecialStr(afterClean);}return str;}/*** 判断是否是富文本** @param str 待判断字符串* @return true/false*/private boolean richText(String str) {return str.contains(RICH_TEXT);}/*** 判断是否参数错误** @param richText   是否富文本* @param afterClean 清理后字符* @param str        原字符串* @return true/false*/private boolean paramError(boolean richText, String afterClean, String str) {// 如果包含富文本字符,那么不是参数错误if (richText) {return false;}// 如果清理后的字符和清理前的字符匹配,那么不是参数错误if (Objects.equals(str, afterClean)) {return false;}// 如果仅仅包含可以通过的特殊字符,那么不是参数错误if (Objects.equals(str, this.backSpecialStr(afterClean))) {return false;}// 如果还有......return true;}/*** 转义回特殊字符** @param str 已经通过转义字符* @return 转义后特殊字符*/private String backSpecialStr(String str) {return str.replaceAll("&amp;", "&");}
}

防XSS-请求拦截器

package com.fir.gateway.filter.request;import com.alibaba.fastjson.JSONArray;
import com.alibaba.fastjson.JSONObject;
import com.fir.gateway.config.GlobalConfig;
import com.fir.gateway.config.exception.CustomException;
import com.fir.gateway.config.result.AjaxStatus;
import com.fir.gateway.utils.XssUtils;
import lombok.Data;
import lombok.SneakyThrows;
import lombok.extern.slf4j.Slf4j;
import org.apache.commons.lang3.StringUtils;
import org.springframework.boot.context.properties.ConfigurationProperties;
import org.springframework.cloud.gateway.filter.GatewayFilterChain;
import org.springframework.cloud.gateway.filter.GlobalFilter;
import org.springframework.core.Ordered;
import org.springframework.http.HttpMethod;
import org.springframework.http.server.reactive.ServerHttpRequest;
import org.springframework.stereotype.Component;
import org.springframework.util.LinkedMultiValueMap;
import org.springframework.util.MultiValueMap;
import org.springframework.web.server.ServerWebExchange;
import org.springframework.web.util.UriComponentsBuilder;
import reactor.core.publisher.Mono;import javax.annotation.Resource;
import java.io.UnsupportedEncodingException;
import java.net.URI;
import java.net.URLEncoder;
import java.nio.charset.StandardCharsets;
import java.util.Collections;
import java.util.List;
import java.util.Map;
import java.util.Set;/*** XSS过滤** @author lieber*/
@Data
@Slf4j
@ConfigurationProperties("config.form.xss")
@Component
public class XssFormFilter implements GlobalFilter, Ordered {/*** 网关参数配置*/@Resourceprivate GlobalConfig globalConfig;@Overridepublic Mono<Void> filter(ServerWebExchange exchange, GatewayFilterChain chain) {log.info("xss攻击验证:start");boolean xss = globalConfig.isXss();if (xss) {// 白名单路由判断ServerHttpRequest request = exchange.getRequest();String path = request.getPath().toString();List<String> whiteUrls = globalConfig.getWhiteUrls();if (whiteUrls.contains(path)) {log.info("xss攻击验证:true,白名单");return chain.filter(exchange);}ServerHttpRequest req = exchange.getRequest();String method = req.getMethodValue();ServerHttpRequest builder = req.mutate().build();if (HttpMethod.GET.matches(method)) {builder = change(exchange, builder);} else if (HttpMethod.POST.matches(method)) {builder = change(exchange, builder);}exchange = exchange.mutate().request(builder).build();log.info("xss攻击验证:true");} else {log.info("xss攻击验证:true,验证已关闭");}return chain.filter(exchange);}/*** 获取请求参数等信息进行过滤处理** @param exchange          请求* @param serverHttpRequest 请求* @return 处理结束的参数*/@SneakyThrowsprivate ServerHttpRequest change(ServerWebExchange exchange, ServerHttpRequest serverHttpRequest) {// 获取原参数URI uri = serverHttpRequest.getURI();// 更改参数ServerHttpRequest request = exchange.getRequest();MultiValueMap<String, String> query = request.getQueryParams();String originalQuery = JSONObject.toJSONString(query);MultiValueMap<String, String> newQueryParams = new LinkedMultiValueMap<>();if (StringUtils.isNoneBlank(originalQuery)) {// 执行XSS清理log.info("{} - XSS清理,处理前参数:{}", uri.getPath(), originalQuery);Set<String> strings = query.keySet();for (String key : strings) {List<String> v = query.get(key);String newV = XssUtils.INSTANCE.cleanXss(JSONObject.toJSONString(v));List<String> newVList = JSONArray.parseArray(newV, String.class);for (String string:newVList) {String encodedString = URLEncoder.encode(string, StandardCharsets.UTF_8.toString());newQueryParams.add(key, encodedString);}}originalQuery = JSONObject.toJSONString(newQueryParams);log.info("{} - XSS清理,处理后参数:{}", uri.getPath(), originalQuery);}URI newUri = UriComponentsBuilder.fromUri(uri).query(null).queryParams(newQueryParams).build(true).toUri();return exchange.getRequest().mutate().uri(newUri).build();}@Overridepublic int getOrder() {return -180;}
}

前端

身为一个后端程序员,本次终于不用再写前端代码了,难得难得。。。
此次前端无需更改任何内容。

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

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

相关文章

Visual Studio Code插件

文章目录 工具类AIChinese (Simplified) (简体中文)cmake集Code RunnerGitLens — Git superchargedPath IntellisenseTodo TreeBookmarks &#xff08;书签&#xff09;markdownclangd 美化类Output Colorizer &#xff08;输出窗口彩色&#xff09;Doxygen Documentation Gen…

安装harbor出现问题: Running 1/1 ✘ Network harbor_harbor Error

安装harbor出现问题&#xff1a; [] Running 1/1 ✘ Network harbor_harbor Error 0.2s failed to create network harbor_harbor: Error response from daemon: Fa…

K8s 搭建 FileBeat+ELK 分布式日志收集系统 以及 KQL 语法介绍

一、K8s FileBeat ELK 介绍 ELK&#xff0c;即Elasticsearch、Logstash和Kibana三个开源软件的组合&#xff0c;是由Elastic公司提供的一套完整的日志管理解决方案。Elasticsearch是一个高度可扩展的开源全文搜索和分析引擎&#xff0c;它允许你快速地、近乎实时地存储、搜索…

力扣654. 最大二叉树

Problem: 654. 最大二叉树 文章目录 题目描述思路复杂度Code 题目描述 思路 对于构造二叉树这类问题一般都是利用先、中、后序遍历&#xff0c;再将原始问题分解得出结果 1.定义递归函数build&#xff0c;每次将一个数组中的最大值作为当前子树的根节点构造二叉树&#xff1b;…

牛客NC391 快乐数【simple 模拟法 Java/Go/PHP】

题目 题目链接&#xff1a; https://www.nowcoder.com/practice/293b9ddd48444fa493dd17da0feb192d 思路 直接模拟即可Java代码 import java.util.*;public class Solution {/*** 代码中的类名、方法名、参数名已经指定&#xff0c;请勿修改&#xff0c;直接返回方法规定的值…

Linux-应用编程学习笔记(二、文件I/O、标准I/O)

一、文件I/O基础 文件 I/O 指的是对文件的输入/输出操作&#xff0c;就是对文件的读写操作。Linux 下一切皆文件。 1.1 文件描述符 在 open函数执行成功的情况下&#xff0c; 会返回一个非负整数&#xff0c; 该返回值就是一个文件描述符&#xff08;file descriptor&#x…

解禁谷歌等浏览器禁止网站使用麦克等媒体设备

1、浏览器地址栏输入chrome://flags/ 微软的chromium内核的edge浏览器&#xff0c;既可以输入&#xff1a;chrome://flags/ &#xff0c;也可以输入edge://flags/ 2、打开后&#xff0c;界面如下 3、输入搜索&#xff0c;unsafe&#xff0c;并启用、输入需要启用的网址

对AI 感兴趣的小伙伴

如图&#xff0c;欢迎来玩儿&#xff01; 欢迎来玩儿

Python异常处理:打造你的代码防弹衣!

Hi&#xff0c;我是阿佑&#xff0c;上文咱们讲到——揭秘Python的魔法&#xff1a;装饰器的超能力大揭秘 ‍♂️✨&#xff0c;阿佑将带领大家通过精准捕获异常、使用with语句和上下文管理器、以及异常链等高级技巧来增强代码的健壮性。就像为代码穿上防弹衣&#xff0c;保护它…

生活小区火灾预警新篇章:泵吸式可燃气体报警器的检定与运用

在现代化的生活小区中&#xff0c;燃气设备广泛应用于居民的日常生活之中&#xff0c;但同时也带来了潜在的火灾风险。 可燃气体报警器作为一种安全监测设备&#xff0c;能够及时检测到燃气泄漏等安全隐患&#xff0c;并在达到预设的阈值时发出警报&#xff0c;提醒居民采取相…

SpringBoot Redis 扩展高级功能

环境&#xff1a;SpringBoot2.7.16 Redis6.2.1 1. Redis消息发布订阅 Spring Data 为 Redis 提供了专用的消息传递集成&#xff0c;其功能和命名与 Spring Framework 中的 JMS 集成类似。Redis 消息传递大致可分为两个功能区域&#xff1a; 信息发布 信息订阅 这是一个通常…

北斗短报文终端 | 什么是北斗短报文功能?如何实现北斗短报文通信?

北斗短报文功能是指通过北斗卫星进行短报文通信的功能。这种功能允许用户在没有移动通信信号覆盖的偏远山区、海洋、沙漠等地带&#xff0c;通过北斗短报文终端发送和接收文本信息&#xff0c;进行基本的数据通信。 北斗短报文功能是指北斗卫星导航系统特有的双向报文通信功能。…

urllib_post请求_百度翻译

打开百度翻译&#xff0c;并打开控制台&#xff0c;输入spider&#xff0c;然后在网络中找到对应的接口&#xff0c;可以看出&#xff0c;该url是post请求 在此案例中找到的接口为sug&#xff0c;依据为&#xff1a; 可以看到&#xff0c;传递的数据为kw : XXX&#xff0c; 所…

[Linux]服务管理

一.服务的概念&#xff0c;状态&#xff0c;查看系统服务 服务(service)本质就是进程 如(mysqld&#xff0c;sshd 防火墙等) 是运行在后台的&#xff0c;通常都会监听某个端口&#xff0c;等待其它程序的请求 -------比如mysqld&#xff0c;防火墙等&#xff0c;因此我们又称为…

3D瓦片地图组件上线|提供DEM数据接入,全方位呈现三维地图地形!

在用户调研中&#xff0c;我们了解到很多用户自身的可视化项目&#xff0c;需要在垂直空间上表现一些业务&#xff0c;例如&#xff1a;3D地形效果&#xff0c;数据底板建设等&#xff0c;而传统的地图效果不满足此用户需求。瓦片地图能够无限加载大地图&#xff0c;以更三维的…

【Linux】在Ubuntu 16.04上安装Gerrit + PostgreSQL + Apache服务

Gerrit是一个基于Git版本控制系统的运行于Web浏览器上的Code Review工具&#xff0c;本文叙述如何在Ubuntu 16.04上安装Gerrit服务。&#xff08;当然安装Gerrit的方法有很多&#xff0c;本文只是其中之一&#xff09; 文章目录 前提安装PostgreSQL数据库并创建用户下载、配置和…

【飞舞的花瓣】飞舞的花瓣代码||樱花代码||表白代码(完整代码)

关注微信公众号「ClassmateJie」有完整代码以及更多惊喜等待你的发现。 简介/效果展示 这段代码是一个HTML页面&#xff0c;其中包含一个canvas元素和相关的JavaScript代码。这个页面创建了一个飘落花瓣的动画效果。 代码【获取完整代码关注微信公众号「ClassmateJie」回复“…

一步将 CentOS 7.X 原地升级并迁移至 RHEL 7.9

《OpenShift / RHEL / DevSecOps 汇总目录》 在《在离线环境中将 CentOS 7.X 原地升级并迁移至 RHEL 7.9》一文中为了实现从 CentOS 7.X 原地升级并迁移至 RHEL 7.9&#xff0c;我们第一步先将一个测试环境 CentOS 7.5 升级到 CentOS 7.9&#xff0c;然后在第二步使用 convert2…

抖音运营_如何开抖店

截止20年8月&#xff0c;抖音的日活跃数高达6亿。 20年6月&#xff0c;上线抖店 &#xff08;抖音官方电商&#xff09; 一 抖店的定位和特色 1 一站式经营 帮助商家进行 商品交易、店铺管理、客户服务 等全链路的生意经营 2 多渠道拓展 抖音、今日头条、西瓜、抖音火山版…

【Python】用于发送电子邮件的标准库smtplib和构建邮件主体、添加附件、设置收件人的email

欢迎来到《小5讲堂》 这是《Python》系列文章&#xff0c;每篇文章将以博主理解的角度展开讲解。 温馨提示&#xff1a;博主能力有限&#xff0c;理解水平有限&#xff0c;若有不对之处望指正&#xff01; 目录 插件介绍邮件代码扩展知识点文章推荐 插件介绍 smtplib 是 Pytho…