mdrip
is a markdown code block extractor.
To extract and print all code blocks below your current directory:
mdrip print .
Pipe that into /bin/bash -e
to have the effect of a test, or better yet
try the test
command:
mdrip test .
This fails only if an extracted markdown code block fails. Use a label to be selective about which blocks to run.
To give demos with tmux
, or generally use your browser and tmux
as a markdown code block IDE, try:
mdrip serve --port 8080 .
In the browser, while focused on a code block, hit the Enter
key.
The app posts the block's ID to the server, and the server sends
the corresponding code block to tmux
via its api.
-
Download a build from the release page.
-
Install via the Go tool:
go install github.com/monopole/mdrip/v2@latest
-
Run via
docker
:image=monopole/mdrip:latest docker run $image version # try 'help' instead of 'version' to see options. docker run \ --publish 8080:8080 \ --mount type=bind,source=`pwd`,target=/mnt \ $image serve /mnt
For something to work with, download this busted Go tutorial:
repo=https://raw.githubusercontent.com/monopole/mdrip
curl -O $repo/master/assets/bustedGoTutorial.md
This markdown has code blocks showing how to write, compile
and run a Go program in your TMPDIR
.
Extract the blocks to stdout
:
mdrip print bustedGoTutorial.md
Some code blocks in this markdown have labels; these are visible as HTML comments preceding the blocks in the raw markdown.
Use a label to extract a subset of blocks:
mdrip print --label goCommand bustedGoTutorial.md
Test the code from the markdown in a subshell:
mdrip test bustedGoTutorial.md
echo $?
The above command should show an error, and exit with non-zero status, because the tutorial has errors.
Fix the error:
cp bustedGoTutorial.md goTutorial.md
sed -i 's|badecho |echo |' goTutorial.md
Try the fix:
mdrip test goTutorial.md
echo $?
There's another error. Fix it:
sed -i 's|comment this|// comment this|' goTutorial.md
There are now two changes:
diff bustedGoTutorial.md goTutorial.md
Test the new file:
mdrip test goTutorial.md
echo $?
The return code should be zero.
You can run a block in your current shell to, say, set current environment variables as specified in the markdown:
eval "$(mdrip print --label setEnv goTutorial.md)"
echo $greeting
The upshot is that adding a line like
mdrip test --label {someLabel} {filePath}
to your CI/CD test framework covers the markdown code block execution path determined by that label.
The {filepath}
argument defaults to your current directory (.
),
but it can be
- a path to a file,
- a path to a directory,
- a GitHub URL in the style
gh:{user}/{repoName}
, - or a particular file or a directory in the
repo, e.g.
gh:{user}/{repoName}/foo/bar
.
Add labels to a code block by preceding the block with a one-line HTML comment, e.g:
<!-- @sayHello @mississippi @tutorial01 --> ``` echo hello ```
Labels are just words beginning with @
in the comment.
The first label on a block is slightly special in that it is treated as the block's name for various purposes. If no labels are present, a block name is generated for these purposes.
mdrip
and tmux
provide a handy way to develop and demonstrate
command line procedures.
Render a markdown web app like this:
mdrip serve --port 8000 goTutorial.md
Visit it at localhost:8000.
Hit the n
key for navigation tools.
Hit ?
to see all key controls.
The handy aspect is provided by tmux
.
If there's a running instance of tmux
, the server
will send code blocks to it when you hit Enter
.
Fire up tmux
, then try this README
directly:
mdrip serve gh:monopole/mdrip/README.md
To see what using a full tree of markdown looks like, generate some content with:
mdrip writemd /tmp/mdTestData
then serve it:
mdrip serve /tmp/mdTestData
mdrip
encourages literate programming via markdown.
It lets one run or test code (shell commands) that is otherwise embedded in explanatory content (markdown).
One can use here documents to incorporate any programming language into tested markdown - as in the busted Go tutorial discussed above. That tutorial could have covered C, C++, Rust, etc.
Place commands that the reader would want to execute directly (with no edits) in fenced code blocks.
In contrast, code-style text that is not intended for copy/paste execution,
e.g. alternative commands with fake arguments or example output,
should be in a fenced code block indented via a
block quote. Block quotes are ignored by mdrip
.
Eschew adding prompts to code blocks. The following code snippet is easy to copy/paste:
echo hello du -sk
But this is not:
$ echo hello $ du -sk