gfmrun
gfmrun copied to clipboard
GitHub-Flavored Markdown Runner
gfmrun
A (GitHub-Flavored Markdown Runner). Runs stuff inside code gates (maybe!).
This project is not intended to fully replace the example code running capabilities
of any given language's tooling, but instead to fill a gap. In the case of a Go
repository, for example, it can be handy to have some verifiable code examples
in the README.md
and example functions in *_test.go
files.
Explicitly Supported Languages
- bash
- go
- java
- javascript
- json
- python
- ruby
- shell
- sh
- zsh
Bash
If a code example has a declared language of bash
, then gfmrun
will write
the source to a temporary file and run it via whatever executable is first in
line to respond to bash
.
for i in {97..99} ; do
echo "$i problems" >&2
done
echo "Begot by all the supernova (${0})"
exit 0
Go
If a code example has a declared language of go
and the first line is package main
, then gfmrun
will write the source to a temporary file, build it, and run
it. It is worth noting that go run
is not used, as this executes a child
process of its own, thus making process management and exit success detection
all the more complex.
package main
import (
"fmt"
"os"
"golang.org/x/example/stringutil"
)
func main() {
fmt.Printf("---> %v\n", os.Args[0])
fmt.Println("we could make an entire album out of this one sound")
fmt.Println(stringutil.Reverse("[SQUEAK INTENSIFIES]"))
}
package main
import (
"log"
)
func main() {
log.Fatal("we can handle errors too")
}
Java
If a code example has a declared language of java
and a line matching ^public class ([^ ]+)
, then gfmrun
will write the source to a temporary file, build
it, and run the class by name.
public class GalacticPerimeter {
public static void main(String[] args) {
System.out.println(System.getenv("FILE"));
System.out.println("Awaken the hive");
}
}
JavaScript (assumed node.js compatible)
If a code example has a declared language of javascript
, then gfmrun
will
write the source to a temporary file and run it via whatever executable is first
in line to respond to node
.
var main = function() {
console.log("they won't stop at dancin, no");
console.log("they won't stop at dancin");
};
if (require.main == module) {
main();
}
JSON
If a code example has a declared language of json
, then gfmrun
will write the
source to a temporary file and "run" it via the node
executable for
validation.
{
"no": "output",
"levels": [
8000,
9000,
9001
]
}
Python
If a code example has a declared language of python
, then gfmrun
will write
the source to a temporary file and run it via whatever executable is first in
line to respond to python
.
from __future__ import print_function
import sys
def main():
print('lipstick ringo dance all night {!r}!'.format(sys.argv))
return 0
if __name__ == '__main__':
sys.exit(main())
Ruby
If a code example has a declared language of ruby
, then gfmrun
will write
the source to a temporary file and run it via whatever executable is first in
line to respond to ruby
.
def main
puts $0
puts "get out of the path of the king"
return 0
end
if $0 == __FILE__
exit main
end
Shell
If a code example has a declared language that can be mapped to the linguist
definition of shell
, then gfmrun
will write the source to a temporary file
and run it via whatever executable is first in line to respond to bash
.
if [ 0 -eq 0 ] ; then
echo "Just the way you like it, yes"
echo "just the way you like it. (${0})"
fi
exit 0
Sh
If a code example has a declared language of sh
, then gfmrun
will write
the source to a temporary file and run it via whatever executable is first in
line to respond to sh
.
if [ 5 -eq 3 ] ; then
echo "YOU FOUND THE MARBLE IN THE OATMEAL"
else
echo "Saddle up preacher, don't look back. (${0})"
fi
exit 0
Zsh
If a code example has a declared language of zsh
, then gfmrun
will write
the source to a temporary file and run it via whatever executable is first in
line to respond to zsh
.
printf "Kiss me.\nJust kiss me.\n(${0})\n"
bye 0
Implicitly Supported Languages
If a code example's declared language can be matched to one of the explicitly supported languages listed above via the linguist languages definition, then the matched language's runner will be used.
This example declares node
, but will be run via the javascript
frob, which
happens to use the node
executable anyway:
if (1 / 1 == 1) {
console.log("kiss me, Nefertiti");
}
Disabling automatic download of languages.yml
By default, the linguist languages definition is downloaded if not present.
This behavior can be disabled by passing the --no-auto-pull
/ -N
flag or
setting a GFMRUN_NO_AUTO_PULL=true
environment variable.
Tag annotation comments
gfmrun
supports the use of JSON tags embedded in comments preceding code
blocks, e.g. (just pretend ^
are backticks):
<!-- { "awesome": "maybe", "all on one line": "yep" } -->
^^^ go
package lolmain
// ... stuff
^^^
<!-- {
"wow": "cool",
"multiple": "lines"
} -->
^^^ go
package lolmain
// ... stuff
^^^
<!-- {
"super": "advanced",
"whitespace after the comment": "mindblown"
} -->
^^^ go
package lolmain
// ... stuff
^^^
"output"
tag
Given a regular expression string value, asserts that the program output (stdout) matches.
"error"
tag
Given a regular expression string value, asserts that the program error (stderr) matches.
"args"
tag
Given a string array, run the program with the value as command line arguments.
"interrupt"
tag
Given either a truthy or duration string value, interrupts the program via
increasingly serious signals (INT
, HUP
, TERM
, and finally KILL
). If the
value is truthy, then the default duration is used (3s). If the value is a
string duration, then the parsed duration is used. This tag is intended for
use with long-lived example programs such as HTTP servers.
"os"
tag
Given either a string or array of strings, skips the program if the current OS does not match. When absent, no filter is applied.
Examples
No tag annotations, expected to be short-lived and exit successfully:
var _ = 1 / 1;
Annotated with an "output"
JSON tag that informs gfmrun
to verify the example
program's output:
console.log("Ohai from the land of GitHub-Flavored Markdown :wave:");
Annotated with an "interrupt"
JSON tag that informs gfmrun
to interrupt the
example program after a specified duration, which implies that the exit code is
ignored (not Windows-compatible):
package main
import (
"fmt"
"net/http"
)
func main() {
http.HandleFunc("/", func(w http.ResponseWriter, req *http.Request) {
w.WriteHeader(http.StatusOK)
fmt.Fprintf(w, "Why Hello From Your Friendly Server Example :bomb:\n")
})
http.ListenAndServe(":8990", nil)
}
Similar to the above example, but the "interrupt"
tag is truthy rather than a
specific interval, which will result in the default interrupt duration being
used (3s). It is also annotated with an "output"
tag that takes
precedence over the "interrupt"
tag's behavior of ignoring the exit code:
package main
import (
"fmt"
"net/http"
)
func main() {
http.HandleFunc("/", func(w http.ResponseWriter, req *http.Request) {
w.WriteHeader(http.StatusOK)
fmt.Fprintf(w, "Hello Again From Your Friendly Server Example :bomb:\n")
})
fmt.Println(":bomb:")
http.ListenAndServe(":8989", nil)
}