Подскажите пожалуйста, как мне получить IP-адрес клиента в ASP.NET при использовании MVC 6. Request.ServerVariables["REMOTE_ADDR"]
Не работает.
Подскажите пожалуйста, как мне получить IP-адрес клиента в ASP.NET при использовании MVC 6. Request.ServerVariables["REMOTE_ADDR"]
Не работает.
Ответы:
API был обновлен. Не уверен, когда это изменилось, но, по словам Дэмиена Эдвардса в конце декабря, теперь вы можете сделать это:
var remoteIpAddress = request.HttpContext.Connection.RemoteIpAddress;
RemoteIpAddress
всегда null
для меня, когда я публикую его веб-сайт на IIS и регистрирую это в файле.
В project.json добавьте зависимость к:
"Microsoft.AspNetCore.HttpOverrides": "1.0.0"
В Startup.cs
, в Configure()
метод добавить:
app.UseForwardedHeaders(new ForwardedHeadersOptions
{
ForwardedHeaders = ForwardedHeaders.XForwardedFor |
ForwardedHeaders.XForwardedProto
});
И, конечно же:
using Microsoft.AspNetCore.HttpOverrides;
Затем я мог бы получить IP с помощью:
Request.HttpContext.Connection.RemoteIpAddress
В моем случае при отладке в VS я всегда получал локальный хост IpV6, но при развертывании на IIS я всегда получал удаленный IP.
Некоторые полезные ссылки: Как получить IP-адрес клиента в ASP.NET CORE? и RemoteIpAddress всегда равен нулю
Это ::1
может быть из-за:
Завершение соединений в IIS, который затем перенаправляется на Kestrel, веб-сервер v.next, поэтому соединения с веб-сервером действительно выполняются с localhost. ( https://stackoverflow.com/a/35442401/5326387 )
Некоторая резервная логика может быть добавлена для обработки присутствия балансировщика нагрузки.
Кроме того, путем проверки X-Forwarded-For
заголовок в любом случае устанавливается даже без балансировщика нагрузки (возможно, из-за дополнительного слоя Kestrel?):
public string GetRequestIP(bool tryUseXForwardHeader = true)
{
string ip = null;
// todo support new "Forwarded" header (2014) https://en.wikipedia.org/wiki/X-Forwarded-For
// X-Forwarded-For (csv list): Using the First entry in the list seems to work
// for 99% of cases however it has been suggested that a better (although tedious)
// approach might be to read each IP from right to left and use the first public IP.
// http://stackoverflow.com/a/43554000/538763
//
if (tryUseXForwardHeader)
ip = GetHeaderValueAs<string>("X-Forwarded-For").SplitCsv().FirstOrDefault();
// RemoteIpAddress is always null in DNX RC1 Update1 (bug).
if (ip.IsNullOrWhitespace() && _httpContextAccessor.HttpContext?.Connection?.RemoteIpAddress != null)
ip = _httpContextAccessor.HttpContext.Connection.RemoteIpAddress.ToString();
if (ip.IsNullOrWhitespace())
ip = GetHeaderValueAs<string>("REMOTE_ADDR");
// _httpContextAccessor.HttpContext?.Request?.Host this is the local host.
if (ip.IsNullOrWhitespace())
throw new Exception("Unable to determine caller's IP.");
return ip;
}
public T GetHeaderValueAs<T>(string headerName)
{
StringValues values;
if (_httpContextAccessor.HttpContext?.Request?.Headers?.TryGetValue(headerName, out values) ?? false)
{
string rawValues = values.ToString(); // writes out as Csv when there are multiple.
if (!rawValues.IsNullOrWhitespace())
return (T)Convert.ChangeType(values.ToString(), typeof(T));
}
return default(T);
}
public static List<string> SplitCsv(this string csvList, bool nullOrWhitespaceInputReturnsNull = false)
{
if (string.IsNullOrWhiteSpace(csvList))
return nullOrWhitespaceInputReturnsNull ? null : new List<string>();
return csvList
.TrimEnd(',')
.Split(',')
.AsEnumerable<string>()
.Select(s => s.Trim())
.ToList();
}
public static bool IsNullOrWhitespace(this string s)
{
return String.IsNullOrWhiteSpace(s);
}
Предполагается, _httpContextAccessor
был предоставлен через DI.
Вы можете использовать IHttpConnectionFeature
для получения этой информации.
var remoteIpAddress = httpContext.GetFeature<IHttpConnectionFeature>()?.RemoteIpAddress;
httpContext.GetFeature<IHttpConnectionFeature>()
всегда будь null
.
В ASP.NET 2.1, в StartUp.cs добавить эти службы:
services.AddHttpContextAccessor();
services.TryAddSingleton<IActionContextAccessor, ActionContextAccessor>();
а затем сделать 3 шага:
Определите переменную в вашем контроллере MVC
private IHttpContextAccessor _accessor;
DI в конструктор контроллера
public SomeController(IHttpContextAccessor accessor)
{
_accessor = accessor;
}
Получить IP-адрес
_accessor.HttpContext.Connection.RemoteIpAddress.ToString()
Вот как это делается.
::1
это локальный хост в IPv6. Эквивалент IPv4127.0.0.1
В моем случае у меня есть приложение DotNet Core 2.2 Web, работающее на DigitalOcean с докером и nginx в качестве обратного прокси-сервера. С помощью этого кода в Startup.cs я могу получить IP-адрес клиента
app.UseForwardedHeaders(new ForwardedHeadersOptions
{
ForwardedHeaders = ForwardedHeaders.All,
RequireHeaderSymmetry = false,
ForwardLimit = null,
KnownNetworks = { new IPNetwork(IPAddress.Parse("::ffff:172.17.0.1"), 104) }
});
:: ffff: 172.17.0.1 был ip, который я получал перед использованием
Request.HttpContext.Connection.RemoteIpAddress.ToString();
Сначала в .Net Core 1.0 Добавить using Microsoft.AspNetCore.Http.Features;
в контроллер. Затем внутри соответствующего метода:
var ip = HttpContext.Features.Get<IHttpConnectionFeature>()?.RemoteIpAddress?.ToString();
Я прочитал несколько других ответов, которые не удалось скомпилировать, поскольку он использовал строчный httpContext, что привело к добавлению VS с использованием Microsoft.AspNetCore.Http вместо соответствующего использования или с HttpContext (компилятор также вводит в заблуждение).
получить IP-адрес и имя хоста в ядре .net
поместите этот код в контроллер
Следуй этим шагам:
var addlist = Dns.GetHostEntry(Dns.GetHostName());
string GetHostName = addlist.HostName.ToString();
string GetIPV6 = addlist.AddressList[0].ToString();
string GetIPV4 = addlist.AddressList[1].ToString();
Я обнаружил, что некоторые из вас обнаружили, что IP-адрес, который вы получаете: ::: 1 или 0.0.0.1
Это проблема из-за того, что вы пытаетесь получить IP с вашего собственного компьютера, и из-за путаницы в C #, которая пытается вернуть IPv6.
Итак, я реализую ответ от @Johna ( https://stackoverflow.com/a/41335701/812720 ) и @David ( https://stackoverflow.com/a/8597351/812720 ), спасибо им!
а вот к решению:
добавить пакет Microsoft.AspNetCore.HttpOverrides в ваши ссылки (зависимости / пакеты)
добавить эту строку в Startup.cs
public void Configure(IApplicationBuilder app, IHostingEnvironment env)
{
// your current code
// start code to add
// to get ip address
app.UseForwardedHeaders(new ForwardedHeadersOptions
{
ForwardedHeaders = ForwardedHeaders.XForwardedFor | ForwardedHeaders.XForwardedProto
});
// end code to add
}
чтобы получить IP-адрес, используйте этот код в любом из ваших Controller.cs
IPAddress remoteIpAddress = Request.HttpContext.Connection.RemoteIpAddress;
string result = "";
if (remoteIpAddress != null)
{
// If we got an IPV6 address, then we need to ask the network for the IPV4 address
// This usually only happens when the browser is on the same machine as the server.
if (remoteIpAddress.AddressFamily == System.Net.Sockets.AddressFamily.InterNetworkV6)
{
remoteIpAddress = System.Net.Dns.GetHostEntry(remoteIpAddress).AddressList
.First(x => x.AddressFamily == System.Net.Sockets.AddressFamily.InterNetwork);
}
result = remoteIpAddress.ToString();
}
и теперь вы можете получить IPv4-адрес из remoteIpAddress или результат
попробуй это.
var host = Dns.GetHostEntry(Dns.GetHostName());
foreach (var ip in host.AddressList)
{
if (ip.AddressFamily == AddressFamily.InterNetwork)
{
ipAddress = ip.ToString();
}
}
Запуск .NET core
(3.1.4) IIS
за балансировщиком нагрузки не работал с другими предлагаемыми решениями.
Чтение X-Forwarded-For
заголовка вручную .
IPAddress ip;
var headers = Request.Headers.ToList();
if (headers.Exists((kvp) => kvp.Key == "X-Forwarded-For"))
{
// when running behind a load balancer you can expect this header
var header = headers.First((kvp) => kvp.Key == "X-Forwarded-For").Value.ToString();
ip = IPAddress.Parse(header);
}
else
{
// this will always have a value (running locally in development won't have the header)
ip = Request.HttpContext.Connection.RemoteIpAddress;
}
Я запускаю автономный пакет, используя пакет хостинга .
При запуске ASP.NET Core 2.1 за обратным прокси-сервером Traefik в Ubuntu, мне нужно установить IP-адрес шлюза KnownProxies
после установки официального Microsoft.AspNetCore.HttpOverrides
пакета
var forwardedOptions = new ForwardedHeadersOptions {
ForwardedHeaders = ForwardedHeaders.XForwardedFor,
};
forwardedOptions.KnownProxies.Add(IPAddress.Parse("192.168.3.1"));
app.UseForwardedHeaders(forwardedOptions);
Согласно документации , это требуется, если обратный прокси-сервер не работает на localhost. docker-compose.yml
Из Traefik назначил статический IP - адрес:
networks:
my-docker-network:
ipv4_address: 192.168.3.2
В качестве альтернативы, этого должно быть достаточно, чтобы убедиться, что здесь определена известная сеть, чтобы указать ее шлюз в .NET Core.
httpContext.GetFeature<IHttpConnectionFeature>().RemoteIpAddress