springcloud-网关(gateway)

springcloud-网关(gateway)

概述

\Spring Cloud Gateway旨在提供一种简单而有效的方式来路由到API,并为其提供跨领域的关注,如:安全、监控/指标和容错

常用术语

  • Route(路由): 网关的基本构件。它由一个ID、一个目的地URI、一个谓词(Predicate)集合和一个过滤器(Filter)集合定义。如果集合谓词为真,则路由被匹配。
  • Predicate(谓词): 这是一个 Java 8 Function Predicate。输入类型是 [Spring Framework ServerWebExchange]。这让你可以在HTTP请求中的任何内容上进行匹配,比如header或查询参数。
  • Filter(过滤器): 这些是 [GatewayFilter] 的实例,已经用特定工厂构建。在这里,你可以在发送下游请求之前或之后修改请求和响应。

网关工作原理

客户端向 Spring Cloud Gateway 发出请求。如果Gateway处理程序映射确定一个请求与路由相匹配,它将被发送到Gateway Web处理程序。这个处理程序通过一个特定于该请求的过滤器链来运行该请求。过滤器被分割的原因是,过滤器可以在代理请求发送之前和之后运行逻辑。所有的 "pre" (前)过滤器逻辑都被执行。然后发出代理请求。在代理请求发出后,"post" (后)过滤器逻辑被运行。

image-20231213121300712

业务架构图

Spring Cloud:Feign负载均衡及熔断&Gateway网关的路由及过滤器&Gateway跨域&Spring Cloud Config分布式配置中心&Spring Cloud Bus ...

开发流程

引入依赖

gateway框架需要引入loadbalancer,否则在访问的时候会出现503异常。

<dependency>
    <groupId>org.springframework.cloud</groupId>
    <artifactId>spring-cloud-starter-loadbalancer</artifactId>
</dependency>
<dependency>
    <groupId>org.springframework.cloud</groupId>
    <artifactId>spring-cloud-starter-gateway</artifactId>
</dependency>

bootstrap.yml配置

配置nacos

如果不需要nacos的配置,可以不引入nacos的config依赖

<dependency>
    <groupId>com.alibaba.cloud</groupId>
    <artifactId>spring-cloud-starter-alibaba-nacos-discovery</artifactId>
</dependency>
spring:
  application:
    name: ssc-cloud-gateway
  cloud:
    nacos:
      discovery:
        server-addr: 192.168.201.81:7777
        namespace: ssc-cloud-id
        group: DEV
        register-enabled: true
配置gateway
spring:
  cloud:
    gateway:
      discovery:
        locator:
          enabled: true
      routes:
        - id : ssc-order-id               #路由编号(自定义)
#          uri: https://www.bilibili.com/   #跳转地址(网页跳转),lb--负载调整(微服务)
          uri: lb://ssc-cloud-older
          predicates:
            - Path=/ssc-older/**       #url 的映射名称(让客户看到的)
          filters:
            - StripPrefix=1
        - id: ssc-paycenter-id               #路由编号(自定义)
          uri: lb://ssc-cloud-paycenter
          predicates:
            - Path=/ssc-pay/**
          filters:
            - StripPrefix=1
路由配置规则
序号名称解释
1id⾃定义的路由 ID,保持唯⼀
2uri目标服务地址:以lb:// 开头则表示要去向的微服务名称
3predicates路由条件
4Filter(过滤器)过滤器是路由转发请求时所经过的过滤逻辑,可⽤于修改请求、响应内容。
predicates断言条件
  • Path

    实例: - Path=/ssc-older/

    routes:
        - id : ssc-older-id          
          uri: lb://ssc-cloud-older
          predicates:
            - Path=/ssc-older/**    
    

    解释: 当请求的路径为ssc-older开头的时,转发到ssc-cloud-older微服服务器上,ssc-cloud-older是navcos中微服务注册的名称。

    image-20231213175938386

  • Before

    - Before=2017-01-20T17:42:47.789-07:00[Asia/Shanghai]

    在某个时间之前的请求才会被转发到 http://localhost:9001 服务器上

  • After

    - After=2017-01-20T17:42:47.789- 07:00[Asia/Shanghai]

    在某个时间之后的请求才会被转发

  • Between

    - Between=2017-01-20T17:42:47.789-07:00[Asia/Shanghai],

    2017-01- 21T17:42:47.789-07:00[Asia/Shanghai]

    在某个时间段之间的才会被转发

  • Cookie

    - Cookie=sesionId,ssc-test

    名为ss-old的表单或者满⾜正则old的表单才会被匹配到 进⾏请求转发

    routes:
        - id : ssc-older-id          
          uri: lb://ssc-cloud-older
          predicates:
            - Cookie=sesionId,ssc-test   
    
  • Header

    - Header=X-Request-Id

    携带参数X-Request-Id的请求头才会匹配

  • Host

    - Host=www.ssc.older.com

    当主机名为www.ssc.older.com的时候直接转发到指定服务器。

  • Method

    - Method=GET

    只有GET⽅法才会匹配转发请求,还可以限定POST、PUT等。

    routes:
        - id : ssc-older-id          
          uri: lb://ssc-cloud-older
          predicates:
            - Method=GET,POST  
    
过滤器规则
过滤规则实例说明
PrefixPath- PrefixPath=/app在请求路径前加上app
PrefixPath- PrefixPath=/app在请求路径前加上app
SetPathSetPath=/app/{path}通过模板设置路径,转发的规则时会在路径前增加 app,{path}表示原请求路径
RedirectTo重定向
RemoveRequestHea去掉某个请求头信息\1. PrefixPath
StripPrefixPath=/ssc-pay/**去掉ssc-pay才是真实路径

网关的案例

使用网关进行用户鉴权过滤。

image-20231220095437516

鉴权微服务

image-20231220095804335

api模块

api模块为外界提供接口(openfeign调用),为网关调用需要基于reactor开发。

pom
<?xml version="1.0" encoding="UTF-8"?>
<project xmlns="http://maven.apache.org/POM/4.0.0"
         xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
         xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
    <modelVersion>4.0.0</modelVersion>
    <parent>
        <groupId>com.wnhz.ssc.cloud.authority</groupId>
        <artifactId>ssc-cloud-authority</artifactId>
        <version>0.0.1-SNAPSHOT</version>
    </parent>

    <groupId>com.wnhz.ssc.cloud.authority.api</groupId>
    <artifactId>ssc-cloud-authority-api</artifactId>

    <dependencies>
        <dependency>
            <groupId>com.playtika.reactivefeign</groupId>
            <artifactId>feign-reactor-webclient</artifactId>
            <version>3.0.3</version>
        </dependency>

        <dependency>
            <groupId>com.playtika.reactivefeign</groupId>
            <artifactId>feign-reactor-cloud</artifactId>
            <version>3.0.3</version>
        </dependency>

        <dependency>
            <groupId>com.playtika.reactivefeign</groupId>
            <artifactId>feign-reactor-spring-configuration</artifactId>
            <version>3.0.3</version>
        </dependency>
    </dependencies>
</project>
openfeign接口
@ReactiveFeignClient(value = "ssc-cloud-authority",url = "http://localhost:12121")
public interface IAuthorityFeign {

    @GetMapping("/api/authority/login")
    Mono<HttpResp> login(@RequestParam("username") String username,
                              @RequestParam("password") String password);

    @PostMapping("/api/authority/verifyToken")
    Mono<HttpResp> verifyToken(String token);
}

业务接口(sevice)

pom.xml
<?xml version="1.0" encoding="UTF-8"?>
<project xmlns="http://maven.apache.org/POM/4.0.0"
         xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
         xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
    <modelVersion>4.0.0</modelVersion>
    <parent>
        <groupId>com.wnhz.ssc.cloud.authority</groupId>
        <artifactId>ssc-cloud-authority</artifactId>
        <version>0.0.1-SNAPSHOT</version>
    </parent>

    <groupId>com.wnhz.ssc.cloud.authority.service</groupId>
    <artifactId>ssc-cloud-authority-service</artifactId>

    <dependencies>

        <dependency>
            <groupId>com.wnhz.ssc.cloud.springcloud</groupId>
            <artifactId>ssc-cloud-springcloud</artifactId>
            <version>0.0.1-SNAPSHOT</version>
        </dependency>

        <dependency>
            <groupId>com.github.xiaoymin</groupId>
            <artifactId>knife4j-openapi2-spring-boot-starter</artifactId>
        </dependency>
        <dependency>
            <groupId>com.wnhz.ssc.cloud</groupId>
            <artifactId>ssc-cloud-tools</artifactId>
            <version>0.0.1-SNAPSHOT</version>
        </dependency>
        <dependency>
            <groupId>com.wnhz.ssc.cloud</groupId>
            <artifactId>ssc-cloud-ssm</artifactId>
            <version>0.0.1-SNAPSHOT</version>
        </dependency>

        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-data-redis</artifactId>
        </dependency>

    </dependencies>
</project>
配置

image-20231220100840317

bootstrap.yml

image-20231220100632172

image-20231220100711263

spring:
  cloud:
    nacos:
      config:
        server-addr: ${spring.cloud.nacos.discovery.server-addr}
        namespace: ${spring.cloud.nacos.discovery.namespace}
        file-extension: yaml
        extension-configs:
          - data-id: db_ssc.yaml
            group: DEV
            refresh: true
          - data-id: redis.yaml
            group: DEV
            refresh: true
      discovery:
        server-addr: 192.168.201.107
        namespace: ssc-cloud-id
        group: DEV
  application:
    name: ssc-cloud-authority
application
  • application-dev.yml

    server:
      port: 12121
    knife4j:
      enable: true
    logging:
      level:
        com.wnhz: debug
    
  • application.yml

    spring:
      profiles:
        active: dev
    
启动类
@EnableDiscoveryClient
@SpringBootApplication
public class AuthorityApp {
    public static void main(String[] args) {
        SpringApplication.run(AuthorityApp.class);
    }
}
dao
@Mapper
public interface ILoginDao extends BaseMapper<Login> {
}
异常

image-20231220101139895

/**
 * 鉴权异常
 */
public class AuthorityException extends RuntimeException{
    public AuthorityException(String message) {
        super(message);
    }
}
/**
 * 用户名或密码错误异常
 */
public class BadCredentialsException extends AuthorityException{
    public BadCredentialsException(String message) {
        super(message);
    }
}
service
  • 接口

    public interface ILoginService {
        Login loginCheck(String username, String password);
        List<Login> findByUsername(String username);
    }
    
  • 实现类

    package com.wnhz.ssc.cloud.authority.service.impl;
    
    import com.baomidou.mybatisplus.core.conditions.query.LambdaQueryWrapper;
    import com.wnhz.ssc.cloud.authority.dao.ILoginDao;
    import com.wnhz.ssc.cloud.authority.exception.BadCredentialsException;
    import com.wnhz.ssc.cloud.authority.exception.UsernameEmptyException;
    import com.wnhz.ssc.cloud.authority.exception.UsernameNotFoundException;
    import com.wnhz.ssc.cloud.authority.service.ILoginService;
    import com.wnhz.ssc.domain.entity.po.Login;
    import org.springframework.beans.factory.annotation.Autowired;
    import org.springframework.stereotype.Service;
    import org.springframework.util.StringUtils;
    
    import java.util.List;
    import java.util.Objects;
    
    @Service
    public class LoginServiceImpl implements ILoginService {
        @Autowired
        private ILoginDao loginDao;
    
        @Override
        public Login loginCheck(String username, String password) {
            if (!StringUtils.hasText(username)) throw new UsernameEmptyException("用户名为空异常");
            LambdaQueryWrapper<Login> lambdaQueryWrapper
                    = new LambdaQueryWrapper<>();
            lambdaQueryWrapper.eq(Login::getUsername, username);
            List<Login> logins = findByUsername(username);
            Login login = logins.stream()
                    .filter(lg -> username.equals(lg.getUsername())).findFirst().get();
            if (Objects.isNull(login)) throw new BadCredentialsException("用户名|密码错误");
            return login;
        }
    
        /**
         * 判读用户名是否存在
         * @param username 用户名
         * @return 所有查询服务用户名的用户集合
         */
        @Override
        public List<Login> findByUsername(String username) {
            List<Login> logins = loginDao.selectList(new LambdaQueryWrapper<Login>().eq(Login::getUsername, username));
            if (Objects.isNull(logins) || logins.isEmpty()) throw new UsernameNotFoundException("用户名不存在异常");
            return logins;
        }
    }
    
controller
package com.wnhz.ssc.cloud.authority.controller;

import com.wnhz.ssc.cloud.authority.service.ILoginService;
import com.wnhz.ssc.cloud.tools.exception.jwt.JwtException;
import com.wnhz.ssc.cloud.tools.exception.jwt.JwtExpiredException;
import com.wnhz.ssc.cloud.tools.jwt.JwtUtil;
import com.wnhz.ssc.common.result.HttpResp;
import com.wnhz.ssc.domain.entity.po.Login;
import io.swagger.annotations.Api;
import io.swagger.annotations.ApiOperation;
import lombok.extern.slf4j.Slf4j;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.data.redis.core.RedisTemplate;
import org.springframework.data.redis.core.StringRedisTemplate;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;
import reactor.core.publisher.Mono;

import java.util.HashMap;
import java.util.Map;
import java.util.Objects;
import java.util.concurrent.TimeUnit;

@Api(tags = "鉴权模块api")
@RestController
@RequestMapping("/api/authority")
@Slf4j
public class AuthorityController {

    @Autowired
    private ILoginService loginService;
    @Autowired
    private StringRedisTemplate stringRedisTemplate;
    private final String REDIS_PREFIX="token:";

    /**
     * 用户登录
     *
     * @param username
     * @param password
     * @return
     */
    @ApiOperation(value = "login", notes = "用户登录验证")
    @GetMapping("/login")
    public HttpResp login(String username, String password) {
        try {
            Login login = loginService.loginCheck(username, password);
            log.debug("用户名登录成功:{}",login);
            Map<String, Object> map = new HashMap<>();
            map.put("username", username);
            String token = JwtUtil.getInstance().creatToken(
                    map, 1000 * 30, login.getSign());
            stringRedisTemplate.opsForValue().set(REDIS_PREFIX+token, login.getSign(),
                    10, TimeUnit.MINUTES);
            log.debug("token产生成功,并存入redis中:{}",token);
            return HttpResp.success(new String[]{"ssc_login_token", token});
        } catch (Exception e) {
            return HttpResp.failed(e.getMessage());
        }
    }

    /**
     * 验证token
     *
     * @param token
     * @return
     */
    @ApiOperation(value = "verifyToken", notes = "token验证")
    @GetMapping("/verifyToken")
    public Mono<HttpResp> verifyToken(String token) {
        String sign = stringRedisTemplate.opsForValue().get(REDIS_PREFIX+token);
        log.debug("从redis中获取token的值: {}", sign);
        if (Objects.isNull(sign)) {
            return Mono.just(HttpResp.failed("token不存在"));
        }
        try {
            JwtUtil.getInstance().verifyToken(token, sign);
            return Mono.just(HttpResp.success("token验证成功"));
        } catch (JwtExpiredException e) {//token已过期,判断是否续期
            return refreshToken(token, sign);
        } catch (JwtException e) {
            return Mono.just(HttpResp.failed(e.getMessage()));
        }
    }

    private Mono<HttpResp> refreshToken(String originalToken, String sign) {
        Long expireLeftTime = stringRedisTemplate.getExpire(REDIS_PREFIX+originalToken);
        log.debug("redis中token剩余时间:{}", expireLeftTime);
        if (expireLeftTime > 0) {//token续期
            String newToken = JwtUtil.getInstance().cloneToken(originalToken, expireLeftTime);
            stringRedisTemplate.delete(REDIS_PREFIX+originalToken);
            stringRedisTemplate.opsForValue().set(REDIS_PREFIX+newToken, sign);
            log.debug("新token产生成功,续期完成,token==> {}",newToken);
            return Mono.just(HttpResp.success(new String[]{"ssc_login_token", newToken}));
        } //token已经过期,redis也过期
        return Mono.just(HttpResp.failed("token已经过期"));
    }
}

全局网关

负责对鉴权控制

pom

<?xml version="1.0" encoding="UTF-8"?>
<project xmlns="http://maven.apache.org/POM/4.0.0"
         xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
         xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
    <parent>
        <artifactId>wnhz-ssc-cloud</artifactId>
        <groupId>com.wnhz.ssc.cloud</groupId>
        <version>0.0.1-SNAPSHOT</version>
    </parent>
    <modelVersion>4.0.0</modelVersion>

    <groupId>com.wnhz.ssc.cloud.gateway</groupId>
    <artifactId>ssc-cloud-gateway</artifactId>

    <dependencies>
        <dependency>
            <groupId>com.wnhz.ssc.cloud.springcloud</groupId>
            <artifactId>ssc-cloud-springcloud</artifactId>
            <version>0.0.1-SNAPSHOT</version>
        </dependency>
        <dependency>
            <groupId>com.wnhz.ssc.cloud.common</groupId>
            <artifactId>ssc-cloud-common</artifactId>
            <version>0.0.1-SNAPSHOT</version>
        </dependency>
        <dependency>
            <groupId>org.springframework.cloud</groupId>
            <artifactId>spring-cloud-starter-gateway</artifactId>
        </dependency>
        <dependency>
            <groupId>com.playtika.reactivefeign</groupId>
            <artifactId>feign-reactor-spring-configuration</artifactId>
            <version>3.0.3</version>
            <scope>compile</scope>
        </dependency>

        <dependency>
            <groupId>com.wnhz.ssc.cloud.authority.api</groupId>
            <artifactId>ssc-cloud-authority-api</artifactId>
            <version>0.0.1-SNAPSHOT</version>
        </dependency>


    </dependencies>

    <build>
        <plugins>
            <plugin>
                <groupId>org.springframework.boot</groupId>
                <artifactId>spring-boot-maven-plugin</artifactId>
                <configuration>
                    <outputDirectory>../out</outputDirectory>
                </configuration>
            </plugin>
        </plugins>
    </build>
</project>

配置

bootstrap.yml
spring:
  main:
    web-application-type: reactive
  application:
    name: ssc-cloud-gateway
  cloud:
    nacos:
      discovery:
        server-addr: 192.168.201.81:7777
        namespace: ssc-cloud-id
        group: DEV
        register-enabled: true
      config:
        enabled: off

    gateway:
      discovery:
        locator:
          enabled: true
      routes:
        - id : ssc-order-id               #路由编号(自定义)
#          uri: https://www.bilibili.com/   #跳转地址(网页跳转),lb--负载调整(微服务)
          uri: lb://ssc-cloud-older
          predicates:
            - Path=/ssc-older/**       #url 的映射名称(让客户看到的)
          filters:
            - StripPrefix=1
        - id: ssc-paycenter-id               #路由编号(自定义)
          uri: lb://ssc-cloud-paycenter
          predicates:
            - Path=/ssc-pay/**
          filters:
            - StripPrefix=1
        - id: ssc-authority-id               #路由编号(自定义)
          uri: lb://ssc-cloud-authority
          predicates:
            - Path=/ssc-auth/**
          filters:
            - StripPrefix=1

全局过滤器

package com.wnhz.ssc.cloud.gateway.filter;

import com.fasterxml.jackson.databind.ObjectMapper;
import com.wnhz.ssc.cloud.authority.feign.IAuthorityFeign;
import com.wnhz.ssc.common.result.HttpResp;
import com.wnhz.ssc.common.result.RespCode;
import lombok.SneakyThrows;
import lombok.extern.slf4j.Slf4j;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.cloud.gateway.filter.GatewayFilterChain;
import org.springframework.cloud.gateway.filter.GlobalFilter;
import org.springframework.cloud.gateway.filter.NettyWriteResponseFilter;
import org.springframework.context.annotation.Lazy;
import org.springframework.core.Ordered;
import org.springframework.core.io.buffer.DataBuffer;
import org.springframework.http.HttpHeaders;
import org.springframework.http.HttpStatus;
import org.springframework.http.server.reactive.ServerHttpRequest;
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.nio.charset.StandardCharsets;
import java.util.List;
import java.util.Objects;

@Component
@Slf4j
public class SscGlobalGatewayFilter implements GlobalFilter, Ordered {

    @Lazy
    @Autowired
    private IAuthorityFeign authorityFeign;

    private final String LOGIN_TOKEN="ssc_login_token";

    @SneakyThrows
    @Override
    public Mono<Void> filter(ServerWebExchange exchange, GatewayFilterChain chain) {
        ServerHttpRequest request = exchange.getRequest();
        ServerHttpResponse response = exchange.getResponse();
        log.debug("我进入过滤中.....");
        ObjectMapper objectMapper = new ObjectMapper();
        DataBuffer dataBuffer = null;

        String requestURI =request.getURI().getPath() ;
        log.debug("请求uri:{}",requestURI);

       if(requestURI.endsWith("/authority/login")){//客户访问登录业务,直接跳转验证
           return chain.filter(exchange);
       }

       //1. Token不存在,客户未登录状态访问系统
        if(!isLogin(request)) {//如果客户没有登录
            log.debug("客户还没有登录,重定向到登录(baidu)页面");
            String redirectUrl = "https://www.baidu.com";
            log.debug("将客户请求重定向到指定页面: {}", redirectUrl);
            response.getHeaders().set(HttpHeaders.LOCATION, redirectUrl);
            //303状态码表示由于请求对应的资源存在着另一个URI,应使用GET方法定向获取请求的资源
            response.setStatusCode(HttpStatus.SEE_OTHER);
            response.getHeaders().add("Content-Type", "text/plain;charset=UTF-8");
            return response.setComplete();
        }
        //2. token存在,调用验证微服务验证token是否有效
        String token = getToken(request);
        HttpResp block = authorityFeign.verifyToken(token).block();
        if(block.getCode()== RespCode.FAILED.getCode()){ //token验证失败
            response.getHeaders().add("Content-Type", "application/json;charset=UTF-8");
            dataBuffer = response.bufferFactory().wrap(
                    objectMapper.writeValueAsString(
                                    block)
                            .getBytes(StandardCharsets.UTF_8));
            return response.writeWith(Mono.just(dataBuffer));
        }
        return chain.filter(exchange);
    }

    @Override
    public int getOrder() {
        return NettyWriteResponseFilter.WRITE_RESPONSE_FILTER_ORDER - 1;
    }

    /**
     * 客户未登录,没有token
     * @param request
     * @return
     */
    private boolean isLogin(ServerHttpRequest request){
        HttpHeaders headers = request.getHeaders();
        log.debug("header===> {}", headers);
        List<String> loginToken = headers.get(LOGIN_TOKEN);
        if(Objects.nonNull(loginToken)){
            return true;
        }
        return false;
    }

    /**
     * 从请求中获取token
     * @param request
     */
    private String getToken(ServerHttpRequest request){
        HttpHeaders headers = request.getHeaders();
        log.debug("header===> {}", headers);
        List<String> tokens = headers.get(LOGIN_TOKEN);
        String token = tokens.get(0);
        return token;
    }
}

启动类

package com.wnhz.ssc.cloud.gateway;

import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
import org.springframework.cloud.client.discovery.EnableDiscoveryClient;
import reactivefeign.spring.config.EnableReactiveFeignClients;

@SpringBootApplication
@EnableDiscoveryClient
@EnableReactiveFeignClients(basePackages = "com.wnhz.ssc.cloud.authority.feign")
public class GatewayApp {
    public static void main(String[] args) {
        SpringApplication.run(GatewayApp.class);
    }
}

HttpHeaders headers = request.getHeaders();
    log.debug("header===> {}", headers);
    List<String> tokens = headers.get(LOGIN_TOKEN);
    String token = tokens.get(0);
    return token;
}

}


### 启动类

```java
package com.wnhz.ssc.cloud.gateway;

import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
import org.springframework.cloud.client.discovery.EnableDiscoveryClient;
import reactivefeign.spring.config.EnableReactiveFeignClients;

@SpringBootApplication
@EnableDiscoveryClient
@EnableReactiveFeignClients(basePackages = "com.wnhz.ssc.cloud.authority.feign")
public class GatewayApp {
    public static void main(String[] args) {
        SpringApplication.run(GatewayApp.class);
    }
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mfbz.cn/a/399816.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

软考 系统分析师系列知识点之企业信息化规划(1)

所属章节&#xff1a; 第7章. 企业信息化战略与实施 第2节. 企业信息化规划 企业信息化建设是一项长期而艰巨的任务&#xff0c;不可能在短时间内完成。因此&#xff0c;企业信息化建设必然会分解成各个相对独立的项目&#xff0c;在不同时期分别实施&#xff0c;从而建立多个…

1902_野火FreeRTOS教程内核在STM32中用到的2个中断PENDSV和SYSTICK

1902_野火FreeRTOS教程内核在STM32中用到的2个中断PENDSV和SYSTICK 全部学习汇总&#xff1a; g_FreeRTOS: FreeRTOS学习笔记 (gitee.com) 上面是涉及到的源代码&#xff0c;而这次需要分析的就是78、79行的两个中断。首先&#xff0c;需要确认NVIC_SYSPRI2寄存器的作用。 进一…

jquery写组件滑动人机验证组件

jquery组件&#xff0c;虽然 jquery 语法古老&#xff0c;但是写好了用起来真的很爽啊&#xff0c;本文用滑动人机验证给大家做个详细教程&#xff08;直接复制代码就可以用噢o(*&#xffe3;▽&#xffe3;*)ブ&#xff09; 第一步 先看下组件本身 component.js (function() {…

QT中的多线程有什么作用?

概述 在学习QT线程的时候我们首先要知道的是QT的主线程&#xff0c;也叫GUI线程&#xff0c;意如其名&#xff0c;也就是我们程序的最主要的一个线程&#xff0c;主要负责初始化界面并监听事件循环&#xff0c;并根据事件处理做出界面上的反馈。但是当我们只限于在一个主线程上…

Python urllib模块学习

HTTP协议 HTTP 协议&#xff1a;一般指HTTP(超文本传输)协议。 HTTP是为Web浏览器和Web服务器之间的通信而设计的&#xff0c;基于TCP/IP通信协议嘞传递数据。 HTTP消息结构 客户端请求消息 客户端发送一个HTTP请求到服务器的请求消息包括以下格式 请求行(request line)请求…

基于Java开发的个人视频网站的搭建与实现[附源码]

基于Java开发的个人视频网站的搭建与实现[附源码] &#x1f345; 作者主页 央顺技术团队 &#x1f345; 欢迎点赞 &#x1f44d; 收藏 ⭐留言 &#x1f4dd; &#x1f345; 文末获取源码联系方式 &#x1f4dd; &#x1f345; 查看下方微信号获取联系方式 承接各种定制系统 &am…

威来国际教育:留学服务的全新标杆,打造无忧留学体验

在当今全球化日益加深的背景下&#xff0c;留学已成为众多年轻人拓宽视野、提升个人能力的重要手段。随着留学市场的不断扩大和留学目的地的多样化&#xff0c;家庭和学生在选择留学服务时更加注重专业性和全面性。 在这一领域&#xff0c;威来国际教育凭借其深厚的背景和专业的…

mysql 2-20

TEXT类型 枚举类型 SET类型 二进制字符串类型 BLOB类型 注意事项 JSON类型 提取数据 空间类型 选择建议 约束

Spark 离线开发框架设计与实现

一、背景 随着 Spark 以及其社区的不断发展&#xff0c;Spark 本身技术也在不断成熟&#xff0c;Spark 在技术架构和性能上的优势越来越明显&#xff0c;目前大多数公司在大数据处理中都倾向使用 Spark。Spark 支持多种语言的开发&#xff0c;如 Scala、Java、Sql、Python 等。…

创业者看到这3000多箱磁吸春联滞销面临销毁一定要吸取教训!

2月18日&#xff0c;浙江金华一个工厂&#x1f3ed;3000多箱龙年磁吸春联&#xff0c; 因为滞销&#xff0c;加上春联中含有龙元素和日期而面临报废销毁&#xff0c; 造成了数十万的损失以及大量的资源浪费。 —————————— 而引起广泛的社会讨论&#x1f5e3;️&…

《Solidity 简易速速上手小册》第8章:高级 Solidity 概念(2024 最新版)

文章目录 8.1 高级数据类型和结构8.1.1 基础知识解析更深入的理解实际操作技巧 8.1.2 重点案例&#xff1a;构建一个去中心化身份系统案例 Demo&#xff1a;创建去中心化身份系统案例代码DecentralizedIdentityContract.sol 测试和验证拓展案例 8.1.3 拓展案例 1&#xff1a;管…

C++力扣题目 121--买卖股票的最佳时机 122-- 买卖股票的最佳时机II 123--买卖股票的最佳时机III 188--买卖股票的最佳时机IV

121. 买卖股票的最佳时机 力扣题目链接(opens new window) 给定一个数组 prices &#xff0c;它的第 i 个元素 prices[i] 表示一支给定股票第 i 天的价格。 你只能选择 某一天 买入这只股票&#xff0c;并选择在 未来的某一个不同的日子 卖出该股票。设计一个算法来计算你所…

基于FPGA的二维DCT变换和逆变换verilog实现,包含testbench

目录 1.算法运行效果图预览 2.算法运行软件版本 3.部分核心程序 4.算法理论概述 5.算法完整程序工程 1.算法运行效果图预览 数据导入到matlab显示图像 2.算法运行软件版本 vivado2019.2 matlab2022a 3.部分核心程序 timescale 1ns / 1ps // // Company: // Engineer:…

阿里云国际-在阿里云服务器上快速搭建幻兽帕鲁多人服务器

幻兽帕鲁是最近流行的新型生存游戏。该游戏一夜之间变得极为流行&#xff0c;同时在线玩家数量达到了200万。然而&#xff0c;幻兽帕鲁的服务器难以应对大量玩家的压力。为解决这一问题&#xff0c;幻兽帕鲁允许玩家建立专用服务器&#xff0c;其提供以下优势&#xff1a; &am…

如何在Ubuntu部署Emlog,并将本地博客发布至公网可远程访问

文章目录 前言1. 网站搭建1.1 Emolog网页下载和安装1.2 网页测试1.3 cpolar的安装和注册 2. 本地网页发布2.1 Cpolar临时数据隧道2.2.Cpolar稳定隧道&#xff08;云端设置&#xff09;2.3.Cpolar稳定隧道&#xff08;本地设置&#xff09; 3. 公网访问测试总结 前言 博客作为使…

使用Docker部署Docker-Compose-Ui工具并实现公网访问

文章目录 1. 安装Docker2. 检查本地docker环境3. 安装cpolar内网穿透4. 使用固定二级子域名地址远程访问 Docker Compose UI是Docker Compose的web界面。这个项目的目标是在Docker Compose之上提供一个最小的HTTP API&#xff0c;同时保持与Docker Compose CLI的完全互操作性。…

十大顶级电脑分区恢复软件,不用重装系统直接分区磁盘

与分区相关的问题总是令人不愉快&#xff0c;但解决它们并不像看起来那么困难。您只需要使用可用的最佳分区恢复软件&#xff0c;例如本文列出的 10 种解决方案。配备功能强大的分区数据恢复软件&#xff0c;无论分区损坏有多严重&#xff0c;您都应该能够立即恢复数据。 我们如…

【git 使用】使用 git rebase -i 修改任意的提交信息/合并多个提交

修改最近一次的提交信息的方法有很多&#xff0c;可以参考这篇文章&#xff0c;但是对于之前的提交信息进行修改只能使用 rebase。 修改提交信息 假设我们想修改下面这个提交信息&#xff0c;想把【登录】改成【退出登录】步骤如下 运行 git rebase -i head~3 打开了一个文本…

数据结构与算法:栈

朋友们大家好啊&#xff0c;在链表的讲解过后&#xff0c;我们本节内容来介绍一个特殊的线性表&#xff1a;栈&#xff0c;在讲解后也会以例题来加深对本节内容的理解 栈 栈的介绍栈进出栈的变化形式 栈的顺序存储结构的有关操作栈的结构定义与初始化压栈操作出栈操作获取栈顶元…

Fluter学习3 - Dart 空安全

Dart 空安全&#xff1a; 空类型操作符 (?)空值合并操作符 (??)空值断言操作符 (!)延迟初始化 (late) 1、空类型操作符 (?) 当你想要根据一个表达式是否为 null 来执行某个操作时&#xff0c;你可以使用 (?)语法&#xff1a;expression1?.expression2如果 expression1…
最新文章