Makefile: A Comprehensive Guide with Examples for .NET and Beyond
Learn
make
util with a practical example of a C# project.
Not that far ago, I learned about the make
utility and the Makefile
. Although that util is very old by computer science standards (1977), it is still extremely helpful today in 2025! In this article, I want to share with you my findings about this utility and give you an example of how to use it with a modern project.
The util was originally designed to help with compiling and building C projects, but the way it was built allows it to be useful for roughly any programming language there is. In this article, we will play around with a .NET project and Docker, but I believe you will still be able to grasp an overall idea regardless of the programming language you are using.
Preparing the Project and Writing Our First Command (Recipe)
Let's start by creating a new minimalistic ASP .NET Core Project. Let's call it Playground
:
dotnet new web --name Playground
We will make just two small modifications to make our experiment look a little bit nicer. First of all, let's make our logs occupy just one line per log:
builder.Logging.AddSimpleConsole(c => c.SingleLine = true);
Secondly, let's return an object (JSON) instead of just text from the default endpoint:
app.MapGet("/", () => new {
message = "Hello World!"
});
Here's how our Program.cs
should look after the changes:
var builder = WebApplication.CreateBuilder(args);
builder.Logging.AddSimpleConsole(c => c.SingleLine = true);
var app = builder.Build();
app.MapGet("/", () => new {
message = "Hello World!"
});
app.Run();
Now, let's add our hero of the day - Makefile
. For now, it will have just a single command to run our project:
run:
dotnet run
And here's what make run
:
For now, it may not seem like much, since we could've just executed dotnet run
in our shell directly. We will do something much more fun in the next section, but now let's take time to reflect on why abstracting our command under make
util may be already beneficial.
It all comes down to changes. Let's say folder structure changes, a command line argument is needed, or even the whole project is rewritten in another programming language. We can just update the script under the run
command and call our familiar make run
to get our project up and going.
Exploring the Power of Makefile: Adding curl
Command
In the first section, I've provided the code for Program.cs
, so we can be sure we have the exact same code in place. However, we will unlikely have different ports exposed, since the dotnet new web
command picks a random port for a newly created project. In my case, it was 5154
How about we let Makefile
get us back synced, by making the same curl
command, using our own port? Here's how mine looks:
curl:
curl http://localhost:5154
Now, if you've used your own port we should be able to run exactly the same commands for our experiments: make run
and make curl
. Here's what the result might look like:
We were able to test our application responsiveness by running the commands in two shells side-by-side. But how about we start the project, wait a little bit until it is ready, make a request, and then bring it back to the foreground?
This is where make
comes in handy again. We can use a single file with multiple commands, basically calling each other like real programming language functions.
Let's draft an example recipe (as a single line inside a makefile command is called):
It's generally considered a better practice to call another make command via
$(MAKE)
. But since in our case the benefits of$(MAKE)
are not applicable we will keep it simple.
rurl-naive:
make run & sleep 4 && make curl && fg
Here's what we will get after running it with make rurl-naive
:
As you may see, we get an error: /bin/sh: line 0: fg: no job control
. This happens due to the fact that make
runs a non-interactive shell, for each line of the command. A non-interactive shell basically means a shell without job control, which, in turn, means that we can not bring a job to the foreground.
We will fix our command, but now we have a hanging dotnet run
process. This is especially problematic because while the port is occupied we will not be able to start another dotnet run
process. Gladly, we can use Makefile
to fix that problem, too. Let's write a command killing a process by the occupied port.
Don't forget that
5154
is my generated port and you may likely have some other port
kill:
kill `lsof -t -i:5154`
And here's what happens if we run make kill
:
One more make
command in our chest. Let's fix the buggy one in the next section!
Fixing our rurl
Command: Making it Work with Job Control
As you may remember, our initial implementation of rurl
failed because make
util creates a non-interactive shell. We can overcome this by running another shell inside the make
shell and making it interactive!
If it sounds rough for you like it did for me, don't worry! We'll just need to wrap our command with bash -c -i
(with -i
making our subshell interactive). Here's the updated command:
rurl:
bash -c -i 'make run & sleep 2 && make curl && fg'
And here's what we will get from running it:
We were able to successfully test our API, returning it back to the main thread. However, right now it looks pretty ugly, since curl
prints the result unformatted and doesn't add a line break, so everything looks kind of messy. Let's fix it in the next section.
Using Httpyac to Make It Beautiful
There's a much prettier alternative to curl - the tool is called httpyac
. I've written a dedicated article about it, but what the way we use it in this article should be intuitively clean without any additional preparation. Yet to follow the steps in the tutorial you would need to install the httpyac CLI (don't worry it's very simple).
We'll start by creating a .http
file:
GET http://localhost:5154/
Now, let's add a command that will run all (yes, here it means one) requests from the file:
Take a moment to appreciate how much less typing we would need afterward (just
make yac
)
yac:
httpyac send .http --all
Here's how running the script might look like:
And now let's combine the script we had before with httpyac instead of curl:
play:
bash -c -i 'make run & sleep 2 && make yac && fg'.
Now, we will be able to get a beautifully looking result of debugging our application with a simple make play
command:
This command finalizes the article, let's just do a quick recap and call it a day!
Recap
In this article, we've created a Makefile
for running an ASP .NET Core application and executing a test request against it, all in one shell. Here's how the file looks assembled:
run:
dotnet run
curl:
curl http://localhost:5154
rurl-naive:
make run & sleep 4 && make curl && fg
rurl:
bash -c -i 'make run & sleep 2 && make curl && fg'
yac:
httpyac send .http --all
play:
bash -c -i 'make run & sleep 2 && make yac && fg'
kill:
kill `lsof -t -i:5154`
The article and the playground are parts of the nice-shell repository, trying to help your shell experience be nicer. Don't hesitate to give the repository a star! ⭐
Claps for this article are also appreciated! 😊