OCR em PDF — Extraia Texto de PDFs Digitalizados com PDF Oxide
Extraia texto de PDFs digitalizados com OCR embutido. A partir da v0.3.27, o OCR está disponível em todos os bindings de linguagem — Python, Node.js, Go, C# e Rust — por meio de uma camada FFI unificada (pdf_ocr_engine_create, pdf_ocr_page_needs_ocr, pdf_ocr_extract_text).
Python
from pdf_oxide import PdfDocument
doc = PdfDocument("scanned.pdf")
text = doc.extract_text_ocr(0)
print(text)
Node.js
const { PdfDocument, OcrEngine } = require("pdf-oxide");
const doc = new PdfDocument("scanned.pdf");
const ocr = new OcrEngine();
if (ocr.pageNeedsOcr(doc, 0)) {
console.log(ocr.extractText(doc, 0));
}
ocr.close();
doc.close();
Go
import pdfoxide "github.com/yfedoseev/pdf_oxide/go"
doc, _ := pdfoxide.Open("scanned.pdf")
defer doc.Close()
ocr, _ := pdfoxide.NewOcrEngine()
defer ocr.Close()
if ocr.NeedsOcr(doc, 0) {
text, _ := ocr.ExtractTextWithOcr(doc, 0)
fmt.Println(text)
}
C#
using PdfOxide.Core;
using PdfOxide.Ocr;
using var doc = PdfDocument.Open("scanned.pdf");
using var ocr = new OcrEngine();
if (ocr.PageNeedsOcr(doc, 0))
{
Console.WriteLine(ocr.ExtractText(doc, 0));
}
Rust
use pdf_oxide::PdfDocument;
use pdf_oxide::ocr::{OcrEngine, OcrConfig, OcrExtractOptions, extract_text_with_ocr};
let mut doc = PdfDocument::open("scanned.pdf")?;
let config = OcrConfig::default();
let engine = OcrEngine::new("models/det.onnx", "models/rec.onnx", "models/dict.txt", config)?;
let options = OcrExtractOptions::default();
let text = extract_text_with_ocr(&mut doc, 0, Some(&engine), options)?;
println!("{text}");
Java
import fyi.oxide.pdf.PdfDocument;
try (PdfDocument doc = PdfDocument.open("scanned.pdf")) {
// Lean-tier bindings have no OCR engine handle — extractTextAuto
// routes scanned pages through OCR automatically (graceful fallback).
String text = doc.extractTextAuto(0);
System.out.println(text);
}
PHP
<?php
use PdfOxide\PdfDocument;
$doc = PdfDocument::open("scanned.pdf");
// No OCR engine handle in the lean tier — extractTextAuto routes
// scanned pages through OCR automatically (graceful fallback).
echo $doc->extractTextAuto(0);
Ruby
require "pdf_oxide"
doc = PdfOxide::PdfDocument.open("scanned.pdf")
# No OCR engine handle in the lean tier — extract_text_auto routes
# scanned pages through OCR automatically (graceful fallback).
puts doc.extract_text_auto(0)
C++
#include <pdf_oxide/pdf_oxide.hpp>
auto doc = pdf_oxide::Document::open("scanned.pdf");
auto engine = pdf_oxide::OcrEngine::create("det.onnx", "rec.onnx", "dict.txt");
if (doc.ocr_page_needs_ocr(0)) {
std::string text = doc.ocr_extract_text(0, &engine);
std::cout << text << "\n";
}
Swift
import PdfOxide
let doc = try Document.open("scanned.pdf")
let engine = try OcrEngine.create(
detModelPath: "det.onnx", recModelPath: "rec.onnx", dictPath: "dict.txt")
if try doc.ocrPageNeedsOcr(0) {
let text = try doc.ocrExtractText(0, engine: engine)
print(text)
}
Kotlin
import fyi.oxide.pdf.PdfDocument
PdfDocument.open("scanned.pdf").use { doc ->
// Lean-tier bindings have no OCR engine handle — extractTextAuto
// routes scanned pages through OCR automatically (graceful fallback).
println(doc.extractTextAuto(0))
}
Dart
import 'package:pdf_oxide/pdf_oxide.dart';
final doc = PdfDocument.open('scanned.pdf');
final engine = OcrEngine.create('det.onnx', 'rec.onnx', 'dict.txt');
if (doc.pageNeedsOcr(0)) {
print(doc.ocrExtractText(0, engine));
}
engine.close();
doc.close();
R
library(pdfoxide)
doc <- pdf_open("scanned.pdf")
engine <- pdf_ocr_engine_create("det.onnx", "rec.onnx", "dict.txt")
if (pdf_ocr_page_needs_ocr(doc, 0)) {
text <- pdf_ocr_extract_text(doc, 0, engine)
cat(text)
}
Julia
using PdfOxide
doc = open_document("scanned.pdf")
engine = ocr_engine_create("det.onnx", "rec.onnx", "dict.txt")
if page_needs_ocr(doc, 0)
text = ocr_extract_text(doc, 0, engine)
println(text)
end
Zig
const pdf_oxide = @import("pdf_oxide");
const a = std.heap.page_allocator;
var doc = try pdf_oxide.Document.open("scanned.pdf");
defer doc.deinit();
var engine = try pdf_oxide.OcrEngine.create("det.onnx", "rec.onnx", "dict.txt");
defer engine.deinit();
if (try doc.ocrPageNeedsOcr(0)) {
const text = try doc.ocrExtractText(a, 0, engine);
defer a.free(text);
std.debug.print("{s}\n", .{text});
}
Scala
import fyi.oxide.pdf.PdfDocument
val doc = PdfDocument.open("scanned.pdf")
// Lean-tier bindings have no OCR engine handle — extractTextAuto
// routes scanned pages through OCR automatically (graceful fallback).
println(doc.extractTextAuto(0))
doc.close()
Clojure
(require '[pdf-oxide.core :as pdf])
(with-open [doc (pdf/open "scanned.pdf")]
;; Lean-tier bindings have no OCR engine handle — the AutoExtractor
;; routes scanned pages through OCR automatically (graceful fallback).
(let [ax (pdf/auto-extractor doc)]
(println (pdf/auto-text ax))))
Objective-C
#import "POXPdfOxide.h"
NSError *err = nil;
POXDocument *doc = [POXDocument openPath:@"scanned.pdf" error:&err];
POXOcrEngine *engine = [POXOcrEngine createWithDetModelPath:@"det.onnx"
recModelPath:@"rec.onnx"
dictPath:@"dict.txt"
error:&err];
if ([doc pageNeedsOcr:0 error:&err]) {
NSString *text = [doc ocrExtractText:0 engine:engine error:&err];
NSLog(@"%@", text);
}
Elixir
{:ok, doc} = PdfOxide.open("scanned.pdf")
{:ok, engine} = PdfOxide.ocr_engine("det.onnx", "rec.onnx", "dict.txt")
case PdfOxide.ocr_page_needs_ocr(doc, 0) do
{:ok, true} ->
{:ok, text} = PdfOxide.ocr_extract_text(doc, 0, engine)
IO.puts(text)
_ ->
:ok
end
O PDF Oxide inclui o PaddleOCR via ONNX Runtime — sem necessidade de instalar o Tesseract, sem dependências de sistema, sem chamadas a subprocessos. O motor OCR roda diretamente dentro do processo. Suporta as famílias de modelos PP-OCRv3, PP-OCRv4 e PP-OCRv5.
Observação: OCR não está disponível em WebAssembly (requer o ONNX Runtime nativo). Para Go / Node.js / C# / Rust, compile com a feature
ocr. Os wheels Python já vêm com OCR habilitado por padrão.
OCR de PDF em Python Sem Tesseract
A maioria das soluções de OCR em PDF para Python exige a instalação do Tesseract como dependência do sistema — uma configuração complexa que varia entre sistemas operacionais e ambientes de CI. O PDF Oxide inclui os modelos PaddleOCR diretamente no wheel Python:
- Sem dependências de sistema —
pip install pdf_oxideé tudo que você precisa - Sem chamadas a subprocessos — OCR roda nativamente via ONNX Runtime
- Três famílias de modelos — PP-OCRv3, PP-OCRv4 e PP-OCRv5
- Detecção automática de páginas — identifica quais páginas são digitalizadas e quais têm texto
Comparação: PDF Oxide OCR vs PyMuPDF + Tesseract
| PDF Oxide | PyMuPDF + Tesseract | |
|---|---|---|
| Instalação | pip install pdf_oxide |
pip install pymupdf + Tesseract do sistema |
| Motor OCR | PaddleOCR (ONNX) | Tesseract (subprocesso) |
| Complexidade da configuração | Uma linha | Instalação do Tesseract específica por SO |
| CI/Docker | Sem configuração extra | Requer apt-get install tesseract-ocr |
| Modelos incluídos | Sim (no wheel) | Não (download separado) |
Instalação
Python
pip install pdf_oxide
Os modelos OCR já estão incluídos no wheel. Não são necessários downloads adicionais.
Rust
[dependencies]
pdf_oxide = { version = "0.3", features = ["ocr"] }
Go
go build -tags ocr ./...
Node.js
npm install pdf-oxide --build-from-source -- --features ocr
C#
O pacote NuGet já vem com OCR habilitado nos binários padrão para Linux / macOS / Windows — nenhuma configuração adicional necessária.
Quando Usar OCR
A maioria dos PDFs contém texto incorporado que extract_text() processa em 0,8ms por página. O OCR é necessário apenas para:
- Documentos digitalizados — documentos em papel escaneados para PDF
- PDFs apenas com imagens — PDFs criados a partir de fotos ou capturas de tela
- PDFs com texto como imagem — alguns geradores rasterizam o texto
- Páginas híbridas — páginas com texto nativo e regiões de imagem digitalizada
Versões dos Modelos PP-OCR
O PDF Oxide suporta três gerações de modelos PaddleOCR. A configuração padrão funciona com PP-OCRv3 e PP-OCRv4. Os modelos de servidor PP-OCRv5 exigem uma estratégia de redimensionamento diferente.
PP-OCRv3 / PP-OCRv4 (Padrão)
Modelos otimizados para mobile que reduzem as imagens para caber no tamanho máximo de lado. Ideais para a maioria dos documentos.
- Modelo de detecção: DBNet++ (leve)
- Modelo de reconhecimento: SVTR
- Estratégia de redimensionamento:
MaxSide— reduz o lado mais longo para 960px - Melhor para: documentos padrão, implantação mobile/edge
Python
from pdf_oxide import OcrConfig, OcrEngine
# Default config works with v3/v4 models
config = OcrConfig()
engine = OcrEngine("det_v4.onnx", "rec_v4.onnx", "dict.txt", config)
Rust
use pdf_oxide::ocr::{OcrConfig, OcrEngine};
// Default config: MaxSide { max_side: 960 }
let config = OcrConfig::default();
let engine = OcrEngine::new("det_v4.onnx", "rec_v4.onnx", "dict.txt", config)?;
PP-OCRv5 (Servidor)
Modelos de nível servidor que preservam alta resolução ampliando as imagens quando necessário. Significativamente mais precisos em documentos com texto denso ou letras pequenas.
- Modelo de detecção: DBNet++ (servidor, maior)
- Modelo de reconhecimento: SVTR-v5
- Estratégia de redimensionamento:
MinSide— garante que o lado mais curto tenha pelo menos 64px, limite máximo de 4000px - Melhor para: extração de alta precisão, ambientes de servidor, texto denso
Python
from pdf_oxide import OcrConfig, OcrEngine
# v5 config: high-resolution input for server models
config = OcrConfig(use_v5=True)
engine = OcrEngine("det_v5.onnx", "rec_v5.onnx", "dict_v5.txt", config)
Rust
use pdf_oxide::ocr::{OcrConfig, OcrEngine};
// v5 config: MinSide { min_side: 64, max_side_limit: 4000 }
let config = OcrConfig::v5();
let engine = OcrEngine::new("det_v5.onnx", "rec_v5.onnx", "dict_v5.txt", config)?;
Comparação de Modelos
| Característica | PP-OCRv3/v4 | PP-OCRv5 |
|---|---|---|
| Estratégia de redimensionamento | MaxSide (reduzir para 960px) |
MinSide (ampliar, limite de 4000px) |
| Resolução de entrada | Menor (mais rápido) | Maior (mais preciso) |
| Tamanho do modelo de detecção | ~3 MB | ~12 MB |
| Tamanho do modelo de reconhecimento | ~12 MB | ~25 MB |
| Melhor para | Mobile, edge, documentos padrão | Servidor, texto denso, letras pequenas |
OcrConfig |
OcrConfig() / OcrConfig::default() |
OcrConfig(use_v5=True) / OcrConfig::v5() |
Detecção do Tipo de Página
O PDF Oxide classifica automaticamente as páginas para determinar se o OCR é necessário. A função extract_text_ocr() faz isso internamente, mas você também pode detectar os tipos de página manualmente.
Detecção Automática de Páginas Digitalizadas
Python
from pdf_oxide import PdfDocument
doc = PdfDocument("mixed.pdf")
for i in range(doc.page_count()):
text = doc.extract_text(i)
if len(text.strip()) < 50:
# Likely scanned — use OCR
text = doc.extract_text_ocr(i)
print(f"Page {i + 1} (OCR): {text[:100]}...")
else:
print(f"Page {i + 1} (text): {text[:100]}...")
Rust
use pdf_oxide::PdfDocument;
use pdf_oxide::ocr::{detect_page_type, PageType, OcrEngine, OcrConfig, OcrExtractOptions, extract_text_with_ocr};
let mut doc = PdfDocument::open("mixed.pdf")?;
let engine = OcrEngine::new("det.onnx", "rec.onnx", "dict.txt", OcrConfig::default())?;
for i in 0..doc.page_count() {
let page_type = detect_page_type(&mut doc, i)?;
match page_type {
PageType::NativeText => {
let text = doc.extract_text(i)?;
println!("Page {} (native): {}...", i + 1, &text[..100.min(text.len())]);
}
PageType::ScannedPage => {
let text = extract_text_with_ocr(&mut doc, i, Some(&engine), OcrExtractOptions::default())?;
println!("Page {} (OCR): {}...", i + 1, &text[..100.min(text.len())]);
}
PageType::HybridPage => {
// Has both native text and scanned images — merges both sources
let text = extract_text_with_ocr(&mut doc, i, Some(&engine), OcrExtractOptions::default())?;
println!("Page {} (hybrid): {}...", i + 1, &text[..100.min(text.len())]);
}
}
}
Java
import fyi.oxide.pdf.PdfDocument;
try (PdfDocument doc = PdfDocument.open("mixed.pdf")) {
for (int i = 0; i < doc.pageCount(); i++) {
// extractTextAuto classifies each page and routes scanned
// pages through OCR automatically (graceful fallback).
System.out.printf("Page %d: %s%n", i + 1, doc.extractTextAuto(i));
}
}
PHP
<?php
use PdfOxide\PdfDocument;
$doc = PdfDocument::open("mixed.pdf");
for ($i = 0; $i < $doc->pageCount(); $i++) {
// extractTextAuto classifies each page and routes scanned
// pages through OCR automatically (graceful fallback).
printf("Page %d: %s\n", $i + 1, $doc->extractTextAuto($i));
}
Ruby
require "pdf_oxide"
doc = PdfOxide::PdfDocument.open("mixed.pdf")
doc.page_count.times do |i|
# extract_text_auto classifies each page and routes scanned
# pages through OCR automatically (graceful fallback).
puts "Page #{i + 1}: #{doc.extract_text_auto(i)}"
end
C++
#include <pdf_oxide/pdf_oxide.hpp>
auto doc = pdf_oxide::Document::open("mixed.pdf");
auto engine = pdf_oxide::OcrEngine::create("det.onnx", "rec.onnx", "dict.txt");
for (int i = 0; i < doc.page_count(); ++i) {
std::string text = doc.ocr_page_needs_ocr(i)
? doc.ocr_extract_text(i, &engine) // scanned / hybrid → OCR
: doc.extract_text(i); // native text
std::cout << "Page " << (i + 1) << ": " << text << "\n";
}
Swift
import PdfOxide
let doc = try Document.open("mixed.pdf")
let engine = try OcrEngine.create(
detModelPath: "det.onnx", recModelPath: "rec.onnx", dictPath: "dict.txt")
for i in 0..<(try doc.pageCount()) {
let text = try doc.ocrPageNeedsOcr(i)
? doc.ocrExtractText(i, engine: engine) // scanned / hybrid → OCR
: doc.extractText(i) // native text
print("Page \(i + 1): \(text)")
}
Kotlin
import fyi.oxide.pdf.PdfDocument
PdfDocument.open("mixed.pdf").use { doc ->
for (i in 0 until doc.pageCount()) {
// extractTextAuto classifies each page and routes scanned
// pages through OCR automatically (graceful fallback).
println("Page ${i + 1}: ${doc.extractTextAuto(i)}")
}
}
Dart
import 'package:pdf_oxide/pdf_oxide.dart';
final doc = PdfDocument.open('mixed.pdf');
final engine = OcrEngine.create('det.onnx', 'rec.onnx', 'dict.txt');
for (var i = 0; i < doc.pageCount; i++) {
final text = doc.pageNeedsOcr(i)
? doc.ocrExtractText(i, engine) // scanned / hybrid → OCR
: doc.extractText(i); // native text
print('Page ${i + 1}: $text');
}
engine.close();
doc.close();
R
library(pdfoxide)
doc <- pdf_open("mixed.pdf")
engine <- pdf_ocr_engine_create("det.onnx", "rec.onnx", "dict.txt")
for (i in seq_len(pdf_page_count(doc)) - 1) {
text <- if (pdf_ocr_page_needs_ocr(doc, i)) {
pdf_ocr_extract_text(doc, i, engine) # scanned / hybrid -> OCR
} else {
pdf_extract_text(doc, i) # native text
}
cat(sprintf("Page %d: %s\n", i + 1, text))
}
Julia
using PdfOxide
doc = open_document("mixed.pdf")
engine = ocr_engine_create("det.onnx", "rec.onnx", "dict.txt")
for i in 0:(page_count(doc) - 1)
text = page_needs_ocr(doc, i) ?
ocr_extract_text(doc, i, engine) : # scanned / hybrid -> OCR
extract_text(doc, i) # native text
println("Page $(i + 1): $text")
end
Zig
const pdf_oxide = @import("pdf_oxide");
const a = std.heap.page_allocator;
var doc = try pdf_oxide.Document.open("mixed.pdf");
defer doc.deinit();
var engine = try pdf_oxide.OcrEngine.create("det.onnx", "rec.onnx", "dict.txt");
defer engine.deinit();
var i: i32 = 0;
const n = try doc.pageCount();
while (i < n) : (i += 1) {
const text = if (try doc.ocrPageNeedsOcr(i))
try doc.ocrExtractText(a, i, engine) // scanned / hybrid → OCR
else
try doc.extractText(a, i); // native text
defer a.free(text);
std.debug.print("Page {d}: {s}\n", .{ i + 1, text });
}
Scala
import fyi.oxide.pdf.PdfDocument
val doc = PdfDocument.open("mixed.pdf")
for (i <- 0 until doc.pageCount) {
// extractTextAuto classifies each page and routes scanned
// pages through OCR automatically (graceful fallback).
println(s"Page ${i + 1}: ${doc.extractTextAuto(i)}")
}
doc.close()
Clojure
(require '[pdf-oxide.core :as pdf])
(with-open [doc (pdf/open "mixed.pdf")]
;; The AutoExtractor classifies each page and routes scanned pages
;; through OCR automatically (graceful fallback).
(println (pdf/auto-text (pdf/auto-extractor doc))))
Objective-C
#import "POXPdfOxide.h"
NSError *err = nil;
POXDocument *doc = [POXDocument openPath:@"mixed.pdf" error:&err];
POXOcrEngine *engine = [POXOcrEngine createWithDetModelPath:@"det.onnx"
recModelPath:@"rec.onnx"
dictPath:@"dict.txt"
error:&err];
NSInteger n = [doc pageCountError:&err];
for (NSInteger i = 0; i < n; i++) {
NSString *text = [doc pageNeedsOcr:i error:&err]
? [doc ocrExtractText:i engine:engine error:&err] // scanned / hybrid → OCR
: [doc extractText:i error:&err]; // native text
NSLog(@"Page %ld: %@", (long)(i + 1), text);
}
Elixir
{:ok, doc} = PdfOxide.open("mixed.pdf")
{:ok, engine} = PdfOxide.ocr_engine("det.onnx", "rec.onnx", "dict.txt")
{:ok, n} = PdfOxide.page_count(doc)
for i <- 0..(n - 1) do
{:ok, text} =
case PdfOxide.ocr_page_needs_ocr(doc, i) do
{:ok, true} -> PdfOxide.ocr_extract_text(doc, i, engine) # scanned / hybrid -> OCR
_ -> PdfOxide.extract_text(doc, i) # native text
end
IO.puts("Page #{i + 1}: #{text}")
end
Variantes de PageType (Rust)
| Variante | Descrição |
|---|---|
NativeText |
Página com texto incorporado — OCR desnecessário |
ScannedPage |
Página totalmente digitalizada (imagem grande, sem texto ou mínimo) — OCR completo |
HybridPage |
Página com texto nativo e imagens digitalizadas — combina texto nativo com resultados do OCR |
O auxiliar needs_ocr() retorna true tanto para ScannedPage quanto para HybridPage:
use pdf_oxide::ocr::needs_ocr;
if needs_ocr(&mut doc, 0)? {
let text = extract_text_with_ocr(&mut doc, 0, Some(&engine), OcrExtractOptions::default())?;
}
Como Funciona
- O PDF Oxide renderiza a página para uma imagem internamente (a 300 DPI)
- A imagem é redimensionada conforme a estratégia de detecção (
MaxSidepara v3/v4,MinSidepara v5) - DBNet++ localiza regiões de texto como caixas delimitadoras quadrilaterais
- SVTR lê os caracteres de cada região detectada
- Os resultados são montados em texto com ordenação por ordem de leitura
- Para páginas híbridas, o texto OCR é mesclado com o texto nativo
Todo o pipeline roda dentro do processo via ONNX Runtime. Sem binários externos, sem chamadas a subprocessos, sem arquivos temporários.
Configuração do OCR
Python
from pdf_oxide import OcrConfig, OcrEngine
# Default (v3/v4)
config = OcrConfig()
# PP-OCRv5 server models
config = OcrConfig(use_v5=True)
# Custom thresholds
config = OcrConfig(
det_threshold=0.5, # Detection confidence (0.0-1.0)
box_threshold=0.7, # Box confidence (0.0-1.0)
rec_threshold=0.6, # Recognition confidence (0.0-1.0)
num_threads=8, # ONNX Runtime threads
max_candidates=500, # Max text regions
)
# v5 with custom thresholds
config = OcrConfig(use_v5=True, det_threshold=0.4, num_threads=8)
engine = OcrEngine("det.onnx", "rec.onnx", "dict.txt", config)
Rust
use pdf_oxide::ocr::{OcrConfig, OcrConfigBuilder, DetResizeStrategy};
// Default (v3/v4): MaxSide { max_side: 960 }
let config = OcrConfig::default();
// PP-OCRv5: MinSide { min_side: 64, max_side_limit: 4000 }
let config = OcrConfig::v5();
// Custom builder
let config = OcrConfig::builder()
.det_threshold(0.5)
.box_threshold(0.7)
.rec_threshold(0.6)
.num_threads(8)
.max_candidates(500)
.detect_styles(true) // Enable style detection from OCR geometry
.build();
// Custom resize strategy
let config = OcrConfig::builder()
.det_resize_strategy(DetResizeStrategy::MinSide {
min_side: 128,
max_side_limit: 6000,
})
.build();
DetResizeStrategy (Rust)
Controla como as imagens de entrada são redimensionadas antes de o modelo de detecção ser executado.
| Variante | Campos | Descrição |
|---|---|---|
MaxSide |
max_side: u32 (padrão: 960) |
Reduz a imagem para que o lado mais longo caiba em max_side. Padrão para PP-OCRv3/v4. |
MinSide |
min_side: u32 (padrão: 64), max_side_limit: u32 (padrão: 4000) |
Amplia para que o lado mais curto tenha pelo menos min_side, com limite em max_side_limit. Padrão para PP-OCRv5. |
Campos de OcrConfig
| Campo | Tipo | Padrão | Descrição |
|---|---|---|---|
det_threshold |
f32 |
0.3 |
Limiar de probabilidade de detecção |
box_threshold |
f32 |
0.6 |
Limiar de confiança da caixa |
rec_threshold |
f32 |
0.5 |
Limiar de confiança do reconhecimento |
det_max_side |
u32 |
960 |
Dimensão máxima da imagem (compatível com v3/v4) |
det_resize_strategy |
DetResizeStrategy |
MaxSide { 960 } |
Estratégia de redimensionamento da imagem |
rec_target_height |
u32 |
48 |
Altura alvo para recortes de reconhecimento |
num_threads |
usize |
4 |
Threads de inferência do ONNX Runtime |
unclip_ratio |
f32 |
1.5 |
Taxa de expansão da caixa |
max_candidates |
usize |
1000 |
Máximo de regiões de texto a detectar |
detect_styles |
bool |
true |
Detectar estilos de fonte pela geometria do OCR |
det_model_path |
Option<PathBuf> |
None |
Caminho para modelo de detecção personalizado |
rec_model_path |
Option<PathBuf> |
None |
Caminho para modelo de reconhecimento personalizado |
dict_path |
Option<PathBuf> |
None |
Caminho para dicionário de caracteres personalizado |
Modelos Personalizados
Use seus próprios modelos ONNX em vez dos embutidos:
Rust
use pdf_oxide::ocr::OcrConfig;
let config = OcrConfig::builder()
.det_model_path("models/custom_det.onnx")
.rec_model_path("models/custom_rec.onnx")
.dict_path("models/custom_dict.txt")
.build();
Detecção de Estilos
Quando detect_styles está habilitado (padrão), o PDF Oxide infere estilos de fonte (negrito, nível de cabeçalho) a partir da geometria do OCR — tamanho do texto, espaçamento e posição. Isso melhora a saída da conversão para Markdown em páginas digitalizadas.
let config = OcrConfig::builder()
.detect_styles(true) // Infer styles from text geometry
.build();
OCR vs Tesseract
| Característica | PDF Oxide OCR | Tesseract (via PyMuPDF) |
|---|---|---|
| Instalação | pip install pdf_oxide |
Pacote do sistema + pytesseract |
| Dependências do sistema | Nenhuma | Binário do Tesseract necessário |
| Runtime | ONNX (in-process) | Chamada a subprocesso |
| Versões de modelos | PP-OCRv3, v4, v5 | Tesseract LSTM |
| Idiomas | Multilíngue | Requer pacotes de idioma |
| Complexidade da configuração | Zero | Moderada |
| Modelo de detecção | DBNet++ | Interno do Tesseract |
| Modelo de reconhecimento | SVTR / SVTR-v5 | Tesseract LSTM |
| Suporte a alta resolução | Estratégia MinSide (v5) |
Configuração de DPI |
| Detecção de tipo de página | Automática (nativo/digitalizado/híbrido) | Manual |
DPI Personalizado
Controle a resolução de renderização ao converter páginas PDF em imagens para OCR:
Python
from pdf_oxide import PdfDocument
doc = PdfDocument("scanned.pdf")
# Default is 300 DPI — good balance of accuracy and speed
text = doc.extract_text_ocr(0)
# Higher DPI for better accuracy on fine print
text = doc.extract_text_ocr(0) # DPI configured via OcrExtractOptions in Rust
Rust
use pdf_oxide::ocr::OcrExtractOptions;
// Higher DPI = better accuracy but slower
let options = OcrExtractOptions::default().with_dpi(300.0);
// Lower DPI = faster but less accurate
let options = OcrExtractOptions::default().with_dpi(150.0);
Estrutura da Saída OCR (Rust)
O método OcrEngine::ocr_image() retorna resultados detalhados com pontuações de confiança por span:
use pdf_oxide::ocr::OcrEngine;
let engine = OcrEngine::new("det.onnx", "rec.onnx", "dict.txt", Default::default())?;
let output = engine.ocr_image(&image)?;
// Full text in reading order
println!("{}", output.text_in_reading_order());
// Per-span details
for span in &output.spans {
println!("Text: '{}' (confidence: {:.2})", span.text, span.confidence);
println!(" Bounding box: {:?}", span.bounding_rect());
println!(" Per-char confidence: {:?}", span.char_confidences);
}
// Overall confidence
println!("Total confidence: {:.2}", output.total_confidence);
Campos de OcrOutput
| Campo / Método | Tipo | Descrição |
|---|---|---|
spans |
Vec<OcrSpan> |
Todas as regiões de texto reconhecidas |
total_confidence |
f32 |
Confiança média de todos os spans |
text() |
String |
Todo o texto concatenado com espaços |
text_in_reading_order() |
String |
Texto ordenado por posição (cima para baixo, esquerda para direita) |
Campos de OcrSpan
| Campo | Tipo | Descrição |
|---|---|---|
text |
String |
Texto reconhecido |
polygon |
[[f32; 2]; 4] |
Caixa delimitadora quadrilateral (4 cantos) |
confidence |
f32 |
Confiança geral (0.0–1.0) |
char_confidences |
Vec<f32> |
Pontuações de confiança por caractere |
Processamento OCR em Lote
Processe um diretório de PDFs digitalizados:
Python
from pdf_oxide import PdfDocument, PdfError
from pathlib import Path
pdf_dir = Path("scans/")
output_dir = Path("text-output/")
output_dir.mkdir(exist_ok=True)
for pdf_path in pdf_dir.glob("*.pdf"):
try:
doc = PdfDocument(str(pdf_path))
pages = []
for i in range(doc.page_count()):
text = doc.extract_text(i)
if len(text.strip()) < 50:
text = doc.extract_text_ocr(i)
pages.append(text)
out_path = output_dir / pdf_path.with_suffix(".txt").name
out_path.write_text("\n\n".join(pages), encoding="utf-8")
except PdfError as e:
print(f"Error: {pdf_path.name}: {e}")
Rust
use pdf_oxide::PdfDocument;
use pdf_oxide::ocr::{OcrEngine, OcrConfig, OcrExtractOptions, extract_text_with_ocr, needs_ocr};
use std::fs;
use std::path::Path;
let engine = OcrEngine::new("det.onnx", "rec.onnx", "dict.txt", OcrConfig::default())?;
let options = OcrExtractOptions::default();
for entry in fs::read_dir("scans/")? {
let path = entry?.path();
if path.extension().map_or(false, |e| e == "pdf") {
let mut doc = PdfDocument::open(path.to_str().unwrap())?;
let mut all_text = String::new();
for i in 0..doc.page_count() {
let text = if needs_ocr(&mut doc, i)? {
extract_text_with_ocr(&mut doc, i, Some(&engine), options.clone())?
} else {
doc.extract_text(i)?
};
all_text.push_str(&text);
all_text.push_str("\n\n");
}
let out_path = Path::new("text-output/")
.join(path.file_stem().unwrap())
.with_extension("txt");
fs::write(out_path, &all_text)?;
}
}
OCR Paralelo (Python)
from pdf_oxide import PdfDocument
from multiprocessing import Pool
from pathlib import Path
def ocr_pdf(pdf_path: str) -> dict:
doc = PdfDocument(pdf_path)
text = ""
for i in range(doc.page_count()):
text += doc.extract_text_ocr(i) + "\n"
return {"file": pdf_path, "text": text}
pdf_files = [str(p) for p in Path("scans/").glob("*.pdf")]
with Pool(4) as pool:
results = pool.map(ocr_pdf, pdf_files)
OCR para Markdown
Converta páginas digitalizadas para Markdown:
Python
from pdf_oxide import PdfDocument
doc = PdfDocument("scanned-report.pdf")
for i in range(doc.page_count()):
md = doc.to_markdown(i, detect_headings=True)
if len(md.strip()) < 50:
# Scanned page — OCR then format
text = doc.extract_text_ocr(i)
md = text # OCR output is plain text
print(f"--- Page {i + 1} ---")
print(md)
Rust
use pdf_oxide::PdfDocument;
use pdf_oxide::ocr::{OcrEngine, OcrConfig, OcrExtractOptions, needs_ocr, extract_text_with_ocr};
let mut doc = PdfDocument::open("scanned-report.pdf")?;
let engine = OcrEngine::new("det.onnx", "rec.onnx", "dict.txt", OcrConfig::default())?;
for i in 0..doc.page_count() {
let text = if needs_ocr(&mut doc, i)? {
extract_text_with_ocr(&mut doc, i, Some(&engine), OcrExtractOptions::default())?
} else {
doc.to_markdown(i, &Default::default())?
};
println!("--- Page {} ---\n{}", i + 1, text);
}
Java
import fyi.oxide.pdf.PdfDocument;
try (PdfDocument doc = PdfDocument.open("scanned-report.pdf")) {
for (int i = 0; i < doc.pageCount(); i++) {
String md = doc.toMarkdown(i);
if (md.strip().length() < 50) {
// Scanned page — auto-routing returns OCR text (plain).
md = doc.extractTextAuto(i);
}
System.out.printf("--- Page %d ---%n%s%n", i + 1, md);
}
}
PHP
<?php
use PdfOxide\PdfDocument;
$doc = PdfDocument::open("scanned-report.pdf");
for ($i = 0; $i < $doc->pageCount(); $i++) {
$md = $doc->toMarkdown($i);
if (strlen(trim($md)) < 50) {
// Scanned page — auto-routing returns OCR text (plain).
$md = $doc->extractTextAuto($i);
}
printf("--- Page %d ---\n%s\n", $i + 1, $md);
}
Ruby
require "pdf_oxide"
doc = PdfOxide::PdfDocument.open("scanned-report.pdf")
doc.page_count.times do |i|
md = doc.to_markdown(i)
if md.strip.length < 50
# Scanned page — auto-routing returns OCR text (plain).
md = doc.extract_text_auto(i)
end
puts "--- Page #{i + 1} ---\n#{md}"
end
C++
#include <pdf_oxide/pdf_oxide.hpp>
auto doc = pdf_oxide::Document::open("scanned-report.pdf");
auto engine = pdf_oxide::OcrEngine::create("det.onnx", "rec.onnx", "dict.txt");
for (int i = 0; i < doc.page_count(); ++i) {
std::string text = doc.ocr_page_needs_ocr(i)
? doc.ocr_extract_text(i, &engine) // scanned / hybrid → OCR
: doc.to_markdown(i); // native → Markdown
std::cout << "--- Page " << (i + 1) << " ---\n" << text << "\n";
}
Swift
import PdfOxide
let doc = try Document.open("scanned-report.pdf")
let engine = try OcrEngine.create(
detModelPath: "det.onnx", recModelPath: "rec.onnx", dictPath: "dict.txt")
for i in 0..<(try doc.pageCount()) {
let text = try doc.ocrPageNeedsOcr(i)
? doc.ocrExtractText(i, engine: engine) // scanned / hybrid → OCR
: doc.toMarkdown(i) // native → Markdown
print("--- Page \(i + 1) ---\n\(text)")
}
Kotlin
import fyi.oxide.pdf.PdfDocument
PdfDocument.open("scanned-report.pdf").use { doc ->
for (i in 0 until doc.pageCount()) {
var md = doc.toMarkdown(i)
if (md.trim().length < 50) {
// Scanned page — auto-routing returns OCR text (plain).
md = doc.extractTextAuto(i)
}
println("--- Page ${i + 1} ---\n$md")
}
}
Dart
import 'package:pdf_oxide/pdf_oxide.dart';
final doc = PdfDocument.open('scanned-report.pdf');
final engine = OcrEngine.create('det.onnx', 'rec.onnx', 'dict.txt');
for (var i = 0; i < doc.pageCount; i++) {
final text = doc.pageNeedsOcr(i)
? doc.ocrExtractText(i, engine) // scanned / hybrid → OCR
: doc.toMarkdown(i); // native → Markdown
print('--- Page ${i + 1} ---\n$text');
}
engine.close();
doc.close();
R
library(pdfoxide)
doc <- pdf_open("scanned-report.pdf")
engine <- pdf_ocr_engine_create("det.onnx", "rec.onnx", "dict.txt")
for (i in seq_len(pdf_page_count(doc)) - 1) {
text <- if (pdf_ocr_page_needs_ocr(doc, i)) {
pdf_ocr_extract_text(doc, i, engine) # scanned / hybrid -> OCR
} else {
pdf_to_markdown(doc, i) # native -> Markdown
}
cat(sprintf("--- Page %d ---\n%s\n", i + 1, text))
}
Julia
using PdfOxide
doc = open_document("scanned-report.pdf")
engine = ocr_engine_create("det.onnx", "rec.onnx", "dict.txt")
for i in 0:(page_count(doc) - 1)
text = page_needs_ocr(doc, i) ?
ocr_extract_text(doc, i, engine) : # scanned / hybrid -> OCR
to_markdown(doc, i) # native -> Markdown
println("--- Page $(i + 1) ---\n$text")
end
Zig
const pdf_oxide = @import("pdf_oxide");
const a = std.heap.page_allocator;
var doc = try pdf_oxide.Document.open("scanned-report.pdf");
defer doc.deinit();
var engine = try pdf_oxide.OcrEngine.create("det.onnx", "rec.onnx", "dict.txt");
defer engine.deinit();
var i: i32 = 0;
const n = try doc.pageCount();
while (i < n) : (i += 1) {
const text = if (try doc.ocrPageNeedsOcr(i))
try doc.ocrExtractText(a, i, engine) // scanned / hybrid → OCR
else
try doc.toMarkdown(a, i); // native → Markdown
defer a.free(text);
std.debug.print("--- Page {d} ---\n{s}\n", .{ i + 1, text });
}
Scala
import fyi.oxide.pdf.PdfDocument
val doc = PdfDocument.open("scanned-report.pdf")
for (i <- 0 until doc.pageCount) {
var md = doc.toMarkdown(i)
if (md.trim.length < 50) {
// Scanned page — auto-routing returns OCR text (plain).
md = doc.extractTextAuto(i)
}
println(s"--- Page ${i + 1} ---\n$md")
}
doc.close()
Clojure
(require '[pdf-oxide.core :as pdf])
(with-open [doc (pdf/open "scanned-report.pdf")]
;; The AutoExtractor routes scanned pages through OCR automatically.
(println (pdf/auto-text (pdf/auto-extractor doc))))
Objective-C
#import "POXPdfOxide.h"
NSError *err = nil;
POXDocument *doc = [POXDocument openPath:@"scanned-report.pdf" error:&err];
POXOcrEngine *engine = [POXOcrEngine createWithDetModelPath:@"det.onnx"
recModelPath:@"rec.onnx"
dictPath:@"dict.txt"
error:&err];
NSInteger n = [doc pageCountError:&err];
for (NSInteger i = 0; i < n; i++) {
NSString *text = [doc pageNeedsOcr:i error:&err]
? [doc ocrExtractText:i engine:engine error:&err] // scanned / hybrid → OCR
: [doc toMarkdown:i error:&err]; // native → Markdown
NSLog(@"--- Page %ld ---\n%@", (long)(i + 1), text);
}
Elixir
{:ok, doc} = PdfOxide.open("scanned-report.pdf")
{:ok, engine} = PdfOxide.ocr_engine("det.onnx", "rec.onnx", "dict.txt")
{:ok, n} = PdfOxide.page_count(doc)
for i <- 0..(n - 1) do
{:ok, text} =
case PdfOxide.ocr_page_needs_ocr(doc, i) do
{:ok, true} -> PdfOxide.ocr_extract_text(doc, i, engine) # scanned / hybrid -> OCR
_ -> PdfOxide.to_markdown(doc, i) # native -> Markdown
end
IO.puts("--- Page #{i + 1} ---\n#{text}")
end
Considerações de Desempenho
O OCR é significativamente mais lento que a extração de texto:
| Operação | Velocidade Típica |
|---|---|
| Extração de texto | 0,8ms por página |
| OCR (v3/v4) | 200–1.000ms por página |
| OCR (v5 servidor) | 500–2.000ms por página |
A velocidade do OCR depende da complexidade da página, resolução da imagem, densidade do texto e versão do modelo. PP-OCRv5 é mais lento, mas mais preciso. Para grandes lotes, considere o processamento paralelo (veja Processamento OCR em Lote acima).
Carregar Modelos a Partir de Bytes (Rust)
use pdf_oxide::ocr::{OcrEngine, OcrConfig};
let det_bytes = std::fs::read("models/det.onnx")?;
let rec_bytes = std::fs::read("models/rec.onnx")?;
let dict = std::fs::read_to_string("models/dict.txt")?;
let engine = OcrEngine::from_bytes(&det_bytes, &rec_bytes, &dict, OcrConfig::default())?;
Páginas Relacionadas
- Extração de Texto — extração de texto padrão
- Conversão para Markdown — Markdown com detecção de cabeçalhos
- Renderização de Páginas — renderizar páginas como imagens (usado internamente pelo OCR)
- Processamento em Lote — padrões de processamento paralelo
- Extrair Texto de PDF — guia de extração de texto