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.
If a code example has a declared language of bash
, then gfxmr
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
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"
)
func main() {
fmt.Printf("---> %v\n", os.Args[0])
fmt.Println("we could make an entire album out of this one sound")
}
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");
}
}
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();
}
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
]
}
If a code example has a declared language of python
, then gfxmr
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())
If a code example has a declared language of ruby
, then gfxmr
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
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
If a code example has a declared language of sh
, then gfxmr
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
If a code example has a declared language of zsh
, then gfxmr
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
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");
}
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.
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
^^^
Given a regular expression string value, asserts that the program output (stdout) matches.
Given a string array, run the program with the value as command line arguments.
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.
Given either a string or array of strings, skips the program if the current OS does not match. When absent, no filter is applied.
No tag annotations, expected to be short-lived and exit successfully:
var _ = 1 / 1;
Annotated with an "output"
JSON tag that informs gfxmr
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)
}