Remotely connect to .net core self hosted web api
Asked Answered
F

8

85

I have a simple .net core web api with one action:

[Route("[action]")]
public class APIController : Controller
{
    // GET api/values
    [HttpGet]
    public string Ping()
    {
        return DateTime.Now.ToString();
    }
}

If I run this via dotnet run I get

Hosting environment: Production
Content root path: C:\Users\xxx\Documents\Visual Studio 2015\Projects\SelfHostTest\src\SelfHostTest
Now listening on: http://localhost:5000
Application started. Press Ctrl+C to shut down.

Going to the browser and typing in http://localhost:5000/ping results in a successful return of the current time. However going to a remote machine (same LAN) and trying to access the service via http://odin:5000/ping results in a 404 error. (Odin is the name of the machine running the web api in a console via dotnet run).

Both server (and client!) firewalls are turned off. I can ping "odin" successfully.

Any ideas what simple step I am missing here. I've tried this at home and at work with no success.

Favorable answered 27/9, 2016 at 18:52 Comment(2)
What happens if you try with IP address of 'odin'? Ex: 192.111.11.2:5000 ? May be this can help you.Manisa
same thing tried 192.168.1.2 , both on server and on client. 127.0.0.1 works on the server though!Favorable
D
108

My guess is that the issue isn't in your controller, it is in program.cs. You need to modify the construction of your WebHost

var host = new WebHostBuilder()
.UseKestrel()
.UseContentRoot(Directory.GetCurrentDirectory())
.UseUrls("http://localhost:5000", "http://odin:5000", "http://192.168.1.2:5000")
.UseIISIntegration()
.UseStartup<Startup>()
.Build();

Unless you add the UseUrls line, Kestrel isn't going to listen outside of localhost. This makes sense, because in a normal situation Kestrel will be sitting behind a reverse proxy like IIS or NGNIX and doesn't need to bind to external URLs.

Dyslalia answered 27/9, 2016 at 20:6 Comment(7)
Thanks - this works! Guess this sort of config really needs to be in a config file of some sort though... ?Favorable
Either that or you can use the special ASPNETCORE_URLS environment variable. I wouldn't want to be listening directly with Kestrel in production, though. It really is best to have it sit behind something else.Dyslalia
In addition, you can use .UseUrls("http://*:5000") instead to accept any incoming requests from port 5000!Heroic
How would you implement the WebHostBuilder without Startup since with Dotnet 6 there is not Startup. Do you jsut remove the UseStartup method?Tie
This does not work for me still. I tried both UseUrls and applicationUrl in the config from the answer below. My port is explicitly open in the firewall. Still I can only access server from the local machine, including by its local ip (192.168.0.101)Carrier
I ended up using netsh portproxy: netsh interface portproxy add v4tov4 listenport=5216 127.0.0.1 5216 listenaddress=192.168.0.101 tcpCarrier
As mentioned by LOST, remember to add a firewall rule to allow access to port 5000 from the remote machine.Dwelt
D
81

The best way is to adjust the launchSettings.json, which is located inside the Properties folder.

Change

"applicationUrl": "https://localhost:5001"

to

"applicationUrl": "https://0.0.0.0:5001"

This allows the Kestrel Web Server to listen for traffic from all Network Interfaces.

Distinguishing answered 5/1, 2020 at 18:15 Comment(4)
There are two applicationUrl part in asp.net core 3.1 where do I add this part?Anderlecht
You should add this to whichever profile you are using to launch your application.Archaeozoic
Worked for me! Thanks! Updated profiles { ProjectName { applicationUrl: "https://0.0.0.0:7036;http://0.0.0.0:5036" } } in launchSettings.json, and when I started the application, I was prompted by Windows to allow private and public networks. I did, and now I can access it from my phone too. Thanks!Huygens
Jesus, this info was hard to findSidky
S
24

You can simply do the following to create your WebHost, this will allow remote connections to kestrel.

var host = WebHost.CreateDefaultBuilder(args)
                .UseUrls("http://0.0.0.0:80")
                .UseStartup<Startup>()
                .Build();

After using the following code I still wasn't able to access my API remotely, I had to disable the network adapters created by Docker in the windows control panel (Control Panel\Network and Internet\Network Connections)

Snowstorm answered 3/3, 2018 at 15:26 Comment(1)
In the program.cs fileSnowstorm
Q
11

In my case (.NET core 2.1) I had to modify the Properties/launchSettings.json file.

Set the applicationUrl to a list of allowed urls separated by semicolons like this

"applicationUrl": "https://localhost:5001;http://odin:5000"

Hope this helps someone out there.

Quintero answered 27/8, 2018 at 3:19 Comment(1)
This worked for me with .net core 3.1. Note that it does NOT work when specifying ip addresses.Hydrant
R
7

For us, it also worked (i don't know if that is a new feature of of core 3.1) it that way:

 public static IHostBuilder CreateHostBuilder(string[] args) =>
        Host.CreateDefaultBuilder(args)
            .ConfigureWebHostDefaults(webBuilder =>
            {
                webBuilder.UseKestrel();
                webBuilder.UseIIS();
                webBuilder.UseUrls("http://*:8080");
                webBuilder.UseStartup<Startup>();
            });
Romola answered 31/8, 2021 at 12:39 Comment(1)
this works suspiciously good :)Waki
C
5

Another way to solve this problem is editing "applicationhost.config" file. in project folder -> .vs (Hidden folder) -> config open "applicationhost.config" file. Under sites section, site name="your project name" in Bindings node add another binding and change localhost with your "IP/Domain" on "bindingInformation", like This:

<site name="project_name" id="2">
    <application path="/" applicationPool="Clr4IntegratedAppPool">
     <virtualDirectory path="/" physicalPath="D:\Projects\project_directory" />
    </application>
    <bindings>
     <binding protocol="http" bindingInformation="*:5000:localhost" />
     <binding protocol="http" bindingInformation="*:5000:192.168.1.2" />
     <binding protocol="http" bindingInformation="*:5000:odin" />
    </bindings>
</site>

remember Visual Studio must be run as Administrator.

Chopin answered 4/3, 2018 at 18:11 Comment(1)
Yes "Remember Visual Studio must be run as Administrator." und you wont loose hours in stupidity like me. :)Nitrosamine
M
1

You can use command line argument

dotnet .\SelfHostTest.dll --urls="http://*:5000/ping"
Mclaren answered 17/12, 2022 at 19:27 Comment(0)
N
0

There is a more accurate way when there are multi IP addresses available on the local machine. Connect a UDP socket and read its local endpoint:

string localIP;
using (Socket socket = new Socket(AddressFamily.InterNetwork, SocketType.Dgram, 0))
{
    socket.Connect("8.8.8.8", 65530);
    IPEndPoint endPoint = socket.LocalEndPoint as IPEndPoint;
    localIP = endPoint.Address.ToString();
}
Nitz answered 19/12, 2017 at 11:1 Comment(0)

© 2022 - 2024 — McMap. All rights reserved.