gateway-02 springcloud集成oauth2,网关验证token

一.项目架构

gateway-02 springcloud集成oauth2,网关验证token_第1张图片

此文章内容只应用到 oauth2服务 网关服务 还一个算是基础业务服务吧。

最近抽空写了一个身份认证平台的代码,支持企业微信、oauth2、cas、jwt简易方式等方式进行接入记录一下gateway网关的验证逻辑。

二.gateway网关验证逻辑和代码

(1)验证逻辑

用户前端(登陆操作)

->oauth服务(登录验证成功返回一个jwt增强的token和其他的一些扩展信息)

->前端接收token请求header添加token(请求后端接口操作)

->网关拦截请求(1.拦截请求2.判断请求头3.验证token4.验证成功添加网关key(这个可以根据实际业务来考虑需不需要)5.转发请求到相应的服务)

(2)核心代码

1.网关拦截器:拦截请求并进行白名单路径匹配和token验证

package com.gateway.application.filter;

import com.gateway.application.route.RouteContants;
import com.gateway.application.utils.ResponseProvider;
import com.gateway.application.utils.TokenUtils;
import io.jsonwebtoken.Claims;
import lombok.extern.slf4j.Slf4j;
import org.springframework.cloud.gateway.filter.GatewayFilterChain;
import org.springframework.cloud.gateway.filter.GlobalFilter;
import org.springframework.core.io.buffer.DataBuffer;
import org.springframework.http.HttpStatus;
import org.springframework.http.server.PathContainer;
import org.springframework.http.server.reactive.ServerHttpRequest;
import org.springframework.http.server.reactive.ServerHttpResponse;
import org.springframework.stereotype.Component;
import org.springframework.util.AntPathMatcher;
import org.springframework.util.ObjectUtils;
import org.springframework.web.server.ServerWebExchange;
import reactor.core.publisher.Flux;
import reactor.core.publisher.Mono;

import java.nio.charset.StandardCharsets;
import java.util.UUID;

/**
 * 网关拦截器
 * 
 * @Author: zy
 * @Date: 2022/1/17 11:45
 */
@Component
@Slf4j
public class GatewayFilter implements GlobalFilter {

    private final AntPathMatcher antPathMatcher = new AntPathMatcher();

    /**
     * 全局拦截器
     * @param exchange
     * @param chain
     * @return
     */
    @Override
    public Mono filter(ServerWebExchange exchange, GatewayFilterChain chain) {
        ServerHttpResponse resp = exchange.getResponse();
        ServerHttpRequest request = exchange.getRequest();
        PathContainer pathContainer = request.getPath().pathWithinApplication();
        //if white route
        if (isSkip(pathContainer.toString())) {
            ServerHttpRequest.Builder mutate = request.mutate();
            mutate.header("ggp-token", UUID.randomUUID().toString());
            return chain.filter(exchange.mutate().request(mutate.build()).build());
        }
        String token = request.getHeaders().getFirst(RouteContants.AUTH_KEY);
        if (ObjectUtils.isEmpty(token)) {
            return unAuth(resp, "缺失令牌,鉴权失败");
        }
        String realToken = token.replace("bearer ", "");
        Claims claims = TokenUtils.parseJWT(realToken);
        if(ObjectUtils.isEmpty(realToken) || ObjectUtils.isEmpty(claims)){
            return unAuth(resp,"请求未授权");
        }
        if(ObjectUtils.isEmpty(claims.getExpiration()) || TokenUtils.isTokenExpired(claims.getExpiration())){
            return unAuth(resp,"令牌已失效");
        }
        // 添加gatewayKey,防止下游接口直接被访问
        ServerHttpRequest.Builder mutate = request.mutate();
        mutate.header("ggp-token", UUID.randomUUID().toString());
        return chain.filter(exchange.mutate().request(mutate.build()).build());
    }

    private boolean isSkip(String path) {
        return RouteContants.getDefaultWhiteRoute().stream().anyMatch(pattern -> antPathMatcher.match(pattern, path));
    }

    private Mono unAuth(ServerHttpResponse resp, String msg) {
        resp.setStatusCode(HttpStatus.UNAUTHORIZED);
        resp.getHeaders().add("Content-Type", "application/json;charset=UTF-8");
        String result = "";
        result = ResponseProvider.unAuth(msg).toString();
        DataBuffer buffer = resp.bufferFactory().wrap(result.getBytes(StandardCharsets.UTF_8));
        return resp.writeWith(Flux.just(buffer));
    }
}

2.jwt token验证(只是示例(私钥方式) jwt验证根据实际情况编写)

package com.gateway.application.utils;

import io.jsonwebtoken.Claims;
import io.jsonwebtoken.Jwts;
import lombok.extern.slf4j.Slf4j;

import java.util.Date;

/**
 * @Author: zy
 * @Date: 2022/2/25 16:39
 */
@Slf4j
public class TokenUtils {

    /**
     * token密钥暂时写死在这
     * @return
     */
    private static String getSecret() {
        return "secret";
    }

    /**
     * 验证access_token
     * @param jsonWebToken
     * @return
     */
    public static Claims parseJWT(String jsonWebToken) {
        try {
            return Jwts.parser().setSigningKey(getSecret().getBytes()).parseClaimsJws(jsonWebToken).getBody();
        } catch (Exception e) {
            log.error(e.getMessage());
            return null;
        }
    }

    /**
     * 验证token是否过期失效
     *
     * @param expirationTime
     * @return
     */
    public static boolean isTokenExpired(Date expirationTime) {
        return expirationTime.before(new Date());
    }
}

3gateway路由匹配配置

gateway-02 springcloud集成oauth2,网关验证token_第2张图片

你可能感兴趣的:(微服务,gateway,java,spring,cloud)