前后端分离项目通过header参数鉴权

5 篇文章 0 订阅
4 篇文章 0 订阅
  • 前端,在请求头加上鉴权参数
const apiKey = '******'
const echo = '随机字符串';
const timestamp = new Date().getTime();
headers: {
   'Content-Type': 'application/json;charset=utf-8',
    'Cache-Control': 'no-cache;max-age=0',
    'Pragma': 'no-cache',
    'x-echo-key': echo,
    'x-timestamp-key': timestamp,
    'x-signature-key': 加密函数(apiKey + timestamp + echo),
    'access-token': '校验登录的token',
    'user-id': '用户ID'
}
  • 后端,拦截器配置类(AuthInterceptorConfig.java)
@Configuration
public class AuthInterceptorConfig implements WebMvcConfigurer {

    @Bean
    public AuthInterceptor initAuthInterceptor(){
        return new AuthInterceptor();
    }

    @Override
    public void addInterceptors(InterceptorRegistry registry) {
        registry.addInterceptor(initAuthInterceptor())
                .addPathPatterns("/col/**") //需要拦截的URL
                .excludePathPatterns("/login/**"); //不需要拦截的URL
    }
}
  • 后端,拦截器处理类(AuthInterceptor.java)
public class AuthInterceptor implements HandlerInterceptor {
    private final static Logger logger = LoggerFactory.getLogger(AuthInterceptor.class);
    @Value("${app.x-api-key}")
    private String x_api_key;
    @Autowired
    private RedisService redisService;

    @Override
    public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {
        response.setCharacterEncoding("UTF-8");
        response.setContentType("text/html;charset=utf-8");

        String servletPath = request.getServletPath();
        String xEchoKey = request.getHeader("x-echo-key");
        String xTimestampKey = request.getHeader("x-timestamp-key");
        String xSignatureKey = request.getHeader("x-signature-key");
        String accessToken = request.getHeader("access-token");
        String userId = request.getHeader("user-id");

        //处理特殊请求URL
        if (servletPath.contains("/**")) {
            String queryString = request.getQueryString();
            Map<String, String> urlParams = StringUtil.queryStrToMap(queryString);
            xEchoKey = urlParams.get("x-echo-key");
            xTimestampKey = urlParams.get("x-timestamp-key");
            xSignatureKey = urlParams.get("x-signature-key");
            accessToken = urlParams.get("access-token");
            userId = urlParams.get("user-id");
        }

        //1.接口参数非空验证
        if (org.apache.commons.lang3.StringUtils.isEmpty(xEchoKey) || org.apache.commons.lang3.StringUtils.isEmpty(xTimestampKey) || org.apache.commons.lang3.StringUtils.isEmpty(xSignatureKey)) {
            response.getWriter().println("非法请求,接口参数验证错误");
            return false;
        }

        //2.签名检查
        String signature = MD5.gtMD5Code(x_api_key + xTimestampKey + xEchoKey);
        if (!xSignatureKey.equals(signature)) {
            response.getWriter().println("非法请求,签名校验失败");
            return false;
        }

        //3.检查时间戳
        long timestampNow = System.currentTimeMillis();
        long timestamp = Long.parseLong(xTimestampKey);
        long minutesDiff = (timestampNow - timestamp) / (1000 * 60);
        //5分钟内有效(自行设置)
        if (Math.abs(minutesDiff) > 5) {
            response.getWriter().println("非法请求,接口时间戳校验失败");
            return false;
        }

		//token已经在login系统时存到redis中
        String authCode = redisService.getStr("token_".concat(userId));
        if (null == authCode) {
            response.getWriter().println("会话已过期,请重新打开应用");
            return false;
        } else  {
			if (!accessToken.equals(authCode)) {
				response.getWriter().println("会话已过期,请重新打开应用");
            	return false;
			}
		}

        return true;
    }

    @Override
    public void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler, ModelAndView modelAndView) throws Exception {

    }

    @Override
    public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) throws Exception {

    }
}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值