Skip to content

mcandre/jelly

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 

History

14 Commits
 
 
 
 

Repository files navigation

jelly: the JSON build system

SUMMARY

A basic build system cobbled together from the jq JSON utility and a handful of POSIX standard commands.

EXAMPLE

$ 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

CONFIGURATION

The build system is configured with a jq script named jelly (no file extension). The file should feature chmod +x permissions.

Create a task

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.

Usage menu

#!/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.

Dry Run

For debugging, temporarily replace | sh with | cat to see the command strings that jelly would have executed.

Further Research

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.

REQUIREMENTS

SEE ALSO

  • 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

🍮

About

the JSON build system

Resources

Stars

Watchers

Forks

Releases

No releases published

Packages

No packages published

Languages