当前位置:首页 > 技术 > 正文内容

微服务网关限流与鉴权

访客 技术 2026年5月26日 3

微服务网关设计与实现

  1. 微服务网关概述

在分布式系统中,不同微服务通常运行在不同的网络地址上。如果客户端直接调用这些服务的接口,会面临以下问题:

  • 客户端需要多次请求多个微服务,增加了复杂性。
  • 可能存在跨域请求问题。
  • 认证逻辑分散在每个微服务中,难以统一管理。
  • 随着项目迭代,重构微服务架构变得困难。

这些问题可以通过引入网关来解决。网关作为客户端和服务器之间的中间层,可以集中处理安全、性能监控和认证授权等功能。

1.1 网关技术选型

常见的网关技术包括:

  • Nginx: 高性能HTTP和反向代理服务器。
  • Zuul: Netflix开发的基于JVM的路由和服务负载均衡器。
  • Spring Cloud Gateway: Spring生态系统中的新一代网关解决方案,支持路径重写和断路器功能。
  1. Spring Cloud Gateway 实现

2.1 环境搭建

以下是基于Spring Cloud Gateway构建微服务网关的基本步骤:

  1. 创建Maven工程,并添加依赖项:
<dependency>
    <groupId>org.springframework.cloud</groupId>
    <artifactId>spring-cloud-starter-gateway</artifactId>
</dependency>
<dependency>
    <groupId>org.springframework.cloud</groupId>
    <artifactId>spring-cloud-starter-netflix-hystrix</artifactId>
</dependency>
<dependency>
    <groupId>org.springframework.cloud</groupId>
    <artifactId>spring-cloud-starter-netflix-eureka-client</artifactId>
</dependency>
  1. 编写启动类:
@SpringBootApplication
@EnableEurekaClient
public class GatewayApplication {
    public static void main(String[] args) {
        SpringApplication.run(GatewayApplication.class, args);
    }
}
  1. 配置文件:
spring:
  application:
    name: sysgateway
  cloud:
    gateway:
      routes:
      - id: goods
        uri: lb://goods
        predicates:
        - Path=/goods/**
        filters:
        - StripPrefix=1
server:
  port: 9101
eureka:
  client:
    service-url:
      defaultZone: http://127.0.0.1:6868/eureka

2.2 跨域支持

通过修改application.yml文件,可以轻松实现跨域支持:

spring:
  cloud:
    gateway:
      globalcors:
        cors-configurations:
          '[/**]':
            allowedOrigins: "*"
            allowedMethods:
            - GET
            - POST
            - PUT
            - DELETE

2.3 自定义过滤器

可以通过自定义过滤器实现IP黑白名单、URL拦截等功能:

@Component
public class IpFilter implements GlobalFilter, Ordered {

    @Override
    public Mono<Void> filter(ServerWebExchange exchange, GatewayFilterChain chain) {
        ServerHttpRequest request = exchange.getRequest();
        InetSocketAddress address = request.getRemoteAddress();
        System.out.println("Request IP: " + address.getHostName());
        return chain.filter(exchange);
    }

    @Override
    public int getOrder() {
        return 1;
    }
}

@Component
public class UrlFilter implements GlobalFilter, Ordered {

    @Override
    public Mono<Void> filter(ServerWebExchange exchange, GatewayFilterChain chain) {
        String url = exchange.getRequest().getURI().getPath();
        System.out.println("Accessing URL: " + url);
        return chain.filter(exchange);
    }

    @Override
    public int getOrder() {
        return 2;
    }
}
  1. 网关限流

3.1 限流算法简介

常用的限流算法有令牌桶算法。其核心思想是通过控制令牌生成速率限制请求频率。

3.2 基于Redis的限流实现

以下是基于Redis的限流配置:

  1. 添加Redis依赖:
<dependency>
    <groupId>org.springframework.boot</groupId>
    <artifactId>spring-boot-starter-data-redis-reactive</artifactId>
</dependency>
  1. 定义KeyResolver:
@Bean
public KeyResolver ipKeyResolver() {
    return exchange -> Mono.just(exchange.getRequest().getRemoteAddress().getHostName());
}
  1. 更新application.yml:
spring:
  cloud:
    gateway:
      routes:
      - id: goods
        uri: lb://goods
        predicates:
        - Path=/goods/**
        filters:
        - name: RequestRateLimiter
          args:
            key-resolver: "#{@ipKeyResolver}"
            redis-rate-limiter.replenishRate: 1
            redis-rate-limiter.burstCapacity: 1
  1. BCrypt 密码加密

BCrypt是一种安全的密码加密算法,支持加盐机制防止彩虹表攻击。

// 加密示例
String hashedPassword = BCrypt.hashpw("123456", BCrypt.gensalt());

// 校验示例
boolean matches = BCrypt.checkpw("123456", hashedPassword);
System.out.println("Password matches: " + matches);
  1. JWT 实现微服务鉴权

JSON Web Token (JWT) 是一种轻量级的安全信息传递标准,广泛用于微服务鉴权。

5.1 JWT 构造与解析

一个JWT由三部分组成:头部、载荷和签名。

// 创建Token
String token = Jwts.builder()
    .setId("123")
    .setSubject("user")
    .setIssuedAt(new Date())
    .signWith(SignatureAlgorithm.HS256, "secret")
    .compact();

// 解析Token
Claims claims = Jwts.parser()
    .setSigningKey("secret")
    .parseClaimsJws(token)
    .getBody();

5.2 网关鉴权实现

通过自定义过滤器验证JWT令牌:

@Component
public class AuthFilter implements GlobalFilter {

    @Override
    public Mono<Void> filter(ServerWebExchange exchange, GatewayFilterChain chain) {
        String token = exchange.getRequest().getHeaders().getFirst("Authorization");
        if (token == null || !isValidToken(token)) {
            exchange.getResponse().setStatusCode(HttpStatus.UNAUTHORIZED);
            return exchange.getResponse().setComplete();
        }
        return chain.filter(exchange);
    }

    private boolean isValidToken(String token) {
        try {
            Jwts.parser().setSigningKey("secret").parseClaimsJws(token).getBody();
            return true;
        } catch (Exception e) {
            return false;
        }
    }
}

相关文章

Linux crontab 详解

1) crontab 是什么cron 是 Linux 的定时任务守护进程;crontab 是用来编辑/查看“按时间周期执行命令”的表(cron table)。常见两类:用户 crontab:每个用户一份(crontab -e 编辑)系统级 crontab / cron.d:可指定执行用户(/etc/crontab、/etc/cron.d/*)2) crontab 时间...

富文本里可以允许的 HTML 属性

一、所有标签默认允许的安全属性(极少)class        (可选)id           (通常建议禁用)title️ 注意:id 容易被滥用做锚点注入,很多系统直接禁用class 允许的话最好只允许固定前缀(如 editor-*)二、a 标签允许属性<a href="" t...

Mac 安装 Node.js 指南

方法一:通过官网安装包(最简单,适合初学者)如果你只是想快速安装并开始使用,这是最直接的方法。访问 Node.js 官网。页面会显示两个版本:LTS (Recommended For Most Users):长期支持版,最稳定。建议选这个。Current:最新特性版,包含最新功能但可能不够稳定。下载 .pkg 安装包并运行。按照安装向导点击“下一步”即可完成。方法二:使用 Homebrew 安装(...

Dom\HTML_NO_DEFAULT_NS 的副作用:自动加闭合标签

在使用Dom\HTMLDocument时,Dom\HTML_NO_DEFAULT_NS 将禁止在解析过程中设置元素的命名空间, 此设置是为了与DOMDocument向后兼容而存在的。当使用它时,已知的一个副作用就是:自动加闭合标签例如 </img> 为什么会这样?当你使用:Dom\HTML_NO_DEFAULT_NS文档会变成 无命名空间模式,此时内部更接近 XML...

Laravel 事件和监听器创建

在 Laravel 中,使用 Artisan 命令创建 Events(事件) 和 Listeners(监听器) 是非常高效的。你可以通过以下几种方式来实现:1. 手动创建单个 Event如果你只想创建一个事件类,可以使用 make:event 命令:Bashphp artisan make:event UserRegistered执行后,文件将生成在 app/Even...

自定义域名解析神器 dnsmasq

什么是 dnsmasq?dnsmasq 是一个轻量级、功能强大的网络服务工具,专为小型和中等规模网络设计。它是一个综合的网络基础设施解决方案[1]。dnsmasq 能做什么?功能说明应用场景DNS 转发与缓存将 DNS 查询转发到上游服务器(ISP、Google DNS 等),并在本地缓存结果加快 DNS 查询速度,减少外部 DNS 流量本地 DNS解析本地网络设备的主机名,无需编辑&n...

发表评论

访客

◎欢迎参与讨论,请在这里发表您的看法和观点。