.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,需要显式启用转发头解析机制。

相关推荐
雪可问春风16 小时前
docker环境部署
运维·docker·容器
lwx91485216 小时前
Linux-Shell算术运算
linux·运维·服务器
翻斗包菜16 小时前
PostgreSQL 日常维护完全指南:从基础操作到高级运维
运维·数据库·postgresql
somi716 小时前
ARM-驱动-02-Linux 内核开发环境搭建与编译
linux·运维·arm开发
海的透彻16 小时前
nginx启动进程对文件的权限掌控
运维·chrome·nginx
路溪非溪17 小时前
Linux驱动开发中的常用接口总结(一)
linux·运维·驱动开发
此刻觐神17 小时前
IMX6ULL开发板学习-01(Linux文件目录和目录相关命令)
linux·服务器·学习
航Hang*17 小时前
第3章:Linux系统安全管理——第2节:部署代理服务
linux·运维·服务器·开发语言·笔记·系统安全
fengfuyao98517 小时前
VC++基于服务器的点对点文件传输实例
服务器·开发语言·c++
favour_you___17 小时前
epoll惊群问题与解决
服务器·网络·tcp/ip·epoll