Skip to content

Latest commit

 

History

History
819 lines (560 loc) · 31.7 KB

README.md

File metadata and controls

819 lines (560 loc) · 31.7 KB

Argument Maps in YAML

Argument Map Example 1 in mindmup format
Collapsible Section: Table of Contents

This repository contains some tools for working with argument maps written in a relatively simple YAML format, described below. For example, the map above was generated from this YAML:

"Brunellus is irrational":
  r1:
    "Brunellus is a donkey.":
    "All donkeys are irrational.": 
  o1:
    "Brunellus studied in Paris.":
    "-Most who study in Paris are rational.":

The map above is also a hyper-link: click on it to open a read-only version of the map in Mindmup's Argument Visualization Mode.

One of the tools is a filter for Pandoc, a universal document converter. This will allow you to convert markdown files with embedded argument map code blocks into pdf or html containing a graphical or even interactive representation of the argument map.

Collapsible Section: Installation

Installation

See CHANGELOG.md for change notes.

This has been tested on Debian only. I believe the Lua code should be portable, but I'm not familiar enough with Lua to know for sure. Some of the supporting code e.g. contents of /scripts may work on other Linux distros or even MacOS, but will not work on Windows.

  1. You can place argmap2mup.lua, argmap2tikz.lua, and mup2argmap.lua somewhere in your PATH e.g. ~/bin.

    Or you leave them in place and put symbolic links to them into your PATH, as has been done in scripts/install.sh. This script also removes the .lua extension from the links to reduce command line typing, however the examples below keep the .lua extension for clarity.

  2. Place pandoc-argmap.lua in the filters folder inside your pandoc data directory, e.g. $HOME/.local/share/pandoc/filters/pandoc-argmap.lua.

  3. For additional functionality, see Dependencies section.

Dependencies

Tested with:

  • Lua 5.3.4
  • LuaRocks 3.7.0
  • Pandoc 2.9.2.1-0; 2.6

argmap2mup.lua, argmap2tikz.lua, and mup2argmap.lua require:

  1. To install these dependencies, you can use the included rockspec file. Navigate to the package directory and run luarocks make --only-deps argmap-[version number].rockspec depending on latest version.

  2. For lyaml, you may need to install libyaml-dev or yaml (conda).

  3. argmap2mup.lua and mup2argmap.lua also depend on the command line utility gdrive for Google Drive integration. Follow the link for installation instructions. (Note that on linux the 386 version is more likely to work: see prasmussen/gdrive#597).

  4. Converting the tikz code generated by argmap2tikz.lua to PDF requires several TeX packages, some of which you may need to install if they are not already available on your system. e.g. texlive-latex-extra and texlive-luatex from apt-get.

  5. pandoc-argmap.lua depends on Pandoc(tested with v2.9.2.1 and v2.6), and on argmap2lua and argmap2tikz.lua. It also depends on pdf2svg for conversion to svg, and ImageMagick's convert for conversion to png.

  6. To display mindmup JSON files on a web page as an interactive mindmap, without using the MindMup website, see Installing mapjs.

See environment.yml for conda environment export of non-lua dependencies and version numbers.


Getting Started

Note: since this readme was created, mindmup no longer seems to read all the mup files generated by argmap2mup.lua (see Issue #11). However, legacy mindmup - mapjs can still be used to read them.

See scripts/bash_aliases_argmap.sh for bash commands to simplify execution. Note that some use the mindmup mapjs repo for displaying the mup file using their legacy open source code.

There are some example files in the examples folder in the repository. Here are some things to try:

$ cat examples/example.yml | argmap2mup.lua > output/example.mup: generate a MindMup map from example.yml. Import that map into MindMup to work with.

$ argmap2mup.lua -u examples/example-2.yml: generate a MindMup map and upload it to your Google Drive. You can then go to Google Drive and open the map in MindMup to work with it.

Create an argument map on Mindmup and download it as a MindMup file (File → Download As → MindMup). Now convert it to YAML:

mup2argmap.lua download.mup > download.yml

Next, try generating a PDF image from example.yml:

cat examples/example.yml | argmap2tikz.lua -s > output/example.tex
lualatex output/example

Finally, try generating an HTML document from a markdown document containing an embedded argument map:

pandoc examples/example.md -o output/example.html --lua-filter pandoc-argmap.lua

How to represent Argument Maps in YAML

The goal here is to describe a spec for argument maps that is relatively easy for humans to read and write. The spec we use is much less ambitious than other specs that I know of, like Argdown and the Argument Interchange Format.

For our purposes, an argument map consists of claims and reasons: each claim can be supported by zero or more reasons; each reason consists of zero or more claims.

We represent each claim as a key-value pair:

  • the key is the content of the claim;
  • the value is a (possibly empty) list of reasons.

So here is how we represent the claim that Brunellus is irrational unsupported by any reasons:

"Brunellus is irrational": {}

We also represent each reason as a key-value pair:

  • the key is an identifier;
  • the value is a (possibly empty) list of claims.

For example, here is a reason that consists of two claims:

  reason1:
    "Brunellus is a donkey.": {}
    "All donkeys are irrational.": {} 

There are two kinds of reasons: supporting reasons and reasons against (i.e., objections). We use the identifier to represent this difference:

  • a reason whose key begins with an o or - is an objection;
  • all other reasons are supporting reasons.

So that takes us back to our original example:

"Brunellus is irrational":
  r1:
    "Brunellus is a donkey.": {}
    "All donkeys are irrational.": {}
  o1:
    "Brunellus studied in Paris.": {}
    "-Most who study in Paris are rational.": {}

Note the '-' before "Most who study..". Claims are either explicit or implicit. We use the key to represent this:

  • a claim whose key begins with a - is implicit;
  • all other claims are explicit.

We also include a way to attach a single note to each claim, and a way to label reasons and specify their relative strength.

Collapsible Section: Argmap Syntax Rules

Markdown Formatting

Each claim key is processed by Pandoc. This means that you should be able to get away with using simple markdown inside the claim keys.

"Brunellus is *probably* 90% H~2~O":

Keep in mind that MindMup offers very limited support for formatted text. So, for MindMup maps, formatting will be converted to Pandoc's "plain" output:

Brunellus is _probably_ 90% H₂O

For TikZ maps, formatting will be converted to latex:

Brunellus is \emph{probably} 90\% H\textsubscript{2}O

Unique Identifiers

Note that the reasons for a given claim cannot share the same identifier.

Something like this will lead to unpredictable behaviour:

"Brunellus is irrational":
  reason: {}
  reason: {}

The resulting map will only display a single reason, and which one it will choose in unpredictable.

Instead, you need to ensure that the keys are distinct:

"Brunellus is irrational.":
  reason1: {}
  reason2: {}

Reasons for distinct claims can have the same identifier. So this is okay:

"Brunellus is a donkey.": 
  reason1: {}
"All donkeys are irrational.":
  reason1: {}

The same holds for claims: each of the claims that make up a given reason must have distinct keys. So don't write something like this:

"Brunellus is irrational":
  r1:
    "Brunellus is a donkey.": {}
    "Brunellus is a donkey.": {}

Since the key is the content of the claim, this restriction makes sense: you would not want to represent the same premise twice as part of a single reason. But, of course, the same premise can occur as part of two distinct reasons:

"Brunellus is irrational":
  r1:
    "Brunellus is a donkey.": {}
    "All donkeys are irrational.": {}
  r2:
    "Brunellus is a donkey.": {}
    "Most donkeys I've met have been irrational.": {}

Empty Lists

Note that every claim is represented as key-value pair, where the value is a list of reasons. If there are no reasons offered for or against a claim, that means that the value is an empty list. You can represent this empty list explicitly if you like:

    "Brunellus is a donkey.": {}

Or you can leave it implicit, omitting the '{}'. But you still must put the colon (':') after the claim:

    "Brunellus is a donkey.": 

In practice, the implicit empty list notation is preferred, because it means you don't have to delete the braces before adding reasons in support of a claim.

Keys and Quotation

YAML allows just about any string to be a key, but you do have to be careful about quotes and line-wrapping. If your key has no special characters, like ':', you get away with omitting the quotes:

    Brunellus is a donkey.: 

Multiline keys are also possible, but the format is a bit complicated (1, 2)

?
 "Brunellus is a donkey who wanted a longer tail, and he went to Paris
 to study, but he couldn't remember the city's name."
:
  r1:
    ?
     "The wikipedia says that Brunellus is a donkey who wanted a longer
     tail, and he went to Paris to study, but he couldn't remember the
     city's name."
    :
    ?
     "The wikipedia is a reliable source for information about donkeys
     in medieval literature."
    :

Notes, Labels, and Strength

(Currently, notes are only supported for MindMup input and output, and do not appear in TikZ output.)

You can attach one note to any claim (if you attach more than one note, only one of them will be processed). A note is represented by a key-value pair, where the key is the string "note" and the value is the content of the note. To attach a note to a claim, add it to the list of reasons:

Brunellus is irrational:
  r1:
    Brunellus is a donkey.: 
      note: |
        Brunellus means "brown one" in Latin, and was the name of a stock character in medieval logic texts.
    All donkeys are irrational.: 
  note: |
    Although donkeys *are* notoriously stubborn, it seems a bit much to suppose that irrationality is the defining feature of a donkey.

Notes are also processed by Pandoc, allowing the use of simple markdown.

You can attach a label or strength to any reason. A label is a key-value pair, where the key is the string "label" and the value is the content of the label. A strength is a key-value pair, where the key is "strength" and the value is a number between 1 (weak) and 5 (strong). Labels and strengths are added to the list of a reason's claims:

Brunellus is irrational:
  r1:
    Brunellus is a donkey.: 
    All donkeys are irrational.:
    label: argumentum ad asinum
    strength: 4
  o1:                                            
    Brunellus studied in Paris.: 
    -Most who study in Paris are rational.:
    label: argumentum ad parisiensis
    strength: 2


argmap2mup.lua

argmap2mup.lua is a pipe for converting YAML maps to JSON encoded MindMup maps.

It takes as its input the first CLI argument that is not an option, or, if there is no such argument, STDIN.

$ cat examples/example.yml | argmap2mup.lua | jq
{
  ⋮
  "ideas": {
    "1": {
      "attr": {},
      "title": "Brunellus is irrational",
      ⋮
}

(Here I've used jq to pretty-print the JSON output.)

The following options are available:

-u, --upload: Upload to Google Drive.

If this option is selected, then, instead of dumping the map to STDOUT, it is uploaded to Google Drive, and its Google Drive ID is returned.

$  argmap2mup.lua -u examples/example.yml
1e4HAl1iHPKBiKZ_BI_yBw7rXYbuvMsC2

If you have connected your MindMup account to Google Drive, you can use this Google Drive ID to construct a URL that allows you to open the file directly in MindMup. The format for the URL is: https://drive.mindmup.com/map/ID

-g ID, --gdrive_id ID: Update the file with ID on Google Drive.

By default, argmap2mup.lua creates a new file on your Google Drive every time you run it. If you have already uploaded a map, and would prefer to update that map rather than create a new one, use this option to specify its Google Drive ID. If you don't know its Google Drive ID, use gdrive list to find it. This command implies --upload:

cat examples/example.yml | argmap2mup.lua -g 1e4HAl1iHPKBiKZ_BI_yBw7rXYbuvMsC2

-f ID, --folder ID: upload to Google Drive folder with ID

For example, I prefer to upload all of my automatically generated maps to a folder called argmaps.

p, --public: Mark the uploaded file as shareable.

By default, files uploaded to Google Drive are private and unshared. Use this flag to mark them as shareable instead. This option also implies --upload.

-n, --name: Specify a name for your map.

This name is used to set the title attribute within the map itself. It is also used as the name of the file on Google Drive. If this option is omitted, and so no name is specified, the conclusion of the argument will be used as the name. No sanity checks are performed on this: no doubt you can break things by specifying weird things as the name.

-h, --help: Display a brief synopsis of these options.

For more information about the structure of the .mup/.json output format, see: Data Format · mindmup/mapjs Wiki.

mup2argmap.lua

mup2argmap.lua is a pipe for converting MindMup maps into the argmap YAML format.

It takes as its input the first CLI argument that is not an option, or, if there is no such argument, STDIN.

cat examples/example.mup | mup2argmap.lua
mup2argmap.lua examples/example.mup

The following options are available:

-g ID, --gdrive_id ID: read the file with the specified Google Drive ID.

This option takes precedence any files specified on the command line or anything piped to STDIN.

-e, --embed: wrap output in a Pandoc markdown code block with attributes, suitable for embedding.

-h, --help: Display a brief synopsis of these options.

argmap2tikz.lua

argmap2tikz.lua is a pipe for converting YAML maps to TikZ pictures, suitable for embedding in LaTeX. It takes as its input the first CLI argument that is not an option, or, if there is no such argument, STDIN.

$ cat examples/example.yml | argmap2tikz.lua

\begin{tikzpicture}

  {
    c1/"Brunellus is irrational
"[claim]
      --[opposingedge]
      r2/"" [opposing] // [ tree layout ] {

\end{tikzpicture}      

Note that it uses TikZ's graph support, and TikZ's graph support only works with lualatex. So you need to use lualatex instead of pdflatex or xelatex to convert the tikz code to PDF.

Also note that the resulting tikz code requires some settings in your preamble. For the required settings, see the --includes option below.

The following options are available:

-s, --standalone: generate a standalone LaTeX file.

$ argmap2tikz.lua -s examples/example.yml

\documentclass[tikz]{standalone}
    \usepackage{tikz}
    \usetikzlibrary{graphs,graphdrawing}

\begin{document}
\begin{tikzpicture}

\end{tikzpicture}
\end{document}

This standalone LaTeX file is suitable for generating a PDF image of the map, which can then be used as is or converted into other formats using other tools.

e.g.

argmap2tikz.lua -s examples/example.yml > output/example.tex
lualatex output/example
convert output/example.pdf output/example.png

-i, --includes: dump lines suitable for including in a LaTeX preamble

-t, --template: dump lines suitable for including in a LaTeX preamble in a Pandoc template.

If you want to use the tikz code inside your own LaTeX file, include the output of --includes in your preamble:

$ argmap2tikz.lua -i

    \usepackage{tikz}
    \usetikzlibrary{graphs,graphdrawing}

    \usepackage{varwidth}
    \newcommand{\argmapmaxnodewidth}{15em}

If you want to use the tikz code with Pandoc, include the output of --template in a custom Pandoc LaTeX template, and set the metadata variable argmaps to true, either in your document's YAML metadata block, or using the --metadata option.

$ argmap2tikz.lua -t

$if(argmaps)$
    \usepackage{tikz}
    \usetikzlibrary{graphs,graphdrawing}

    \usepackage{varwidth}
    \newcommand{\argmapmaxnodewidth}{15em}
$endif$

-h, --help: Display a brief synopsis of these options.

Collapsible Section: pandoc-argmap.lua

pandoc-argmap.lua

pandoc-argmap.lua is a Pandoc Lua filter for converting YAML argument maps embedded in a Pandoc document, using argmap2mup.lua and argmap2tikz.lua.

If you're not familiar with Pandoc, see the official: Pandoc - Getting started with pandoc.

Default Google Folder

I put a line like this at the top of my personal copy of pandoc-argmap.lua:

local gdriveFolder = "11w-foIj3p_FWSUROEX0VJg1KsslhJR0m"

This tells Pandoc to upload all the maps it generates to a specific folder on my Google Drive.

Embedding Maps in Markdown

Embedded argument maps are represented as code blocks with the class argmap.

    ``` {.argmap}
    "Brunellus is irrational":
      r1:
        "Brunellus is a donkey.":
        "All donkeys are irrational.":
      o1:
        "Brunellus studied in Paris.":
        "-Most who study in Paris are rational.":
    ```

As per Pandoc's markdown format for Fenced code blocks, you can use:

  • # for an id
  • . for classes
  • attribute="value" for assigning values.

If you want your argmap to be converted into mapjs format, then add the attribute to="js". See section Generating mapjs Interactive Argument Maps for more details.

A name and Google Drive ID can also be specified as attributes of the code block:

    ``` {.argmap name="A Donkey Argument" gid="1X6uD8KyrSawW2qSqhmPaNtuSy-6eEK6g"}
    "Brunellus is irrational":
      r1:
        "Brunellus is a donkey.":
        "All donkeys are irrational.": 
      o1:
        "Brunellus studied in Paris.":
        "-Most who study in Paris are rational.":
    ```

Converting Maps Embedded in Markdown

To convert a markdown document containing argument maps, use the --lua-filter option with Pandoc:

pandoc examples/example.md -o test/output/example.html --lua-filter pandoc-argmap.lua

By default, this will generate a png image representing the argmap.

Alternatively, the next two sections describe how to generate an svg image or a mapjs interactive map.

HTML5 Output

If the output format is html5, and you are not using the mapjs output format, then the filter generates individual pdf files for each map, and uses pdf2svg to convert them to svg, which are embedded directly into the html file.

Displaying Argmaps with mapjs

You can display mindmup JSON files on a web page as an interactive mindmap without using the MindMup website, by using a customised version of mapjs.

Here is an example:

Argument Map Example 1 in mapjs format

mapjs is an earlier, open source version of mindmup, and while it does not look as attractive, it provides similar functionality.

See Generating Interactive Argument Maps mapjs for details of how to use it.

Uploading Maps to Google Drive

By default, pandoc-argmap.lua does the following:

  1. Replaces each embedded YAML map with a TikZ generated image of the map;
  2. Generates a MindMup map;
  3. Uploads it to Google Drive, and links the image to the MindMup url.

It will not do this when displaying in argmapjs format.

Alternatively, if your output format is markdown and you add the code block attribute "tidy=true", the filter will skip step '1. Replacing the map with a TikZ image' above, and will just do steps 2 and 3.

e.g.

    ``` {.argmap tidy=true}
    "Brunellus is irrational":
    ```

In addition, if the Google Drive gid attribute for the file is added to the code block, it will associate the map in your markdown file with a specific map on your Google Drive. This will ensure that future conversions will update that map, rather than cluttering your Google Drive with a new map every time.

Generating PDFs

If you want to generate a PDF file using LaTeX, you will also need to use the --pdf-engine=lualatex option, and use a custom template with the --template:

pandoc examples/example.md -o output/examples/example.pdf --lua-filter pandoc-argmap.lua --pdf-engine lualatex --template examples example-template.latex

If you prefer, you could use the --include-in-header option instead of a custom template:

argmap2tikz.lua -i > output/header.tex
pandoc examples/example.md -o output/example.pdf --lua-filter pandoc-argmap.lua --pdf-engine lualatex --include-in-header output/header.tex

LaTeX and PDF output

If the output format is latex, the filter directly embeds the TikZ image code into the latex file as raw latex. So if you want to use Pandoc to directly produce a PDF using one of its latex engines, you will need to specify --pdf-engine lualatex and use a custom template or the --include-in-header option.

pandoc examples/example.md -o output/example.pdf --lua-filter pandoc-argmap.lua --template custom.latex --pdf-engine lualatex

Output in All Other Formats

For all other formats, it uploads the mindmup file to Google Drive and generates individual pdf files for each map, and then converts them to png using ImageMagick's convert command. It replaces the code block with a paragraph containing the generated image, linked to the mindmup file.

Expandable Section: Generating Interactive Argument Maps (mapjs)

Generating Interactive Argument Maps (mapjs)

Argument Map Example 1 in mapjs format, embedded in html page

This interactive mindmap format is an alternative way to display your argument maps in html pages.

Installing mapjs

I have added a custom branch of repo s6mike/mapjs-webpack-example at custom to a sub-folder, map-js.

To set this up, you will need to:

  1. Install node.js and then run npm --prefix [path to mapjs folder] install to grab the dependencies.

  2. Call npm run --prefix [path to mapjs folder] pack-js to build the app with webpack.

This builds the client side JavaScript file which activates the maps: mapjs-example/site/main.js.

Creating html pages containing mapjs with Pandoc

You can convert an argmap inside a markdown file into html containing mapjs by doing the following:

1a ) Either adding attribute to="js" to the code block:

  ``` {.argmap name="An argument about a donkey" to="js"}
  "Brunellus is irrational":
    r1:
      "Brunellus is a donkey.":
      "All donkeys are irrational.":
    o1:
      "Brunellus studied in Paris.":
      "-Most who study in Paris are rational.":
  ```

b) Or you can set a default value to: value for all argmap blocks in the document by adding this to the document metadata:

argmap:
  to: js

e.g.

    ---
    title: An Example
    argmaps: true
    argmap:
      to: js
    ---

    This is an example of an argument about a donkey!

    ``` {.argmap name="An argument about a donkey"}
    "Brunellus is irrational":
      r1:
        "Brunellus is a donkey.":
        "All donkeys are irrational.":
      o1:
        "Brunellus studied in Paris.":
        "-Most who study in Paris are rational.":
    ```

2 ) The simplest way to convert your argmap code block into the mapjs format is to use the pandoc-argmap.lua filter and provided Pandoc template to convert the markdown document into html containing mapjs.

e.g.

pandoc input/example-updated.md --template /pandoc-templates/mapjs/mapjs-main-html5.html --metadata=mapjs-output-js:/mapjs-example/site/main.js --metadata=css:mapjs-default-styles.css --lua-filter=/src/pandoc-argmap.lua > output.html

3 ) Open output.html in a browser to see the map.

If you can see the map controls but not the map, check the console for errors.

If you get a CORS origin error, you can fix it easily in Chrome by opening it with the command line option --allow-file-access-from-files. Other browser should have similar solutions.

Note that this disables a security feature, so best to use this option only when viewing mapjs files.

Using mapjs in your own projects

You can adapt the above for your own DIY approach:

1 ) Create a JSON file using src/argmap2mup, and put it somewhere accessible from your web page.

2 ) Embed it into an html page using html like this, replacing [JSON file path] with the path:

<div id="container_argmap1" class="container_argmapjs">
  <script type="application/json" class="argmap_json" src="[JSON file path]"></script>
</div>

The container and script classes must be as above. The container div id can be as you wish.

3 ) Add the mapjs script built from webpack: <script src="/mapjs-example/site/main.js"></script>

4 ) Add the included stylesheet to ensure the mapjs is formatted properly: <link rel="stylesheet" href="mapjs-default-styles.css" />

I have written scripts to help with this workflow, you may be able to adapt them for your purposes. See scripts/bash-aliases-argmap.sh.

See mapjs README.md for a customised version of the original (and brief) mapjs documentation.

For more information about the structure of the mapjs output format (.json/.mup), see: Data Format · mindmup/mapjs Wiki.

See Displaying Argmaps with mapjs for other ways to use mapjs with argmap.


Troubleshooting

  1. Error sh: 1: webpack: not found: Install node.js, navigate to folder mapjs and run npm install.

License

This project is licensed under the MIT License - see the LICENSE.md file for details.

Contributing

I could use all the help I can get :)

So if you'd like to help make argument maps more effective, have a look at CONTRIBUTING.md.

Relationships to Other Projects

This repo inherits from:

1 ) https://github.com/dsanson/argmap: The source of most of the functionality and documentation.

Changes include: Issue fixes (potentially caused by different lua version), and renamed files, added scripts, environment and config files for development convenience.

2 ) https://github.com/mindmup/mapjs-webpack-example: To display mindmup JSON files on a web page as an interactive mindmap, without using the MindMup website. This is an earlier, open source version of mindmup, customised to display the output from argmap2mup by passing the filename to the webpack build as an environment variable. See Using mapjs to display argmaps section for more details.

Changes include: moving the functionality to a sub-folder, and getting start.js to dynamically read JSON files from the web page its on, instead of building them into webpack.


Uses Semantic Versioning 2.0.0 and Conventional Commits 1.0.0.

Though documentation is not yet precise and comprehensive! Lua code is well documented, but the bash scripts still need to be properly documented.

Note that test files, and bash script functions beginning with __ are not considered part of a public API, and therefore may change during patch updates without warning.