just
version 0.11.2+ now supports duplicate recipes where the last
one defined is the one that is executed.
To enable this feature add this setting to your file(s): set allow-duplicate-recipes
A pre-processor for the "just" command line utility. https://github.com/casey/just
Please forgive the disorganization - thie README is still under development.
Casey's just
utility is such a useful tool; but, it has one thing that bothered me - the inability to inherently include recipes from other files. For those of us who work on many projects in parallel the ability to share recipes (effective code reuse / modularization) is an important feature. Its advantages are:
- provides for smaller more easily maintained files
- more cohesive file content
- shared recipes between projects
Allowing this flexibility also opens up some use cases which are not supported by the just
utility - the ability to over-ride a recipe, for example. Duplicate recipes (and aliases) are treated by just
as errors. Allowing the inclusion of multiple files into a single project specific justfile
can lead to having duplicate recipes. These are easily found and can be dealt with manually.
- System Environment Variables
- Install
- Shell Configuration and My Conventions
- The Just List aka Help/Usage Output
- Process
- Path to a File to be Included
- Limitations
- Error Messages
- Testing
justprep
in both its Ruby Gem form and its binary executable via Crystal is controlled by the following environment variables.
THis is a string of one or more keywords to use to designate a file to be included into the output file passed on to just
The default is 'include import require with' where each keyword is a designation in the input file for an inclusionary action on a designated file path. You do not have to use all or any. Personnally I've pretty much standardized on "with" as my perfered keyword. You may use whatever makes you happy.
This specifies the name of the file that is to be used as input into justprep
to be processed into the output file that goes to just
.
The default value is "main.just" but you can use whatenver filename you want.
Remember this is a filename and not a path.
This is the name of the file that is created by justprep
which is passed on to just
The default is "justfile" because that is the default that just
uses. Its possible to use other file names but you have to tell just
via its --justfile
command line option the name of the file.
Remember this also is just a file name and not a path. It will be the name of the file that is created by justprep
in the same directory where the JUSTPREP_FILENAME_IN
is located.
First you need just
which is easy enough to install (unless you are not on a Mac or other great platform) using the brew
package manager:
$ brew install just
justprep
is available as a Ruby Gem. To install the gem do:
$ gem install justprep
Also notice that in this repository there is a version of justprep
implemented in Crystal. I've used Crystal version 1.0.0 to compile it for my use.
To use the Crystal executable instead of the Ruby Gem version you will need to do a few more things. First have Crystal installed.
$ brew install crystal
$ git clone http://github.com/MadBomber/justprep
$ cd justprep
$ just install
The install
recipe will compile and then move the executable to the default location /usr/local/bin
If you do not have /usr/local/bin
as part of your $PATH, or you just want the executable installed in a different directory that is in your $PATH you can do this:
$ just install ~/bin
~/bin
is just where I put mine. You can put yours wherever you want.
See what other recipes are in this justfile
by doing this:
$ just
In your favorite shell, setup the system environment variables and an alias that invokes justprep
before just
alias jj='justprep && just'
# These values are the defaults. Replace with your choice or if
# you like the defaults, you do not need the setup the shell variables.
export JUSTPREP_KEYWORDS='include import require with'
export JUSTPREP_FILENAME_IN=main.just
export JUSTPREP_FILENAME_OUT=justfile
This is my configuration. If you are using the just
and thinking about justprep
, you are more than capable of coming up with your own conventions for configuring shell tools.
In my .bashrc
file I source all files in my home directory that have the pattern .bashrc__*
# ... <snip>
########################################################
## Setup additional Plugins/Packages
for file in ~/.bashrc__* ; do
if [ -f $file ]; then
echo -n "Loading $file ... "
source $file
echo "done."
fi
done
# ... <snip>
so what I did is create a file .bashrc__just
that holds my configuration for justprep
and just
- Here is what it looks like:
# ~/.bashrc_just
# brew install just
# gem install justprep or its crystal version
# Use the crystal version of justprep
alias jj='~/bin/justprep && just'
export JUSTPREP_KEYWORDS='include import require with'
export JUSTPREP_FILENAME_IN=main.just
export JUSTPREP_FILENAME_OUT=justfile
In my home directory I have a file named .justfile
which has the following content:
# ~/.justfile
set positional-arguments := true
# List available recipes
@list:
echo
echo "Available Recipes at"
echo `pwd`
echo "are:"
echo
just -l --list-prefix 'jj ' --list-heading ''
echo
Now in every main.just
file in all my projects I have it start like this to take advantage of the standardized recipe list format:
# ~/**/main.just
with ~/,justfile
# ... other inclusionary directives
# ... project-specific recipes, etc.
I keep going back and fornth over naming the first recipe (the default recipe) in my justfiles. The just
command line option that is used in my default recipe is called list. I then to think of it as help. As the default recipe it is the one that is executed whenever just
is invoked without specifying a recipe.
Here is what shows up in my terminal when I execute just
in this repository's root directory:
$ just
Available Recipes at
/Users/dewayne/sandbox/git_repos/madbomber/justprep
are:
just build # Build both Ruby Gem and Crystal versions
just bump level # Bump version level: major.minor.patch
just help # Displays this list of available recipes
just install # Install both Ruby Gem and Crystal versions
just set version # Set the version to major.minor.patch
just test # Test both the Ruby Gem and the Crystal Executable
just version # Show current source version
There are more recipes than this in the repository root ($RR) than just these. The others are hidden. Any just
recipe that begins with the underscore character '+' is defined as a hidden recipe and will not be included in the just -l
command.
justprep
reads the $JUSTPREP_FILENAME_IN
file from the current working directory. If there is not one there, it then goes up the directory hierarchy until it finds one or until there is no more directory places to look.
Once the input file is found, justprep
reads its entire contents into memory. It then starts looking for any $JUSTPREP_KEYWORDS
in the contents. For each one it finds it takes the stuff after the keyword to the end of line as a path to a file that is to have its contents included. That content is insert into the same place as the keyword line.
As part of that insert a header and a footer line are added to the imported file's contents as a way to signal from where the content came.
Once the entire input file has been processed, its contents with its recently inclued content from other files, is written out to the $JUSTPREP_FILENAME_OUT
file.
# ~/sandbox/git_repos/madbomber/main.just
with ~/.justfile
@hello:
echo "Hello World"
# ~/sandbox/git_repos/madbomber/main.just
# >>> /Users/dewayne/.justfile
# ~/.justfile
# brew install just
# gem install justprep
# alias jj='justprep && just'
set positional-arguments := true
# List available recipes
@list:
echo
echo "Available Recipes at"
echo `pwd`
echo "are:"
echo
just -l --list-prefix 'jj ' --list-heading ''
echo
# <<< /Users/dewayne/.justfile
@hello:
echo "Hello World"
If there are any errors detected by the just
utility, they will show and reference the exact line number in the output file. Its easy enought to find the source of the error even if it is in one of the included files due to the header and footer banners placed before and after the inclued file content.
The path to the file to be included, imported, required or with'ed etc. follows the inclusionary keyword.
<keyword> <space> <path-to-file-to-be-included>
It can be an absolute path (begins with a '/' character) or relative. If relative, then "Relative to what?" is a good question. A good answer is "Relative to the location of the $JUSTPREP_FILENAME_IN
file." That is the file location that just included the relative file.
The path to the file to be included can make use of system environment variables in either the ~
if present as the first character of the file path is taken as a shortcut to the user's home directory.
with ~/.justfile
include $HOME/.justfile
import /Users/dewayne/.justfile
require ../${PROJECT}_common_tasks/${ENVIRONMENT}_tasks.just
Noticed I used multiple spaces after the keyword. There only needs to be one; but, I like using extra spaces to align the file paths becuase it makes it easier for me to see what is intended.
- filenames may not contain
just
variables - the keyword must start in column 1 and be followed by at least one space
- included files may not contain inclusionary keywords - only single-level inclusions are allowed - you cannot include a file that also includes another file
- the file must be present and not a directory otherwise an error message is sent to STDERR and the exit code is set to 1
When a filename does not exist, an ERROR message will be written to STDERR. The exit code will be set to 1.
There is also a WARNING message that comes to STDERR when justprep
does not find a JUSTPREP_FILENAME_IN file it the current directory hierarchy. The exit code is still set to zero which allows the just
program to execute.
Any errors in syntax, duplication of recipes etc. from just will will relative to the output file from justprep
which may be found in the included files. Review the outfile file at the line number provided. If it is in an inclued file, you can easily find out which file requires the fix by looking at the header/foot banners that surround the included file content.
Here is the error:
$ jj
error: Expected character `=`
|
10 | !output: 'Hello world'
| ^
... and here is the output file which generated the error:
$ cat -n justfile
1
2 # >>> /Users/dewayne/Documents/sandbox/git_repos/madbomber/error.just
3
4 world: hello
5 echo 'world'
6
7 hello:
8 echo -n 'hello '
9
10 !output: 'Hello world'
11
12 # <<< /Users/dewayne/Documents/sandbox/git_repos/madbomber/error.just
13
14 doit: world
You can see that line 10 is clearly within the banners giving the absolute path to the file which was included that contains the error noted by the just
tool.
From the repository root directory use just test
to execute the tests for both the Ruby Gem version and the Crystal version. Here is the output that I get:
jj test
crystal build --no-debug --release -p -o bin/justprep justprep.cr version.cr
justprep 1.0.1 built to pkg/justprep-1.0.1.gem.
justprep (1.0.1) installed.
Tests PASSED
In the justprep/test
there is a file named expected.txt
that contains the expected results of the various test scripts. It is compared via diff
to the file results.txt
that is generated from the test scripts. If there is any difference the test has failed. On failure the difference between results.txt and expected.txt will be echo'ed to the terminal.
This repository contains two versions of the justprep
pre-processor. One is implemented as a Ruby Gem. The other is implemented in Crystal. While Crystal shares some of the syntax as Ruby, not all Ruby code is directly usable by Crystal which is too bad; but, they are close enough to be fun.
TODO: Put stuff about the Ruby Gem here ...
TODO: Put stuff about the Crystal Version here ...