gateway网关实现AK/SK接口签名认证

gateway网关实现AK/SK接口签名认证

自定义filter

@Slf4j
@Component
public class AccessKeyFilter implements WebFilter {

    public static final String API_CODE = "api-code";
    public static final String TIME_STAMP = "time-stamp";
    public static final String SIGNATURE = "signature";
    public static final String ACCESS_KEY = "access-key";
    public static final String ORG_CODE = "org-code";
    private final Map<String,String> validateUrl = new HashMap<String,String>(){
        {
            put("test", "/api/v1/test");
        };
    };
    private int expireSeconds = 300;
    private Sign sign = SecureUtil.sign(SignAlgorithm.SHA256withECDSA);
    @Autowired
    private OrgSecretKeyMapper orgSecretKeyMapper;

    private String resolveBodyFromRequest(ServerWebExchange webExchange){
        // 获取请求体
        Flux<DataBuffer> body = webExchange.getRequest().getBody();
        AtomicReference<String> bodyRef = new AtomicReference<>();
        body.subscribe(buffer -> {
            CharBuffer charBuffer = StandardCharsets.UTF_8.decode(buffer.asByteBuffer());
            DataBufferUtils.release(buffer);
            bodyRef.set(charBuffer.toString());
        });
        return bodyRef.get();
    }

    @Override
    public Mono<Void> filter(ServerWebExchange exchange, WebFilterChain chain) {
        ServerHttpRequest request = exchange.getRequest();
        String url = request.getURI().getPath();
        HttpHeaders headers = request.getHeaders();
        // 跳过不需要验证的路径
        if (!StringUtils.matches(url, validateUrl.values().stream().collect(Collectors.toList()))) {
            return chain.filter(exchange);
        }
        // 获取api code
        String apiCode = headers.getFirst(API_CODE);
        String timeStr = headers.getFirst(TIME_STAMP);
        String signature = headers.getFirst(SIGNATURE);
        String accessKey = headers.getFirst(ACCESS_KEY);
        String orgCode = headers.getFirst(ORG_CODE);
        // 校验apiCode是否匹配
        if (StringUtils.isEmpty(apiCode)
            ||!validateUrl.containsKey(apiCode)
            ||!StringUtils.isMatch(validateUrl.get(apiCode),url)){
            String formatMsg = StringUtils.format("apiCode[{}],url[{}]不匹配", apiCode, url);
            log.error(formatMsg);
            return ServletUtils.webFluxResponseWriter(exchange.getResponse(), formatMsg);
        }
        // 校验时间
        if(StringUtils.isEmpty(timeStr)){
            String formatMsg = StringUtils.format("请求头缺少{}:[{}]", TIME_STAMP, timeStr);
            log.error(formatMsg);
            return ServletUtils.webFluxResponseWriter(exchange.getResponse(), formatMsg);
        }
        try{
            Date expireTime= DateUtil.offset(new Date(Long.parseLong(timeStr)), DateField.SECOND,expireSeconds);
            Date currentDate = new Date();
            if(DateUtil.compare(expireTime, currentDate)<0) {
                String formatMsg = StringUtils.format("请求时间戳[{}]已超过有效期{}秒,服务器当前时间[{}]",
                    timeStr,expireSeconds,DateUtil.format(currentDate,DatePattern.PURE_DATETIME_PATTERN));
                log.error(formatMsg);
                return ServletUtils.webFluxResponseWriter(exchange.getResponse(),formatMsg);
            }
        }catch (Exception e){
            String formatMsg = StringUtils.format("请求时间戳[{}]校验失败", timeStr);
            log.error(formatMsg,e);
            return ServletUtils.webFluxResponseWriter(exchange.getResponse(),formatMsg);
        }
        // 校验orgCode和accessKey
        if (StringUtils.isEmpty(signature)||StringUtils.isEmpty(accessKey)||StringUtils.isEmpty(orgCode)){
            String formatMsg = StringUtils.format("请求头参数校验失败,请检查是否缺少参数[{},{},{}]",SIGNATURE,ACCESS_KEY,ORG_CODE);
            log.error("{},{}:{},{}:{},{}:{}",formatMsg,SIGNATURE,signature,ACCESS_KEY,accessKey,ORG_CODE,orgCode);
            return ServletUtils.webFluxResponseWriter(exchange.getResponse(),formatMsg);
        }
        String secretKey = orgSecretKeyMapper.querySecretKey(orgCode, accessKey);
        if (StringUtils.isEmpty(secretKey)){
            log.error("校验失败,密钥不存在,{}:{},{}:{}",ACCESS_KEY,accessKey,ORG_CODE,orgCode);
            return ServletUtils.webFluxResponseWriter(exchange.getResponse(),"校验失败");
        }
        try {
            String rspStr = resolveBodyFromRequest(exchange);
            JSONObject obj = JSONObject.parseObject(rspStr);
            SortedMap<String, Object> sortedMap = new TreeMap<String, Object>();
            sortedMap.putAll(obj);
            String signHeader = StringUtils.format("{}={}",
                TIME_STAMP, timeStr);
            String signParam = sortedMap.entrySet().stream().map(
                e -> String.join("=", e.getKey(), e.getValue().toString())).collect(Collectors.joining("&"));
            HMac mac = new HMac(HmacAlgorithm.HmacSHA256,(signHeader+"&"+signParam).getBytes(StandardCharsets.UTF_8));
            String digestBase64 = mac.digestBase64(signHeader + "&" + signParam, StandardCharsets.UTF_8, false);
            if (StringUtils.equals(digestBase64,signature)) {
                log.info("签名通过");
                return chain.filter(exchange);
            } else {
                log.info("参数校验出错");
                return ServletUtils.webFluxResponseWriter(exchange.getResponse(), "参数校验出错");
            }
        } catch (Exception e) {
            log.error("签名验证异常",e);
            return ServletUtils.webFluxResponseWriter(exchange.getResponse(), e.getMessage());
        }
    }
}

  • 0
    点赞
  • 9
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 2
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

ableyang~

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值