zoukankan      html  css  js  c++  java
  • 使用路由网关的全局过滤功能

    概述

    全局过滤器作用于所有的路由,不需要单独配置,我们可以用它来实现很多统一化处理的业务需求,比如权限认证,IP 访问限制等等。

    注意:截止博客发表时间 2019 年 01 月 10 日,Spring Cloud Gateway 正式版为 2.0.2 其文档并不完善,并且有些地方还要重新设计,这里仅提供一个基本的案例

    详见:Spring Cloud Gateway Documentation

    #声明周期

    img

    Spring Cloud Gateway 基于 Project Reactor 和 WebFlux,采用响应式编程风格,打开它的 Filter 的接口 GlobalFilter 你会发现它只有一个方法 filter。

    #创建全局过滤器

    实现 GlobalFilter, Ordered 接口并在类上增加 @Component 注解就可以使用过滤功能了,非常简单方便

    package com.snake.hello.spring.cloud.gateway.filters;
    
    import com.fasterxml.jackson.core.JsonProcessingException;
    import com.fasterxml.jackson.databind.ObjectMapper;
    import com.google.common.collect.Maps;
    import org.springframework.cloud.gateway.filter.GatewayFilterChain;
    import org.springframework.cloud.gateway.filter.GlobalFilter;
    import org.springframework.core.Ordered;
    import org.springframework.core.io.buffer.DataBuffer;
    import org.springframework.http.HttpStatus;
    import org.springframework.http.server.reactive.ServerHttpResponse;
    import org.springframework.stereotype.Component;
    import org.springframework.web.server.ServerWebExchange;
    import reactor.core.publisher.Mono;
    
    import java.util.Map;
    
    /**
     * 鉴权过滤器
     */
    @Component
    public class AuthFilter implements GlobalFilter, Ordered {
        @Override
        public Mono<Void> filter(ServerWebExchange exchange, GatewayFilterChain chain) {
            String token = exchange.getRequest().getQueryParams().getFirst("token");
    
            if (token == null || token.isEmpty()) {
                ServerHttpResponse response = exchange.getResponse();
    
                // 封装错误信息
                Map<String, Object> responseData = Maps.newHashMap();
                responseData.put("code", 401);
                responseData.put("message", "非法请求");
                responseData.put("cause", "Token is empty");
    
                try {
                    // 将信息转换为 JSON
                    ObjectMapper objectMapper = new ObjectMapper();
                    byte[] data = objectMapper.writeValueAsBytes(responseData);
    
                    // 输出错误信息到页面
                    DataBuffer buffer = response.bufferFactory().wrap(data);
                    response.setStatusCode(HttpStatus.UNAUTHORIZED);
                    response.getHeaders().add("Content-Type", "application/json;charset=UTF-8");
                    return response.writeWith(Mono.just(buffer));
                } catch (JsonProcessingException e) {
                    e.printStackTrace();
                }
            }
    
            return chain.filter(exchange);
        }
    
        /**
        * 设置过滤器的执行顺序
        * @return 
        */
        @Override
        public int getOrder() {
            return Ordered.LOWEST_PRECEDENCE;
        }
    }
    

    #测试过滤器

    浏览器访问:http://localhost:9000/nacos-consumer/echo/app/name 网页显示

    img

    浏览器访问:http://localhost:9000/nacos-consumer/echo/app/name?token=123456 网页显示

    Hello Nacos Discovery nacos-consumer i am from port 8082
    

    #附:Spring Cloud Gateway Benchmark

    Spring 官方人员提供的网关基准测试报告 GitHub

    Proxy Avg Latency Avg Req/Sec/Thread
    gateway 6.61ms 3.24k
    linkered 7.62ms 2.82k
    zuul 12.56ms 2.09k
    none 2.09ms 11.77k

    #说明

    • 这里的 Zuul 为 1.x 版本,是一个基于阻塞 IO 的 API Gateway
    • Zuul 已经发布了 Zuul 2.x,基于 Netty,非阻塞的,支持长连接,但 Spring Cloud 暂时还没有整合计划
    • Linkerd 基于 Scala 实现的、目前市面上仅有的生产级别的 Service Mesh(其他诸如 Istio、Conduit 暂时还不能用于生产)。
    等你看到的时候,想变得有一点点不一样
  • 相关阅读:
    HL7及PIX相关的测试工具
    HDU4570----Multi-bit Trie----简单的DP
    hdu2248
    poj 3693 Maximum repetition substring (后缀数组)
    高性能通道
    volyaire重振Infiniband
    利用iWARP/RDMA解决以太网高延迟
    linux 单网卡来绑定多IP实现多网段访问以及多网卡绑定单IP实现负载均衡
    C细节学习
    每2秒获取系统的赋值及内存使用率
  • 原文地址:https://www.cnblogs.com/snake107/p/11920848.html
Copyright © 2011-2022 走看看