hyalo-cli 0.7.1

CLI for exploring and managing Markdown knowledge bases with YAML frontmatter
Documentation

hyalo

Your markdown collection deserves a proper query language.

If you maintain an Obsidian vault, a Zettelkasten, documentation site, or any folder of .md files with YAML frontmatter, you've probably hit the limits of grep and manual editing. Hyalo gives you a fast, structured way to search, filter, and bulk-edit your markdown files from the command line.

What it does

  • Find files by frontmatter properties, tags, body content (regex), section headings, task status, or title
  • Bulk-update metadata — set, remove, or append to properties and tags across hundreds of files at once
  • Move files safely — rename or reorganize files and hyalo rewrites all [[wikilinks]] and [markdown](links) across the vault
  • Fix broken links — detect unresolved links and auto-repair them with fuzzy matching
  • Read content — extract specific sections or line ranges from files
  • Get an overview — see property/tag distributions, task counts, orphan files, and link health at a glance

Why hyalo?

  • Fast. Parallel scanning, streaming I/O, optional snapshot index. Handles 10,000+ file vaults in under a second.
  • Structured output. JSON by default with built-in --jq support. Easy to pipe into scripts, CI, or AI agents.
  • AI-agent friendly. Designed as a tool for Claude Code and other LLM coding agents. One command sets up the integration: hyalo init --claude.
  • Safe mutations. Dry-run mode on all write operations. Preview before committing changes.
  • Cross-platform. Works on macOS, Linux, and Windows. No runtime dependencies.

"Hyalo" — like "obsidian" — is a volcanic glass. The project started as a high-performance CLI for Claude Code to maintain Obsidian-compatible knowledgebases.

Installation

Homebrew (macOS & Linux)

brew tap ractive/tap
brew install ractive/tap/hyalo

Scoop (Windows)

scoop bucket add hyalo https://github.com/ractive/scoop-hyalo
scoop install hyalo

winget (Windows)

winget install ractive.hyalo

Cargo (from crates.io)

cargo install hyalo-cli

Intel Mac users: Homebrew bottles and pre-built binaries are only provided for Apple Silicon. If you're on an Intel Mac, use cargo install above.

Manual download

Download pre-built binaries from the GitHub Releases page. Binaries are available for Linux (x86_64, ARM64, glibc and musl), macOS (Apple Silicon), and Windows (x86_64, ARM64).

Build

cargo build --release

Usage

All commands accept these global flags:

Flag Description
-d/--dir <PATH> Root directory (default: ., override via .hyalo.toml)
--format json|text Output format (default: json, override via .hyalo.toml)
--jq <FILTER> Apply a jq expression to the JSON output (incompatible with --format text)
--count Print total as bare integer — shortcut for --jq '.total' (list commands only)
--hints / --no-hints Enable/disable drill-down command hints (default: on)
--site-prefix <PREFIX> Override site prefix for resolving root-absolute links
--index <PATH> Use a pre-built snapshot index (see Snapshot Index)
-q/--quiet Suppress warnings on stderr

All JSON output uses a consistent envelope: {"results": <payload>, "total": N, "hints": [...]}. total is present for list commands (find, tags summary, properties summary, backlinks). hints is always present (empty [] when --no-hints). --jq operates on the full envelope, e.g. --jq '.results[].file' or --jq '.total'.

Most flags have short aliases for quick interactive use:

Short Long Available in
-d --dir all commands
-e --regexp find
-p --property find, set, remove, append
-t --tag find, set, remove
-s --section find, read
-f --file find, read, set, remove, append, task, backlinks, mv
-g --glob find, set, remove, append, properties summary, properties rename, tags summary, tags rename, summary, links fix
-n --limit find
-n --recent summary
-l --lines read
-l --line task read, task toggle, task set-status
-s --status task set-status
-o --output create-index

Glob patterns use standard shell semantics: * matches within a single directory, ** matches across directory boundaries. For example, *.md matches top-level files only, while **/*.md matches all .md files recursively.

Configuration

Place a .hyalo.toml file in your working directory to set defaults for global flags:

# .hyalo.toml
dir = "./my-vault"   # default: "."
format = "text"      # default: "json"
hints = false        # default: true (set to false to suppress drill-down hints)
site_prefix = "docs" # override auto-derived prefix for absolute link resolution

All fields are optional. CLI flags always take precedence over config values. If .hyalo.toml is missing, hyalo silently uses built-in defaults; if the file is present but cannot be read or is malformed/invalid, hyalo warns on stderr and falls back to the built-in defaults.

Use --no-hints to explicitly disable hints when the config file enables them.

Absolute link resolution (site prefix)

Documentation sites often use root-absolute links like /docs/guides/setup.md. Hyalo resolves these by stripping a site prefix — turning /docs/guides/setup.md into the vault-relative path guides/setup.md.

By default, hyalo auto-derives the prefix from the last component of --dir:

--dir ../vscode-docs/docs  →  prefix = "docs"   (/docs/foo.md → foo.md)
--dir /home/me/wiki        →  prefix = "wiki"    (/wiki/foo.md → foo.md)
--dir .                    →  prefix = current directory name (e.g. "wiki")

Override when the directory name doesn't match the URL prefix:

# Directory is "content/" but links use "/docs/..." prefix
hyalo --site-prefix docs --dir ./content find --fields links

# Disable absolute-link resolution entirely
hyalo --site-prefix "" find --fields links

Precedence: --site-prefix flag > site_prefix in .hyalo.toml > auto-derived from --dir.

init

Initialize hyalo in the current project. Creates a .hyalo.toml config file with a dir setting pointing to your markdown directory.

# Basic init — creates .hyalo.toml
hyalo init

# Specify the markdown directory explicitly
hyalo init --dir my-vault

# Also set up Claude Code integration (skill + CLAUDE.md hint)
hyalo init --claude

Without --dir, hyalo auto-detects common documentation directories (docs/, knowledgebase/, wiki/, notes/, content/, pages/) by looking for a subdirectory that contains .md files. Falls back to . if none is found.

With --claude, hyalo additionally:

  • Creates .claude/skills/hyalo/SKILL.md so Claude Code automatically uses hyalo for markdown operations
  • Appends a hyalo usage hint to .claude/CLAUDE.md

All steps are idempotent — existing files are skipped, and duplicate hints are not added.

find

Search and filter files. Returns a JSON envelope {"results": [...], "total": N, "hints": [...]} where each item in results contains frontmatter properties, tags, sections, tasks, and links.

# All files
hyalo find

# Files with broken links (unresolved wikilinks or markdown links)
hyalo find --broken-links

# Content search (case-insensitive substring)
hyalo find "retry backoff"
hyalo find "retry" --tag research

# Regex content search (case-insensitive by default)
hyalo find --regexp "retry.*backoff"
hyalo find -e "TODO|FIXME|HACK"
hyalo find -e "fn\s+\w+_test" --tag rust

# Filter by property (operator: =, !=, >, >=, <, <=, existence, absence, or regex)
hyalo find --property status=draft
hyalo find --property status!=done
hyalo find --property priority>=3
hyalo find --property status          # existence check (has this property)
hyalo find --property '!status'       # absence check (missing this property)
hyalo find --property 'title~=draft'  # regex match on property value (unanchored)
hyalo find --property 'title~=/^Draft/i'  # regex with flags (i = case-insensitive)
hyalo find --property status=draft --property topic=cli   # AND

# Filter by tag (prefix-matches hierarchy: --tag inbox matches inbox/processing)
hyalo find --tag inbox

# Filter by task status
hyalo find --task todo    # open tasks
hyalo find --task done    # completed tasks
hyalo find --task any     # any tasks

# Filter by section heading (case-insensitive substring match by default)
hyalo find --section "Tasks" --task todo          # matches "Tasks", "Tasks [4/4]", etc.
hyalo find --section "## Design" "TODO"           # content search scoped to level-2 Design sections
hyalo find --section "# Introduction" --fields sections  # level-pinned: only # Introduction, not ## Introduction
hyalo find --section "Tasks" --section "Notes"    # OR: match either section
hyalo find --section "/DEC-03[12]/"               # regex section match

# Scope to file(s) (--file is repeatable)
hyalo find --file path/to/note.md
hyalo find --file a.md --file b.md
hyalo find --glob "notes/*.md"
hyalo find --glob '!**/draft-*'      # exclude files matching a pattern (glob negation)

# Control returned fields (default: all except properties-typed and backlinks)
hyalo find --fields properties,tags
hyalo find --fields sections,tasks,links
hyalo find --fields properties-typed     # [{name, type, value}] array instead of {key: value} map
hyalo find --fields backlinks --file my-note.md    # show who links to this note
hyalo find --fields properties,backlinks           # combine with other fields

# Sort and limit
hyalo find --sort modified --limit 10
hyalo find --sort modified --reverse --limit 5      # newest first
hyalo find --sort title                             # sort by title (frontmatter or first H1)
hyalo find --sort date                              # sort by frontmatter date
hyalo find --sort property:priority --reverse       # highest priority first

read

Read the body content of a markdown file, optionally filtered by section or line range. Defaults to plain text output.

# Read full body (text output)
hyalo read --file path/to/note.md

# Read a specific section
hyalo read --file path/to/note.md --section "Proposal"
hyalo read --file path/to/note.md --section "## Proposal"

# Read a line range (1-based, inclusive)
hyalo read --file path/to/note.md --lines 5:10
hyalo read --file path/to/note.md --lines 5:
hyalo read --file path/to/note.md --lines :10

# Show frontmatter only
hyalo read --file path/to/note.md --frontmatter

# JSON output
hyalo read --file path/to/note.md --format json
hyalo read --file path/to/note.md --format json --jq '.results.content'

properties

Subcommand group for property operations.

# Aggregate summary of unique property names with inferred types and file counts
hyalo properties summary
hyalo properties summary --glob "notes/*.md"

# Bulk rename a property key across all files
hyalo properties rename --from old-key --to new-key
hyalo properties rename --from old-key --to new-key --glob "notes/*.md"

tags

Subcommand group for tag operations.

# Aggregate summary of unique tags with file counts
hyalo tags summary
hyalo tags summary --glob "notes/*.md"

# Bulk rename a tag across all files
hyalo tags rename --from old-tag --to new-tag
hyalo tags rename --from old-tag --to new-tag --glob "notes/*.md"

summary

High-level vault overview: file counts, property and tag aggregates, status groups, tasks, link health (total and broken links with source locations), orphan files (fully isolated — no links in or out), and recently modified files.

hyalo summary
hyalo summary --glob "notes/*.md"
hyalo summary --recent 5          # control how many recent files to show (default: 10)
hyalo summary --depth 1           # collapse subdirectories beyond depth 1
hyalo summary --format text
hyalo summary --jq '.results.tasks.total'
hyalo summary --jq '.results.orphans.files'  # list fully isolated files (no links in or out)
hyalo summary --format text --hints

set

Set (create or overwrite) frontmatter properties and/or add tags across one or more files.

hyalo set --property status=done --file path/to/note.md
hyalo set --property status=active --glob "notes/*.md"
hyalo set --tag cli --file path/to/note.md
hyalo set --property status=done --tag reviewed --file path/to/note.md

# Set a list-type (YAML sequence) property
hyalo set --property 'authors=[Alice, Bob, Charlie]' --file path/to/note.md

# Multi-file targeting (--file is repeatable)
hyalo set --property status=reviewed --file a.md --file b.md

# Bulk-update: set status on files matching a filter
hyalo set --property status=completed --where-property status=done --glob '**/*.md'

# Add tag to files matching a tag filter
hyalo set --tag reviewed --where-tag research --glob '**/*.md'

remove

Remove frontmatter properties and/or tags from file(s).

hyalo remove --property status --file path/to/note.md          # remove property
hyalo remove --property tags=serde --file path/to/note.md      # remove value from list
hyalo remove --tag cli --file path/to/note.md
hyalo remove --property status --glob "draft/*.md"

# Remove tag from files matching a property filter
hyalo remove --tag deprecated --where-property status=completed --glob '**/*.md'

remove --property K (no value) removes the property entirely. remove --property K=V removes V from a list property, or removes the property if it is a scalar matching V.

append

Append values to list properties, promoting scalars to lists if needed.

hyalo append --property tags=serde --file path/to/note.md
hyalo append --property tags=serde --glob "crates/*.md"

# Append to list property on files matching a tag
hyalo append --property aliases=old-name --where-tag renamed --glob '**/*.md'

task

Read, toggle, or set the status of a single task checkbox by line number.

hyalo task read --file path/to/note.md --line 42
hyalo task toggle --file path/to/note.md --line 42
hyalo task set-status --file path/to/note.md --line 42 --status /

Tasks are markdown checkboxes (- [ ], - [x], - [/], etc.) in the file body. Checkboxes inside fenced code blocks and %%comment%% blocks are ignored.

backlinks

Reverse link lookup — find all files that link to a given file. Scans all .md files in the vault and builds an in-memory link graph, then returns every incoming link (both [[wikilinks]] and [markdown](links)) pointing to the target file.

hyalo backlinks --file path/to/note.md

JSON output (default):

{
  "results": {
    "file": "path/to/note.md",
    "backlinks": [
      {
        "source": "index.md",
        "line": 5,
        "target": "note"
      },
      {
        "source": "journal/2026-03-20.md",
        "line": 12,
        "target": "note",
        "label": "project notes"
      }
    ]
  },
  "total": 2,
  "hints": []
}

Text output (--format text):

2 backlinks to path/to/note.md:
  index.md:5
  journal/2026-03-20.md:12 ("project notes")

The label field (and the parenthesised text in text mode) appears only for aliased wikilinks ([[target|label]]) and titled markdown links ([label](target.md)).

mv

Move or rename a markdown file and update all links across the vault. Builds an in-memory link graph, moves the file on disk, then rewrites all inbound [[wikilinks]] and [markdown](links) in other files that pointed to the old path. Also rewrites relative markdown links inside the moved file whose targets changed due to the new directory context.

hyalo mv --file old/path.md --to new/path.md
hyalo mv --file note.md --to archive/note.md --dry-run   # preview without writing

JSON output (default):

{
  "results": {
    "from": "old/path.md",
    "to": "new/path.md",
    "dry_run": false,
    "updated_files": [
      {
        "file": "index.md",
        "replacements": [
          { "line": 5, "old_text": "[[old/path]]", "new_text": "[[new/path]]" }
        ]
      }
    ],
    "total_files_updated": 1,
    "total_links_updated": 1
  },
  "hints": []
}

Text output (--format text):

Moved old/path.md → new/path.md
Updated 1 link in 1 file:
  index.md:5  [[old/path]] → [[new/path]]

Use --dry-run to preview which files and links would change without modifying anything.

Root-absolute links (e.g. /docs/guides/setup.md) are also rewritten during a move. Hyalo uses the site prefix to map these to vault-relative paths. If mv reports 0 links updated but you expect absolute links to be rewritten, check your --site-prefix setting (see Absolute link resolution).

links

Subcommand group for link operations.

# Preview broken link fixes (dry-run is the default)
hyalo links fix

# Apply fixes to disk
hyalo links fix --apply

# Adjust fuzzy matching threshold (0.0–1.0, default: 0.8)
hyalo links fix --threshold 0.9

# Scope to specific files
hyalo links fix --glob "notes/*.md"

# Skip links that contain Hugo/template syntax
hyalo links fix --ignore-target '{{ ref' --ignore-target '{{ relref'

# Text output
hyalo links fix --format text

links fix detects broken [[wikilinks]] and [markdown](links) across the vault and attempts auto-repair using four strategies (in priority order): case-insensitive exact match, extension mismatch (.md present/absent), unique stem match anywhere in the vault (shortest-path resolution), and Jaro-Winkler fuzzy match above --threshold.

Default is --dry-run (preview only). Pass --apply to write fixes to disk. Use --ignore-target (repeatable) to skip links containing specific substrings — useful for template syntax, external paths, or anchors that aren't real files.

Hints

Add --hints to any read-only command to get suggested drill-down commands:

$ hyalo summary --format text --hints
Files: 32 total
  ".": 5
  "backlog": 7
  "iterations": 12
  "research": 8
Properties: 8 unique
Tags: 15 unique
Status: completed (10), in-progress (2), planned (2)
Tasks: 89/174
Orphans: 3
  "backlog/old-idea.md"
  "research/scratch.md"
  "research/unused-ref.md"

  -> hyalo --dir . properties summary
  -> hyalo --dir . tags summary
  -> hyalo --dir . find --task todo
  -> hyalo --dir . find --property status=in-progress

In JSON mode, hints populate the "hints" array in the standard envelope: {"results": ..., "hints": [{"description": "...", "cmd": "hyalo ..."}]}. The envelope shape is always the same regardless of --hints/--no-hints — only the array contents change. Each hint has a short description and a concrete, copy-pasteable command. Suppressed when combined with --jq.

Snapshot Index

The snapshot index is a MessagePack file that captures a point-in-time snapshot of the vault's metadata (frontmatter, tags, sections, tasks, links) for faster repeated queries. It is short-lived and ephemeral — it becomes stale as soon as any file in the vault is modified outside of hyalo.

Usage:

# Create the index (one disk scan)
hyalo create-index

# Run read-only queries against the index (no disk scan)
hyalo find --property status=in-progress --index .hyalo-index
hyalo summary --index .hyalo-index
hyalo tags summary --index .hyalo-index

# Drop the index when done
hyalo drop-index

When to use: workflows that run many queries in a short window — CI pipelines, automation scripts, LLM tool loops. Create the index at the start, query and mutate against it, then drop it.

Read-only commands (find, summary, tags summary, properties summary, backlinks) skip disk scans entirely when using --index.

Mutation commands (set, remove, append, task, mv, tags rename, properties rename) still read and write individual files on disk, but when --index is provided they also patch the index entry in-place after each mutation — keeping the index current for subsequent queries. This is safe as long as no external tool modifies files in the vault while the index is active.

Never commit .hyalo-index files to version control — they are throwaway artifacts.

Common pitfalls

Mistake Correct usage
--property 'title=~/pat/' (Perl-style =~) --property 'title~=/pat/' (hyalo uses ~=)
--property title~=draft to search all titles --title draft (searches frontmatter title AND H1 headings)
--tag projects expecting substring match --tag project (prefix match: matches project, project/backend, but not projects)
--glob '/absolute/path/*.md' --glob 'relative/*.md' (globs are relative to --dir)
--format text --jq '.total' Remove --format text--jq is incompatible with text format
--count --jq '.results' Use one or the other — --count is a shortcut for --jq '.total'

Benchmarking

On-demand performance benchmarks using Criterion and Hyperfine:

cargo bench --bench micro                # pure-function micro-benchmarks
cargo bench --bench vault                # vault-scale benchmarks (needs obsidian-hub)
./bench-e2e.sh                           # end-to-end CLI benchmarks
./bench-e2e.sh target/release/hyalo /tmp/hyalo-baseline   # A/B comparison

See benches/README.md for full setup, A/B comparison workflows, profiling with samply, and memory measurement.

Releasing

  1. Bump the version in Cargo.toml
  2. Commit: git commit -am "Bump version to X.Y.Z"
  3. Create a GitHub release with tag vX.Y.Z (must match Cargo.toml)

The release workflow automatically builds binaries for all platforms, uploads them to the release, and updates the Homebrew formula.

License

MIT