Skip to main content

Multi-engine document OCR with cascading fallback

Project description

smart-ocr

PyPI Python License

Multi-engine OCR with cascading fallback, quality audit, and figure extraction.

Process academic papers and documents using free local models first, with automatic cloud fallback for failed pages. Extract and describe figures using vision models.

Features

  • Multi-engine OCR — Nougat, DeepSeek, Mistral, Gemini (via dedicated CLI tools)
  • Smart routing — Free local engines first, cloud only when needed
  • Quality audit — Heuristics + LLM review catches garbage text
  • Figure extraction — Renders figures from PDFs, describes with vision models
  • Parallel processing — Process multiple pages/figures concurrently
  • Batch processing — Process entire directories of papers
  • CLI — Live progress, colored panels, cost tracking
  • Modular engines — Each OCR backend is a standalone CLI tool

Quick Start

# Install globally (recommended)
pipx install smart-ocr --python python3.12

# Or install in a project
pip install smart-ocr

# Pull required Ollama models
ollama pull deepseek-r1:32b    # quality audit
ollama pull deepseek-ocr       # local OCR

# Process a paper
smart-ocr paper.pdf

# Process with figure images saved
smart-ocr paper.pdf --save-figures

# Batch process a folder
smart-ocr batch ~/Papers/ --limit 10

Example Output

Processing a 22-page economics paper:

smart-ocr v0.1.0

kuttner_2001_monetary_policy.pdf
22 pages, 1.2 MB
type: academic

(1) primary ocr
    deepseek
    [+] page 1
    [+] page 2
    ...

(2) quality audit
    [!] page 10 (19.2% garbage)

(3) fallback ocr
    gemini
    [+] page 10

(4) figure processing
    [+] fig 1 (p.1): unknown
    [+] fig 2 (p.8): scatter_plot
    [+] fig 3 (p.12): scatter_plot

---

done 22/22 pages
     3 figures
     241.5s
     $0.0002
     deepseek (21) + gemini (1)

-> output/kuttner_2001_monetary_policy/kuttner_2001_monetary_policy.md

Output structure:

output/<doc_stem>/
├── <doc_stem>.md      # Full OCR text + figure descriptions
├── metadata.json      # Stats, engines used, cost
└── figures/           # With --save-figures
    ├── figure_1_page1.png
    └── ...

Output Formats

Use -f/--format to choose the output format:

Format Content Best For
markdown (default) Formatted text with page headers, inline figure descriptions Human reading, documentation
json Structured data with pages, figures, stats, engine info Programmatic access, downstream processing
txt Plain concatenated text with page separators Simple text extraction

JSON format structure

{
  "document": "/path/to/paper.pdf",
  "pages": [
    {
      "page_num": 1,
      "text": "...",
      "status": "success",
      "engine": "deepseek",
      "confidence": null,
      "figures": [
        {
          "figure_num": 1,
          "figure_type": "scatter_plot",
          "description": "A scatter plot showing...",
          "bbox": [100, 200, 400, 500],
          "image_path": "figures/figure_1_page1.png"
        }
      ]
    }
  ],
  "stats": {
    "total_pages": 22,
    "pages_success": 22,
    "figures_detected": 3,
    "total_cost": 0.0002,
    "total_time": 241.5
  }
}

The metadata.json contains:

{
  "document": "/path/to/paper.pdf",
  "stats": {
    "total_pages": 22,
    "pages_success": 22,
    "total_cost": 0.0002,
    "total_time": 241.5
  },
  "engines_used": { "deepseek": 21, "gemini": 1 },
  "figures": 3,
  "pages_needing_reprocessing": []
}

See examples/ for complete processed papers:

  • kuttner_2001/ - 22 pages, 3 figures (monetary policy)
  • sutskever_2014/ - 9 pages (sequence to sequence learning)
  • bernanke_kuttner_2005/ - 37 pages, 6 vector figures (stock market reaction)

Pipeline Architecture

                                    ┌─────────────┐
                                    │  PDF Input  │
                                    └──────┬──────┘
                                           │
                                           ▼
                              ┌────────────────────────┐
                              │  Document Classifier   │
                              │  (academic vs general) │
                              └───────────┬────────────┘
                                          │
          ┌───────────────────────────────┼───────────────────────────────┐
          │                               │                               │
          ▼                               ▼                               ▼
    ACADEMIC PATH                   GENERAL PATH                    ROUTER AGENT
 (Nougat → DeepSeek)             (DeepSeek → Nougat)            (picks available engine)
          │                               │                               │
          └───────────────────────────────┴───────────────────────────────┘
                                          │
                          ┌───────────────┴───────────────┐
                          │   STAGE 1: PRIMARY OCR        │
                          │                               │
                          │  ┌─────────────────────────┐  │
                          │  │ Local Engines (FREE):   │  │
                          │  │                         │  │
                          │  │  • Nougat               │  │
                          │  │    nougat_ocr lib       │  │
                          │  │    model: 0.1.0-small   │  │
                          │  │                         │  │
                          │  │  • DeepSeek             │  │
                          │  │    Ollama               │  │
                          │  │    deepseek-ocr:latest  │  │
                          │  └─────────────────────────┘  │
                          │             OR                │
                          │  ┌─────────────────────────┐  │
                          │  │ Cloud Engines (PAID):   │  │
                          │  │                         │  │
                          │  │  • Gemini               │  │
                          │  │    Google genai SDK     │  │
                          │  │    gemini-3-flash       │  │
                          │  │    ~$0.0002/page        │  │
                          │  │                         │  │
                          │  │  • Mistral              │  │
                          │  │    Mistral SDK          │  │
                          │  │    pixtral-large        │  │
                          │  │    ~$0.001/page         │  │
                          │  └─────────────────────────┘  │
                          └───────────────┬───────────────┘
                                          │
                                   [Page Results]
                                          │
                                          ▼
                          ┌───────────────────────────────┐
                          │   STAGE 2: QUALITY AUDIT      │
                          │                               │
                          │  ┌─────────────────────────┐  │
                          │  │ Heuristics Checker      │  │
                          │  │ (rule-based, instant):  │  │
                          │  │  • Word count ≥ 50      │  │
                          │  │  • Garbage ratio < 15%  │  │
                          │  │  • Avg word length ok   │  │
                          │  │  • No unicode issues    │  │
                          │  │  • No repeated patterns │  │
                          │  └──────────┬──────────────┘  │
                          │             │                 │
                          │     [pages flagged?]          │
                          │             │                 │
                          │             ▼                 │
                          │  ┌─────────────────────────┐  │
                          │  │ Cross-Check (optional): │  │
                          │  │  Try 2nd local engine   │  │
                          │  └──────────┬──────────────┘  │
                          │             │                 │
                          │     [still flagged?]          │
                          │             │                 │
                          │             ▼                 │
                          │  ┌─────────────────────────┐  │
                          │  │ LLM Auditor (FREE):     │  │
                          │  │  Ollama                 │  │
                          │  │  deepseek-r1:32b        │  │
                          │  │  (reasoning model)      │  │
                          │  │  • Can override heur.   │  │
                          │  │  • verdict: acceptable, │  │
                          │  │    needs_review, poor   │  │
                          │  └──────────┬──────────────┘  │
                          └─────────────┼─────────────────┘
                                        │
                         ┌──────────────┴──────────────┐
                         │                             │
                    [all pass]                   [some flagged]
                         │                             │
                         │                             ▼
                         │             ┌───────────────────────────────┐
                         │             │   STAGE 3: FALLBACK OCR       │
                         │             │                               │
                         │             │  Router selects different     │
                         │             │  engine (not primary):        │
                         │             │   Gemini → Mistral →          │
                         │             │   DeepSeek → Nougat           │
                         │             │                               │
                         │             │  Reprocess flagged pages      │
                         │             └───────────────┬───────────────┘
                         │                             │
                         └─────────────┬───────────────┘
                                       │
                                  [All Pages]
                                       │
                                       ▼
                          ┌────────────────────────────┐
                          │  STAGE 4: FIGURE AGENT     │
                          │  (if enabled)              │
                          │                            │
                          │  ┌──────────────────────┐  │
                          │  │ PyMuPDF Extractor:   │  │
                          │  │  • Vector figures    │  │
                          │  │    (charts, plots)   │  │
                          │  │  • IMAGE blocks      │  │
                          │  │  • Embedded images   │  │
                          │  │  • Filter by size    │  │
                          │  │  • Render at 150 DPI │  │
                          │  └──────────┬───────────┘  │
                          │             │              │
                          │             ▼              │
                          │  ┌──────────────────────┐  │
                          │  │ Vision Engine:       │  │
                          │  │  Gemini 3 Flash      │  │
                          │  │  DeepSeek-OCR        │  │
                          │  │  Pixtral-Large       │  │
                          │  └──────────┬───────────┘  │
                          │             │              │
                          │      [Figure Results]      │
                          └─────────────┬──────────────┘
                                        │
                                        ▼
                          ┌─────────────────────────────┐
                          │      OUTPUT ASSEMBLY        │
                          │                             │
                          │  Format: markdown/json/txt  │
                          │                             │
                          │  output/<doc_stem>/         │
                          │    ├─ document.md           │
                          │    ├─ metadata.json         │
                          │    └─ figures/              │
                          │         └─ figure_N.png     │
                          └─────────────────────────────┘

Cost optimization:

  • Local engines first (Nougat, DeepSeek via Ollama) - FREE
  • Quality audit with local Ollama (llama3.2/qwen2.5) - FREE
  • Cloud fallback only for failed pages (Gemini ~$0.0002/page)
  • Typical 22-page paper: $0.0002 total (only 1 page needed cloud)

Requirements

Base:

  • Python 3.11+
  • Ollama with deepseek-r1:32b (for quality audit) or llama3.2/qwen2.5 as lighter alternatives

OCR Engines (install individually based on needs):

Engine Installation Type Cost
DeepSeek pip install deepseek-ocr-cli Local (Ollama) Free
Nougat pip install nougat-ocr-cli Local (Python) Free
Gemini pip install gemini-ocr-cli Cloud API ~$0.0002/page
Mistral pip install mistral-ocr-cli Cloud API ~$0.001/page

Installation examples:

# Install with local engines only (recommended to start)
pip install smart-ocr[local]

# Install with all engines
pip install smart-ocr[all]

# Install specific engines
pip install smart-ocr[deepseek,gemini]

Setup:

# 1. Install Ollama and pull audit model (reasoning model recommended)
ollama pull deepseek-r1:32b  # best quality, or llama3.2 for speed

# 2. If using DeepSeek OCR, pull its model
ollama pull deepseek-ocr:latest

# 3. If using cloud engines, set API keys
export GEMINI_API_KEY="your-key"
export MISTRAL_API_KEY="your-key"

# 4. Check engine status
smart-ocr engines

OCR Engine CLIs

Each OCR backend is a standalone CLI tool that can be used independently:

CLI Commands

# Simple usage
smart-ocr paper.pdf                    # Process a PDF
smart-ocr paper.pdf --save-figures     # Save figure images

# Full options
smart-ocr process paper.pdf [OPTIONS]
  -o, --output PATH      Output file path
  -f, --format           markdown|json|txt
  --primary ENGINE       Force primary engine
  --fallback ENGINE      Force fallback engine
  --no-audit             Skip quality audit
  --no-figures           Skip figure processing
  --save-figures         Save figure images to disk
  --figures-engine       gemini|deepseek|mistral (default: auto-select)
  --timeout SECONDS      Timeout per page/figure (default: 300)
  --workers N            Parallel workers (default: 4, use 1 for sequential)

# Batch process directory
smart-ocr batch ~/Papers/ [OPTIONS]
  --limit N              Process first N files
  --save-figures         Save all figure images
  --figures-engine       gemini|deepseek|mistral (default: auto-select)
  --timeout SECONDS      Timeout per page/figure (default: 300)
  --workers N            Parallel workers (default: 4)

# Check engines
smart-ocr engines

# Check audit system
smart-ocr audit-status

Parallel Processing

smart-ocr can process multiple pages and figures in parallel to speed up large documents:

# Fast processing with 8 parallel workers (recommended for M3 Pro/Max with 32GB+ RAM)
smart-ocr paper.pdf --workers 8

# Conservative processing for limited RAM
smart-ocr paper.pdf --workers 2

# Sequential processing (most reliable, slowest)
smart-ocr paper.pdf --workers 1

# Extended timeout for very complex pages (10 min per page)
smart-ocr paper.pdf --timeout 600

Hardware recommendations:

RAM Workers Notes
8GB 1-2 Sequential or minimal parallelism
16GB 2-4 Good for most documents
32GB 4-6 Fast processing
64GB+ 6-8 Maximum parallelism

Parallel processing uses ThreadPoolExecutor for concurrent page/figure processing. Each worker loads the page image into memory, so higher worker counts require more RAM.

Timeout configuration:

The default 300s (5 min) timeout works for most pages. For documents with complex figures or slow API responses, increase the timeout:

# smart-ocr.yaml
parallel_pages: 4
parallel_figures: 2
figure_timeout: 600  # 10 min per figure

nougat:
  timeout: 600
deepseek:
  timeout: 600

Configuration

Create smart-ocr.yaml in your project or home directory:

# Engine selection
primary_engine: deepseek
fallback_engine: gemini
figures_engine: gemini  # For figure descriptions

# Quality audit
audit:
  enabled: true
  min_word_count: 50
  garbage_threshold: 0.15

# Figure processing
include_figures: true
save_figures: false
figures_max_total: 25
figures_max_per_page: 3
figure_timeout: 180  # seconds per figure description

# Parallel processing (adjust based on your hardware)
parallel_pages: 4    # pages processed concurrently
parallel_figures: 2  # figures described concurrently

# Output
output_dir: output
output_format: markdown

# Engine-specific timeouts (seconds)
nougat:
  timeout: 300
deepseek:
  timeout: 300
gemini:
  timeout: 300
mistral:
  timeout: 300

Architecture

See docs/ARCHITECTURE.md for detailed module documentation.

src/smart_ocr/
├── cli.py              # Click CLI
├── core/
│   ├── config.py       # AgentConfig dataclass
│   ├── document.py     # PDF loading
│   └── result.py       # OCRResult, PageResult, FigureResult
├── engines/
│   ├── base.py         # BaseEngine ABC
│   ├── deepseek.py     # Ollama/DeepSeek
│   ├── gemini.py       # Google Gemini
│   ├── mistral.py      # Mistral AI
│   └── nougat.py       # Nougat (academic)
├── audit/
│   ├── heuristics.py   # Garbage detection
│   └── llm_audit.py    # Ollama-based review
├── pipeline/
│   ├── processor.py    # 4-stage pipeline
│   └── router.py       # Engine selection
└── ui/                 # Rich console output

License

MIT

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

smart_ocr-0.2.4.tar.gz (568.2 kB view details)

Uploaded Source

Built Distribution

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

smart_ocr-0.2.4-py3-none-any.whl (74.9 kB view details)

Uploaded Python 3

File details

Details for the file smart_ocr-0.2.4.tar.gz.

File metadata

  • Download URL: smart_ocr-0.2.4.tar.gz
  • Upload date:
  • Size: 568.2 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/6.2.0 CPython/3.11.14

File hashes

Hashes for smart_ocr-0.2.4.tar.gz
Algorithm Hash digest
SHA256 72c1c97ff07ffeaf5c8dd2caebd04f53bf19f4ee3114888e92e8ba201a100157
MD5 41501e1f211187ea451240f1938e8362
BLAKE2b-256 ce431d2585ae83f3b73003f891d803091cc374341ca66d2db5a9eab6b5ef0fa4

See more details on using hashes here.

File details

Details for the file smart_ocr-0.2.4-py3-none-any.whl.

File metadata

  • Download URL: smart_ocr-0.2.4-py3-none-any.whl
  • Upload date:
  • Size: 74.9 kB
  • Tags: Python 3
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/6.2.0 CPython/3.11.14

File hashes

Hashes for smart_ocr-0.2.4-py3-none-any.whl
Algorithm Hash digest
SHA256 500c6f031cfaedd5b4af19284dedfab1bd47190f17408cc700166db5974a5233
MD5 3b5cf1ae013ec85c8ea5770714e22118
BLAKE2b-256 35882c46d19bda7609a5a983b7aa3e6ed916717b8332f7a47265be81c04e460b

See more details on using hashes here.

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