CLI reference#
ovellum <subcommand> [flags]
Run via npx ovellum, via the package binary after install, or via your
package manager's task runner.
Subcommands#
| Subcommand | Status | Summary |
| ---------- | --------- | ------------------------------------------------------------------------ |
| init | available | Scaffold a new project (config + starter content + .gitignore entry). |
| build | available | Run the configured pipeline (parse + generate + merge, or build a site). |
| dev | available | Build, watch, serve, and live-reload connected browsers — the one-command dev loop. |
| watch | available | Build, then rebuild on every change under input/ (debounced 300 ms). |
| serve | available | Serve the built site over HTTP. No watch, no live reload. |
| check | available | Validate config + check for broken internal links + flag unsafe URLs. |
| orphans | planned | List / inspect / reattach quarantined manual blocks. |
| clean | planned | Remove auto-generated outputs while preserving manual files. |
ovellum init#
Scaffold a new project in the current (or given) directory. Refuses to
clobber an existing ovellum.config.json unless --force is passed.
Synopsis#
ovellum init [--cwd <dir>] [--yes] [--force]
Flags#
| Flag | Type | Default | Notes |
| ------------- | ------- | --------------- | ------------------------------------------------------------------------------------ |
| --cwd <dir> | path | process.cwd() | Project root. |
| --yes, -y | boolean | false | Non-interactive: accept every default. Useful in CI / smoke tests. |
| --force | boolean | false | Overwrite an existing ovellum.config.json. By default the command exits with 2. |
Prompts (interactive)#
- Project name — defaults to
package.json#nameor the folder name. - Mode —
manual(default),auto, orhybrid. - Site title — defaults to a title-cased project name.
- Description — used for
<meta name="description">. - (manual) Content dir / Output dir / Generate landing page?
- (auto / hybrid)
tsconfig/ Output dir. - Default theme —
auto,light, ordark.
Output#
Writes only files that don't already exist (unless --force):
ovellum.config.json<input>/index.md(manual + hybrid modes only) with a friendly starter..gitignore— appends<output>/and.orphans/if absent.
Prints a numbered next-steps list keyed to the chosen mode.
Exit codes#
| Code | Meaning |
| ----- | ---------------------------------------------------------------------- |
| 0 | Project initialised. |
| 2 | ovellum.config.json already exists; re-run with --force to replace. |
| 130 | User cancelled the prompts (Ctrl-C). |
ovellum build#
Resolves the project's ovellum.config.*, runs the configured pipeline,
writes output to disk, prints a summary.
Synopsis#
ovellum build [--cwd <dir>] [--config <path>]
Flags#
| Flag | Type | Default | Notes |
| ----------------- | ---- | --------------- | --------------------------------------------------------------- |
| --cwd <dir> | path | process.cwd() | Project root. All paths in the config resolve relative to this. |
| --config <path> | path | auto-discovered | Skip discovery and load this file directly. |
Behaviour by mode#
auto#
- Parse
input/to aDocProjectIR. - Render IR to Markdown.
- Write each output, overwriting any existing file.
hybrid (default)#
Same as auto, then for each generated file:
- If the existing output file is present, read it.
- If it carries
<!-- @manual:start -->blocks, run the merger. - Any block whose anchor no longer exists is written to
protect.orphanDir.
manual#
- Walk
input/for.mdfiles. - Render each to HTML (Markdown is sanitised — see Security).
- Build a sidebar nav and breadcrumb trail.
- Wrap each page in the default template (topbar, sidebar, ToC, prev/next, page meta).
- Write pretty URLs to
output/. - Copy
assets/ovellum.css+assets/ovellum.jsfrom the bundled template. - When
site.baseUrlis set, emitsitemap.xml. - When
site.search.enabledistrue, run Pagefind against the output and emitdist/pagefind/.
Summary output#
Auto / hybrid#
ovellum build complete in 207ms
config: .../ovellum.config.json
mode: hybrid
sources: 2 ← input files parsed
written: 2 file(s) ← Markdown files written
merged: 1 file(s) ← hybrid only: files where a manual block was spliced
orphans: 0 ← hybrid only: blocks whose anchor disappeared
warnings: 0
→ docs/format.md
→ docs/user.md
quarantined: ← only printed when orphans > 0
↪ .ovellum/orphans/2026-05-15_src-format.ts-padZero.md
Manual#
ovellum build complete in 207ms
config: .../ovellum.config.json
mode: manual
output: dist/
pages: 5
warnings: 0
→ / (dist/index.html)
→ /configuration/ (dist/configuration/index.html)
→ /getting-started/ (dist/getting-started/index.html)
→ /guides/deploying/ (dist/guides/deploying/index.html)
→ /guides/theming/ (dist/guides/theming/index.html)
Exit codes#
| Code | Meaning |
| ---- | ----------------------------------------------------------- |
| 0 | Success. |
| 1 | Build error (parser failure, write failure, unknown mode). |
| 2 | Reserved for --strict (warnings promoted to errors). |
| 3 | ConfigError — config schema invalid, file not found, etc. |
stderr carries per-warning lines (warning: …). stdout carries the summary.
Examples#
# Build from the current directory
npx ovellum build
# Build a different project
npx ovellum build --cwd ./website
# Bypass config discovery
npx ovellum build --config ./config/ovellum.prod.ts
ovellum dev#
The combined build + watch + serve + live-reload loop. The one command you want running while writing.
Synopsis#
ovellum dev [--cwd <dir>] [--config <path>] [--port <n>] [--host <addr>]
Flags#
| Flag | Type | Default | Notes |
| ----------------- | ------- | ----------- | ------------------------------------------------------------------------------------ |
| --cwd <dir> | path | cwd | Project root. |
| --config <path> | path | auto | Skip discovery and load this file directly. |
| --port <n> | integer | 3000 | Starting port. If busy, auto-bumps up to 19 ports forward before giving up. |
| --host <addr> | string | 127.0.0.1 | Bind address. Pass 0.0.0.0 to expose on the local network. |
Behaviour#
- Loads the config and resolves
config.output(the build'sdist/dir). - Starts an HTTP server bound to
--host:--port. - Runs an initial build, then watches
input/and the config file for changes (same debounce asovellum watch— 300 ms). - On every successful rebuild, pushes a
reloadevent over Server-Sent Events to every connected browser tab; the injected client script callslocation.reload(). Ctrl-Cshuts down both the watcher and the server cleanly.
The injected reload script is added only for HTML responses, only when
dev is the running command. ovellum build output is never modified.
Output#
ovellum dev starting from .../ovellum.config.json
built 17 page(s) in 720ms
watching content for changes…
local: http://127.0.0.1:3000/
press Ctrl-C to exit.
After a save:
changed: content/getting-started.md
built 17 page(s) in 60ms
Exit codes#
| Code | Meaning |
| ---- | ------------------------------------------------------------------ |
| 0 | Clean shutdown (Ctrl-C). |
| 1 | Mode unsupported (dev is manual-mode only today). |
| 3 | Config invalid. |
Examples#
# Default: localhost:3000
npx ovellum dev
# Pick a port
npx ovellum dev --port 4000
# Expose to the LAN (useful for mobile testing)
npx ovellum dev --host 0.0.0.0
# Multi-site monorepo
npx ovellum dev --cwd ./website
ovellum serve#
Pure static-file server, no watching. Useful for previewing a production build exactly as it'll be served, or wiring into a process manager that handles rebuilds elsewhere.
Synopsis#
ovellum serve [--cwd <dir>] [--config <path>] [--port <n>] [--host <addr>]
Flags are identical to ovellum dev. The server reads from
config.output; if that directory doesn't exist, serve exits with
1 and points you at ovellum build or ovellum dev.
Differences vs. ovellum dev#
| | dev | serve |
|---|---|---|
| Initial build | yes (via watcher) | no — requires existing dist/ |
| Watches files | yes | no |
| Injects reload script | yes | no |
| Cache headers | no-store | public, max-age=0 |
If you only want the server (e.g. you're running ovellum watch in
another shell yourself), serve is the right command.
ovellum check#
Validation pass only — no writes. Loads config, walks every .md file
under input/, extracts links via remark (so fenced code blocks are
correctly ignored), and verifies:
- Every internal link resolves to a real page URL in the sidebar nav.
- No link uses an unsafe URL scheme (
javascript:,vbscript:,data:,file:). Even thoughrenderMarkdownstrips these at render time,checkflags them here so authors can remove them at the source.
Synopsis#
ovellum check [--cwd <dir>] [--config <path>]
Output#
Clean:
ovellum check complete in 76ms
config: .../ovellum.config.json
mode: manual
pages: 14
broken links: 0
unsafe schemes: 0
With issues:
ovellum check complete in 87ms
config: .../ovellum.config.json
mode: manual
pages: 14
broken links: 1
unsafe schemes: 1
details:
content/getting-started.md:42 [SECURITY] unsafe URL scheme 'javascript:' — link will be stripped by the HTML sanitizer (raw: javascript:alert(1))
content/getting-started.md:112 broken internal link to /no/such/page/ (raw: /no/such/page/)
Exit codes#
0clean1one or more issues found3config invalid
Scope today#
Manual mode only. Hybrid and auto modes exit 1 with a "not yet
supported" message — broken-link coverage for the auto-generated half
of the pipeline is on the roadmap.
Frontmatter validation, required-fields checking, and orphan listing for hybrid mode are deferred.
ovellum watch#
Build, then watch input/ (and the config file) for changes and rebuild
on every change. Debounced at 300 ms with chokidar's
awaitWriteFinish enabled so partial writes don't trigger a half-state
rebuild.
For the common "rebuild + serve + auto-refresh" loop, you almost
certainly want ovellum dev instead. watch is the
primitive — useful when you want to run a different server (a CDN
emulator, a reverse proxy, your own process manager) or pipe build
notifications somewhere.
Synopsis#
ovellum watch [--cwd <dir>] [--config <path>]
Behaviour#
- An initial build runs once on start.
- Changes to any file under
input/re-trigger the same pipeline. - Changes to the config file itself reload it before the next build.
Ctrl-Cshuts the watcher down cleanly.
No HTTP server, no live reload — pair with ovellum serve in another
terminal, or hit a different static server of your choice.
Planned subcommands#
ovellum orphans#
Browse .ovellum/orphans/:
- default: list with metadata
--stale: filter to orphans older thanprotect.orphanRetentiondays--interactive: reattach / delete / skip prompts
ovellum clean#
Removes auto-generated files (identified by ovellum: true frontmatter)
while preserving manual files. Dry-run by default; --confirm actually
deletes. Does not touch .ovellum/orphans/ (those are committed
manual writing).
Global flags (planned)#
| Flag | Notes |
| ----------- | -------------------------------------------------- |
| --strict | Promote warnings to errors; exit 2. |
| --verbose | Print debug output (parser stages, merge details). |
--cwd and --config are available on build, check, and watch
today; they'll be promoted to global once more subcommands land.