Jak uzyskać adres IP klienta w rdzeniu ASP.NET?

205

Czy możesz dać mi znać, jak uzyskać adres IP klienta w ASP.NET, gdy używasz MVC 6. Request.ServerVariables["REMOTE_ADDR"]nie działa.

eadam
źródło
Przykład:httpContext.GetFeature<IHttpConnectionFeature>().RemoteIpAddress
Kiran Challa
Przyczyna wstrzymania jest dostępna u dołu pytania: „Istnieje zbyt wiele możliwych odpowiedzi lub dobre odpowiedzi byłyby zbyt długie dla tego formatu. Dodaj szczegóły, aby zawęzić zestaw odpowiedzi lub wyodrębnić problem, na który można odpowiedzieć w kilka akapitów. ”
Kevin Brown
12
Nie sądzę, żeby odpowiedź była wystarczająco długa. To proste, co robimy w starszych wersjach MVC. Właśnie zapytałem, jak to zrobić w najnowszej wersji beta. Po drugie, jeśli jest wiele odpowiedzi, dlaczego nikt nie wskaże mi JEDNEJ z nich. Rzeczywiście, MVC 6 to wersja beta i myślę, że nawet nie przeczytali pytania poprawnie przed jego trzymaniem.
eadam
Edytuj swoje pytanie, pokazując więcej tego, czego próbowałeś i wyraźne dane wejściowe / wyjściowe. nie narzekaj tylko, że twoje pytanie zostało zamknięte bez przynajmniej próby jego poprawy.
10
Tylko programiści z tej samej domeny powinni zamknąć pytanie. Każdy, kto pracuje w asp.net 5.0, po prostu będzie wiedział, o co proszę w ułamku sekundy. Reszta z was karze mnie tylko za proste pytanie.
eadam

Odpowiedzi:

275

Interfejs API został zaktualizowany. Nie jestem pewien, kiedy to się zmieniło, ale według Damiena Edwardsa pod koniec grudnia możesz teraz to zrobić:

var remoteIpAddress = request.HttpContext.Connection.RemoteIpAddress;
David Peden
źródło
9
To RemoteIpAddressjest zawsze nulldla mnie, kiedy publikuję stronę internetową w IIS i loguję to do pliku.
A-Sharabiani
3
Wygląda na to, że problem został naprawiony w rc 2
Jon
12
Zawsze dostaję 127.0.0.1, nawet gdy łączę się zdalnie
frostymarvelous
33
Zwraca mi to „:: 1”, czyli format IPv6. Jak inni widzą 127.0.0.1?
Derek Greer,
4
Czy ktoś jeszcze zwraca lokalny adres IP swojego serwera IIS?
dave317
73

W project.json dodaj zależność do:

"Microsoft.AspNetCore.HttpOverrides": "1.0.0"

W Startup.csw Configure()metodzie dodatku:

  app.UseForwardedHeaders(new ForwardedHeadersOptions
        {
            ForwardedHeaders = ForwardedHeaders.XForwardedFor |
            ForwardedHeaders.XForwardedProto
        });  

I oczywiście:

using Microsoft.AspNetCore.HttpOverrides;

Następnie mogę uzyskać adres IP za pomocą:

Request.HttpContext.Connection.RemoteIpAddress

W moim przypadku podczas debugowania w VS zawsze otrzymywałem localhost IpV6, ale po wdrożeniu na IIS zawsze otrzymywałem zdalne IP.

Kilka przydatnych linków: Jak uzyskać adres IP klienta w rdzeniu ASP.NET? a adres RemoteIpAddress jest zawsze zerowy

::1Jest być może ze względu na:

Zakończenie połączeń w usługach IIS, które następnie przekazują do Kestrel, serwera sieci web v.next, więc połączenia z serwerem sieciowym faktycznie pochodzą z hosta lokalnego. ( https://stackoverflow.com/a/35442401/5326387 )

Johna
źródło
6
To jest prawidłowa odpowiedź, która jest również udokumentowana w oficjalnej dokumentacji dotyczącej odwrotnych serwerów proxy: docs.microsoft.com/en-us/aspnet/core/host-and-deploy/...
Melvyn
8
należy zaznaczyć, że przed użyciem aplikacji UseAuthentication () należy dodać „app. linia, jeśli używasz
wcięcia
1
Działa to doskonale i przetestowałem na lokalnie hostowanych usługach IIS i na platformie Azure. Działa w obu miejscach.
ThomasCle
72

Można dodać pewną logikę zastępczą, aby obsłużyć obecność modułu równoważenia obciążenia.

Ponadto, po sprawdzeniu, X-Forwarded-Fornagłówek i tak jest ustawiany nawet bez modułu równoważenia obciążenia (być może z powodu dodatkowej warstwy pustki?):

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);
}

Zakłada się, że _httpContextAccessordostarczono przez DI.

Crokusek
źródło
2
To jest właściwa odpowiedź. Nie ma jednego sposobu na odzyskanie adresu IP, szczególnie gdy twoja aplikacja znajduje się za Nginx, modułem równoważenia obciążenia lub czymś podobnym. Dzięki!
Feu
@feu, ponieważ AspNetCore może zastąpić właściwość RemoteIpAddress z nagłówkami takimi jak X-Forwarded-For.
Mark Lopez
@crokusek ... próbuje dostosować twoje rozwiązanie, ale VS zmusza mnie do klasy enkapsulacji tego kodu statycznie. Czy masz ten kod w swoim projekcie aplikacji internetowej lub w bibliotece klas w rozwiązaniu?
dinotom
Pierwsze 2 metody powinny znajdować się w instancji, która udostępnia __httpContextAccessor (lub ją dostosowuje). Drugie 2-łańcuchowe metody zostały pobrane z osobnej klasy rozszerzeń statycznych.
crokusek
To dobre rozwiązanie, zwłaszcza gdy aplikacja korzysta z Kestrel i jest hostowana z Nginx w systemie Linux.
Timothy Macharia,
16

Możesz użyć IHttpConnectionFeaturedo uzyskania tej informacji.

var remoteIpAddress = httpContext.GetFeature<IHttpConnectionFeature>()?.RemoteIpAddress;
Kiran Challa
źródło
Dziękuję za odpowiedź!
eadam
2
Czy to działa w przypadku hostingu Kestrel? W moim demo httpContext.GetFeature<IHttpConnectionFeature>()zawsze bądź null.
Jerry Bian
2
@JerryBian zgodnie z tym dokumentem: github.com/aspnet/Docs/blob/master/aspnet/fundamentals/… , IHttpConnectionFeature nie jest jeszcze obsługiwany w Kestrel (jeszcze).
qbik
@JerryBian jest teraz
David Peden
Musiał być przestarzały - wersja @feradz działa dla mnie w RC-1
fiat
12
var remoteIpAddress = HttpContext.Features.Get<IHttpConnectionFeature>()?.RemoteIpAddress;
feradz
źródło
6
Zbyt skomplikowane. MVC już to wywołuje wewnętrznie i poddaje HttpContext.Connection.RemoteIpAddress.
Fred
@Fred - twoja wersja zwraca mi wartość null w RC-1 - IIS i Kestrel
fiat
7

To działa dla mnie (DotNetCore 2.1)

[HttpGet]
public string Get() 
{
    var remoteIpAddress = HttpContext.Connection.RemoteIpAddress;
    return remoteIpAddress.ToString();
}
MC9000
źródło
7

W ASP.NET 2.1, w StartUp.cs Dodaj te usługi:

services.AddHttpContextAccessor();
services.TryAddSingleton<IActionContextAccessor, ActionContextAccessor>();

a następnie wykonaj 3 kroki:

  1. Zdefiniuj zmienną w kontrolerze MVC

    private IHttpContextAccessor _accessor;
  2. DI do konstruktora kontrolera

    public SomeController(IHttpContextAccessor accessor)
    {
        _accessor = accessor;
    }
  3. Pobierz adres IP

    _accessor.HttpContext.Connection.RemoteIpAddress.ToString()

Tak to się robi.

hojjat.mi
źródło
2
To daje mi :: 1. Asp.Net Core 2.2., Na hoście lokalnym.
Stian
::1jest hostem lokalnym w IPv6. Odpowiednik IPv4127.0.0.1
Andy
3

W moim przypadku aplikacja internetowa DotNet Core 2.2 działa na DigitalOcean z dokerem i nginx jako odwrotnym proxy. Dzięki temu kodowi w Startup.cs mogę uzyskać adres IP klienta

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 był ip, który otrzymywałem przed użyciem

Request.HttpContext.Connection.RemoteIpAddress.ToString();
gorums
źródło
2

Najpierw w .Net Core 1.0 Dodaj using Microsoft.AspNetCore.Http.Features;do kontrolera Następnie w odpowiedniej metodzie:

var ip = HttpContext.Features.Get<IHttpConnectionFeature>()?.RemoteIpAddress?.ToString();

Przeczytałem kilka innych odpowiedzi, których nie udało się skompilować, ponieważ używałem małych liter httpContext, co spowodowało, że VS dodał, używając Microsoft.AspNetCore.Http, zamiast odpowiedniego użycia lub z HttpContext (kompilator również wprowadza w błąd).

Chłopak
źródło
2

Uzyskaj adres IP i nazwę hosta w rdzeniu .net

umieść ten kod w kontrolerze

Wykonaj następujące kroki:

var addlist = Dns.GetHostEntry(Dns.GetHostName());
string GetHostName = addlist.HostName.ToString();
string GetIPV6 = addlist.AddressList[0].ToString();
string GetIPV4 = addlist.AddressList[1].ToString();
Shreyash Banawala
źródło
2

Odkryłem, że niektórzy z was stwierdzili, że otrzymany adres IP to ::: 1 lub 0.0.0.1

Jest to problem, ponieważ próbujesz uzyskać adres IP z własnego komputera i zamieszanie w C #, które próbują zwrócić IPv6.

Tak więc implementuję odpowiedź z @Johna ( https://stackoverflow.com/a/41335701/812720 ) i @David ( https://stackoverflow.com/a/8597351/812720 ), dzięki nim!

i tutaj do rozwiązania:

  1. dodaj pakiet Microsoft.AspNetCore.HttpOverrides w swoich referencjach (zależności / pakiety)

  2. dodaj tę linię w 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
    
    }
  3. aby uzyskać adres IP, użyj tego kodu w dowolnym pliku 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();
    }

a teraz możesz uzyskać adres IPv4 ze zdalnego adresu IP lub wyniku

Sruit A.Suk
źródło
1

Spróbuj tego.

var host = Dns.GetHostEntry(Dns.GetHostName());
        foreach (var ip in host.AddressList)
        {
            if (ip.AddressFamily == AddressFamily.InterNetwork)
            {
                 ipAddress = ip.ToString();
            }
        }
Kunal Burangi
źródło
0

Uruchomienie .NET core(3.1.4) IISza modułem równoważenia obciążenia nie działało z innymi sugerowanymi rozwiązaniami.

Czyta ręcznie X-Forwarded-Fornagłówek.

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;
}

Używam autonomiczny pakiet użyciu pakiet hostingu .

Aage
źródło
0

Uruchamiając ASP.NET Core 2.1 za odwrotnym proxy Traefik na Ubuntu, muszę ustawić adres IP bramy KnownProxiespo zainstalowaniu oficjalnego Microsoft.AspNetCore.HttpOverridespakietu

        var forwardedOptions = new ForwardedHeadersOptions {
            ForwardedHeaders = ForwardedHeaders.XForwardedFor,
        };
        forwardedOptions.KnownProxies.Add(IPAddress.Parse("192.168.3.1"));
        app.UseForwardedHeaders(forwardedOptions);

Zgodnie z dokumentacją jest to wymagane, jeśli odwrotne proxy nie działa na localhost. docker-compose.ymlZ Traefik ma przypisany statyczny adres IP:

networks:
  my-docker-network:
    ipv4_address: 192.168.3.2

Ewentualnie powinno wystarczyć upewnienie się, że zdefiniowano tutaj znaną sieć, aby określić jej bramę w .NET Core.

Lew
źródło