A basic build system cobbled together from the jq JSON utility and a handful of POSIX standard commands.
$ cd example
$ ./jelly
Pass
$ ./jelly test
Pass
$ ./jelly lint
./good-2.json
{
"count": 3
}
./bad.json
parse error: Expected another key-value pair at line 3, column 1
./good-1.json
{
"count": 3
}
$ ./jelly help
Usage: ./jelly [<task>]
Tasks:
help
lint
test
The build system is configured with a jq script named jelly
(no file extension). The file should feature chmod +x
permissions.
jelly tasks are created with shell command strings, indexed by task name in a single JSON document.
A task may specify multiple shell commands through the conventional semicolon (;
) shell command delimiter.
Note that additional quoting may be required for certain characters, according to traditional C-style backslash syntax.
#!/bin/sh
unset IFS
set -euf
jq -r '.[($ARGS.positional[0] // "test")]' --args "$@" <<HERE | sh
{
"test": "echo 'Pass'",
"lint": "find . -type f -iname '*.json' -print -execdir jq . \\"{}\\" +"
}
HERE
Above, we see a stripped down jelly build system, with just two tasks, test
and lint
.
The three opening lines #!/bin/sh
, unset IFS
, set -euf
setup a robust, POSIX compliant shell script wrapper around an inner jq script. This allows jelly to automatically process task commands with | sh
.
Note that the JSON configuration resides inside of a shell heredoc. In addition to any JSON string escapes, certain special characters like backslash (\
) and dollar ($
) in your task shell commands, will require additional backslash escapes.
Naturally, you will want to replace uninformative echo 'Pass'
test command with some real commands relevant to your specific testing needs.
The jq expression // "test"
declares the test
task as the default, when no task is named when jelly runs.
#!/bin/sh
unset IFS
set -euf
jq -r '.[($ARGS.positional[0] // "test")]' --args "$@" <<HERE | sh
{
"test": "echo 'Pass'",
"lint": "find . -type f -iname '*.json' -print -execdir jq . \\"{}\\" +",
"help": "printf \\"Usage: ./jelly [<task>]\\\\n\\\\nTasks:\\\\n\\\\n\\"; tail -r ./jelly | tail -n +2 | tail -r | tail -n +6 | sed 's/\\\\\\\\\\\\\\\\/\\\\\\\\/g' | jq -r 'keys | .[]'"
}
HERE
The canned help
task generates a usage menu. It works by extracting the task names from its own jelly script.
For debugging, temporarily replace | sh
with | cat
to see the command strings that jelly would have executed.
There is probably a way for json to perform the simple text substitution duties on raw string text input, which would remove the need for sed in usage menu generation.
We should improve validation so that typos are caught and transformed into a request for the usage menu. Currently, invalid task names sent to ./jelly
result in empty output. Strange output like that is ver bad to send to sh
. Perhaps POSIX has something to say about what sh
should even do with null output. That may not be well defined behavior, or present a security risk to some operating systems. In any case, the user deserves a more specific error message.
A looping mechanism would add support for processing multiple targets in a single ./jelly <task> <task> <task>
... invocation.
GNU head
features support for -n
with negative values, removing the need for inefficient tail -r
file content reversal.
Much JSON related work will be simplified when the format officially adopts a comment syntax, preferably one that integrates well with UNIX shebangs.
- Inspiration from nobuild, a convention for C/C++ build systems
- bashate, a shell script style linter
- bb, a build system for (g)awk projects
- beltaloada, a guide to writing build systems for (POSIX) sh
- booty for JS/Node.js/altJS
- Gradle, a build system for JVM projects
- lake, a Lua task runner
- Leiningen + [lein-exec](https://github.com/kumarshantanu/
- lichen, a sed task runner lein-exec), a Clojure task runner
- Mage, a task runner for Go projects
- mian, a task runner for (Chicken) Scheme Lisp
- npm, Grunt, Node.js task runners
- POSIX make, a task runner standard for C/C++ and various other software projects
- Rake, a task runner for Ruby projects
- Rebar3, a build system for Erlang projects
- rez builds C/C++ projects
- sbt, a build system for Scala projects
- Shake, a task runner for Haskell projects
- ShellCheck, a shell script linter with a rich collection of rules for promoting safer scripting
- slick, a linter to enforce stricter, unextended POSIX sh syntax compliance
- stank, a collection of POSIX-y shell script linters
- tinyrick for Rust projects
- yao, a task runner for Common LISP projects
🍮