6
6

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?

Gemini で数式対応 OCR

Last updated at Posted at 2025-01-02

PDF を画像に変換して、Gemini でテキストを抽出する方法について解説します。数式部分は TeX 形式で出力します。

サンプル

アインシュタインの 1905 年の論文から一部を抜粋して変換した例を示します。

  • Einstein, A. (1905), Ist die Trägheit eines Körpers von seinem Energieinhalt abhängig?. Ann. Phys., 323: 639-641. (PDF)

1905_18_639-641

OCR 出力
da $C$ sich während der Lichtaussendung nicht ändert. Wir
erhalten also:
$K_0 - K_1 = L \left\{ \frac{1}{\sqrt{1 - (\frac{v}{V})^2}} - 1 \right\}.$

Die kinetische Energie des Körpers in bezug auf ($\xi$, $\eta$, $\zeta$) nimmt
infolge der Lichtaussendung ab, und zwar um einen von den
Qualitäten des Körpers unabhängigen Betrag. Die Differenz
$K_0 - K_1$ hängt ferner von der Geschwindigkeit ebenso ab wie
die kinetische Energie des Elektrons (l. c. § 10).
Unter Vernachlässigung von Größen vierter und höherer
Ordnung können wir setzen:
$K_0 - K_1 = \frac{L}{V^2} \frac{v^2}{2}.$

da $C$ sich während der Lichtaussendung nicht ändert. Wir erhalten also:

K_0 - K_1 = L \left\{ \frac{1}{\sqrt{1 - (\frac{v}{V})^2}} - 1 \right\}.

Die kinetische Energie des Körpers in bezug auf ($\xi$, $\eta$, $\zeta$) nimmt infolge der Lichtaussendung ab, und zwar um einen von den Qualitäten des Körpers unabhängigen Betrag. Die Differenz $K_0 - K_1$ hängt ferner von der Geschwindigkeit ebenso ab wie die kinetische Energie des Elektrons (l. c. § 10).

Unter Vernachlässigung von Größen vierter und höherer Ordnung können wir setzen:

K_0 - K_1 = \frac{L}{V^2} \frac{v^2}{2}.

この論文は有名な $E=mc^2$ につながります。

Poppler

Poppler は、PDF ファイルを扱うためのオープンソースライブラリです。PDF の表示、変換、解析などの機能を提供しています。

Poppler には PDF から画像に変換する pdftoppm コマンドが付属していますが、本記事ではファイル名のフォーマット指定やガンマ補正を行うためスクリプトで実装します。

準備

使用している OS のパッケージなどで Poppler をインストールしてください。

その後、Python から Poppler の機能を利用するための pdf2image ライブラリをインストールします。uv での例を示します。

uv python pin 3.10
uv init
uv add pdf2image

変換スクリプト

PDF の各ページを PNG 画像に変換するスクリプトです。

pdf2png.py
import argparse

parser = argparse.ArgumentParser(description='Convert PDF to images')
parser.add_argument('input_pdf', help='Input PDF file')
parser.add_argument('--output-dir', default=None, help='Output directory')
parser.add_argument('--gamma', type=int, default=None, help='Gamma correction value')
args = parser.parse_args()

import os, pdf2image
from tqdm import tqdm
from PIL import Image, ImageEnhance

# PDFを画像に変換
pages = pdf2image.convert_from_path(args.input_pdf)

# 出力ディレクトリの指定がない場合は入力 PDF の拡張子を除いた名前
output_dir = args.output_dir or os.path.splitext(args.input_pdf)[0]

# 出力ディレクトリが存在しない場合は作成
if not os.path.exists(output_dir):
    os.makedirs(output_dir)

# 各ページを処理
for i, page in enumerate(tqdm(pages, desc="Converting pages"), 1):
    img = page.convert("RGB")

    # ガンマ補正
    if args.gamma:
        enhancer = ImageEnhance.Contrast(img)
        img = enhancer.enhance(args.gamma)

    # 保存
    output_path = os.path.join(output_dir, f"{i:03d}.png")
    img.save(output_path, "PNG")

使用方法

PDF ファイルを指定すれば、拡張子を除いた名前のディレクトリを作成して、ページごとに連番で PNG ファイルを生成します。

uv run pdf2png.py input.pdf

出力ディレクトリは --output-dir オプションで指定できます。スキャンした画像が薄い場合、ガンマ補正を行うことができます。

uv run pdf2png.py input.pdf --output-dir pages --gamma 1.5

OCR

生成された PNG 画像からテキストを抽出するために Gemini API を使用します。必要なライブラリをインストールします。

uv add google-generativeai

Google AI Studio に登録して、API キーを取得してください。それを環境変数 GEMINI_API_KEY にセットしてください。

1 ページ 1 リクエストです。Gemini には無料枠があるため、1 日につき 1,500 リクエストまで処理できます。超過した場合はエラーになりますが、自動課金はされません。

無料枠での入出力は学習に利用される可能性があります。機密情報は渡さないようにご注意ください。

スクリプト

指定したディレクトリの PNG ファイルを OCR 処理します。数式を含むテキストの場合、$ で囲ったTeX 数式で出力します。

ocr.py
import argparse

parser = argparse.ArgumentParser()
parser.add_argument("work_dir", help="Working directory")
parser.add_argument("-o", dest="output", default=None, help="Output filename")
args = parser.parse_args()

import sys, os, base64
import google.generativeai as genai
from glob import glob
from tqdm import tqdm

prompt = "Please transcribe the text exactly as it is written, without translating it. Please use $TeX$ to write mathematical equations. Please only return the results, and do not include any comments."
output = args.output or (args.work_dir + ".txt")

model = genai.GenerativeModel(
    model_name="models/gemini-2.0-flash-exp",
    generation_config={
        "temperature": 0.5,
        "top_p": 0.95,
        "top_k": 40,
        "max_output_tokens": 8192,
        "response_mime_type": "text/plain",
    }
)

texts = []
for png in tqdm(sorted(glob(args.work_dir + "/*.png"))):
    txt = os.path.splitext(png)[0] + ".txt"
    if os.path.exists(txt):
        with open(txt, "r", encoding="utf-8") as f:
            texts.append(f.read())
        continue
    try:
        with open(png, "rb") as f:
            png_data = {
                "mime_type": "image/png",
                "data": base64.standard_b64encode(f.read()).decode("utf-8"),
            }
        response = model.generate_content([png_data, prompt])
        with open(txt, "w", encoding="utf-8") as f:
            f.write(response.text)
        texts.append(response.text)
    except Exception as e:
        print(e, file=sys.stderr)

text = "\n--------\n\n".join(t.rstrip() + "\n" for t in texts)
with open(output, "w", encoding="utf-8") as f:
    f.write(text)
print("Output:", output)

使用方法

PNG ファイルがあるディレクトリを指定すれば、ページごとにテキストファイルを生成します。すべてのページの処理が完了すれば、結合してディレクトリ名に .txt を付加したテキストファイルに出力します。途中で中断した場合でも、既に処理済みのファイルはスキップするため、効率的に再開できます。

uv run ocr.py pages

オプションで出力ファイル名が指定できます。

uv run ocr.py pages -o result.txt

google-generativeai==0.8.3 では、終了時に以下の警告が出ます。実害はないので無視してください。

WARNING: All log messages before absl::InitializeLog() is called are written to STDERR
E0000 00:00:1735850262.936540   10965 init.cc:229] grpc_wait_for_shutdown_with_timeout() timed out.

参考

model.generate_content に画像データが渡せることを知りました。

リファレンスのサンプルコードにも載っていました。

response = model.generate_content([{'mime_type': 'application/pdf', 'data': doc_data}, prompt])

このリファレンスは生ビールさんにご教示いただきました。

関連記事

TeX から作成された PDF ファイルは OCR を行わなくても文字が抽出できます。その応用として、論文を翻訳するツールを紹介します。

6
6
0

Register as a new user and use Qiita more conveniently

  1. You get articles that match your needs
  2. You can efficiently read back useful information
  3. You can use dark theme
What you can do with signing up
6
6

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?