alonswartz / notesium

Markdown notes system with bi-directional links, Vim integration and local webapp
https://www.notesium.com
MIT License
60 stars 5 forks source link
d3 evergreen-notes fzf graph-visualization markdown note-taking vim zettelkasten

Notesium - A simple yet powerful system for networked thought

Writing does not make intellectual endeavours easier, it makes them possible. Deepen understanding, insight, and allow for structure to emerge organically by linking notes.

Build status Latest Release License

image: web interface screenshot


Notesium (pronounced no-tes-ee-im), is an opiniated, simplistic yet powerful system for creating, storing and accessing knowledge by means of notes.

It aspires and is designed to:

Warning: Experimental, proof-of-concept. Here be dragons!

Table of contents

Features

Screenshots

[Web] Panels: compact notes list and graph panels image: web compact-list graph panels

[Web] Panels: labels, detailed notes list and note metadata sidepanels image: web labels detailed-list metadata panels

[Web] Graph: visualize the relationships between notes with an interactive force graph image: web graph fullscreen

[Web] Graph: filter notes with emphasized matches, and preview note image: web graph fullscreen preview

[Web] List: prefixed with associated labels and sorted alphabetically image: web prefixed label

[Web] List: prefixed with modification date and sorted per modification time image: web prefixed mtime

[Web] Links: display all links image: web links all

[Web] Links: display links related to the current note image: web links related

[Web] Links: link insertion triggered by [[ image: web link insertion

[Web] Lines: full text search image: web full text search

[Vim] List: prefixed with associated labels and sorted alphabetically image: vim prefixed label

[Vim] List: prefixed with modification date and sorted per modification time image: vim prefixed mtime

[Vim] Links: display all links image: vim links all

[Vim] Links: display links related to the current note image: vim links related

[Vim] Links: link insertion triggered by [[ image: vim link insertion

[Vim] Lines: full text search (light theme) image: vim full text search light

CLI

Notesium is primarily tested and supported on Linux, with only preliminary tests performed on macOS and Windows.

Installation

Download the latest release for your platform, making sure to give it executable permissions and available in your PATH.

# Example for Linux 64-bit
curl -sLO https://github.com/alonswartz/notesium/releases/latest/download/notesium-linux-amd64
curl -sLO https://github.com/alonswartz/notesium/releases/latest/download/checksums.txt
sha256sum --check --ignore-missing checksums.txt && rm checksums.txt
chmod +x notesium-linux-amd64
mv notesium-linux-amd64 $HOME/.local/bin/notesium
# Example for macOS 64-bit
curl -sLO https://github.com/alonswartz/notesium/releases/latest/download/notesium-darwin-amd64
curl -sLO https://github.com/alonswartz/notesium/releases/latest/download/checksums.txt
shasum -a 256 -c checksums.txt --ignore-missing && rm checksums.txt
chmod +x notesium-darwin-amd64
mv notesium-darwin-amd64 $HOME/bin/notesium
# Example for Windows 64-bit (in PowerShell)
Invoke-WebRequest -Uri "https://github.com/alonswartz/notesium/releases/latest/download/notesium-windows-amd64.exe" -OutFile "./notesium-windows-amd64.exe"
Invoke-WebRequest -Uri "https://github.com/alonswartz/notesium/releases/latest/download/checksums.txt" -OutFile "./checksums.txt"
$hash = Get-FileHash -Path "./notesium-windows-amd64.exe" -Algorithm SHA256
$checksums = Get-Content "./checksums.txt"
if ($checksums -match $hash.Hash) { Write-Output "Checksum verified"; Remove-Item "./checksums.txt" } else { Write-Error "Checksum verification failed!" }
New-Item -ItemType Directory -Path "$env:USERPROFILE\AppData\Local\Microsoft\WindowsApps" -Force
Move-Item -Path "./notesium-windows-amd64.exe" -Destination "$env:USERPROFILE\AppData\Local\Microsoft\WindowsApps\notesium.exe"

Or build from source.

git clone https://github.com/alonswartz/notesium.git
cd notesium
./web/app/make.sh all
go build -ldflags "
    -X main.gitversion=$(git describe --tags --long --always --dirty) \
    -X main.buildtime=$(date -u +%Y-%m-%dT%H:%M:%SZ)"
ln -s $(pwd)/notesium $HOME/.local/bin/notesium

Shell completion

Bash completion is available for all notesium commands and option flags. Add the following line to your $HOME/.bashrc or similar location.

command -v notesium >/dev/null && source <(notesium extract completion.bash)

Usage

$ notesium help
Usage: notesium COMMAND [OPTIONS]

Commands:
  home              Print path to notes directory
  new               Print path for a new note
    --verbose       Output key:value pairs of related info
    --ctime=        Use specified ctime instead of now (YYYY-MM-DDThh:mm:ss)
  list              Print list of notes
    --color         Color code prefix using ansi escape sequences
    --labels        Limit list to only label notes (ie. one word title)
    --orphans       Limit list to notes without outgoing or incoming links
    --sort=WORD     Sort list by date or alphabetically (ctime|mtime|alpha)
    --prefix=WORD   Prefix title with date or linked label (ctime|mtime|label)
    --date=FORMAT   Date format for ctime/mtime prefix (default: 2006-01-02)
  links [filename]  Print list of links
    --color         Color code using ansi escape sequences
    --outgoing      Limit list to outgoing links related to filename
    --incoming      Limit list to incoming links related to filename
    --dangling      Limit list to broken links
  lines             Print all lines of notes (ie. fulltext search)
    --color         Color code prefix using ansi escape sequences
    --prefix=title  Prefix each line with note title
  stats             Print statistics
    --color         Color code using ansi escape sequences
    --table         Format as table with whitespace delimited columns
  web               Start web server
    --webroot=PATH  Path to web root to serve (default: embedded webroot)
    --open-browser  Launch default web browser with web server URL
    --stop-on-idle  Automatically stop when no activity is detected
    --port=INT      Port for web server to listen on (default: random)
    --no-check      Disable daily new version checks
    --writable      Allow writing of notes in NOTESIUM_DIR via API
  extract [path]    Print list of embedded files or contents of file path
  version           Print version
    --verbose       Output key:value pairs of related info
    --check         Check if a newer version is available

Environment:
  NOTESIUM_DIR      Path to notes directory (default: $HOME/notes)

Web

Notesium ships with an embedded web interface which is self contained and runs locally. To allow writing of notes, be sure to specify the --writable flag.

notesium web --writable --open-browser

The web interface stores state and preferences such as editor settings, panel state, start-of-week, and sorting preferences. The state is port-specific, meaning that to retain them across sessions, the same port must be used. For example:

notesium web --writable --open-browser --port=8080

Keybindings

Global

Binding Comment
space n n Open new note for editing
space n d Open new or existing daily note
space n w Open new or existing weekly note
space n l Finder: List with prefixed label, sorted alphabetically
space n m Finder: List with prefixed date modified, sorted by mtime
space n c Finder: List with prefixed date created, sorted by ctime
space n k Finder: Links related to active note (or all if none open)
space n s Finder: Full text search across all notes
space n g Open fullscreen force graph view

Note tabs

Binding Comment
C-h Switch to note tab on the left of the active note tab
C-l Switch to note tab on the right of the active note tab
C-^ C-6 Switch to previously active note tab

For more keybindings see below.

Finder

The finder is typically opened using global keybindings or icons in the ribbon, but is also seamlessly integrated throughout the web interface.

Keybindings

Binding Comment
C-p Toggle preview
C-j Select next entry (down)
C-k Select previous entry (up)
Enter Submit selected entry
Esc Dismiss finder

Search syntax

Token Match Type Comment
word exact-match Items that include word
foo bar multiple exact match (AND) Items that include both foo AND bar

Editor modes

Default mode

Mode Binding Comment
none Tab Enter editing mode (focus active note)
all C-s Save note
edit [[ Insert selected note link via Finder (mtime sorted)
edit Alt-k Insert selected note link via Finder (mtime sorted)
edit Shift-Tab Auto-indent current line or selection
edit C-] Indent current line or selection
edit C-[ Dedent current line or selection
edit Esc Exit editing mode (unfocus)

Vim mode

The editor supports Vim mode, which attempts to emulate the most useful features of Vim as faithfully as possible, but is not a complete implementation. It does however feature the following:

Mode Binding Comment
none Tab Enter normal mode (focus active note)
all C-s Save note and set normal mode
all C-l C-h C-6 Note tab keybinds passthrough
normal space n <char> Global keybinds passthrough
normal ge gx Open link under cursor
insert [[ Insert selected note link via Finder (mtime sorted)
insert Alt-k Insert selected note link via Finder (mtime sorted)
command :set [no]wrap Set line wrapping
command :set [no]conceal Set conceal formatting

Prefer the real thing? Notesium supports Vim integration.

Table formatting and navigation

The editor will recognize when the cursor is placed within a table structure (identified by lines starting with the | character), and provide formatting and navigation.

Mode Binding Comment
table Tab Format table and advance column (right)
table Shift-Tab Navigate to previous column (left)
table Alt-Arrow Navigate rows and columns

Syntax highlighting and concealment

The editor is configured to syntax highlight based on Markdown formatting.

By default, formatting characters such as those used for bold, italic and code will be concealed except on the active-line. In addition, links will also be concealed and only display the title. This setting can be toggled using the icon in the note sidebar.

The following table lists the Markdown syntax of elements that will be formatted and/or syntax highlighted, and indicates whether concealment of the formatting characters is supported.

Syntax Concealment Comment
# heading 1 No If on first line, text after # is considered the note title
## heading 2 No Heading level corresponds to the number of # symbols
*italic text* Yes Italic, _italic_ also supported
**bold text** Yes Bold, __bold__ also supported
***italic and bold*** Yes Italic and bold, ___italic and bold___ also supported
> blockquote No Block quote
- list item No Unordered list
`code` Yes Inline code
```fenced codeblock``` No Fenced code block
~~strikethrough~~ Yes Strikethrough
--- Yes Horizontal rule. Displayed as full-width line when concealed
[link title](64218088.md) Yes Inline link. Displayed as link title when concealed

Vim

Notesium does not supply a Vim plugin, it is up to the user to write their own Vim commands and configure keybindings. That said, below are some fairly generic commands, with preferences configured in the keybindings.

Example integration

let $NOTESIUM_DIR = trim(system("notesium home"))
let $NOTESIUM_WEEKSTART = 1 "0 Sunday, 1 Monday, ...

autocmd BufRead,BufNewFile $NOTESIUM_DIR/*.md inoremap <expr> [[ fzf#vim#complete({
  \ 'source': 'notesium list --sort=mtime',
  \ 'options': '+s -d : --with-nth 3.. --prompt "NotesiumInsertLink> "',
  \ 'reducer': {l->"[". split(l[0],':1: ')[1] ."](".split(l[0],':')[0].")"},
  \ 'window': {'width': 0.5, 'height': 0.5}})

command! -bang NotesiumNew
  \ execute ":e" system("notesium new")

command! -bang NotesiumWeb
  \ let s:options = "--stop-on-idle --open-browser" |
  \ execute ":silent !nohup notesium web ".s:options." > /dev/null 2>&1 &"

command! -bang -nargs=* NotesiumList
  \ let s:spec = {'dir': $NOTESIUM_DIR, 'options': '+s -d : --with-nth 3..'} |
  \ call fzf#vim#grep(
  \   'notesium list '.join(map(split(<q-args>), 'shellescape(v:val)'), ' '), 0,
  \   &columns > 79 ? fzf#vim#with_preview(s:spec, 'right', 'ctrl-/') : s:spec, <bang>0)

command! -bang -nargs=* NotesiumLinks
  \ let s:spec = {'dir': $NOTESIUM_DIR, 'options': '-d : --with-nth 3..'} |
  \ call fzf#vim#grep(
  \   'notesium links '.join(map(split(<q-args>), 'shellescape(v:val)'), ' '), 0,
  \   &columns > 79 ? fzf#vim#with_preview(s:spec, 'right', 'ctrl-/') : s:spec, <bang>0)

command! -bang -nargs=* NotesiumSearch
  \ let s:spec = {'dir': $NOTESIUM_DIR, 'options': '-d : --with-nth 3..'} |
  \ call fzf#vim#grep(
  \   'notesium lines '.join(map(split(<q-args>), 'shellescape(v:val)'), ' '), 0,
  \   &columns > 79 ? fzf#vim#with_preview(s:spec, 'right', 'ctrl-/') : s:spec, <bang>0)

command! -bang -nargs=* NotesiumDaily
  \ let s:cdate = empty(<q-args>) ? strftime('%Y-%m-%d') : <q-args> |
  \ let s:output = system('notesium new --verbose --ctime='.s:cdate.'T00:00:00') |
  \ let s:filepath = matchstr(s:output, 'path:\zs[^\n]*') |
  \ execute 'edit ' . s:filepath |
  \ if getline(1) =~ '^\s*$' |
  \   let s:epoch = matchstr(s:output, 'epoch:\zs[^\n]*') |
  \   call setline(1, '# ' . strftime('%b %d, %Y (%A)', s:epoch)) |
  \ endif

command! -bang -nargs=* NotesiumWeekly
  \ let s:date = empty(<q-args>) ? strftime('%Y-%m-%d') : <q-args> |
  \ let s:output = system('notesium new --verbose --ctime='.s:date.'T00:00:01') |
  \ let s:epoch = str2nr(matchstr(s:output, 'epoch:\zs[^\n]*')) |
  \ let s:day = strftime('%u', s:epoch) |
  \ let s:startOfWeek = empty($NOTESIUM_WEEKSTART) ? 1 : $NOTESIUM_WEEKSTART |
  \ let s:diff = (s:day - s:startOfWeek + 7) % 7 |
  \ let s:weekBegEpoch = s:epoch - (s:diff * 86400) |
  \ let s:weekBegDate = strftime('%Y-%m-%d', s:weekBegEpoch) |
  \ let s:output = system('notesium new --verbose --ctime='.s:weekBegDate.'T00:00:01') |
  \ let s:filepath = matchstr(s:output, 'path:\zs[^\n]*') |
  \ execute 'edit ' . s:filepath |
  \ if getline(1) =~ '^\s*$' |
  \   let s:weekFmt = s:startOfWeek == 0 ? '%U' : '%V' |
  \   let s:yearWeekStr = strftime('%G: Week' . s:weekFmt, s:weekBegEpoch) |
  \   let s:weekBegStr = strftime('%a %b %d', s:weekBegEpoch) |
  \   let s:weekEndStr = strftime('%a %b %d', s:weekBegEpoch + (6 * 86400)) |
  \   let s:title = printf('# %s (%s - %s)', s:yearWeekStr, s:weekBegStr, s:weekEndStr) |
  \   call setline(1, s:title) |
  \ endif

nnoremap <Leader>nn :NotesiumNew<CR>
nnoremap <Leader>nd :NotesiumDaily<CR>
nnoremap <Leader>nw :NotesiumWeekly<CR>
nnoremap <Leader>nl :NotesiumList --prefix=label --sort=alpha --color<CR>
nnoremap <Leader>nm :NotesiumList --prefix=mtime --sort=mtime --color<CR>
nnoremap <Leader>nc :NotesiumList --prefix=ctime --sort=ctime --color --date=2006-01<CR>
nnoremap <Leader>nb :NotesiumLinks --incoming <C-R>=expand("%:t")<CR><CR>
nnoremap <Leader>nk :NotesiumLinks --color <C-R>=expand("%:t")<CR><CR>
nnoremap <Leader>ns :NotesiumSearch --prefix=title --color<CR>
nnoremap <silent> <Leader>nW :NotesiumWeb<CR>

" overrides for journal
if $NOTESIUM_DIR =~ '**/journal/*'
  nnoremap <Leader>nl :NotesiumList --prefix=label --sort=mtime --color<CR>
endif

Keybindings

Mode Binding Comment
insert [[ Opens note list, insert selection as markdown formatted link
normal <Leader>nn Opens new note for editing
normal <Leader>nd Opens new or existing daily note
normal <Leader>nw Opens new or existing weekly note
normal <Leader>nl List with prefixed label, sorted alphabetically (mtime if journal)
normal <Leader>nm List with prefixed date modified, sorted by mtime
normal <Leader>nc List with prefixed date created in custom format, sorted by ctime
normal <Leader>nb List all notes linking to this note (backlinks)
normal <Leader>nk List all links related to this note
normal <Leader>ns Full text search
normal <Leader>nW Opens browser with web view (auto stop webserver on idle)
fzf C-k C-j Move up and down in fzf window
fzf Enter Open selection
fzf C-t C-x C-v Open selection in new tab, split, vertical split
fzf C-/ Toggle preview
fzf Shift-Tab Multiple selection
normal ge Open the link under the cursor (vim-markdown)
normal [[ ]] Jump back and forward between headings (vim-markdown)

Fzf search syntax

Token Match Type Comment
sbtrkt fuzzy-match Items that fuzzy match sbtrkt
'word exact-match Items that include word
^word prefix exact-match Items that start with word
word$ suffix exact-match Items that end with word
!word inverse exact-match Items that do not include word
!^word inverse prefix exact-match Items that do not start with word
!word$ inverse suffix exact-match Items that do not end with word
foo bar multiple exact match (AND) Items that include both foo AND bar
foo \| bar multiple exact match (OR) Items that include either foo OR bar

Related Vim settings

" junegunn/fzf.vim
let $FZF_DEFAULT_OPTS="--reverse --filepath-word --no-separator --no-scrollbar "
let g:fzf_buffers_jump = 1
let g:fzf_layout = { 'window': { 'width': 0.85, 'height': 0.85 } }
let g:fzf_colors = {
    \ 'fg':      ['fg', 'Normal'],
    \ 'bg':      ['bg', 'Normal'],
    \ 'hl':      ['fg', 'Comment'],
    \ 'fg+':     ['fg', 'CursorLine', 'CursorColumn', 'Normal'],
    \ 'bg+':     ['bg', 'CursorLine', 'CursorColumn'],
    \ 'hl+':     ['fg', 'Statement'],
    \ 'info':    ['fg', 'PreProc'],
    \ 'pointer': ['fg', 'Exception'],
    \ 'marker':  ['fg', 'Keyword'],
    \ 'spinner': ['fg', 'Label'],
    \ 'header':  ['fg', 'Comment'] }

" preservim/vim-markdown
let g:vim_markdown_folding_style_pythonic = 1
let g:vim_markdown_folding_level = 2
let g:vim_markdown_frontmatter = 1
let g:vim_markdown_auto_insert_bullets = 0
let g:vim_markdown_new_list_item_indent = 0
let g:vim_markdown_toc_autofit = 1
let g:vim_markdown_conceal_code_blocks = 0

let g:markdown_fenced_languages = ['json', 'sh', 'shell=bash']
hi def link mkdHeading htmlH1
autocmd FileType markdown setlocal conceallevel=2

Custom URI protocol

Notesium does not support handling the custom URI protocol notesium:// itself directly, as this is very dependent on user preference. It is up to the user to write their own script.

Example integration

An example integration can been found in contrib/xdg-urxvt-nvim.sh, which supports opening the notes listing as well as opening a note for editing - in a new urxvt terminal window and neovim instance.

xdg-open notesium:///home/user/notes
xdg-open notesium:///home/user/notes/625d563f.md

For Vim users, opening the listing is useful when integrated with a launcher or desktop keybinding. Opening a note for editing is useful, for example, when using the web/app or web/graph.

Handler registration

To register the x-scheme-handler with the desktop environment, create the file $HOME/.local/share/applications/notesium.desktop with contents similar to the following (don't forget to update the Exec path):

[Desktop Entry]
Name=Notesium
Exec=/home/github/alonswartz/notesium/contrib/xdg-urxvt-nvim.sh %u
MimeType=x-scheme-handler/notesium;
Type=Application
Terminal=false
NoDisplay=true

Design assumptions and rationale

Filenames are 8 hexidecimal digits

There are only two hard things in Computer Science: cache invalidation and naming things. ~ Phil Karlton

Naming is hard. With regards to notes, common conventions are to use the current date, or note title, or both in concatenated form.

Notesium addresses this by using the UNIX epoch time, and further encoding it in hexidecimal, resulting in 8 characters for the identifier (e.g., 625d563f.md). This can later be easily decoded and formatted depending on the use case.

The .md extension is required so external tools can easily identify the filetype for syntax highlighting, as well as limit the processing of files by the parser.

Completely flat directory structure

It is better to prefer associative ontologies to hierarchical taxonomies.

Folders are often thought of as categories, but notes don't always fit neatly into just one category, which could result in decision paralysis when creating a new note, and can prematurely constrain what may emerge.

Links would also be broken when renaming a folder or moving a note from one folder to another.

Notesium assumes a flat directory structure, having all notes be siblings to one another, in one directory. Utilizing bi-directional links allows for the structure to emerge over time. Additionally, the label convention allows for a meta-hierarchical taxonomy to be created which can be useful in certain circumstances.

Titles are inferred from the first line

Note titles are like APIs, and when titled well they become an abstraction.

Notesium assumes note titles are on the first line of the note, in markdown H1 format, for example: # this is the title of a note.

Notes with one-word titles are considered labels

Even though tags are an ineffective association structure, Notesium supports the concept of labels, but there is no special label syntax. Instead, a label is just a regular note, and only considered a label if its title is one-word.

This becomes useful, for example, when issuing the following list command, each note title will be prefixed with its associated label (multiple times for each associated label), color coded, and sorted alphabetically, effectively creating a hierarchical taxonomy listing, which can further be filtered and searched.

notesium list --prefix=label --sort=alpha --color

Links are inline

Notes should be densely linked, allowing for structure to emerge organically, and may even help you see unexpected connections which may surprise you.

Notesium assumes note links use the inline markdown syntax, for example: [link text](625d563f.md). This makes it easier to parse, and simple to insert links with a keybinding.

Even though links are short, for an improved reading experience in Vim consider enabling conceallevel (see Related Vim settings). This is implemented in the web app and enabled by default.

Deterministic periodic notes convention

Notesium itself does not directly support periodic notes, instead the embedded Web app and Vim integration support daily and weekly notes using a creation-time convention (00:00:00 for daily and 00:00:01 on the first day of the week for weekly).

This convention results in deterministic, unique filenames, enabling instant identification of a specific periodic note, without the need for manual organization or special formatting - eliminating overhead and the need to search the entire notes directory for keywords or regular expressions.

Additionally, past and future periodic notes can also be created using this convention while retaining chronological order.

$ notesium new --ctime=2024-07-04T00:00:00
/home/user/notesium/notes/6685bbd0.md

$ notesium list --sort=ctime --prefix=ctime --date=15:04:05 | grep 00:00:00
6685bbd0.md:1: 00:00:00 Jul 04, 2024 (Thursday)
6681c750.md:1: 00:00:00 Jul 01, 2024 (Monday)
668075d0.md:1: 00:00:00 Jun 30, 2024 (Sunday)

Versioning

Versions are specified using git tags, based on semantic versioning, and use the format MAJOR.MINOR.PATCH. While the MAJOR version is 0, MINOR version bumps are considered MAJOR bumps per the semver spec.

Regression tests

Test suites for all commands and their options (except for --color) are included in the tests/ directory, along with fixtures acting as a notes corpus. The tests are dependent on bats-core.

Some tests are dependent on the modification datetime of the notes (sorting and prefixing), in which case the test suite will duplicate the corpus to a temporary directory and modify the mtime deterministically.

These test suites can be paused prior to teardown for manual inspection and additional testing by setting the PAUSE environmental variable.

# run all test suites
bats tests

# run a specific test suite
bats tests/list.bats

# run a subset of tests within a specific test suite
bats tests/list.bats --filter "mtime"

# run a specific test suite, but pause prior to teardown (--tap recommended)
PAUSE=y bats tests/list.bats --tap

Inspiration and thanks

License

The MIT License (MIT)

Copyright (c) 2023-2024 Alon Swartz