zoukankan      html  css  js  c++  java
  • Spring Cloud Gateway 聚合 Swagger

    Gateway 聚合 swagger 虽然网上已经有了很多处理方法,但是参差不齐有很多坑,还是记录一下。

    1. 环境说明

    • 网关依赖 GateWay 2.2.5.RELEASE
    • 注册中心依赖 nacos-discovery 2.2.1.RELEASE
    • Spring Boot v2.3.0.RELEASE

    2. 一般服务配置Swagger

    2.1 引入swagger依赖

    <!--swagger-->
    <dependency>
        <groupId>com.spring4all</groupId>
        <artifactId>swagger-spring-boot-starter</artifactId>
        <version>1.9.1.RELEASE</version>
    </dependency>
    

    2.2 启动类加入@EnableSwagger2Doc注解

    @EnableSwagger2Doc
    @SpringBootApplication
    public class SystemApplication {
        public static void main(String[] args) {
            SpringApplication.run(SystemApplication.class, args);
        }
    }
    

    完成以上俩个步骤,一般服务就可以使用swagger了。

    3. Gateway 网关配置

    3.1 引入swagger依赖

    <!--swagger-->
    <dependency>
        <groupId>com.spring4all</groupId>
        <artifactId>swagger-spring-boot-starter</artifactId>
        <version>1.9.1.RELEASE</version>
    </dependency>
    

    3.2 添加SwaggerHandler、SwaggerResourceConfig两个类

    SwaggerHandler代码如下:

    package com.gct.gateway.swagger;
    
    import org.springframework.beans.factory.annotation.Autowired;
    import org.springframework.http.HttpStatus;
    import org.springframework.http.ResponseEntity;
    import org.springframework.web.bind.annotation.GetMapping;
    import org.springframework.web.bind.annotation.RestController;
    import reactor.core.publisher.Mono;
    import springfox.documentation.swagger.web.*;
    
    import java.util.Optional;
    
    /**
     * @author: NewMeanning
     * @create: 2020-12-24 16:57
     **/
    @RestController
    public class SwaggerHandler {
    
        @Autowired(required = false)
        private SecurityConfiguration securityConfiguration;
    
        @Autowired(required = false)
        private UiConfiguration uiConfiguration;
    
        private final SwaggerResourcesProvider swaggerResources;
    
        @Autowired
        public SwaggerHandler(SwaggerResourcesProvider swaggerResources) {
            this.swaggerResources = swaggerResources;
        }
    
    
        @GetMapping("/swagger-resources/configuration/security")
        public Mono<ResponseEntity<SecurityConfiguration>> securityConfiguration() {
            return Mono.just(new ResponseEntity<>(
                    Optional.ofNullable(securityConfiguration).orElse(SecurityConfigurationBuilder.builder().build()), HttpStatus.OK));
        }
    
        @GetMapping("/swagger-resources/configuration/ui")
        public Mono<ResponseEntity<UiConfiguration>> uiConfiguration() {
            return Mono.just(new ResponseEntity<>(
                    Optional.ofNullable(uiConfiguration).orElse(UiConfigurationBuilder.builder().build()), HttpStatus.OK));
        }
    
        @GetMapping("/swagger-resources")
        public Mono<ResponseEntity> swaggerResources() {
            return Mono.just((new ResponseEntity<>(swaggerResources.get(), HttpStatus.OK)));
        }
    
        @GetMapping("/")
        public Mono<ResponseEntity> swaggerResourcesN() {
            return Mono.just((new ResponseEntity<>(swaggerResources.get(), HttpStatus.OK)));
        }
    
        @GetMapping("/csrf")
        public Mono<ResponseEntity> swaggerResourcesCsrf() {
            return Mono.just((new ResponseEntity<>(swaggerResources.get(), HttpStatus.OK)));
        }
    }
    

    SwaggerResourceConfig代码如下:

    package com.gct.gateway.swagger;
    
    import lombok.AllArgsConstructor;
    import lombok.extern.slf4j.Slf4j;
    import org.springframework.beans.factory.annotation.Autowired;
    import org.springframework.beans.factory.annotation.Value;
    import org.springframework.cloud.gateway.config.GatewayProperties;
    import org.springframework.cloud.gateway.route.RouteLocator;
    import org.springframework.cloud.gateway.support.NameUtils;
    import org.springframework.context.annotation.Primary;
    import org.springframework.stereotype.Component;
    import springfox.documentation.swagger.web.SwaggerResource;
    import springfox.documentation.swagger.web.SwaggerResourcesProvider;
    
    import java.util.ArrayList;
    import java.util.HashSet;
    import java.util.List;
    import java.util.Set;
    
    /**
     * @author: NewMeanning
     * @create: 2020-12-24 16:56
     **/
    @Slf4j
    @Component
    @Primary
    @AllArgsConstructor
    public class SwaggerResourceConfig implements SwaggerResourcesProvider {
    
        /**
         * swagger2默认的url后缀
         */
        private static final String SWAGGER2URL = "/v2/api-docs";
    
        /**
         * 网关路由
         */
        private final RouteLocator routeLocator;
    
        /**
         * 网关应用名称
         */
        @Value("${spring.application.name}")
        private String self;
    
        @Autowired
        public SwaggerResourceConfig(RouteLocator routeLocator) {
            this.routeLocator = routeLocator;
        }
    
        @Override
        public List<SwaggerResource> get() {
            List<SwaggerResource> resources = new ArrayList<>();
            List<String> routeHosts = new ArrayList<>();
            // 由于我的网关采用的是负载均衡的方式,因此我需要拿到所有应用的serviceId
            // 获取所有可用的host:serviceId
            routeLocator.getRoutes().filter(route -> route.getUri().getHost() != null)
                    .filter(route -> !self.equals(route.getUri().getHost()))
                    .subscribe(route -> routeHosts.add(route.getUri().getHost()));
    
            // 记录已经添加过的server,存在同一个应用注册了多个服务在nacos上
            Set<String> dealed = new HashSet<>();
            routeHosts.forEach(instance -> {
                // 拼接url,样式为/serviceId/v2/api-info,当网关调用这个接口时,会自动通过负载均衡寻找对应的主机
                String url = "/" + instance + SWAGGER2URL;
                if (!dealed.contains(url)) {
                    dealed.add(url);
                    SwaggerResource swaggerResource = new SwaggerResource();
                    swaggerResource.setUrl(url);
                    swaggerResource.setName(instance);
                    resources.add(swaggerResource);
                }
            });
            return resources;
        }
    }
    

    3.3 application配置

    只需要指定spring.application.name即可,名称随意;另外无需配置router,配置如下:

    # 端口号
    server:
      port: 88
    
    spring:
      cloud:
        nacos:
          discovery:
            ## 指定nacos server地址
            server-addr: localhost:8848
        gateway:
          discovery:
            locator:
              enabled: true  # 启用网关
      ## 本服务名称
      application:
        name: gateway
    
  • 相关阅读:
    Asp.Net Core混合使用cookie和JwtBearer认证方案
    验证来自JWT的User.Identity
    asp.net core cookie和jwt简单的登录认证
    asp.net core 6.0 访问IConfiguration
    在 ASP.NET Core 中将依赖项注入到视图
    在 ASP.NET Core 上配置 DefaultScheme 和 DefaultChallengeScheme 有什么意义?混合身份认证问题
    WPF基础之路由事件五
    .NET framework 4.5新特性预览:核心语言 新增 功能和改进
    WPF基础之路由事件三
    Brush In WPF
  • 原文地址:https://www.cnblogs.com/nm666/p/14185577.html
Copyright © 2011-2022 走看看