Skip to main content
Version: 0.4.0

LLM Wiki Q&A

A durable, editable wiki for an LLM agent, built entirely on Skardi primitives: every agent verb (write, open, grep, ls, log) is one pipeline YAML plus a CLI alias, and the whole thing runs on candle() inline embeddings, pg_knn / sqlite_knn, pg_fts / sqlite_fts, and RRF hybrid search — all in the same SQL. The demo shows what full data autonomy looks like for an agent: it curates a compounding knowledge base itself, with no external orchestration code.

Two flavours ship side by side:

  • Server / PostgreSQL + pgvector — the "production" shape: each verb is a REST pipeline, invoked over HTTP. Useful when the agent is running somewhere else (Claude Desktop, a web backend, a managed runtime) and talks to skardi across the wire.
  • CLI / SQLite + sqlite-vec + FTS5 — the "drop into any agent" shape: same pipeline YAMLs, but invoked as skardi <verb> shell commands with no server, no Docker, no HTTP. This is the MVP story — any agent with a Bash tool (Claude Code, Cursor, custom loops) gets the full wiki loop locally in a few minutes.

Both flavours use the same pipeline YAML format, because the whole point of the Skardi design is that one declaration is every agent-facing surface: REST today, shell today, skills soon, MCP soon after. For the thinking behind that, read docs/agent_data_plane.md.


Server version — skardi-server + PostgreSQL + pgvector

A data layer for Karpathy's LLM Wiki idea, built on Skardi. The wiki is stored in a single PostgreSQL table that holds markdown content + pgvector embedding on the same row, so every page is atomically searchable via both full-text search and semantic vector search.

Rather than chunked RAG, the LLM agent maintains a persistent, compounding knowledge artifact — entity pages, concept pages, summaries, an index — and uses the endpoints below as its file-system-like primitives (open, write, grep, ls, log).

  • Vector search — candle (bge-small-en-v1.5) embeddings + pgvector KNN (pg_knn)
  • Full-text search — PostgreSQL tsvector / websearch_to_tsquery (pg_fts)
  • Hybrid search — RRF (Reciprocal Rank Fusion) merging both results in SQL
  • Atomic editswiki-create (INSERT) and wiki-update (UPDATE) both re-embed the page inline with candle() in a single statement, so content and vector stay in sync
                    ┌─────────────────────────────────┐
│ Write Path │
│ │
markdown ───────► │ INSERT (wiki-create) or │
(by slug) │ UPDATE (wiki-update) │
│ SET content, │
│ embedding = candle(...) │
│ │
│ ─► row is now visible to │
│ both pg_fts and pg_knn │
└─────────────────────────────────┘

┌─────────────────────────────────┐
│ Read Path │
│ │
query ──────────► │ pg_knn() (top 80) │──┐
│ pg_fts() (top 60) │──┤ RRF merge
│ │ │
│ FULL OUTER JOIN ON slug + RRF │◄─┘
│ ORDER BY rrf_score DESC │
└─────────────────────────────────┘

Because content and embedding live on the same row, a single upsert keeps FTS and vector in sync — no second store, no cross-store consistency problem.

Quick Start

1. Start PostgreSQL with pgvector

docker run --name wiki-postgres \
-e POSTGRES_DB=wikidb \
-e POSTGRES_USER=skardi_user \
-e POSTGRES_PASSWORD=skardi_pass \
-p 5432:5432 \
-d pgvector/pgvector:pg16

2. Create the schema and indexes

docker exec -i wiki-postgres psql -U skardi_user -d wikidb << 'EOF'
CREATE EXTENSION IF NOT EXISTS vector;

CREATE TABLE wiki_pages (
slug TEXT PRIMARY KEY, -- e.g. "entity/alan-turing"
title TEXT NOT NULL,
page_type TEXT NOT NULL, -- entity | concept | summary | index | schema
content TEXT NOT NULL, -- markdown body
embedding vector(384), -- bge-small-en-v1.5 dimension
updated_at TIMESTAMPTZ NOT NULL DEFAULT now()
);

CREATE INDEX ON wiki_pages USING hnsw (embedding vector_cosine_ops)
WITH (m = 16, ef_construction = 64);

CREATE INDEX wiki_pages_content_fts_idx
ON wiki_pages
USING GIN (to_tsvector('english', content));

CREATE INDEX wiki_pages_type_idx ON wiki_pages (page_type, updated_at DESC);

CREATE TABLE wiki_log (
id BIGSERIAL PRIMARY KEY,
event_type TEXT NOT NULL, -- ingest | query | lint | note
slug TEXT NOT NULL, -- "" if not page-specific
message TEXT NOT NULL,
created_at TIMESTAMPTZ NOT NULL DEFAULT now()
);
EOF

3. Download the embedding model

pip install huggingface_hub

python -c "
from huggingface_hub import hf_hub_download
import os
model_dir = 'models/generated/bge-small-en-v1.5'
os.makedirs(model_dir, exist_ok=True)
for f in ['model.safetensors', 'config.json', 'tokenizer.json']:
hf_hub_download('BAAI/bge-small-en-v1.5', f, local_dir=model_dir)
print(f'Model downloaded to {model_dir}')
"

4. Set credentials and start the server

export PG_USER="skardi_user"
export PG_PASSWORD="skardi_pass"

cargo run --bin skardi-server --features candle -- \
--ctx demo/llm_wiki/server/ctx.yaml \
--pipeline demo/llm_wiki/server/pipelines/ \
--port 8080

Agent Primitives

The wiki exposes five HTTP endpoints that mirror the verbs an LLM agent needs to maintain a compounding knowledge base. Each one corresponds to one pipeline file under server/pipelines/.

EndpointVerbPipeline
/wiki-create/executewrite (new)server/pipelines/create.yaml
/wiki-update/executewrite (edit)server/pipelines/update.yaml
/wiki-get/executeopenserver/pipelines/get.yaml
/wiki-search-hybrid/executegrepserver/pipelines/search_hybrid.yaml
/wiki-list/executelsserver/pipelines/list.yaml
/wiki-log-append/executelogserver/pipelines/log_append.yaml

DataFusion's SQL planner does not support INSERT ... ON CONFLICT, so create and edit are exposed as two explicit endpoints. The agent's pattern is: try wiki-update first; if it affects zero rows, fall back to wiki-create. Both re-embed the page inline in a single statement, so FTS and vector stay consistent either way.


Write Path: Creating and Editing Pages

Both endpoints re-embed the page inline with candle() in a single SQL statement, so the pgvector column and the FTS content column are written together from the same row.

Create a new page

# Create an entity page
curl -X POST http://localhost:8080/wiki-create/execute \
-H "Content-Type: application/json" \
-d '{
"slug": "entity/alan-turing",
"title": "Alan Turing",
"page_type": "entity",
"content": "# Alan Turing\n\nBritish mathematician and logician who formalized the concepts of algorithm and computation with the Turing machine. Considered a founder of theoretical computer science and artificial intelligence."
}' | jq .

# Create a concept page that references it
curl -X POST http://localhost:8080/wiki-create/execute \
-H "Content-Type: application/json" \
-d '{
"slug": "concept/turing-machine",
"title": "Turing Machine",
"page_type": "concept",
"content": "# Turing Machine\n\nAn abstract computational model introduced by Alan Turing in 1936. Consists of an infinite tape, a head, and a finite state machine; captures the notion of effective computability."
}' | jq .

# Create a summary page
curl -X POST http://localhost:8080/wiki-create/execute \
-H "Content-Type: application/json" \
-d '{
"slug": "summary/foundations-of-computation",
"title": "Foundations of Computation",
"page_type": "summary",
"content": "# Foundations of Computation\n\nThe theoretical basis for modern computing emerged in the 1930s through the work of Church, Turing, and Gödel. The Church–Turing thesis unified lambda calculus and Turing machines as equivalent models of computation."
}' | jq .

Edit an existing page

wiki-update rewrites the row in place and refreshes the embedding, so an edit is atomically reflected in both pg_fts and pg_knn.

curl -X POST http://localhost:8080/wiki-update/execute \
-H "Content-Type: application/json" \
-d '{
"slug": "entity/alan-turing",
"title": "Alan Turing",
"page_type": "entity",
"content": "# Alan Turing\n\nBritish mathematician, logician, and cryptanalyst. Formalized algorithm and computation via the Turing machine; led the Bletchley Park team that broke the Enigma cipher during WWII."
}' | jq .

Under the hood the two pipelines are:

-- wiki-create
INSERT INTO wikidb.public.wiki_pages (slug, title, page_type, content, embedding, updated_at)
VALUES (
{slug}, {title}, {page_type}, {content},
candle('models/generated/bge-small-en-v1.5', {content}),
now()
);

-- wiki-update
UPDATE wikidb.public.wiki_pages
SET title = {title},
page_type = {page_type},
content = {content},
embedding = candle('models/generated/bge-small-en-v1.5', {content}),
updated_at = now()
WHERE slug = {slug};

DataFusion's planner does not support INSERT ... ON CONFLICT, which is why create and update are separate endpoints. The agent pattern is: try wiki-update first; if it reports zero rows affected, fall back to wiki-create.

Bulk-create from a long document — wiki-bulk-create

wiki-create keeps one page = one row, which is the wiki's editing model. But for seeding the wiki — turning a long markdown doc into many pages in one shot — wiki-bulk-create chunks the body inline with chunk('markdown', …), embeds each chunk with candle(), and writes one page per chunk:

curl -X POST http://localhost:8080/wiki-bulk-create/execute \
-H "Content-Type: application/json" \
-d '{
"slug_prefix": "rfc/9110",
"title": "RFC 9110: HTTP Semantics",
"page_type": "rfc",
"content": "<the full RFC body in markdown>",
"chunk_size": 800,
"overlap": 150
}' | jq .

Synthesised slugs follow {slug_prefix}/p<NNN> (rfc/9110/p001, /p002, …) and titles get a #N suffix so each chunk is a distinct page visible to wiki-list, wiki-search-hybrid, etc. Pages later edited via wiki-update keep their slug — the bulk-create path is a one-way seeder, not a re-runnable upsert.

Requires --features rag on skardi-server (the umbrella that bundles embedding + chunking). See server/pipelines/bulk_create.yaml.


Read Path: Agent Retrieval Loop

A typical LLM agent turn looks like:

  1. grep the wiki with hybrid search to find candidate slugs
  2. open each top-ranked page with wiki-get to read the full body
  3. Synthesize the answer, optionally write new pages back
  4. log the activity so the next session knows what happened
curl -X POST http://localhost:8080/wiki-search-hybrid/execute \
-H "Content-Type: application/json" \
-d '{
"query": "who invented the theoretical model of a computer?",
"text_query": "turing machine computation",
"vector_weight": 0.5,
"text_weight": 0.5,
"limit": 10
}' | jq .

Returns slug, title, page_type, and rrf_score for each candidate page. The RRF join is on slug (the wiki's primary key), so there is no cross-store lookup and no id-type conversion.

open — fetch a full page

curl -X POST http://localhost:8080/wiki-get/execute \
-H "Content-Type: application/json" \
-d '{"slug": "entity/alan-turing"}' | jq .

ls — browse by type or prefix

Rebuild index.md, find orphan pages, or list a category:

# All entity pages, newest first
curl -X POST http://localhost:8080/wiki-list/execute \
-H "Content-Type: application/json" \
-d '{
"page_type_pattern": "entity",
"slug_prefix": "%",
"limit": 100
}' | jq .

# Everything under concept/
curl -X POST http://localhost:8080/wiki-list/execute \
-H "Content-Type: application/json" \
-d '{
"page_type_pattern": "%",
"slug_prefix": "concept/%",
"limit": 100
}' | jq .

log — append an activity entry

curl -X POST http://localhost:8080/wiki-log-append/execute \
-H "Content-Type: application/json" \
-d '{
"event_type": "ingest",
"slug": "entity/alan-turing",
"message": "Created from Wikipedia article; cross-linked from concept/turing-machine."
}' | jq .

Pipelines

PipelineEndpointDescription
server/pipelines/create.yaml/wiki-create/executeINSERT a new page; re-embeds with candle() inline
server/pipelines/update.yaml/wiki-update/executeUPDATE an existing page by slug; re-embeds with candle() inline
server/pipelines/bulk_create.yaml/wiki-bulk-create/executeChunk a long doc with chunk('markdown', …), embed each chunk with candle(), write one page per chunk
server/pipelines/get.yaml/wiki-get/executeFetch one page by slug
server/pipelines/search_hybrid.yaml/wiki-search-hybrid/executeRRF hybrid search over pg_knn + pg_fts
server/pipelines/list.yaml/wiki-list/executeFilter pages by page_type + slug prefix, newest first
server/pipelines/log_append.yaml/wiki-log-append/executeAppend to the wiki_log activity log

Relationship to the RAG Demo

This demo is a schema-level evolution of demo/rag/: same stack (Postgres + pgvector + candle + pg_fts + pg_knn + RRF), but the table is keyed by a human-readable slug, carries page metadata (title, page_type), and uses INSERT ... ON CONFLICT so pages can be edited in place. The RAG demo ingests immutable chunks; the LLM Wiki demo ingests a living, editable knowledge base that the LLM itself curates.


Cleanup

docker stop wiki-postgres && docker rm wiki-postgres
pkill -f skardi-server

CLI version — skardi-cli + SQLite + sqlite-vec + FTS5

The same wiki primitives (create, update, get, grep, ls) run end-to-end through skardino server, no Docker, no HTTP. Each primitive is a pipeline YAML under cli/pipelines/ (same format as the server pipelines) and is invoked through a short verb alias defined in cli/aliases.yaml: skardi grep "...", skardi ls, skardi open <slug>, skardi write --slug=... --title=..., etc.

A regular wiki_pages table holds canonical state (slug, title, page_type, content, embedding); AFTER INSERT / AFTER UPDATE triggers fan rows out to an FTS5 virtual table for keyword search and a sqlite-vec vec0 virtual table for KNN, so a single INSERT (or UPDATE) keeps content and embedding in sync. Embeddings are computed inline by the candle() UDF (same model as the server version).

1. Install the CLI with embedding support

cargo install --locked --path crates/cli --features candle

--locked makes cargo honor the checked-in Cargo.lock instead of re-resolving transitive deps, which can otherwise pull a newer crate whose MSRV is higher than your toolchain.

2. Get the sqlite-vec extension

Build or download the vec0 shared library — see the sqlite-vec install guide. Then:

export SQLITE_VEC_PATH=/absolute/path/to/vec0.dylib   # or .so / .dll

# If using the pip package:
export SQLITE_VEC_PATH=$(python -c "import sqlite_vec; print(sqlite_vec.loadable_path())")

3. Download the embedding model

pip install huggingface_hub
python -c "
from huggingface_hub import hf_hub_download
import os
model_dir = 'models/generated/bge-small-en-v1.5'
os.makedirs(model_dir, exist_ok=True)
for f in ['model.safetensors', 'config.json', 'tokenizer.json']:
hf_hub_download('BAAI/bge-small-en-v1.5', f, local_dir=model_dir)
"

4. Create the database

pip install sqlite-vec
python demo/llm_wiki/setup.py

The script loads the sqlite-vec extension via the sqlite_vec Python package and drops any prior demo/llm_wiki/wiki.db. vec0 requires an INTEGER rowid, so the human-readable slug lives on the base wiki_pages table (and as an UNINDEXED FTS5 column) and the integer id carries the JOIN. The script also creates wiki_pages_fts, wiki_pages_vec, the wiki_log activity table, and AFTER INSERT / AFTER UPDATE triggers that keep both mirrors in sync. See setup.py for the schema.

4b. (Optional) Seed sample pages

To explore the demo with non-empty data, seed.py splits the markdown files in data/test_corpus/ (Alice in Wonderland, Jane Eyre, The Art of War) into ~110 paragraph-sized pages, computes a real bge-small-en-v1.5 embedding for each one in Python, and INSERTs them through the same triggers — so skardi grep and skardi ls work end-to-end without writing a page yourself.

pip install transformers torch
python demo/llm_wiki/seed.py

Embeddings use CLS pooling + L2 normalization, matching what the candle() UDF produces, so seeded pages and pages later written via skardi write share the same vector space. Pages are tagged by source (page_type = alice | jane-eyre | art-of-war) so skardi ls --page_type_pattern=alice shows only Alice excerpts.

Re-running seed.py on an already-seeded DB will fail on the UNIQUE(slug) constraint — re-run setup.py first to drop and recreate.

5. Config layout

Everything the CLI needs for the demo lives under cli/:

demo/llm_wiki/cli/
ctx.yaml # registers wiki.db as a SQLite catalog data source
aliases.yaml # short verbs → pipeline bindings
pipelines/ # pipeline YAMLs (one per verb)

cli/ctx.yaml registers one SQLite source in catalog mode, which auto-discovers every table, loads sqlite-vec once on the shared connection pool, and exposes each table under <catalog>.main.<table> for both SQL and sqlite_knn / sqlite_fts lookups:

kind: context

metadata:
name: example-context
version: 1.0.0

spec:
data_sources:
- name: wiki
type: sqlite
path: demo/llm_wiki/wiki.db
access_mode: read_write
hierarchy_level: catalog
options:
extensions_env: SQLITE_VEC_PATH

The pipeline YAMLs in cli/pipelines/ use the same metadata + query shape as the server pipelines, with {param} placeholders for named parameters — just targeting the SQLite stack (sqlite_knn / sqlite_fts / vec_to_binary(candle(...))) instead of pg_knn / pg_fts. Verb → pipeline bindings live in cli/aliases.yaml.

Export the config dir once so the verbs below don't need --ctx on every line. SKARDICONFIG accepts either a config directory (which the CLI looks inside for ctx.yaml, aliases.yaml, and pipelines/) or an individual ctx file. --ctx PATH still works and takes precedence:

export SKARDICONFIG=demo/llm_wiki/cli

6. Set up aliases (bundled for this demo)

The demo ships with cli/aliases.yaml pre-populated so the verbs below just work. You can add more aliases yourself — each alias maps a short verb to a pipeline plus positional/default param bindings:

# Example: a `today` alias that lists only today's pages
skardi alias add today \
--pipeline wiki-list \
--default 'page_type_pattern=%' \
--default 'slug_prefix=%' \
--default 'limit=20' \
--description "List the 20 most recently-touched pages"

skardi alias list
skardi alias show grep
skardi alias remove today

Alias files resolve in this order: --aliases <path>SKARDI_ALIASES env → aliases.yaml next to the active ctx file → ~/.skardi/config/aliases.yaml.

7. write — create a new page

skardi write \
--slug=entity/alan-turing \
--title="Alan Turing" \
--page_type=entity \
--content='# Alan Turing

British mathematician and logician who formalized the concepts of algorithm and computation with the Turing machine.'

The write alias invokes cli/pipelines/create.yaml, which computes the embedding inline with candle(), packs it with vec_to_binary(), and INSERTs the row. The AFTER INSERT trigger then mirrors the row to wiki_pages_fts and wiki_pages_vec atomically.

Why does create.yaml wrap the seed row as SELECT {slug} AS slug, ... FROM (...) instead of using VALUES? DataFusion's INSERT planner currently propagates the INSERT target schema (5 columns) down into any immediate-child VALUES clause and validates row width against it, ignoring the intermediate projection that adds vec_to_binary(candle(...)). The SELECT-wrapper keeps the subquery's own schema in scope so the projection lands the row at full width.

7b. bulk-write — seed many pages from one long markdown doc

write is for hand-curated pages. To seed a long document — a chapter, an RFC, a transcript — bulk-write chunks the body inline with chunk('markdown', …), embeds each chunk with candle(), and inserts one page per chunk. Each call goes through the same AFTER INSERT trigger as write, so the resulting pages are immediately searchable via skardi grep.

skardi bulk-write \
--slug_prefix=rfc/9110 \
--title="RFC 9110: HTTP Semantics" \
--page_type=rfc \
--content="$(cat rfc9110.md)"

Override the chunk shape on the same line:

skardi bulk-write \
--slug_prefix=alice/ch1 \
--title="Alice in Wonderland — Chapter 1" \
--page_type=alice \
--content="$(cat data/test_corpus/alice_sample.md)" \
--chunk_size=400 --overlap=80

Synthesised slugs follow {slug_prefix}/p<NNN> (rfc/9110/p001, /p002, …) and titles get a #N suffix so each chunk is a distinct page visible to skardi ls and skardi grep. Per-page edits stay on rm + write (or direct UPDATE against SQLite) — bulk-write is a one-shot seeder, not a re-runnable upsert. Re-running it on the same slug prefix will fail on the UNIQUE(slug) constraint.

This is the same flow seed.py implements in Python today, but in pure SQL: chunk('markdown', body, ...) replaces the manual paragraph splitter and candle() replaces the Python embedding step. Seeding a fresh corpus needs only setup.py + bulk-write calls — no Python embedding loop.

Install the CLI with the RAG umbrella (bundles embedding UDFs + chunk):

cargo install --locked --path crates/cli --features rag

8. Edit an existing page (rm + write)

DataFusion's UPDATE planner unparses each SET expression back to SQL for the underlying SQLite connection to execute, and it can't currently render a Binary scalar (the packed-f32 embedding from vec_to_binary(candle(...))) as a SQL literal. The portable workaround is delete + re-insert — the AFTER DELETE trigger cleans both mirrors, the AFTER INSERT trigger repopulates them, and the new row picks up a fresh updated_at.

skardi rm entity/alan-turing

skardi write \
--slug=entity/alan-turing \
--title="Alan Turing" \
--page_type=entity \
--content='# Alan Turing

British mathematician, logician, and cryptanalyst who broke the Enigma cipher at Bletchley Park.'

If you'd rather edit the row in place from a SQLite client (e.g. sqlite3), the original AFTER UPDATE trigger is still installed and will refresh both mirrors when an UPDATE wiki_pages SET ... runs against the underlying database directly — only the DataFusion path needs the delete-and-reinsert dance.

9. open — fetch one page by slug

If you ran the optional seed step in 4b, every paragraph in the corpus is already a page. The opening of Alice in Wonderland is at alice/chapter-i-down-the-rabbit-hole/p001:

skardi open alice/chapter-i-down-the-rabbit-hole/p001

Under the hood this runs cli/pipelines/get.yaml: SELECT slug, title, page_type, content, updated_at FROM wiki.main.wiki_pages WHERE slug = {slug}.

10. grep — hybrid search (RRF over FTS + vector)

A single positional arg drives both halves of the RRF merge — semantic nearest-neighbour over sqlite_knn and BM25 over sqlite_fts:

# Pure semantic intent — finds the rabbit-hole paragraphs in Alice ch.1
skardi grep "white rabbit pocket watch" --limit=5

# Same query crosses corpora — top hits mix Alice and Jane Eyre
skardi grep "young girl sent away to her room" --limit=5

The positional arg binds to both {query} (embedded with candle() for sqlite_knn) and {text_query} (via the text_query: "{query}" default in the alias). Override either independently — useful when you want a loose semantic intent paired with strict keyword filters:

skardi grep "Sun Tzu" \
--text_query="strategy OR victory OR planning" \
--vector_weight=0.3 --text_weight=0.7 --limit=5

See cli/pipelines/search_hybrid.yaml for the full RRF merge. Run skardi grep --help to see every param the alias exposes and where each value comes from.

11. ls — browse by type or slug prefix

skardi ls

# All Alice excerpts (page_type tags pages by source after seeding)
skardi ls --page_type_pattern=alice

# Just one chapter
skardi ls --slug_prefix='art-of-war/chapter-i-laying-plans/%'

12. log — append an activity entry

skardi log \
--event_type=ingest \
--slug=entity/alan-turing \
--message="Created from Wikipedia article."

Falling back to raw SQL

skardi run and the aliases above are a thin layer over the pipeline YAMLs. The underlying queries are still plain SQL — if you want to experiment ad-hoc, skardi query --sql "..." works just as before (same exported SKARDICONFIG).

Cleanup

rm demo/llm_wiki/wiki.db