Skip navigation Home Edit Add Diff Zip Upload

Oddµ: A minimal wiki

This program helps you run a minimal wiki, blog, digital garden, memex or Zettelkasten. There is no version history.

It’s well suited as a self-hosted, single-user web application, when there is no need for collaboration on the site itself. Links and email connect you to the rest of the net. The wiki can be public or private. Perhaps it just runs on your local machine, unreachable from the Internet.

It’s well suited as a secondary medium for a close-knit group: collaboration and conversation happens elsewhere, in chat, on social media. The wiki serves as the text repository that results from these discussions. As there are no logins and no version histories, it is not possible to undo vandalism and spam. Only allow people you trust write-access to the site.

It’s well suited as a simple static site generator. There are no plugins.

When Oddµ runs as a web server, it serves all the Markdown files (ending in .md) as web pages. These pages can be edited via the web.

Oddmu adds the following extensions to Markdown: local links [[like this]], hashtags #Like_This and fediverse account links like @alex@alexschroeder.ch.

If your pages don’t provide their own title (# title), the file name (without .md) is used as the title. Subdirectories are created as necessary.

Other files can be uploaded and images (ending in .jpg, .jpeg, .png, .heic or .webp) can be resized when they are uploaded (resulting in .jpg or .png files).

Documentation

This project uses man(1) pages. They are generated from text files using scdoc. These are the files available:

oddmu(1): This man page has a short introduction to Oddmu, its configuration via templates and environment variables, plus points to the other man pages.

oddmu(5): This man page talks about the Markdown and includes some examples for the non-standard features such as table markup. It also talks about the Oddmu extensions to Markdown: wiki links, hashtags and fediverse account links. Local links must use percent encoding for page names so there is a section about percent encoding. The man page also explains how feeds are generated.

oddmu-releases(7): This man page lists all the Oddmu versions and their user-visible changes.

oddmu-version(1): This man page documents the “version” subcommand which you can use to get the installed Oddmu version.

Working locally:

oddmu-links(1): This man page documents the “links” subcommand which you can use to get the outgoing links for a page.

oddmu-list(1): This man page documents the “list” subcommand which you can use to get page names and page titles.

oddmu-replace(1): This man page documents the “replace” subcommand to make mass changes to the files much like find(1), grep(1) and sed(1) or perl(1).

oddmu-search(1): This man page documents the “search” subcommand which you can use to build indexes – lists of page links. These are important for feeds.

oddmu-search(7): This man page documents how search and scoring work.

oddmu-toc(1): This man page documents the “toc” subcommand which you can use to generate a table of contents linking to all the headings on the page.

Reporting:

oddmu-missing(1): This man page documents the “missing” subcommand to list local links that don’t point to any existing pages or files.

oddmu-hashtags(1): This man page documents the “hashtags” subcommand to count the hashtags used from the command line.

Static site generator:

oddmu-html(1): This man page documents the “html” subcommand to generate HTML from Markdown pages from the command line.

oddmu-static(1): This man page documents the “static” subcommand to generate an entire static website from the command line, avoiding the need to run Oddmu as a server. Also great for archiving.

oddmu-notify(1): This man page documents the “notify” subcommand to add links to hashtag pages, index and changes for a given page. This is useful when you edit the Markdown files locally.

Configuration:

oddmu-templates(5): This man page documents how the templates can be changed (how they must be changed) and lists the attributes available for the various templates.

System administration:

oddmu-apache(5): This man page documents how to set up the Apache web server for various common tasks such as using logins to limit what visitors can edit.

oddmu-filter(7): This man page documents how to exclude subdirectories from search and archiving.

oddmu-nginx(5): This man page documents how to set up the freenginx web server for various common tasks such as using logins to limit what visitors can edit.

oddmu.service(5): This man page documents how to setup a systemd unit and have it manage Oddmu. “Great configurability brings great burdens.”

oddmu-webdav(5): This man page documents how to set up the Apache web server so that the wiki can be accessed via Web-DAV.

Leaving:

oddmu-export(1): This man page documents how to export all the pages as one RSS feed so that you can import them all into a new platform that doesn’t use Markdown files.

Building

To build the binary:

go build

The man pages are already built. If you want to rebuild them, you need to have scdoc installed.

make docs

The Makefile in the man directory has targets to create Markdown and HTML files.

The HEIC library uses C code and prevents cross-compilation.

As the repository changed URLs a few times (from GitHub, to self-hosted using cgit to self-hosted using legit), there is no way to install it using go install. You need to git clone the repository and build it locally.

Running

The working directory is where pages are saved and where templates are loaded from. You need a copy of the template files in this directory.

Here’s how to build and run straight from the source directory:

go run .

The program serves the local directory as a wiki on port 8080. Point your browser to http://localhost:8080/ to use it.

Once the oddmu binary is built, you can run it instead:

./oddmu

To read the main man page witihout installing Oddmu:

man -l man/oddmu.1

Installing

This installs oddmu into $HOME/.local/bin and the manual pages into $HOME/.local/share/man/.

make install

Here’s an example using GNU Stow to install it into /usr/local/stow in a way that allows you to uninstall it later:

sudo mkdir /usr/local/stow/oddmu
sudo make install PREFIX=/usr/local/stow/oddmu/
cd /usr/local/stow
sudo stow oddmu

Hacking

If you’re interested in making changes to the code, here’s a high-level introduction to the various source files.

The code of this package is licensed to you under the AGPL-3.0-or-later license. If you do make changes and your site is public, be aware of section 13:

… if you modify the Program, your modified version must prominently offer all users interacting with it remotely through a computer network (if your version supports such interaction) an opportunity to receive the Corresponding Source of your version by providing access to the Corresponding Source from a network server at no charge, through some standard or customary means of facilitating copying of software.

Changing the markup rules

If you want to change the markup rules, your starting point should be parser.go. Make sure you read the documentation of Go Markdown and note that it offers MathJax support (needs a change to the view.html template so that the MathJax Javascript gets loaded) and MMark support, and it shows how extensions can be added.

Filenames and URL path

One of the sad parts of the code is the distinction between path and filepath. On a Linux system, this doesn’t matter. I suspect that it also doesn’t matter on MacOS and Windows because the file systems handle forward slashes just fine. The code still tries to do the right thing. A path that is derived from a URL is a path with slashes. Before accessing a file, it has to be turned into a filepath using filepath.FromSlashes and in the rare case where the inverse happens, use filepath.ToSlashes. Any path received via the URL path uses slashes and needs to be converted to a filepath before passing it to any os function. Any path received within a path/filepath.WalkFunc is a filepath and needs to be converted to use slashes when used in HTML output.

If you need to access the page name in code that is used from a template, you have to decode the path. See the code in diff.go for an example.

HTTP handlers

The URL paths all have the form /action/directory/pagename (with directory being optional and pagename sometimes being optional). If you need to limit access in Apache or nginx or some other web server acting as a reverse proxy, you can do that. See man oddmu-apache and man oddmu-nginx for some configuration examples.

This is how you can prevent some actions by simply not passing them on to Oddmu, or you can require authentication for certain actions. Furthermore, you can do the same for directories, allowing you to use subdirectories as separate sites, each with their own editors.

Templates

The themes folder has some ideas of how to tweak the HTML templates.

Permissions

An unexplored idea would be to parse a config file that has usernames and passwords, groups usernames into roles, and assigns access to the various actions based on these roles. This would obviate the need for a web server acting as a reverse proxy.

Then again, not having to care about roles and permissions has been a relief.

Dependencies

This section lists the non-standard libraries Oddmu uses and their respective licenses.

github.com/gomarkdown/markdown is used to generate the web pages from Markdown. BSD-2-Clause.

github.com/microcosm-cc/bluemonday is used to strip rendered search results of all HTML except for the bold tag. Regular HTML generated from pages is not sanitized. Don’t give people you don’t trust access to your wiki. BSD-3-Clause.

github.com/pemistahl/lingua-go detects languages in order to set the language tag in templates. This in turn can be used by browsers to get hyphenation right. Apache-2.0.

github.com/gabriel-vasile/mimetype is used to sniff the MIME type of files with unknown filename extensions. MIT.

github.com/gen2brain/heic is used to decode HEIC files (the new default file format for photos on iPhones). LGPL-3.0-only.

github.com/disintegration/imaging is used to resize images. MIT.

github.com/edwvee/exiffix is used to rotate images before resizing them if the EXIF data says the image wasn’t taken with the default orientation of the camera. This is necessary because after resizing, the EXIF data is gone. MIT.

github.com/google/subcommands is used for the parsing and documenting of subcommands. Apache-2.0.

github.com/muesli/reflow/wordwrap is used to wrap the search subcommand output. MIT.

github.com/hexops/gotextdiff is used to show a compact unified diff on the command line before doing any replacements. BSD-3-Clause.

github.com/sergi/go-diff/diffmatchpatch is used to show the page diffs on the web. MIT.

github.com/fsnotify/fsnotify is used to watch the filesystem for changes. BSD-3-Clause.

golang.org/x/exp/constraints for the computation of the intersection between two sets of pages. BSD-3-Clause.

github.com/stretchr/testify/assert is used for testing. MIT.

Bugs

If you spot any, contact me.

References

Writing Web Applications provided the initial code for this wiki.