当前位置: 首页 > ai >正文

基本IP保护 Swagger UI 的中间件

实现说明

  1. 配置文件(appsettings.json)

    • 集中管理 Swagger 相关配置,包括允许访问的 IP 列表、文档标题和版本
    • 支持环境特定配置(如 appsettings.Development.json),方便多环境部署
  2. SwaggerSettings 类

    • 强类型绑定配置文件中的 Swagger 节点
    • 提供类型安全的配置访问,避免硬编码字符串键
  3. SwaggerIPFilterMiddleware 中间件

    • 自动拦截 Swagger 相关路径(/swagger 和 /swagger-ui)
    • 支持获取真实客户端 IP(包括反向代理场景)
    • 处理 IP 格式清洗(移除端口号,兼容 IPv4 和 IPv6)
    • 结合日志记录未授权访问尝试
  4. Program.cs 配置

    • 启用配置文件热重载(reloadOnChange: true)
    • 通过 IOptions 模式注入配置,自动感知配置变化
    • 提供扩展方法简化中间件注册

使用方法

  1. 修改 appsettings.json 中的 Swagger:AllowedIPs 添加允许访问的 IP
  2. 无需重启应用,配置会自动生效
  3. 非允许 IP 访问 Swagger 时会收到 403 禁止访问响应。

1、Program.cs

using SwaggerIpFilterDemo;var builder = WebApplication.CreateBuilder(args);// 配置:启用配置文件热重载
builder.Configuration.SetBasePath(builder.Environment.ContentRootPath).AddJsonFile("appsettings.json", reloadOnChange: true, optional: false).AddJsonFile($"appsettings.{builder.Environment.EnvironmentName}.json", reloadOnChange: true, optional: true).AddEnvironmentVariables();// 注册Swagger配置
builder.Services.Configure<SwaggerSettings>(builder.Configuration.GetSection("Swagger"));// 添加控制器服务
builder.Services.AddControllers();// 添加Swagger服务
builder.Services.AddSwaggerGen(c =>
{var swaggerSettings = builder.Configuration.GetSection("Swagger").Get<SwaggerSettings>();c.SwaggerDoc(swaggerSettings.Version, new() { Title = swaggerSettings.Title, Version = swaggerSettings.Version });
});var app = builder.Build();// 开发环境启用详细异常页
if (app.Environment.IsDevelopment())
{app.UseDeveloperExceptionPage();
}// 注册Swagger IP过滤中间件(必须在UseSwagger之前)
app.UseSwaggerIPFilter();// 启用Swagger
app.UseSwagger();
app.UseSwaggerUI(c =>
{var swaggerSettings = app.Services.GetRequiredService<IOptions<SwaggerSettings>>().Value;c.SwaggerEndpoint($"/swagger/{swaggerSettings.Version}/swagger.json", swaggerSettings.Title);
});// 其他中间件
app.UseHttpsRedirection();
app.UseRouting();
app.UseAuthorization();app.MapControllers();app.Run();

2、SwaggerIPFilterMiddleware.cs

namespace SwaggerIpFilterDemo;/// <summary>
/// Swagger IP访问过滤中间件
/// </summary>
public class SwaggerIPFilterMiddleware
{private readonly RequestDelegate _next;private readonly IOptions<SwaggerSettings> _swaggerSettings;private readonly ILogger<SwaggerIPFilterMiddleware> _logger;/// <summary>/// 构造函数/// </summary>public SwaggerIPFilterMiddleware(RequestDelegate next,IOptions<SwaggerSettings> swaggerSettings,ILogger<SwaggerIPFilterMiddleware> logger){_next = next;_swaggerSettings = swaggerSettings;_logger = logger;}/// <summary>/// 中间件执行逻辑/// </summary>public async Task InvokeAsync(HttpContext context){// 只拦截Swagger相关请求if (context.Request.Path.StartsWithSegments("/swagger") || context.Request.Path.StartsWithSegments("/swagger-ui")){// 获取客户端真实IP(支持反向代理场景)var clientIp = GetClientIpAddress(context);// 处理IP格式(移除端口号)var cleanIp = CleanIpAddress(clientIp);// 检查是否为允许的IPvar allowedIps = _swaggerSettings.Value.AllowedIPs;var isAllowed = allowedIps.Contains(cleanIp) || cleanIp == "127.0.0.1" || cleanIp == "::1";if (!isAllowed){_logger.LogWarning("Swagger访问被拒绝,IP: {Ip}", cleanIp);context.Response.StatusCode = StatusCodes.Status403Forbidden;await context.Response.WriteAsync($"禁止访问Swagger文档,您的IP: {cleanIp}");return;}}// 允许访问,继续处理后续中间件await _next(context);}/// <summary>/// 获取客户端真实IP(支持反向代理)/// </summary>private string GetClientIpAddress(HttpContext context){// 优先从反向代理头获取(如Nginx/Apache设置的X-Forwarded-For)var forwardedFor = context.Request.Headers["X-Forwarded-For"].FirstOrDefault();if (!string.IsNullOrEmpty(forwardedFor)){return forwardedFor.Split(',', StringSplitOptions.TrimEntries).First();}// 直接从连接获取IPreturn context.Connection.RemoteIpAddress?.ToString() ?? "未知IP";}/// <summary>/// 清理IP地址(移除端口号)/// </summary>private string CleanIpAddress(string ip){if (string.IsNullOrEmpty(ip)) return ip;// 处理带端口的IP(如 "192.168.1.100:5000" → "192.168.1.100")if (ip.Contains(':')){// 区分IPv6(如"fe80::1:2:3:4:5")和带端口的IPv4var parts = ip.Split(':');return parts.Length > 2 ? ip : parts[0];}return ip;}
}/// <summary>
/// 中间件扩展方法,简化注册
/// </summary>
public static class SwaggerIPFilterMiddlewareExtensions
{public static IApplicationBuilder UseSwaggerIPFilter(this IApplicationBuilder builder){return builder.UseMiddleware<SwaggerIPFilterMiddleware>();}
}

3、SwaggerSettings.cs

namespace SwaggerIpFilterDemo;/// <summary>
/// Swagger配置选项
/// </summary>
public class SwaggerSettings
{/// <summary>/// 允许访问Swagger的IP列表/// </summary>public List<string> AllowedIPs { get; set; } = new List<string>();/// <summary>/// 文档标题/// </summary>public string Title { get; set; } = "API文档";/// <summary>/// 文档版本/// </summary>public string Version { get; set; } = "v1";
}

4、appsettings.json

{"Logging": {"LogLevel": {"Default": "Information","Microsoft.AspNetCore": "Warning"}},"AllowedHosts": "*","Swagger": {"AllowedIPs": ["127.0.0.1",    // 本地IPv4"::1",          // 本地IPv6"192.168.1.100","192.168.1.101"],"Title": "API文档","Version": "v1"}
}

http://www.xdnf.cn/news/19572.html

相关文章:

  • 解决Elasticsearch高亮显示被横线截断的问题
  • 【音视频】WebRTC-NACK
  • sql执行过程
  • WordPress.com 和 WordPress.org 之间的区别说明
  • 大批量文件管理操作的linux与windows系统命令行终端命令
  • SpringMVC —— Spring集成web环境和SpringMVC快速入门
  • 腾讯混元翻译模型Hunyuan-MT-7B开源:小参数量大能量,获得30项国际冠军
  • Windows---DWORD与IPVOID
  • Sentinel vs Resilience4j vs Bucket4j:分布式限流方案对比与实战
  • 【音视频】VP8 与 VP9 技术详解及与 H.264 H.265 的对比
  • 扩散模型驱动的智能设计与制造:下一场工业革命?
  • idea创建类时自动添加文档注释
  • 【技术教程】如何将文档编辑器集成至基于Node.js的网页应用程序中
  • ESLint 相关
  • 单北斗GNSS位移监测技术解析
  • 网络通信IP细节
  • 企业级架构师综合能力项目案例二(项目性能优化方案JVM+数据库+缓存+代码JUC+消息中间件架构+服务熔断降级)
  • 想找Gamma的平替?这几款AI PPT工具值得试试
  • 设计模式:命令模式(Command Pattern)
  • 从 “容器保姆” 到 “云原生王者”:K8s 全方位指南
  • 并发编程——13 线程池ThreadPoolExecutor实战及其原理分析
  • 顶级科学家的AI使用指南:从工具到合作伙伴
  • 华清远见25072班I/O学习day3
  • Redis分层缓存
  • DELPHI 利用OpenSSL实现加解密,证书(X.509)等功能
  • 犀牛派A1上使用Faster Whisper完成音频转文字
  • 哈尔滨云前沿服务器托管与租用服务
  • 科普:为什么在开发板上运行 Qt 程序时需要在命令后加 -platform linuxfb
  • Linux文本处理工具完全指南:cut、sort、uniq、tr、sed与awk详解
  • odps链接表并预测出现程序阻塞导致任务未完成问题排查