Kroki provides a web service that can be used to generate images from a wide range of supported DSL's, for example mermaid, PlantUML, GraphViz and many others.
This plugin allows the user to provide th DSL for an image in one of the kroki supported dialects within a code block. The remark processor will replace the image code block with the generated image.
The include code plugin is normally used in the context of a site generator using the remark markdown processor underneath. For example, to use the plugin within Docusaurus 2,
-
Include
remark-kroki-plugin": "0.1.0"
in the package.json of the docusaurus website project. -
Within docusaurus.config.js, configure the plugin with the following parameters:
- krokiBase: The base URL of the kroki web service. This is usually
https://kroki.io
unless e selfhosted kroki instance shall be used. - lang: The image transformations is applied for all codeblocks with this language.
- imgRefDir: The prefix that will be used for a generated image to create the link.
- imgDir: The directory where the generated image files are stored.
- langAliases: an array of markdown code languages that get treated as a kroki document (See blow)
- krokiBase: The base URL of the kroki web service. This is usually
Within Docusaurus 2 the plugin could be configured as below:
presets: [
[
'@docusaurus/preset-classic',
{
docs: {
sidebarPath: require.resolve('./sidebars.js'),
remarkPlugins: [
[require('remark-kroki-plugin'), { krokiBase: 'https://kroki.io', lang: "kroki", imgRefDir: "../img/kroki", imgDir: "static/img/kroki" }]
],
},
blog: {
showReadingTime: false,
remarkPlugins: [
[require('remark-kroki-plugin'), { krokiBase: 'https://kroki.io', lang: "kroki", imgRefDir: "../img/kroki", imgDir: "static/img/kroki" }]
],
},
theme: {
customCss: [
require.resolve('./src/css/custom.css'),
//require.resolve('./node_modules/prism-themes/themes/prism-cb.css')
],
},
},
],
]
The image generator will be triggered by a code block that uses the configured language within it's language parameter. The plugin will use the imgType
attribute to determine which generator shall be used within the kroki web service.
The generator will POST
the content of the code block to the kroki web service and either report an error
or generate an appropriate SVG file in the imgDir
.
The filename of the generated image will is generated as $(MD5 hash of the image code).svg
. When the file with the calculated filename already exists within the image directory, this will be used as the image file, otherwise
the kroki web service will be called to generate the file.
The user can decide to put generated images under version control so that images generated once can be reused until the text changes.
Generate a mermaid image
```kroki imgType="mermaid" imgTitle="Collaborating containers" graph TD subgraph Shop X Bx(Shop X) --> FX1((Fs X1)) --> Bx Bx --> FX2((Fs X2)) --> Bx end subgraph Shop Y By(Shop Y) --> FY1((Fs Y1)) --> By By --> FY2((Fs Y2)) --> By end subgraph Shop Z Bz(Shop Z) --> FZ1((Fs Z1)) --> Bz Bz --> FZ2((Fs Z2)) --> Bz end A(Data Center) --> Bx --> A A --> By --> A A --> Bz --> A ```
In some cases, it would make a lot of sense to preserve the original language tag, for example mermaid
documents are a first-class citizen in many markdown editors and is even explicitly supported inside popular code hosting services such as GitHub and GitLab.
In such cases, in would make sense to supply the optional langAliases
option as part of the kroki plugin setup like so, to avoid rewriting the language tag for existing code-blocks that work well in other environments not related to remark/kroki:
presets: [[
'@docusaurus/preset-classic',
{
docs: {
sidebarPath: require.resolve('./sidebars.js'),
remarkPlugins: [
[
require('remark-kroki-plugin'),
{
krokiBase: 'https://kroki.io',
lang: "kroki",
langAliases: ["mermaid"],
imgRefDir: "../img/kroki",
imgDir: "static/img/kroki"
}
]
],
}}
]]
This in turn allows for direct inclusing of mermaid blocks in a more straight-forward manner:
```mermaid graph TD subgraph Shop X Bx(Shop X) --> FX1((Fs X1)) --> Bx Bx --> FX2((Fs X2)) --> Bx end subgraph Shop Y By(Shop Y) --> FY1((Fs Y1)) --> By By --> FY2((Fs Y2)) --> By end subgraph Shop Z Bz(Shop Z) --> FZ1((Fs Z1)) --> Bz Bz --> FZ2((Fs Z2)) --> Bz end A(Data Center) --> Bx --> A A --> By --> A A --> Bz --> A ```
It might be good to allow the generated filename to be overwritten with a parameter.
The code has been realized in typescript follwing this excellent tutorial ff.