Skip to content

Latest commit

 

History

History
259 lines (189 loc) · 8.39 KB

0177-geneva.org

File metadata and controls

259 lines (189 loc) · 8.39 KB

geneva

This is yet another project by Max Rottenkolber (@eugeneia_). Geneva is the documentation system. It includes a few subsystems:

  • the core;
  • the markup language;
  • renderers for HTML, LaTeX and plain text.

Geneva separates a document structure from its representation.

Core package provides the way to define a document’s structure using calls to Lisp functions:

POFTHEDAY> (geneva:make-document
            (list
             (geneva:make-section (list "First section")
                (list
                 (geneva:make-paragraph (list "Foo bar"))
                 (geneva:make-paragraph (list "Blah minor"))))
             (geneva:make-section (list "Second section")
                 (list
                  (geneva:make-paragraph (list (geneva:make-bold "Hello World!")))))))
((:SECTION ("First section")
  ((:PARAGRAPH ("Foo bar"))
   (:PARAGRAPH ("Blah minor"))))
 (:SECTION ("Second section")
  ((:PARAGRAPH ((:BOLD "Hello World!"))))))

POFTHEDAY> (geneva.html:render-html * :index-p nil)

As you can see, the document is a bunch of Lisp lists. It easily can be rendered into the HTML:

<SECTION>
  <HEADER>
    <A NAME="section-1">
      <H1><SPAN CLASS="geneva-index">1</SPAN> First section</H1>
    </A>
  </HEADER>
  <P>Foo bar</P>
  <P>Blah minor</P>
</SECTION>

<SECTION>
  <HEADER>
    <A NAME="section-2">
      <H1><SPAN CLASS="geneva-index">2</SPAN> Second section</H1>
    </A>
  </HEADER>
  <P><B>Hello World!</B></P>
</SECTION>

Or you might want to render it into the plain text:

POFTHEDAY> (geneva.plain-text:render-plain-text * :index-p nil)

1 First section

   Foo bar

   Blah minor

2 Second section

   Hello World!

Humans usually prefer to use specialized markup languages. Geneva provides MK2 markup language to define a rich text. For example, text from the previous example can be written like that:

POFTHEDAY> (let ((text "
< First section

  Foo bar

  Blah minor

>

< Second section

  *Hello World!*

>"))
             (with-input-from-string (s text)
               (geneva.mk2:read-mk2 s)))
((:SECTION ("First section")
  ((:PARAGRAPH ("Foo bar"))
   (:PARAGRAPH ("Blah minor"))))
 (:SECTION ("Second section")
  ((:PARAGRAPH ((:BOLD "Hello World!"))))))

There is also a system to process documentation strings into the Geneva document. It can be used to render documentation for your own system. Docstrings can be written in MK2 markup.

Now we’ll create a test Lisp package and fill it with a variable, function and macro.

POFTHEDAY> (defpackage foo
             (:use #:cl)
             (:documentation "This is an example
                              of the package.

                              Documentation can be written in *MK2 format*.

                              And include _rich_ text with links."))

POFTHEDAY> (defun foo::bar (minor)
             "Makes some tranformation.

              *Arguments*:

              {minor} - an object to transform."

             ;; do the job
             (values))

POFTHEDAY> (defvar foo::*blah* :blah
             "Switches between two modes: {:blah} and {:minor}")

POFTHEDAY> (defmacro foo::with-minor (&body body)
             "Runs {body} with {:minor} mode applied."
             `(let ((foo::*blah* :minor))
                ,@body))

POFTHEDAY> (export '(foo::*blah* foo::bar foo::with-minor)
                   :foo)

Now we can build documentation for this package in two simple steps:

POFTHEDAY> (geneva.cl:api-document :foo)
((:SECTION ("foo (Package)")
  ((:PARAGRAPH ("This is an example of the package."))
   (:PARAGRAPH ("Documentation can be written in " (:BOLD "MK2 format") "."))
   (:PARAGRAPH ("And include " (:ITALIC "rich") " text with links."))
   (:SECTION ("*blah* (Variable)")
    ((:PARAGRAPH ((:BOLD "Initial Value:")))
     (:PARAGRAPH ((:FIXED-WIDTH ":BLAH")))
     (:PARAGRAPH
      ("Switches between two modes: " (:FIXED-WIDTH ":blah") " and "
       (:FIXED-WIDTH ":minor")))))
   (:SECTION ("bar (Function)")
    ((:PARAGRAPH ((:BOLD "Syntax:")))
     (:PARAGRAPH ("— Function: " (:BOLD "bar") " " (:ITALIC "minor")))
     (:PARAGRAPH ("Makes some tranformation."))
     (:PARAGRAPH ((:BOLD "Arguments") ":"))
     (:PARAGRAPH ((:FIXED-WIDTH "minor") " - an object to transform."))))
   (:SECTION ("with-minor (Macro)")
    ((:PARAGRAPH ((:BOLD "Syntax:")))
     (:PARAGRAPH
      ("— Macro: " (:BOLD "with-minor") " " (:FIXED-WIDTH "&body") " "
       (:ITALIC "body")))
     (:PARAGRAPH
      ("Runs " (:FIXED-WIDTH "body") " with " (:FIXED-WIDTH ":minor")
       " mode applied.")))))))

POFTHEDAY> (geneva.html:render-html * :index-p nil)

It will render this HTML:

<SECTION><HEADER><A NAME="section-1"><H1><SPAN CLASS="geneva-index">1</SPAN> foo (Package)</H1></A></HEADER><P>This is an example of the package.</P><P>Documentation can be written in <B>MK2 format</B>.</P><P>And include <I>rich</I> text with links.</P><SECTION><HEADER><A NAME="section-1-1"><H2><SPAN CLASS="geneva-index">1.1</SPAN> *blah* (Variable)</H2></A></HEADER><P><B>Initial Value:</B></P><P><CODE>:BLAH</CODE></P><P>Switches between two modes: <CODE>:blah</CODE> and <CODE>:minor</CODE></P></SECTION><SECTION><HEADER><A NAME="section-1-2"><H2><SPAN CLASS="geneva-index">1.2</SPAN> bar (Function)</H2></A></HEADER><P><B>Syntax:</B></P><P>— Function: <B>bar</B> <I>minor</I></P><P>Makes some tranformation.</P><P><B>Arguments</B>:</P><P><CODE>minor</CODE> - an object to transform.</P></SECTION><SECTION><HEADER><A NAME="section-1-3"><H2><SPAN CLASS="geneva-index">1.3</SPAN> with-minor (Macro)</H2></A></HEADER><P><B>Syntax:</B></P><P>— Macro: <B>with-minor</B> <CODE>&amp;body</CODE> <I>body</I></P><P>Runs <CODE>body</CODE> with <CODE>:minor</CODE> mode applied.</P></SECTION></SECTION>

Of cause, you can provide your own CSS stylesheets to make the page looks like you want.

I think Geneva might become a great replacement to reStructured text for documentation of my own libraries. Thank you, @eugeneia_!.

Though, it would be wonderful to add a little extensibility and ability to cross-referencing between different documentation sections.

By the way, in Geneva’s sources I found an interesting way to keep DRY principle. This piece of code reuses (content-values text-token) 5 times.

(defun render-text (text)
  "Render TEXT as HTML."
  (dolist (text-token text)
    (ecase (content-type text-token)
      (:plain (text #1=(content-values text-token)))
      (:bold (b #1#))
      (:italic (i #1#))
      (:fixed-width (code #1#))
      (:url (multiple-value-bind (string url) #1#
              (a [:href (or url string)] (or string url)))))))

I don’t know the proper name of the Lisp’s feature, but it allows to refer to the piece of data-structure defined earslier. The most common usage I’ve seen before is a circular list’s definition:

POFTHEDAY> '#1=(1 2 3 . #1#)
(1 2 3 1 2 3 1 2 3 1 2 3 1 2 3 1 2 3 1 2 3 1 2 3 1 2 3 1 2 3 1 2 3 1 2 3 1 2 3
 1 2 3 1 2 3 1 2 3 1 2 3 1 2 3 1 2 3 1 2 3 1 2 3 1 2 3 1 2 3 1 2 3 1 2 3 1 2 3
 1 2 3 1 2 3 1 2 3 1 2 3 1 2 3 1 2 3 1 2 3 1 2 3 1 2 3 1 2 3 1 2 3 1 2 3 1 2 3
 1 2 3 1 2 3 1 2 3 1 2 3 1 2 3 1 2 3 1 2 3 1 2 3 1 2 3 1 2 3 1 2 3 1 2 3 1 2 3
 1 2 3 1 2 3 1 2 3 1 2 3 1 2 3 1 2 3 1 2 3 1 2 3 1 2 3 1 2 3 1 2 3 1 2 3 1 2 3
 1 2 3 1 2 3 1 2 3 1 2 3 1 2 3 1 2 3 1 2 3 1 2 3 1 2 3 1 2 3 1 2 3 1 2 3 1 2 3
 1 2 3 1 2 3 1 2 3 1 2 3 1 2 3 1 2 3 1 2 3 1 2 3 1 2 3 1 2 3 1 2 3 1 2 3 1 2 3
 1 2 3 1 2 3 1 2 3 1 2 3 1 2 3 1 2 3 1 2 3 1 2 3 1 2 3 1 2 3 1 2 3 1 2 3 1 2 3
 1 2 3 1 2 3 1 2 3 1 2 3 1 2 3 1 2 3 1 2 3 1 2 3 1 2 3 1 2 3 1 2 3 1 2 3 1 2 3
 1 2 3 1 2 3 1 2 3 1 2 3 1 2 3 1 2 3 1 2 3 1 2 3 1 2 3 1 2 3 1 2 3 1 2 3 1 2 3
 1 2 3 1 2 3 1 2 3 1 2 3 1 2 3 1 2 3 1 2 3 1 2 3 1 2 3 1 2 3 1 2 3 1 2 3 1 2 3
 1 2 3 1 2 3 1 2 3 1 2 3 1 2 3 1 2 3 1 2 3 1 2 3 1 2 3 1 2 3 1 2 3 1 2 3 1 2 3
 1 2 3 1 2 3 1 2 3 1 2 3 1 2 3 1 2 3 1 2 3 1 2 3 1 2 3 1 2 3 1 2 3 1 2 3 1 2 3
 1 2 3 1 2 ...)

Update from [2020-11-19 Thu]

I’ve created an example project which uses Geneva and MK2 markup to build a documentation.

It also shows how to render docs using GitHub Actions.