.NET6 解决 Nginx 反向代理后获取客户端真实 IP(始终显示网关 IP)问题

假设现在你的服务器结构是这样:

  • 客户端真实公网 IP:36.112.25.88

  • Nginx 网关 IP:192.20.1.1

  • .NET 6 服务部署在本机 5000 端口

你在代码里这样获取 IP:

cs 复制代码
var ip = HttpContext.Connection.RemoteIpAddress?.ToString();

结果打印出来的永远是

192.20.1.1

而不是

36.112.25.88

一、为什么会出现这种现象?

因为在反向代理结构下:

  • .NET 接收到的请求是来自 Nginx

  • 网络层认为"请求发起者"就是 Nginx

  • 因此 RemoteIpAddress 返回的是网关 IP

换句话说:

反向代理改变了请求的直接来源。

这是一种正常行为,而不是程序错误。


二、真实 IP 在哪里?

Nginx 在转发请求时,会自动把客户端真实 IP 写入请求头:

cs 复制代码
X-Forwarded-For: 36.112.25.88
X-Real-IP: 36.112.25.88

但默认情况下:

ASP.NET Core 不会自动解析这些头。

原因是 HTTP 头可以伪造,框架出于安全考虑默认关闭信任机制。


三、正确解决方案

解决思路:

  1. Nginx 传递真实 IP

  2. .NET 6 显式启用 ForwardedHeaders


第一步:Nginx 反向代理配置

复制代码
cs 复制代码
server {
    listen 80;
    server_name api.test.com;

    location / {
        proxy_pass http://127.0.0.1:5000;

        proxy_set_header Host $host;
        proxy_set_header X-Real-IP $remote_addr;
        proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for;
        proxy_set_header X-Forwarded-Proto $scheme;
    }
}

关键配置:

cs 复制代码
proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for;

第二步:.NET 6 启用 ForwardedHeaders

Program.cs 中配置:

cs 复制代码
using Microsoft.AspNetCore.HttpOverrides;
using System.Net;

var builder = WebApplication.CreateBuilder(args);

builder.Services.Configure<ForwardedHeadersOptions>(options =>
{
    options.ForwardedHeaders =
        ForwardedHeaders.XForwardedFor |
        ForwardedHeaders.XForwardedProto;

    // 指定你的 Nginx 网关 IP
    options.KnownProxies.Add(IPAddress.Parse("192.20.1.1"));
});

var app = builder.Build();

// 必须放在最前面
app.UseForwardedHeaders();

app.MapControllers();

app.Run();

四、配置完成后的结果

请求仍然经过:

cs 复制代码
36.112.25.88 → 192.20.1.1 → .NET

但现在:

  • .NET 会信任来自 192.20.1.1 的转发信息

  • 自动解析 X-Forwarded-For

此时:

cs 复制代码
RemoteIpAddress

获取到的就是:

cs 复制代码
36.112.25.88

五、核心总结

在 Nginx 反向代理结构下:

  • 未配置时 → 获取到的是网关 IP

  • 正确配置后 → 获取到的是真实客户端 IP

本质原因是:

请求的直接来源变成了 Nginx,需要显式启用转发头解析机制。

相关推荐
云边有个稻草人2 小时前
打工人摸鱼新姿势!轻量斗地主服务器,内网穿透让同事远程联机不翻车
运维·服务器·cpolar
阿钱真强道2 小时前
12 JetLinks MQTT直连设备事件上报实战(继电器场景)
linux·服务器·网络·数据库·网络协议
乾元2 小时前
终端安全(EDR):用深度学习识别未知勒索软件
运维·人工智能·网络协议·安全·网络安全·自动化·安全架构
wbs_scy2 小时前
Linux 进阶指令实操指南:文件查看、时间管理、搜索压缩全场景覆盖(附高频案例)
linux·运维·服务器
安科瑞刘鸿鹏172 小时前
高速路灯故障难定位?用 ASL600 实现精确单灯监测与维护预警
运维·网络·物联网·安全
馨谙3 小时前
Ansible 事实(Facts)全面指南:自动化运维中的主机信息管理
运维·ansible
Lethehong3 小时前
实测可用|一文搞定OpenClaw部署,免费kimi-k2.5+飞书远程,新手也能秒上手
linux·运维·服务器·玩转openclaw·云端创意实践
馨谙3 小时前
Ansible处理程序完全指南:实现智能的任务触发机制
运维·ansible
逍遥德3 小时前
Sring事务详解之02.如何使用编程式事务?
java·服务器·数据库·后端·sql·spring
承渊政道3 小时前
Linux系统学习【Linux基础开发工具】
linux·运维·笔记·学习·centos·编辑器