Makefile VS Justfile
The cool stuffs with Makefile
Nothing to say, it’s POSIX so it eveywhere almost by default since 1976 (47 years). So that’s the reference, either you do better or worst than Make.
I can list few points:
Cool that it exists and you should have went through.
Make is “task runner” and “build tool” since it’s capable to not run a target if a dependencies is up-to-date when justfile is just “task runner”. But on the other hand,
Just
just want to be a “task runner”…
The cool stuffs with Justfile
Here a list of what justfile can do natively but not makefile:
just --choose
- will let you choose in interactif mode among the recipes.Define your work dir
just --justfile ~/.user.justfile --working-directory ~
(I am not convince that you can do it with Makefile)Code Precheck is highly appreciate.
1bash: line 1: repository: unbound variable
2error: Backtick failed with exit code 127
3 |
44 | REPOSITORY := `if [ -n $repository ]; then echo "$repository"; else echo "github.com"; fi`
5 | ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
- Automaticly document the recipes if a commented line is set just before the recipe, so when you execute
just --list
, you get:
1Available recipes:
2 env repository='github.com' # env
3 test # Test
- Possibility to make a hidden recipe for documentation, the default recipes (or even to complete this doc). Imagines that you need to create a custom PHONY with a beautifull sed to do the same in makefile…
1_help:
2 @printf "Some Title"
3 @just --list --unsorted
4 @printf "Some Extra infos"
hidden recipes from documentation
Possibility to create aliases for all the recipes automaticly:
1for recipe in `just -f ~/.justfile --summary`; do
2 alias $recipe="just -f ~/.justfile -d. $recipe"
3done
- Possibility to list in different order :
1just --list # sorted in an alphanumeric order
2just --list --unsorted # sorted in the order given in the justfile
Parameterization in makefile will look like
make something -e CHOICE=test
, in justfilejust something test
since inside an justfile, you can define arguments to your recipes.Autocompletion on your recipes
1$ just
2blank -- Args: PROJECT *GROUP # Create a new empty project on remote repository.
3build -- Args: PROJECT NAMESPACE # Build collection locally.
4clone -- Args: PROJECT # Clone a project from repository keeping directory structure for ansible.
5clone_all -- Args: *GROUP # Git clone all projects from your repository, or if argument provided only from specific group.
6init -- Args: PROJECT *GROUP # Create a new ansible collection on repository.
7install -- Args: PROJECT *VERSION # Install a ansible collection. (if PROJECT is an artifact .tar.gz install local)
8local -- Args: PROJECT NAMESPACE # Create a new ansible collection on localhost (not on repository like function below).
9release -- Args: PROJECT *VERSION # Release collection on your repository to the given version in command or in galaxy.yml.
10role -- Args: GROUP PROJECT ROLE # Create a new ansible role inside an existing collection.
Syntax Check. Will point to error in your
justfile
code.Recipes can be written in arbitrary languages, like Python, NodeJS, bash.
just a “task runner” and all the points listed above are going to this purpose.
The Justfile’s limitation
The exported variables
One limitation that I got with justfile is that you can not pass a variable which does not exist. Imagine, you want to set a default behavior but allow your user to define another bebavior. The code below does not work but if you define the var export repository=gitlab.com
. But the point here is to allow the user to not define the variable… But this come from RUST safety paradigm.
1bash: line 1: repository: unbound variable
2error: Backtick failed with exit code 127
3 |
44 | REPOSITORY := `if [ -n $repository ]; then echo "$repository"; else echo "github.com"; fi`
5 | ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
6
7# Get the same error in case the env var is not defined, but still better than above condition.
8REPOSITORY := env_var('REPOSITORY')
Ok, so what I wrote above is not true anymore. This was before, I found this:
1REPOSITORY := env_var_or_default('REPOSITORY', "github.com")
Variables in backtick
Another limitation, again with variable, cannot use variable define before in backtick. This below will generate an error because
1set shell := ["bash", "-uc"]
2
3REPO := "github.com"
4TEST := "https://" + REPO
5TEST2 := `curl https://{{TEST}}`
6
7# Test
8test:
9 #!/usr/bin/env bash
10 echo {{TEST}}
11 echo {{TEST2}}
But limitations listed above seems to come from RUST paradigm for safety and performance.
Makefile Limitations
The documentation of all the PHONY need a PHONY for it. We should look like this:
1.PHONY: prerequis
2## Install all prerequisites for this Ansible Collections.
3prerequis:
4 $(MAKE) -C ./scripts/prerequis all
5
6# keep it at the end of your Makefile
7.DEFAULT_GOAL := show-help
8
9# Inspired by <http://marmelab.com/blog/2016/02/29/auto-documented-makefile.html>
10.PHONY: show-help
11show-help:
12 @echo "$$(tput bold)Available rules:$$(tput sgr0)"
13 @echo
14 @sed -n -e "/^## / { \
15 h; \
16 s/.*//; \
17 :doc" \
18 -e "H; \
19 n; \
20 s/^## //; \
21 t doc" \
22 -e "s/:.*//; \
23 G; \
24 s/\\n## /---/; \
25 s/\\n/ /g; \
26 p; \
27 }" ${MAKEFILE_LIST} \
28 | LC_ALL='C' sort --ignore-case \
29 | awk -F '---' \
30 -v ncol=$$(tput cols) \
31 -v indent=19 \
32 -v col_on="$$(tput setaf 6)" \
33 -v col_off="$$(tput sgr0)" \
34 '{ \
35 printf "%s%*s%s ", col_on, -indent, $$1, col_off; \
36 n = split($$2, words, " "); \
37 line_length = ncol - indent; \
38 for (i = 1; i <= n; i++) { \
39 line_length -= length(words[i]) + 1; \
40 if (line_length <= 0) { \
41 line_length = ncol - indent - length(words[i]) - 1; \
42 printf "\n%*s ", -indent, " "; \
43 } \
44 printf "%s ", words[i]; \
45 } \
46 printf "\n"; \
47 }' \
48 | cat
Conlcusion
As you can see, the list is long and you end up with a beautifull tool which allow you to organize your tasks linked between them, autodocumented, and quite safe.
It tries to avoid the complexity and idiosyncrasie of Makefile
. In some way, Makefile
code is nested with your shell and diving into an existing long script can become tedious.
By the way, one project I did with justfile, AnsiColt.
Comments