網站首頁 編程語言 正文
前言
先給大家看下效果,原本我們的請求是這樣子的
?加密后的數據傳輸是這樣子的
如果這是你想要的效果,那么請繼續往下看
加解密步驟:
1.前端請求前進行加密,然后發送到后端
2.后端收到請求后解密
3.后端返回數據前進行加密
4.前端拿到加密串后,解密數據
加解密算法:
本文用的是國密算法作為參考,當然大家也可以用其它算法進行加解密
國密算法加解密可參照:java/vue使用國密sm2進行數據加密_vue sm2_qq243920161的博客-CSDN博客java/vue使用國密sm2https://blog.csdn.net/qq243920161/article/details/127865091
一、前端請求前進行加密,然后發送到后端
import axios from 'axios';
import { sm2 } from 'sm-crypto';
axios.interceptors.request.use(config => {
// form-data傳參方式不加密
if (config.headers['Content-Type'] === 'application/x-www-form-urlencoded') {
return;
}
// 非body方式傳參,不加密
if (config.data) {
return;
}
// 使用國密算法進行加密
let encryptData = sm2.doEncrypt(JSON.stringify(config.data), '加密公鑰,請提前生成好');
config.data = {
data: encryptData
}
});
以上代碼使用了axios攔截器,對所有請求進行攔截,攔截器里,使用config.data獲取到請求的body進行加密,加密后,把加密后的數據重新賦值到config.data,sm-crypto是國密算法的依賴,使用前npm install?sm-crypto即可
請確保config.data是一個對象或者數組,不要是一個字符串,否則后端獲取body時會失敗
加密成功后,從network就能看到加密的數據了
二、后端收到請求后解密
這里有兩個類直接復制粘貼即可,一個是RequestWrapper,這個類是用來讀取body的,一個是BodyRequestWrapper,這個類是用來解密后,將解密后的數據封裝到request,供Controller層使用,這里直接上代碼了
import javax.servlet.ReadListener;
import javax.servlet.ServletInputStream;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletRequestWrapper;
import java.io.*;
/**
* 用來讀取body
*/
public class RequestWrapper extends HttpServletRequestWrapper {
private final String body;
public RequestWrapper(HttpServletRequest request) {
super(request);
StringBuilder stringBuilder = new StringBuilder();
BufferedReader bufferedReader = null;
InputStream inputStream = null;
try {
inputStream = request.getInputStream();
if (inputStream != null) {
bufferedReader = new BufferedReader(new InputStreamReader(inputStream));
char[] charBuffer = new char[128];
int bytesRead = -1;
while ((bytesRead = bufferedReader.read(charBuffer)) > 0) {
stringBuilder.append(charBuffer, 0, bytesRead);
}
} else {
stringBuilder.append("");
}
} catch (IOException ex) {
} finally {
if (inputStream != null) {
try {
inputStream.close();
} catch (IOException e) {
e.printStackTrace();
}
}
if (bufferedReader != null) {
try {
bufferedReader.close();
} catch (IOException e) {
e.printStackTrace();
}
}
}
body = stringBuilder.toString();
}
@Override
public ServletInputStream getInputStream() throws IOException {
final ByteArrayInputStream byteArrayInputStream = new ByteArrayInputStream(body.getBytes());
ServletInputStream servletInputStream = new ServletInputStream() {
@Override
public boolean isFinished() {
return false;
}
@Override
public boolean isReady() {
return false;
}
@Override
public void setReadListener(ReadListener readListener) {
}
@Override
public int read() throws IOException {
return byteArrayInputStream.read();
}
};
return servletInputStream;
}
@Override
public BufferedReader getReader() throws IOException {
return new BufferedReader(new InputStreamReader(this.getInputStream()));
}
public String getBody() {
return this.body;
}
}
import javax.servlet.ReadListener;
import javax.servlet.ServletInputStream;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletRequestWrapper;
import java.io.BufferedReader;
import java.io.ByteArrayInputStream;
import java.io.IOException;
import java.io.InputStreamReader;
/**
* 用來重新封裝request
*/
public class BodyRequestWrapper extends HttpServletRequestWrapper {
/**
* 存放JSON數據主體
*/
private String body;
public BodyRequestWrapper(HttpServletRequest request, String context) {
super(request);
body = context;
}
@Override
public ServletInputStream getInputStream() throws IOException {
final ByteArrayInputStream byteArrayInputStream = new ByteArrayInputStream(body.getBytes("UTF-8"));
return new ServletInputStream() {
@Override
public int read() throws IOException {
return byteArrayInputStream.read();
}
@Override
public boolean isFinished() {
return false;
}
@Override
public boolean isReady() {
return false;
}
@Override
public void setReadListener(ReadListener listener) {
}
};
}
@Override
public BufferedReader getReader() throws IOException {
return new BufferedReader(new InputStreamReader(this.getInputStream()));
}
}
然后我們需要寫一個請求過濾器,繼承Filter,對所有請求接口進行過濾
import com.alibaba.fastjson2.JSON;
import org.springframework.stereotype.Component;
import org.springframework.util.StringUtils;
import javax.annotation.Resource;
import javax.servlet.*;
import javax.servlet.http.HttpServletRequest;
import java.io.IOException;
/**
* 請求加解密過濾器
*
* @author 猴哥
*/
@Component
public class RequestHandler implements Filter {
/**
* 進行請求加密
*/
@Override
public void doFilter(ServletRequest request, ServletResponse response, FilterChain chain) throws IOException, ServletException {
// form-data不校驗
if ("application/x-www-form-urlencoded".equals(request.getContentType())) {
chain.doFilter(request, response);
return;
}
// 拿到加密串
String data = new RequestWrapper((HttpServletRequest) request).getBody();
if (StringUtils.isEmpty(data)) {
chain.doFilter(request, response);
return;
}
// 解析
String body = Sm2Util.decrypt("解密私鑰", data);
request = new BodyRequestWrapper((HttpServletRequest) request, body);
chain.doFilter(request, response);
}
}
Sm2Util是國密的解密方式,工具類在之前分享的帖子里有,當然,大家可以用自己喜歡的方式進行加解密
java/vue使用國密sm2進行數據加密_vue sm2_qq243920161的博客-CSDN博客java/vue使用國密sm2https://blog.csdn.net/qq243920161/article/details/127865091
這樣就能拿到加密串了
但是有個問題就是,這樣寫的話,所有請求都會走Filter,但是我們只想讓部分請求走Filter怎么辦呢,寫一個配置類就可以了,這樣就可以將url進行過濾
import org.springframework.boot.web.servlet.FilterRegistrationBean;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
/**
* @author 猴哥
*/
@Configuration
public class EncryptionConfiguration {
/**
* 過濾器配置
*/
@Bean
public FilterRegistrationBean<RequestHandler> filterRegistration(RequestHandler requestHandler) {
FilterRegistrationBean<RequestHandler> registration = new FilterRegistrationBean<>();
registration.setFilter(requestHandler);
registration.addUrlPatterns("/plugin/*");
registration.setName("encryptionFilter");
//設置優先級別
registration.setOrder(1);
return registration;
}
}
以上代碼就是將/plugin開頭的url進行攔截,代碼不難,就不用過多解釋了吧
三、后端返回數據前進行加密
代碼如下
import com.alibaba.fastjson.JSON;
import org.springframework.core.MethodParameter;
import org.springframework.http.MediaType;
import org.springframework.http.converter.HttpMessageConverter;
import org.springframework.http.server.ServerHttpRequest;
import org.springframework.http.server.ServerHttpResponse;
import org.springframework.stereotype.Component;
import org.springframework.web.bind.annotation.ControllerAdvice;
import org.springframework.web.servlet.mvc.method.annotation.ResponseBodyAdvice;
import javax.annotation.Resource;
import java.util.ArrayList;
import java.util.List;
/**
* 響應加解密攔截器
*
* @author 猴哥
*/
@Component
@ControllerAdvice
public class ResponseHandler implements ResponseBodyAdvice<Object> {
/**
* 返回true,才會走beforeBodyWrite方法
*/
@Override
public boolean supports(MethodParameter methodParameter, Class<? extends HttpMessageConverter<?>> aClass) {
return true;
}
/**
* 響應加密
*/
@Override
public Object beforeBodyWrite(Object body, MethodParameter methodParameter, MediaType mediaType, Class<? extends HttpMessageConverter<?>> aClass, ServerHttpRequest request, ServerHttpResponse serverHttpResponse) {
// 拿到響應的數據
String json = JSON.toJSONString(body);
// 進行加密
return Sm2Util.encrypt("加密公鑰", json);
}
}
前端即可拿到這樣一個加密數據
四、前端拿到加密串后,解密數據
需要再axios中添加一個響應攔截器,代碼如下
import axios from 'axios';
import { sm2 } from 'sm-crypto';
// 響應攔截器
axios.interceptors.response.use(res => {
res.data = JSON.parse(sm2.doDecrypt(res.data, '解密私鑰'));
console.log('解密出來的數據', res.data);
});
打印如圖所示
原文鏈接:https://blog.csdn.net/qq243920161/article/details/131261863
- 上一篇:沒有了
- 下一篇:沒有了
相關推薦
- 2023-10-30 Spring的BeanFactory與FactoryBean的區別
- 2022-04-24 TypeScript基礎class類教程示例_基礎知識
- 2022-10-21 tomcat8中startup可以啟動tomcat8w無法啟動的問題分析_Tomcat
- 2022-07-07 ASP.Net使用System.Security.Principal模擬用戶_實用技巧
- 2022-02-12 前后端的身份認證--cookie--session--jwt--token
- 2022-04-01 HIVE str_to_map將字符串轉為map格式
- 2023-01-01 MongoDB?Shell常用基本操作命令詳解_MongoDB
- 2023-10-14 C/C++--跨平臺--預定義宏 WIN32、_WIN32、_WIN64
- 欄目分類
-
- 最近更新
-
- window11 系統安裝 yarn
- 超詳細win安裝深度學習環境2025年最新版(
- Linux 中運行的top命令 怎么退出?
- MySQL 中decimal 的用法? 存儲小
- get 、set 、toString 方法的使
- @Resource和 @Autowired注解
- Java基礎操作-- 運算符,流程控制 Flo
- 1. Int 和Integer 的區別,Jav
- spring @retryable不生效的一種
- Spring Security之認證信息的處理
- Spring Security之認證過濾器
- Spring Security概述快速入門
- Spring Security之配置體系
- 【SpringBoot】SpringCache
- Spring Security之基于方法配置權
- redisson分布式鎖中waittime的設
- maven:解決release錯誤:Artif
- restTemplate使用總結
- Spring Security之安全異常處理
- MybatisPlus優雅實現加密?
- Spring ioc容器與Bean的生命周期。
- 【探索SpringCloud】服務發現-Nac
- Spring Security之基于HttpR
- Redis 底層數據結構-簡單動態字符串(SD
- arthas操作spring被代理目標對象命令
- Spring中的單例模式應用詳解
- 聊聊消息隊列,發送消息的4種方式
- bootspring第三方資源配置管理
- GIT同步修改后的遠程分支