Skip to main content

Extractive QA pipeline — no LLM, no training. BM25 + FAISS + Wikidata + cross-encoder re-ranking + roberta-base-squad2.

Project description

watson-lite

A Watson-inspired extractive QA system that runs on a laptop.
No LLM. No trained weights of your own. No paid APIs.

Python Ruff

Why?

LLMs have largely displaced classical QA systems, but that displacement came with trade-offs that matter in practice. watson-lite optimises for the constraints LLMs handle poorly.

Transparency and auditability. Every step in the pipeline is a named, inspectable object: BM25 scores, RRF fusion weights, span extraction logits, graph corroboration flags, calibrated confidence breakdown. You can trace exactly why the answer is what it is. For regulated industries, research workflows, or anywhere an answer must be explained and verified, that audit trail is essential.

No hallucination by construction. Extractive QA cannot invent a fact that is not in a retrieved passage. Every answer is a verbatim span from a public source. The confidence score is a function of extraction score, span agreement across independent retrievals, graph corroboration, and rank signal — all measurable and falsifiable. A language model can be confidently wrong; watson-lite either finds a span or returns confidence 0.

Cost and data sovereignty. Zero token cost, zero API keys, no data sent to a third party. The system runs on a laptop CPU with roughly 670 MB of pretrained models and a live REST connection to Wikipedia — public infrastructure. Anyone who cannot or will not pay per-query or expose queries to a vendor can run this as-is.

A teaching object for classical NLP architecture. The GAP analysis traces every design decision back to the Ferrucci et al. DeepQA papers. Reading the codebase alongside those papers is a curriculum: question analysis → retrieval → hypothesis generation → scoring → confidence calibration. That pipeline is still the skeleton inside every modern RAG system; watson-lite makes the skeleton visible and runnable.

A composable RAG building block. Once a persistent BM25/FAISS index is wired to a domain corpus (see GAP-01 in GAP.md), watson-lite becomes a deterministic retriever and reader that can feed a generation layer or stand alone. The design question is not "does this beat GPT-4?" but "what do you get when you optimise for explainability, zero hallucination, and zero cost instead of raw benchmark score?"

Install

pip install "watson-lite[full]"
python -m spacy download en_core_web_sm

Optional extras:

  • watson-lite[nlp] — spaCy question processing
  • watson-lite[vector] — dense retrieval dependencies
  • watson-lite[rerank] — cross-encoder reranking
  • watson-lite[reader] — extractive QA reader
  • watson-lite[graph] — SPARQL fallback support
  • watson-lite[full] — all runtime features

Usage

CLI

# Single question
watson-lite "Who designed the Eiffel Tower?"
watson-lite "Who was the 44th president of the United States?"

# Interactive mode
watson-lite

# Minimal profile + JSON output
watson-lite --profile minimal --output json "Who designed the Eiffel Tower?"

# Clear cache before running
watson-lite --clear-cache "Who designed the Eiffel Tower?"

# Toggle optional features (ablation-style)
watson-lite --no-vector-retrieval --no-graph-enrichment "Who designed the Eiffel Tower?"

# Query across multiple online datasets
watson-lite --datasets wikipedia,wikibooks "What is Python?"

# Query additional public sources
watson-lite --datasets wikiquote,wikisource,wikinews,pubmed,arxiv "What is Python?"

# Query Elasticsearch
watson-lite \
  --datasets elasticsearch \
  --elasticsearch-url http://localhost:9200 \
  --elasticsearch-index wiki_passages \
  "What is Python?"

# Query Hugging Face datasets-server
watson-lite \
  --datasets huggingface \
  --huggingface-dataset ag_news \
  --huggingface-config default \
  --huggingface-split train \
  "What is Python?"

# Benchmark/eval run from dataset
watson-lite \
  --benchmark-dataset /path/to/benchmark.json \
  --benchmark-output-json /tmp/watson_benchmark.json \
  --benchmark-output-csv /tmp/watson_benchmark.csv

# Full ablation sweep + regression gate against baseline
watson-lite \
  --benchmark-dataset /path/to/benchmark.json \
  --ablation-sweep \
  --regression-check \
  --max-accuracy-drop 0.02 \
  --max-f1-drop 0.02

Benchmark dataset format (.json or .jsonl):

[
  {
    "question": "Who designed the Eiffel Tower?",
    "answers": ["Gustave Eiffel"],
    "evidence_passages": ["designed by Gustave Eiffel"]
  }
]

Python

from watson_lite import WatsonLite

watson = WatsonLite()
answer = watson.answer("Who designed the Eiffel Tower?")

print(answer.answer)        # "Gustave Eiffel"
print(answer.confidence)    # 0.752
print(answer.source)        # "Eiffel Tower"

KPI evaluation

from watson_lite import WatsonLite
from watson_lite.evaluation import BenchmarkLabel, evaluate_kpis

watson = WatsonLite()
answers = [
    watson.answer("Who designed the Eiffel Tower?", verbose=False),
    watson.answer("What is the capital of France?", verbose=False),
]

labels = [
    BenchmarkLabel(
        answers=["Gustave Eiffel"],
        evidence_passages=["designed by Gustave Eiffel"],
    ),
    BenchmarkLabel(
        answers=["Paris"],
        evidence_passages=["capital of France"],
    ),
]

report = evaluate_kpis(answers, labels, recall_k=10, calibration_bins=10)
print(report.answer_success_rate)
print(report.latency_p95_s)
print(report.confidence_calibration_ece)
print(report.confidence_calibration_kl_divergence)
print(report.confidence_calibration_js_divergence)

Each FinalAnswer now includes diagnostics with stage latencies, cache hit/miss counters, retrieval/extraction counts, and top retrieved passages for KPI rollups.

Example output

$ watson-lite "Who was the 44th president of the United States?"

  ANSWER:     Barack Hussein Obama
  CONFIDENCE: 43.6%
  SOURCE:     Barack Obama
  URL:        https://en.wikipedia.org/wiki/Barack Obama

  Confidence breakdown:
    extraction_model: 0.592
    span_agreement: 0.2
    graph_corroboration: 0.0
    passage_rank_signal: 1.0

  Time: 44.60s

API

  • WatsonLite — Main orchestrator. answer(question) runs the full 6-stage pipeline.
  • NLPProcessor — spaCy-based question classification, NER, decomposition.
  • DatasetQueryEngine — Modular dataset querying and aggregation across pluggable providers.
  • BM25Retriever — BM25 retrieval over aggregated online passages.
  • VectorRetriever — Dense vector retrieval (sentence-transformers + FAISS).
  • WikidataGraph — Structured fact enrichment from Wikidata.
  • Ranker — RRF fusion + cross-encoder re-ranking.
  • ExtractiveReader — Span extraction via roberta-base-squad2.
  • ConfidenceScorer — Multi-signal confidence scoring.
  • Cache — SQLite3 cache for Wikipedia/Wikidata/type-coercion responses with TTL expiry, namespace metrics, and bounded-size pruning.

Feature inventory

Core (always on):

  • NLP parse
  • Dataset query engine fetch
  • BM25 retrieve
  • Span extraction
  • Final scoring shell

Optional toggles (default enabled):

  • Vector retrieval (--no-vector-retrieval)
  • Query expansion variants (--no-query-expansion)
  • Wikidata graph enrichment (--no-graph-enrichment)
  • Cross-encoder reranking (--no-cross-encoder-reranking)
  • Question-type bonus (--no-question-type-bonus)
  • Type-coercion signal (--no-type-coercion)

Dataset providers:

  • wikipedia
  • wikibooks
  • wikiquote
  • wikisource
  • wikinews
  • pubmed
  • arxiv
  • openlibrary
  • stackexchange
  • dbpedia
  • oeis
  • elasticsearch (configure with --elasticsearch-url and --elasticsearch-index, or WATSON_LITE_ELASTICSEARCH_URL and WATSON_LITE_ELASTICSEARCH_INDEX)
  • huggingface
    • required: --huggingface-dataset, --huggingface-split
    • optional: --huggingface-config, --huggingface-token
    • env vars: WATSON_LITE_HUGGINGFACE_DATASET, WATSON_LITE_HUGGINGFACE_SPLIT, WATSON_LITE_HUGGINGFACE_CONFIG, WATSON_LITE_HUGGINGFACE_TOKEN

Development

git clone https://github.com/daedalus/watson-lite.git
cd watson_lite
pip install -e ".[test,lint,full]"

# run tests
pytest

# format
ruff format src/ tests/

# lint + type check
prospector --with-tool ruff --with-tool mypy src/

# find unused code
vulture --min-confidence 90 src/

Checked-in benchmark smoke dataset: benchmarks/smoke.json

Architecture

                          +------------------+
                          |  User question   |
                          +------------------+
                                    |
                                    v
                    +-----------------------------+        +-----------------------------+
                    | NLPProcessor                |------->| WikidataGraph               |
                    | - classify question         | entity | - enrich extracted entities |
                    | - extract entities/keywords | names  +-----------------------------+
                    +-----------------------------+                    | graph_results
                          |          |                                 |
                    queries|  sub-    |                                 |
                           | questions|                                 |
                           v          v                                 |
                    +--------------------------+                        |
                    | Query expansion +        |                        |
                    | sub-questions            |                        |
                    +--------------------------+                        |
                                    |                                   |
                                    v                                   |
                    +-----------------------------+                     |
                    | DatasetQueryEngine          |                     |
                    | - Wikipedia REST API        |                     |
                    | - Wikibooks REST API        |                     |
                    +-----------------------------+                     |
                                    |                                   |
                                    v                                   |
                    +-----------------------------+                     |
                    | Parallel retrieval          |                     |
                    | - BM25Retriever             |                     |
                    | - VectorRetriever (FAISS)   |                     |
                    +-----------------------------+                     |
                                    |                                   |
                                    v                                   |
                    +-----------------------------+                     |
                    | Ranker                      |                     |
                    | - RRF fusion                |                     |
                    | - cross-encoder rerank      |                     |
                    +-----------------------------+                     |
                                    |                                   |
                                    v                                   |
                    +-----------------------------+                     |
                    | ExtractiveReader            |                     |
                    | - answer span extraction    |                     |
                    +-----------------------------+                     |
                                    |                                   |
                                    +-------------------+---------------+
                                                        |
                                                        v
                                        +-----------------------------+
                                        | ConfidenceScorer            |
                                        | - extraction score          |
                                        | - span agreement            |
                                        | - graph corroboration       |
                                        | - rank / type-coercion      |
                                        +-----------------------------+
                                                        |
                                                        v
                                        +-----------------------------+
                                        | FinalAnswer + diagnostics   |
                                        +-----------------------------+

SQLite cache (cross-cutting): backs DatasetQueryEngine fetches,
Wikidata lookups, and type-coercion calls with TTL-expiry entries.

Models Used (all pretrained, inference only)

Model Purpose Size
en_core_web_sm spaCy NLP ~12MB
all-MiniLM-L6-v2 Passage embeddings ~90MB
ms-marco-MiniLM-L-6-v2 Cross-encoder reranking ~90MB
deepset/roberta-base-squad2 Extractive span QA ~480MB

Total: ~670MB — runs CPU-only.

Data Sources

  • Wikipedia REST API — Live article retrieval
  • Wikibooks REST API — Live educational content retrieval
  • Wikidata REST API — Structured entity facts (no SPARQL)

Extending

  • Add a domain corpus: Plug a new provider into DatasetQueryEngine.
  • Add more graph sources: Wikidata REST API pattern is reusable.
  • Offline mode: Download Wikipedia dumps and index locally with BM25 + FAISS.

Project details


Download files

Download the file for your platform. If you're not sure which to choose, learn more about installing packages.

Source Distribution

watson_lite-0.1.3.tar.gz (55.2 kB view details)

Uploaded Source

Built Distribution

If you're not sure about the file name format, learn more about wheel file names.

watson_lite-0.1.3-py3-none-any.whl (70.1 kB view details)

Uploaded Python 3

File details

Details for the file watson_lite-0.1.3.tar.gz.

File metadata

  • Download URL: watson_lite-0.1.3.tar.gz
  • Upload date:
  • Size: 55.2 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: twine/6.1.0 CPython/3.13.12

File hashes

Hashes for watson_lite-0.1.3.tar.gz
Algorithm Hash digest
SHA256 b222223bdfd6454a4a084e507db56136b9d77050446a2e092235c735a61a24dc
MD5 6724ab5207c983f17cbc797ad6b34bb5
BLAKE2b-256 0e68bc7aa6eaecc0fb5ddcbea3f9d25baab1b3f9d408929224c1d43d38ba081d

See more details on using hashes here.

Provenance

The following attestation bundles were made for watson_lite-0.1.3.tar.gz:

Publisher: pypi-publish.yml on daedalus/watson_lite

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

File details

Details for the file watson_lite-0.1.3-py3-none-any.whl.

File metadata

  • Download URL: watson_lite-0.1.3-py3-none-any.whl
  • Upload date:
  • Size: 70.1 kB
  • Tags: Python 3
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: twine/6.1.0 CPython/3.13.12

File hashes

Hashes for watson_lite-0.1.3-py3-none-any.whl
Algorithm Hash digest
SHA256 2648239119211cc110263c042bdbb777c5483d4b704afc39df4fd4729491a485
MD5 0fc02a11662152f8aa8eddbb35da22f8
BLAKE2b-256 9b9ec003b2cccae187f8cf20b4a22fd36fecea7edda218592e688ded0b1b52e6

See more details on using hashes here.

Provenance

The following attestation bundles were made for watson_lite-0.1.3-py3-none-any.whl:

Publisher: pypi-publish.yml on daedalus/watson_lite

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

Supported by

AWS Cloud computing and Security Sponsor Datadog Monitoring Depot Continuous Integration Fastly CDN Google Download Analytics Pingdom Monitoring Sentry Error logging StatusPage Status page