SpringBoot前后端分离框架中,在请求头加入签名

SpringBoot前后端分离框架中,要在请求头加入签名(Sign) ,核心是:前端在请求拦截器统一加签,后端通过拦截器统一验签

一、前端实现(Vue/axios):在请求头加签名

1. 安装依赖
bash 复制代码
npm install js-md5 --save
2. 配置签名密钥(建议单独文件)

创建 @/config/signature.js

javascript 复制代码
// 签名密钥(前后端必须一致,建议从环境变量读取)
export const SIGN_SECRET = 'your-strong-secret-key-2026';
// 参数分隔符
export const SIGN_SEPARATOR = '|';
3. 修改请求拦截器(@/utils/request.js
javascript 复制代码
import axios from 'axios';
import Md5 from 'js-md5';
import { SIGN_SECRET, SIGN_SEPARATOR } from '@/config/signature';

const service = axios.create({
  baseURL: process.env.VUE_APP_BASE_API,
  timeout: 10000
});

// 请求拦截器:加入签名
service.interceptors.request.use(config => {
  // 1. 生成时间戳(毫秒)
  const timestamp = Date.now().toString();
  
  // 2. 构建待签字符串(建议:密钥 + 时间戳 + 请求方法 + URL)
  const method = config.method.toUpperCase();
  const url = config.url;
  const stringToSign = [SIGN_SECRET, timestamp, method, url].join(SIGN_SEPARATOR);
  
  // 3. 生成MD5签名(转大写)
  const sign = Md5(stringToSign).toUpperCase();
  
  // 4. 放入请求头
  config.headers['timestamp'] = timestamp;
  config.headers['sign'] = sign;
  
  return config;
}, error => {
  return Promise.reject(error);
});

export default service;

二、后端实现(Java/Spring Boot):拦截器验签

1. 自定义签名拦截器
java 复制代码
package com.ruoyi.framework.interceptor;

import cn.hutool.core.util.StrUtil;
import com.ruoyi.common.exception.ServiceException;
import com.ruoyi.common.utils.ServletUtils;
import com.ruoyi.common.utils.StringUtils;
import org.springframework.beans.factory.annotation.Value;
import org.springframework.stereotype.Component;
import org.springframework.web.servlet.HandlerInterceptor;

import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.util.Arrays;

@Component
public class SignatureInterceptor implements HandlerInterceptor {

    // 从配置文件读取密钥
    @Value("${signature.secret}")
    private String SIGN_SECRET;
    
    private static final String SIGN_SEPARATOR = "|";
    // 签名有效时间(毫秒,例如5分钟)
    private static final long EXPIRE_TIME = 5 * 60 * 1000;

    @Override
    public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) {
        // 1. 从请求头获取参数
        String timestamp = request.getHeader("timestamp");
        String clientSign = request.getHeader("sign");
        String method = request.getMethod();
        String url = request.getRequestURI();

        // 2. 非空校验
        if (StrUtil.hasBlank(timestamp, clientSign)) {
            throw new ServiceException("缺少签名参数");
        }

        // 3. 时间戳过期校验
        long currentTime = System.currentTimeMillis();
        long reqTime = Long.parseLong(timestamp);
        if (Math.abs(currentTime - reqTime) > EXPIRE_TIME) {
            throw new ServiceException("请求已过期");
        }

        // 4. 后端重新生成签名(与前端规则一致)
        String stringToSign = String.join(SIGN_SEPARATOR,
                SIGN_SECRET, timestamp, method, url);
        String serverSign = StrUtil.upper(StrUtil.md5(stringToSign));

        // 5. 比对签名
        if (!serverSign.equals(clientSign)) {
            throw new ServiceException("签名验证失败");
        }

        return true;
    }
}
2. 注册拦截器(WebConfig.java
java 复制代码
package com.ruoyi.framework.config;

import com.ruoyi.framework.interceptor.SignatureInterceptor;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.context.annotation.Configuration;
import org.springframework.web.servlet.config.annotation.InterceptorRegistry;
import org.springframework.web.servlet.config.annotation.WebMvcConfigurer;

@Configuration
public class WebConfig implements WebMvcConfigurer {

    @Autowired
    private SignatureInterceptor signatureInterceptor;

    @Override
    public void addInterceptors(InterceptorRegistry registry) {
        registry.addInterceptor(signatureInterceptor)
                .addPathPatterns("/**") // 拦截所有请求
                .excludePathPatterns(
                        "/login", "/logout", "/captchaImage" // 放行登录等接口
                )
                .order(1); // 优先于登录拦截
    }
}
3. 配置文件(application.yml
yaml 复制代码
# 签名配置
signature:
  secret: your-strong-secret-key-2026

三、安全增强(可选)

  • 加入参数签名:将请求参数(query/body)按key排序后拼接,防止参数篡改。
  • 使用HMAC-SHA256:比MD5更安全。
  • 动态密钥:按用户/应用分配不同密钥。
  • 全局异常处理:统一返回签名错误JSON。

四、常见问题

  • 跨域问题 :在 CorsConfig 中暴露请求头:

    java 复制代码
    config.setExposedHeaders(Arrays.asList("timestamp", "sign"));
  • 放行接口漏配:登录、验证码、WebSocket等必须排除。

相关推荐
不懂的浪漫2 小时前
一次设备映射缓存设计:用多索引 Map 把高频查询从遍历变成直接命中
java·算法·spring·缓存
九转成圣2 小时前
Spring Boot 导出 Excel 最佳实践:从 POI 函数式封装到 EasyExcel 的“降维打击”
spring boot·后端·excel
好家伙VCC2 小时前
# React发散创新:从状态管理到自定义Hook的极致实践与性能优化在现代前端开发
java·javascript·python·react.js·性能优化
liyi_hz20082 小时前
O2OA(翱途) V10 升级说明(三)数据中心:精准查询·严谨权限·优质视图
后端·java-ee·开源软件
eLIN TECE2 小时前
Redis重大版本整理(Redis2.6-Redis7.0)
java·数据库·redis
花千树-0102 小时前
两行注解把企业 RPC 接口变成 AI 工具
java·rpc·langchain·react·function call·ai agent·mcp
迷藏4942 小时前
**绿色AI:用Python构建节能型机器学习模型的实践与优化策略**在人工智能飞速发展的今天,模型训练和
java·人工智能·python·机器学习
刀法如飞2 小时前
一款基于 NestJS 的 DDD 脚手架,开箱即用
javascript·后端·架构
juniperhan2 小时前
Flink 系列第13篇:Flink 生产环境中的并行度与资源配置
java·大数据·数据仓库·分布式·flink