Solución:
La API se ha actualizado. No estoy seguro de cuándo cambió, pero según Damien Edwards a fines de diciembre, ahora puede hacer esto:
var remoteIpAddress = request.HttpContext.Connection.RemoteIpAddress;
En project.json agregue una dependencia a:
"Microsoft.AspNetCore.HttpOverrides": "1.0.0"
En Startup.cs
, en el Configure()
método agregar:
app.UseForwardedHeaders(new ForwardedHeadersOptions
{
ForwardedHeaders = ForwardedHeaders.XForwardedFor |
ForwardedHeaders.XForwardedProto
});
Y por supuesto:
using Microsoft.AspNetCore.HttpOverrides;
Entonces, podría obtener la ip usando:
Request.HttpContext.Connection.RemoteIpAddress
En mi caso, al depurar en VS siempre obtuve IpV6 localhost, pero cuando lo implementé en un IIS siempre obtuve la IP remota.
Algunos enlaces útiles: ¿Cómo obtengo la dirección IP del cliente en ASP.NET CORE? y RemoteIpAddress siempre es nulo
los ::1
es quizás debido a:
Terminación de las conexiones en IIS, que luego reenvía a Kestrel, el servidor web v.next, por lo que las conexiones al servidor web son de hecho desde localhost. (https://stackoverflow.com/a/35442401/5326387)
Se puede agregar alguna lógica de respaldo para manejar la presencia de un Load Balancer.
Además, a través de la inspección, el X-Forwarded-For
El encabezado se establece de todos modos incluso sin un Load Balancer (¿posiblemente debido a una capa adicional de 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);
}
Asume _httpContextAccessor
se proporcionó a través de DI.