Ocrmypdf Readthedocs Io en Stable
Ocrmypdf Readthedocs Io en Stable
Release 0.1.dev83+g072db75
James R. Barlow
2023-09-29
Contents
1 Introduction 3
2 Release notes 7
3 Installing OCRmyPDF 57
6 Cookbook 71
7 PDF optimization 77
9 Advanced features 83
10 Batch processing 89
11 Online deployments 97
12 Performance 99
16 Plugins 111
i
21 Indices and tables 137
Index 141
ii
ocrmypdf Documentation, Release 0.1.dev83+g072db75
OCRmyPDF adds an optical character recognition (OCR) text layer to scanned PDF files, allowing them to be
searched.
PDF is the best format for storing and exchanging scanned documents. Unfortunately, PDFs can be difficult to modify.
OCRmyPDF makes it easy to apply image processing and OCR (recognized, searchable text) to existing PDFs.
Contents 1
ocrmypdf Documentation, Release 0.1.dev83+g072db75
2 Contents
CHAPTER 1
Introduction
OCRmyPDF is an application and library that adds text “layers” to images in PDFs, making scanned image PDFs
searchable. It uses OCR to guess what text is contained in images. It is written in Python. OCRmyPDF supports
plugins that allow customization of its processing steps, and is very tolerant of PDFs that contain scanned images and
“born digital” content that needs no text recognition.
Optical character recognition is technology that converts images of typed or handwritten text, such as in a scanned
document, to computer text that can be selected, searched and copied.
OCRmyPDF uses Tesseract, the best available open source OCR engine, to perform OCR.
PDFs are page description files that attempts to preserve a layout exactly. They contain vector graphics that can contain
raster objects such as scanned images. Because PDFs can contain multiple pages (unlike many image formats) and
can contain fonts and text, it is a good format for exchanging scanned documents.
A PDF page might contain multiple images, even if it only appears to have one image. Some scanners or scanning
software will segment pages into monochromatic text and color regions for example, to improve the compression ratio
and appearance of the page.
Rasterizing a PDF is the process of generating corresponding raster images. OCR engines like Tesseract work with
images, not scalable vector graphics or mixed raster-vector-text graphics such as PDF.
3
ocrmypdf Documentation, Release 0.1.dev83+g072db75
PDF/A is an ISO-standardized subset of the full PDF specification that is designed for archiving (the ‘A’ stands for
Archive). PDF/A differs from PDF primarily by omitting features that would make it difficult to read the file in the
future, such as embedded Javascript, video, audio and references to external fonts. All fonts and resources needed
to interpret the PDF must be contained within it. Because PDF/A disables Javascript and other types of embedded
content, it is probably more secure.
There are various conformance levels and versions, such as “PDF/A-2b”.
Generally speaking, the best format for scanned documents is PDF/A. Some governments and jurisdictions, US Courts
in particular, mandate the use of PDF/A for scanned documents.
Since most people who scan documents are interested in reading them indefinitely into the future, OCRmyPDF gen-
erates PDF/A-2b by default.
PDF/A has a few drawbacks. Some PDF viewers include an alert that the file is a PDF/A, which may confuse some
users. It also tends to produce larger files than PDF, because it embeds certain resources even if they are commonly
available. PDF/A files can be digitally signed, but may not be encrypted, to ensure they can be read in the future.
Fortunately, converting from PDF/A to a regular PDF is trivial, and any PDF viewer can view PDF/A.
OCRmyPDF analyzes each page of a PDF to determine the colorspace and resolution (DPI) needed to capture all of
the information on that page without losing content. It uses Ghostscript to rasterize the page, and then performs OCR
on the rasterized image to create an OCR “layer”. The layer is then grafted back onto the original PDF.
While one can use a program like Ghostscript or ImageMagick to get an image and put the image through Tesseract,
that actually creates a new PDF and many details may be lost. OCRmyPDF can produce a minimally changed PDF as
output.
OCRmyPDF also provides some image processing options, like deskew, which improves the appearance of files and
quality of OCR. When these are used, the OCR layer is grafted onto the processed image instead.
By default, OCRmyPDF produces archival PDFs – PDF/A, which are a stricter subset of PDF features designed for
long term archives. If regular PDFs are desired, this can be disabled with --output-type pdf.
A PDF is similar to an HTML file, in that it contains document structure along with images. Sometimes a PDF does
nothing more than present a full page image, but often there is additional content that would be lost.
A manual process could work like either of these:
1. Rasterize each page as an image, OCR the images, and combine the output into a PDF. This preserves the layout
of each page, but resamples all images (possibly losing quality, increasing file size, introducing compression
artifacts, etc.).
2. Extract each image, OCR, and combine the output into a PDF. This loses the context in which images are used
in the PDF, meaning that cropping, rotation and scaling of pages may be lost. Some scanned PDFs use multiple
images segmented into black and white, grayscale and color regions, with stencil masks to prevent overlap, as
this can enhance the appearance of a file while reducing file size. Clearly, reassembling these images will be
easy. This also loses and text or vector art on any pages in a PDF with both scanned and pure digital content.
4 Chapter 1. Introduction
ocrmypdf Documentation, Release 0.1.dev83+g072db75
In the case of a PDF that is nothing other than a container of images (no rotation, scaling, cropping, one image per
page), the second approach can be lossless.
OCRmyPDF uses several strategies depending on input options and the input PDF itself, but generally speaking it
rasterizes a page for OCR and then grafts the OCR back onto the original. As such it can handle complex PDFs and
still preserve their contents as much as possible.
OCRmyPDF also supports a many, many edge cases that have cropped over several years of development. We support
PDF features like images inside of Form XObjects, and pages with UserUnit scaling. We support rare image formats
like non-monochrome 1-bit images. We warn about files you may not to OCR. Thanks to pikepdf and QPDF, we
auto-repair PDFs that are damaged. (Not that you need to know what any of these are! You should be able to throw
any PDF at it.)
1.6 Limitations
OCRmyPDF is limited by the Tesseract OCR engine. As such it experiences these limitations, as do any other pro-
grams that rely on Tesseract:
• The OCR is not as accurate as commercial OCR solutions.
• It is not capable of recognizing handwriting.
• It may find gibberish and report this as OCR output.
• If a document contains languages outside of those given in the -l LANG arguments, results may be poor.
• It is not always good at analyzing the natural reading order of documents. For example, it may fail to recognize
that a document contains two columns, and may try to join text across columns.
• Poor quality scans may produce poor quality OCR. Garbage in, garbage out.
• It does not expose information about what font family text belongs to.
OCRmyPDF is also limited by the PDF specification:
• PDF encodes the position of text glyphs but does not encode document structure. There is no markup that
divides a document in sections, paragraphs, sentences, or even words (since blank spaces are not represented).
As such all elements of document structure including the spaces between words must be derived heuristically.
Some PDF viewers do a better job of this than others.
• Because some popular open source PDF viewers have a particularly hard time with spaces between words,
OCRmyPDF appends a space to each text element as a workaround (when using --pdf-renderer hocr).
While this mixes document structure with graphical information that ideally should be left to the PDF viewer to
interpret, it improves compatibility with some viewers and does not cause problems for better ones.
Ghostscript also imposes some limitations:
• PDFs containing JBIG2-encoded content will be converted to CCITT Group4 encoding, which has lower com-
pression ratios, if Ghostscript PDF/A is enabled.
• PDFs containing JPEG 2000-encoded content will be converted to JPEG encoding, which may introduce com-
pression artifacts, if Ghostscript PDF/A is enabled.
• Ghostscript may transcode grayscale and color images, either lossy to lossless or lossless to lossy, based on an
internal algorithm. This behavior can be suppressed by setting --pdfa-image-compression to jpeg or
lossless to set all images to one type or the other. Ghostscript has no option to maintain the input image’s
format. (Modern Ghostscript can copy JPEG images without transcoding them.)
• Ghostscript’s PDF/A conversion removes any XMP metadata that is not one of the standard XMP metadata
namespaces for PDFs. In particular, PRISM Metadata is removed.
1.6. Limitations 5
ocrmypdf Documentation, Release 0.1.dev83+g072db75
• Ghostscript’s PDF/A conversion seems to remove or deactivate hyperlinks and other active content.
You can use --output-type pdf to disable PDF/A conversion and produce a standard, non-archival PDF.
Regarding OCRmyPDF itself:
• PDFs that use transparency are not currently represented in the test suite
To the author’s knowledge, OCRmyPDF is the most feature-rich and thoroughly tested command line OCR PDF
conversion tool. If it does not meet your needs, contributions and suggestions are welcome. If not, consider one of
these similar open source programs:
• pdf2pdfocr
• pdfsandwich
Ghostscript recently added three “pdfocr” output devices. They work by rasterizing all content and converting all
pages to a single colour space.
The Docker image ocrmypdf provides a web service front-end that allows files to submitted over HTTP and the
results “downloaded”. This is an HTTP server intended to simplify web services deployments; it is not intended to be
deployed on the public internet and no real security measures to speak of.
In addition, the following third-party integrations are available:
• Nextcloud OCR is a free software plugin for the Nextcloud private cloud software
OCRmyPDF is not designed to be secure against malware-bearing PDFs (see Using OCRmyPDF online). Users should
ensure they comply with OCRmyPDF’s licenses and the licenses of all dependencies. In particular, OCRmyPDF
requires Ghostscript, which is licensed under AGPLv3.
6 Chapter 1. Introduction
CHAPTER 2
Release notes
OCRmyPDF uses semantic versioning for its command line interface and its public API.
OCRmyPDF’s output messages are not considered part of the stable interface - that is, output messages may be
improved at any release level, so parsing them may be unreliable. Use the API to depend on precise behavior.
The public API may be useful in scripts that launch OCRmyPDF processes or that wish to use some of its features for
working with PDFs.
The most recent release of OCRmyPDF is . Any newer versions referred to in these notes may exist the main branch
but have not been tagged yet.
Note: Attention maintainers: these release notes may be updated with information about a forthcoming release that
has not been tagged yet. A release is only official when it’s tagged and posted to PyPI.
2.1 v15.0.1
2.2 v15.0.0
7
ocrmypdf Documentation, Release 0.1.dev83+g072db75
• Dropped support for 32-bit Linux wheels. You must use a 64-bit operating system, and 64-bit versions of
Python, Tesseract and Ghostscript to use OCRmyPDF. Many of our dependencies are dropping 32-bit builds
(e.g. Pillow), and we are following suit. (Maintainers may still build 32-bit versions from source.)
• Changed to trusted release for PyPI publishing.
• pikepdf memory mapping is enabled again for improved performance, now an issue with pikepdf has been fixed.
• ocrmypdf.helpers.calculate_downsample previously had two variants, one that took a PIL.
Image and one that took a tuple[int, int]. The latter was removed.
• The snap version of ocrmypdf is now based on Ubuntu core22.
• We now account situations where a small portion of an image on a page reports a high DPI (resolution). Previ-
ously, the entire page would be rasterized at the highest resolution, which caused performance problems. Now,
the page is rasterized at a resolution based on the average DPI of the page, weighted by the area that each feature
occupies. Typically, small areas of high resolution in PDFs are errors or quirks from the repeated use of assets
and high resolution is not beneficial. #1010#1104#1004#1079#1010
• Ghostscript color conversion strategy is now configurable. #1143
2.3 v14.4.0
• Digitally signed PDFs are now detected. If the PDF is signed, OCRmyPDF will refuse to modify it. Previously,
only encrypted PDFs were detected, not those that were signed but not encrypted. #1040
• In addition, --invalidate-digital-signatures can be used to override the above behavior and mod-
ify the PDF anyway. #1040
• tqdm progress bars replaced with “rich” progress bars. The rich library is a new dependency. Certain APIs that
used tqdm are now deprecated and will be removed in the next major release.
• Improved integration with GitHub Releases. Thanks to @stumpylog.
2.4 v14.3.0
2.5 v14.2.1
• Fixed #977, where images inside Form XObjects were always excluded from image optimization.
2.6 v14.2.0
• Added --tesseract-downsample-above to downsample larger images even when they do not exceed
Tesseract’s internal limits. This can be used to speed up OCR, possibly sacrificing accuracy.
• Fixed resampling AttributeError on older Pillow. #1096
• Removed an error about using Ghostscript on PDFs with that have the /UserUnit feature in use. Previously,
Ghostscript would fail to process these PDFs, but in all supported versions it is now supported, so the error is no
longer needed.
• Improved documentation around installing other language packs for Tesseract.
2.7 v14.1.0
• Added --tesseract-non-ocr-timeout. This allows using Tesseract’s deskew and other non-OCR fea-
tures while disabling OCR using --tesseract-timeout 0.
• Added --tesseract-downsample-large-images. This downsamples larges images that exceed the
maximum image size Tesseract can handle. Large images may still take a long time to process, but this allows
them to be processed if that is desired.
• Fixed #1082, an issue with snap packaged building.
• Change linter to ruff, fix lint errors, update documentation.
2.8 v14.0.4
2.9 v14.0.3
2.10 v14.0.2
2.5. v14.2.1 9
ocrmypdf Documentation, Release 0.1.dev83+g072db75
2.11 v14.0.1
2.12 v14.0.0
2.13 v13.7.0
2.14 v13.6.2
• Added a shim to prevent an “error during error handling” for Python 3.7 and 3.8.
• Modernized some type annotations.
• Improved annotations on our _windows module to help IDEs and mypy figure out what we’re doing.
2.15 v13.6.1
• Require setuptools-scm 7.0.5 to avoid possible issues with source distributions in earlier versions of setuptools-
scm.
• Suppress a spurious warning, improve tests, improve typing and other miscellany.
2.16 v13.6.0
• Added a new initialize plugin hook, making it possible to suppress built-in plugins more easily, among
other possibilities.
• Fixed an issue where unpaper would exit with a “wrong stream” error, probably related to images with an odd
integer width. #887#665
2.17 v13.5.0
• Added a new optimize_pdf plugin hook, making it possible to create plugins that replace or enhance
OCRmyPDF’s PDF optimizer.
• Removed all max version restrictions. Our new policy is to blacklist known-bad releases and only block known-
bad versions of dependencies.
• The naming schema for object that holds all OCR text that OCRmyPDF inserts has changed. This has always
been an implementation detail (and remains so), but possibly, someone was relying on it and would appreciate
the heads-up.
• Cleanup.
2.18 v13.4.7
2.19 v13.4.6
2.20 v13.4.5
2.21 v13.4.4
2.22 v13.4.3
2.17. v13.5.0 11
ocrmypdf Documentation, Release 0.1.dev83+g072db75
2.23 v13.4.2
• Worked around a major regression in Ghostscript 9.56.0 where all OCR text is stripped out of the PDF. It sim-
ply removes all text, even generated by software other than OCRmyPDF. Fortunately, we can ask Ghostscript
9.56.0 to use its old behavior that worked correctly for our purposes. Users must avoid the combination
(Ghostscript 9.56.0, ocrmypdf <13.4.2) since older versions of OCRmyPDF have no way of detecting that
this particular version of Ghostscript removes all OCR text.
• Marked pdfminer 20220319 as supported.
• Fixed some deprecation warnings from recent versions of Pillow and pytest.
• Test suite now covers Python 3.10 (Python 3.10 worked fine before, but was not being tested).
• Docker image now uses debian:bookworm-slim as the base image to fix the Docker image build.
2.24 v13.4.1
• Temporarily make threads rather than processes the default executor worker, due to a persistent deadlock issue
when processes are used. Add a new command line argument --no-use-threads to disable this.
2.25 v13.4.0
2.26 v13.3.0
• Made a harmless but “scary” exception after failing to optimize an image less scary.
• Added a warning if a page image is too large for unpaper to clean. The image is passed through without cleaning.
This is due to a hard-coded limitation in a C library used by unpaper so it cannot be rectified easily.
• We now use better default settings when calling img2pdf.
• We no longer try to optimize images that we failed to save in certain situations.
• We now account for some differences in text output from Tesseract 5 compared to Tesseract 4.
• Better handling of Ghostscript producing empty images when attempting to rasterize page images.
2.27 v13.2.0
• Removed all runtime uses of distutils since it is deprecated in standard library. We previous used distutils.
version to examine version numbers of dependencies at run time, and now use packaging.version for
this. This is a new dependency.
• Fixed an error message advising the user that Ghostscript was not installed being suppressed when this condition
actually happens.
• Fixed an issue with incorrect page number and totals being displayed in the progress bar. This was purely a
display/presentation issue. #876.
2.28 v13.1.1
2.29 v13.1.0
• Changed to using Python concurrent.futures-based parallel execution instead of pools, since futures have now
exceed pools in features.
• If a child worker is terminated (perhaps by the operating system or the user killing it in a task manager), the
parallel task will fail an error message. Previously, the main ocrmypdf process would “hang” indefinitely,
waiting for the child to report.
• Added new argument --tesseract-thresholding to provide control over Tesseract 5’s threshold pa-
rameter.
• Documentation updates and changes. Better documentation for --output-type none, added a few releases
ago. Removed some obsolete documentation.
• Improved bash completions - thanks to @FPille.
2.30 v13.0.0
Breaking changes
• The deprecated module ocrmypdf.leptonica has been removed.
• We no longer depend on Leptonica (liblept) or CFFI (libffi, python3-cffi). (Note that Tesseract
still requires Leptonica; OCRmyPDF no longer directly uses this library.)
• The argument --remove-background is temporarily disabled while we search for an alternative to the
Leptonica implementation of this feature.
• The --threshold argument has been removed, since this also depended on Leptonica. Tesseract 5.x has
implemented improvements to thresholding, so this feature will be redundant anyway.
• --deskew was previous calculated by a Leptonica algorithm. We now use a feature of Tesseract to find the
appropriate the angle to deskew a page. The deskew angle according to Tesseract may differ from Leptonica’s
algorithm. At least in theory, Tesseract’s deskew angle is informed by a more complex analysis than Leptonica,
so this should improve results in general. We also use Pillow to perform the deskewing, which may affect the
appearance of the image compared to Leptonica.
• Support for Python 3.6 was dropped, since this release is approaching end of life.
• We now require pikepdf 4.0 or newer. This, in turn, means that OCRmyPDF requires a system compatible with
the manylinux2014 specification. This change was “forced” by Pillow not releasing manylinux2010 wheels
anymore.
• We no longer provide requirements.txt-style files. Use pip install ocrmypdf[...] instead.
• Bumped required versions of several libraries.
Fixes
2.28. v13.1.1 13
ocrmypdf Documentation, Release 0.1.dev83+g072db75
• Fixed an issue where OCRmyPDF failed to find Ghostscript on Windows even when installed, and would exit
with an error.
• By removing Leptonica, we fixed all issues related to Leptonica on Apple Silicon or Leptonica failing to import
on Windows.
2.31 v12.7.2
• Fixed “invalid version number” error for Tesseract packaging with nonstandard version “5.0.0-rc1.20211030”.
• Fixed use of deprecated importlib.resources.read_binary.
• Replace some uses of string paths with pathlib.Path.
• Fixed a leaked file handle when using --output-type none.
• Removed shims to support versions of pikepdf that are no longer supported.
2.32 v12.7.1
2.33 v12.7.0
• Fixed test suite failure when using pikepdf 3.2.0 that was compiled with pybind11 2.8.0. #843
• Improve advice to user about using --max-image-mpixels if OCR fails for this reason.
• Minor documentation fixes. (Thanks to @mara004.)
• Don’t require importlib-metadata and importlib-resources backports on versions of Python where the standard
library implementation is sufficient. (Thanks to Marco Genasci.)
2.34 v12.6.0
• Implemented --output-type=none to skip producing PDFs for applications that only want sidecar files
(#787).
• Fixed ambiguities in descriptions of behavior of --jbig2-lossy.
• Various improvements to documentation.
2.35 v12.5.0
• Fixed build failure for the combination of PyPy 3.6 and pikepdf 3.0. This combination can work in a source
build but does not work with wheels.
• Accepted bot that wanted to upgrade our deprecated requirements.txt.
• Documentation updates.
• Replace pkg_resources and install dependency on setuptools with importlib-metadata and importlib-resources.
• Fixed regression in hocrtransform causing text to be omitted when this renderer was used.
• Fixed some typing errors.
2.36 v12.4.0
2.37 v12.3.3
2.38 v12.3.2
2.39 v12.3.1
• Fixed issue with selection of text when using the hOCR renderer (#813).
• Fixed build errors with the Docker image by upgrading to a newer Ubuntu. Also set the timezone of this image
to UTC.
2.40 v12.3.0
• Fixed a regression introduced in Pillow 8.3.0. Pillow no longer rounds DPI for image resolutions. We now
account for this (#802).
• We no longer use some API calls that are deprecated in the latest versions of pikepdf.
• Improved error message when a language is requested that doesn’t look like a typical ISO 639-2 code.
• Fixed some tests that attempted to symlink on Windows, breaking tests on a Windows desktop but not usually
on CI.
• Documentation fixes (thanks to @mara004)
2.36. v12.4.0 15
ocrmypdf Documentation, Release 0.1.dev83+g072db75
2.41 v12.2.0
2.42 v12.1.0
• For security reasons we now require Pillow >= 8.2.x. (Older versions will continue to work if upgrading is not
an option.)
• The build system was reorganized to rely on setup.cfg instead of setup.py. All changes should work
with previously supported versions of setuptools.
• The files in requirements/* are now considered deprecated but will be retained for v12. Instead use pip
install ocrmypdf[test] instead of requirements/test.txt, etc. These files will be removed in
v13.
2.43 v12.0.3
• Expand the list of languages supported by the hocr PDF renderer. Several languages were previously considered
not supported, particularly those non-European languages that use the Latin alphabet.
• Fixed a case where the exception stack trace was suppressed in verbose mode.
• Improved documentation around commercial OCR.
2.44 v12.0.2
• Fixed exception thrown when using --remove-background on files containing small images (#769).
• Improve documentation for description of adding language packs to the Docker image and corrected name of
French language pack.
2.45 v12.0.1
2.46 v12.0.0
Breaking changes
• Due to recent security issues in pikepdf, Pillow and reportlab, we now require newer versions of these libraries
and some of their dependencies. (If necessary, package maintainers may override these versions at their discre-
tion; lower versions will often work.)
• We now use the “LeaveColorUnchanged” color conversion strategy when directing Ghostscript to create a
PDF/A. Generally this is faster than performing a color conversion, which is not always necessary.
• OCR text is now packaged in a Form XObject. This makes it easier to isolate OCR from other document content.
However, some poorly implemented PDF text extraction algorithms may fail to detect the text.
• Many API functions have stricter parameter checking or expect keyword arguments were they previously did
not.
• Some deprecated functions in ocrmypdf.optimize were removed.
• The ocrmypdf.leptonica module is now deprecated, due to difficulties with the current strategy of ABI
binding on newer platforms like Apple Silicon. It will be removed and replaced, either by repackaging Leptonica
as an independent library using or using a different image processing library.
• Continuous integration moved to GitHub Actions.
• We no longer depend on pytest_helpers_namespace for testing.
New features
• New plugin hook: get_progressbar_class, for progress reporting, allowing developers to replace the
standard console progress bar with some other mechanism, such as updating a GUI progress bar.
• New plugin hook: get_executor, for replacing the concurrency model. This is primarily to support execu-
tion on AWS Lambda, which does not support standard Python multiprocessing due to its lack of shared
memory.
• New plugin hook: get_logging_console, for replacing the standard way OCRmyPDF outputs its mes-
sages.
• New plugin hook: filter_pdf_page, for modifying individual PDF pages produced by OCRmyPDF.
• OCRmyPDF now runs on nonstandard execution environments that do not have interprocess semaphores, such
as AWS Lambda and Android Termux. If the environment does not have semaphores, OCRmyPDF will auto-
matically select an alternate process executor that does not use semaphores.
• Continuous integration moved to GitHub Actions.
• We now generate an ARM64-compatible Docker image alongside the x64 image. Thanks to @andkrause for
doing most of the work in a pull request several months ago, which we were finally able to integrate now. Also
thanks to @0x326 for review comments.
Fixes
• Fixed a possible deadlock on attempting to flush sys.stderr when older versions of Leptonica are in use.
• Some worker processes inherited resources from their parents such as log handlers that may have also lead to
deadlocks. These resources are now released.
• Improvements to test coverage.
• Removed vestiges of support for Tesseract versions older than 4.0.0-beta1 ( which ships with Ubuntu 18.04).
• OCRmyPDF can now parse all of Tesseract version numbers, since several schemes have been in use.
• Fixed an issue with parsing PDFs that contain images drawn at a scale of 0. (#761)
• Removed a frequently repeated message about disabling mmap.
2.47 v11.7.3
• Exclude CCITT Group 3 images from being optimized. Some libraries OCRmyPDF uses do not seem to handle
this obscure compression format properly. You may get errors or possible corrupted output images without this
fix.
2.47. v11.7.3 17
ocrmypdf Documentation, Release 0.1.dev83+g072db75
2.48 v11.7.2
• Updated pinned versions in main.txt, primarily to upgrade Pillow to 8.1.2, due to recently disclosed security
vulnerabilities in that software.
• The --sidecar parameter now causes an exception if set to the same file as the input or output PDF.
2.49 v11.7.1
• Some exceptions while attempting image optimization were only logged at the debug level, causing them to be
suppressed. These errors are now logged appropriately.
• Improved the error message related to --unpaper-args.
• Updated documentation to mention the new conda distribution.
2.50 v11.7.0
• We now support using --sidecar in conjunction with --pages; these arguments used to be mutually ex-
clusive. (#735)
• Fixed a possible issue with PDF/A-1b generation. Acrobat complained that our PDFs use object streams. More
robust PDF/A validators like veraPDF don’t consider this a problem, but we’ll honor Acrobat’s objection from
here on. This may increase file size of PDF/A-1b files. PDF/A-2b files will not be affected.
2.51 v11.6.2
• Fixed a regression where the wrong page orientation would be produced when using arguments such as
--deskew --rotate-pages (#730).
2.52 v11.6.1
• Fixed an issue with attempting optimize unusually narrow-width images by excluding these images from opti-
mization (#732).
• Remove an obsolete compatibility shim for a version of pikepdf that is no longer supported.
2.53 v11.6.0
• OCRmyPDF will now automatically register plugins from the same virtual environment with an appropriate
setuptools entrypoint.
• Refactor the plugin manager to remove unnecessary complications and make plugin registration more automatic.
• PageContext and PdfContext are now formally part of the API, as they should have been, since they were
part of ocrmypdf.pluginspec.
2.54 v11.5.0
• Fixed an issue where the output page size might differ by a fractional amount due to rounding, when
--force-ocr was used and the page contained objects with multiple resolutions.
• When determining the resolution at which to rasterize a page, we now consider printed text on the page as re-
quiring a higher resolution. This fixes issues with certain pages being rendered with unacceptably low resolution
text, but may increase output file sizes in some workflows where low resolution text is acceptable.
• Added a workaround to fix an exception that occurs when trying to import ocrmypdf.leptonica on
Apple ARM silicon (or potentially, other platforms that do not permit write+executable memory).
2.55 v11.4.5
• Fixed an issue where files may not be closed when the API is used.
• Improved setup.cfg with better settings for test coverage.
2.56 v11.4.4
2.57 v11.4.3
2.58 v11.4.2
2.59 v11.4.1
• Fixed an issue where invalid pages ranges passed using the pages argument, such as “1-0” would cause un-
handled exceptions.
• Accepted a user-contributed to the Synology demo script in misc/synology.py.
• Clarified documentation about change of temporary file location ocrmypdf.io.
2.54. v11.5.0 19
ocrmypdf Documentation, Release 0.1.dev83+g072db75
• Fixed Python wheel tag which was incorrectly set to py35 even though we long since dropped support for Python
3.5.
2.60 v11.4.0
• When looking for Tesseract and Ghostscript, we now check the Windows Registry to see if their installers
registered the location of their executables. This should help Windows users who have installed these programs
to non-standard locations.
• We now report on the progress of PDF/A conversion, since this operation is sometimes slow.
• Improved command line completions.
• The prefix of the temporary folder OCRmyPDF creates has been changed from com.github.ocrmypdf to
ocrmypdf.io. Scripts that chose to depend on this prefix may need to be adjusted. (This has always been an
implementation detail so is not considered part of the semantic versioning “contract”.)
• Fixed #692, where a particular file with malformed fonts would flood an internal message cue by generating so
many debug messages.
• Fixed an exception on processing hOCR files with no page record. Tesseract is not known to generate such files.
2.61 v11.3.4
• Fixed an error message ‘called readLinearizationData for file that is not linearized’ that may occur when pikepdf
2.1.0 is used. (Upgrading to pikepdf 2.1.1 also fixes the issue.)
• File watcher now automatically includes .PDF in addition to .pdf to better support case sensitive file systems.
• Some documentation and comment improvements.
2.62 v11.3.3
• If unpaper outputs non-UTF-8 data, quietly fix this rather than choke on the conversion. (Possibly addresses
#671.)
2.63 v11.3.2
• Explicitly require pikepdf 2.0.0 or newer when running on Python 3.9. (There are concerns about the stability
of pybind11 2.5.x with Python 3.9, which is used in pikepdf 1.x.)
• Fixed another issue related to page rotation.
• Fixed an issue where image marked as image masks were not properly considered as optimization candidates.
• On some systems, unpaper seems to be unable to process the PNGs we offer it as input. We now convert the
input to PNM format, which unpaper always accepts. Fixes #665 and #667.
• DPI sent to unpaper is now rounded to a more reasonable number of decimal digits.
• Debug and error messages from unpaper were being suppressed.
• Some documentation tweaks.
2.64 v11.3.1
• Declare support for new versions: pdfminer.six 20201018 and pikepdf 2.x
• Fixed warning related to --pdfa-image-compression that appears at the wrong time.
2.65 v11.3.0
• The “OCR” step is describing as “Image processing” in the output messages when OCR is disabled, to better
explain the application’s behavior.
• Debug logs are now only created when run as a command line, and not when OCR is performed for an API call.
It is the calling application’s responsibility to set up logging.
• For PDFs with a low number of pages, we gathered information about the input PDF in a thread rather than
process (when there are more pages). When run as a thread, we did not close the file handle to the working PDF,
leaking one file handle per call of ocrmypdf.ocr.
• Fixed an issue where debug messages send by child worker processes did not match the log settings of parent
process, causing messages to be dropped. This affected macOS and Windows only where the parent process is
not forked.
• Fixed the hookspec of rasterize_pdf_page to remove default parameters that were not handled in an expected
way by pluggy.
• Fixed another issue with automatic page rotation (#658) due to the issue above.
2.66 v11.2.1
• Fixed an issue where optimization of a 1-bit image with a color palette or associated ICC that was optimized to
JBIG2 could have its colors inverted.
2.67 v11.2.0
• Fixed an issue with optimizing PNG-type images that had soft masks or image masks. This is a regression
introduced in (or about) v11.1.0.
• Improved type checking of the plugins parameter for the ocrmypdf.ocr API call.
2.68 v11.1.2
• Fixed hOCR renderer writing the text in roughly reverse order. This should not affect reasonably smart PDF
readers that properly locate the position of all text, but may confuse those that rely on the order of objects in the
content stream. (#642)
2.69 v11.1.1
• We now avoid using named temporary files when using pngquant allowing containerized pngquant installs to be
used.
2.64. v11.3.1 21
ocrmypdf Documentation, Release 0.1.dev83+g072db75
2.70 v11.1.0
2.71 v11.0.2
• Fixed #612, TypeError exception. Fixed by eliminating unnecessary repair of input PDF metadata in memory.
2.72 v11.0.1
2.73 v11.0.0
• Project license changed to Mozilla Public License 2.0. Some miscellaneous code is now under MIT license and
non-code content/media remains under CC-BY-SA 4.0. License changed with approval of all people who were
found to have contributed to GPLv3 licensed sections of the project. (#600)
• Because the license changed, this is being treated as a major version number change; however, there are no
known breaking changes in functional behavior or API compared to v10.x.
2.74 v10.3.3
• Fixed a “KeyError: ‘dpi’” error message when using --threshold on an image. (#607)
2.75 v10.3.2
• Fixed a case where we reported “no reason” for a file size increase, when we could determine the reason.
• Enabled support for pdfminer.six 20200726.
2.76 v10.3.1
• Fixed a number of test suite failures with pdfminer.six older than version 20200402.
• Enabled support for pdfminer.six 20200720.
2.77 v10.3.0
• Fixed an issue where we would consider images that were already JBIG2-encoded for optimization, potentially
producing a less optimized image than the original. We do not believe this issue would ever cause an image to
loss fidelity.
• Where available, pikepdf memory mapping is now used. This improves performance.
• When Leptonica 1.79+ is installed, use its new error handling API to avoid a “messy” redirection of stderr which
was necessary to capture its error messages.
• For older versions of Leptonica, added a new thread level lock. This fixes a possible race condition in handling
error conditions in Leptonica (although there is no evidence it ever caused issues in practice).
• Documentation improvements and more type hinting.
2.78 v10.2.1
• Disabled calculation of text box order with pdfminer. We never needed this result and it is expensive to calculate
on files with complex pre-existing text.
• Fixed plugin manager to accept Path(plugin) as a path to a plugin.
• Fixed some typing errors.
• Documentation improvements.
2.79 v10.2.0
2.75. v10.3.2 23
ocrmypdf Documentation, Release 0.1.dev83+g072db75
2.80 v10.1.1
• Fixed OMP_THREAD_LIMIT set to invalid value error messages on some input files. (The error was harmless,
apart from less than optimal performance in some cases.)
2.81 v10.1.0
• Previously, we --clean-final would cause an unpaper-cleaned page image to be produced twice, which
was necessary in some cases but not in general. We now take this optimization opportunity and reuse the image
if possible.
• We now provide PNG files as input to unpaper, since it accepts them, instead of generating PPM files which can
be very large. This can improve performance and temporary disk usage.
• Documentation updated for plugins.
2.82 v10.0.1
2.83 v10.0.0
Breaking changes
• Support for pdfminer.six version 20181108 has been dropped, along with a monkeypatch that made this version
work.
• Output messages are now displayed in color (when supported by the terminal) and prefixes describing the sever-
ity of the message are removed. As such programs that parse OCRmyPDF’s log message will need to be revised.
(Please consider using OCRmyPDF as a library instead.)
• The minimum version for certain dependencies has increased.
• Many API changes; see developer changes.
• The Python libraries pluggy and coloredlogs are now required.
New features and improvements
• PDF page scanning is now parallelized across CPUs, speeding up this phase dramatically for files with a high
page counts.
• PDF page scanning is optimized, addressing some performance regressions.
• PDF page scanning is no longer run on pages that are not selected when the --pages argument is used.
• PDF page scanning is now independent of Ghostscript, ending our past reliance on this occasionally unstable
feature in Ghostscript.
• A plugin architecture has been added, currently allowing one to more easily use a different OCR engine or PDF
renderer from Tesseract and Ghostscript, respectively. A plugin can also override some decisions, such changing
the OCR settings after initial scanning.
• Colored log messages.
Developer changes
• The test spoofing mechanism, used to test correct handling of failures in Tesseract and Ghostscript, has been
removed in favor of using plugins for testing. The spoofing mechanism was fairly complex and required many
special hacks for Windows.
• Code describing the resolution in DPI of images was refactored into a ocrmypdf.helpers.Resolution
class.
• The module ocrmypdf._exec is now private to OCRmyPDF.
• The ocrmypdf.hocrtransform module has been updated to follow PEP8 naming conventions.
• Ghostscript is no longer used for finding the location of text in PDFs, and APIs related to this feature have been
removed.
• Lots of internal reorganization to support plugins.
2.84 v9.8.2
• Fixed an issue where OCRmyPDF would ignore text inside Form XObject when making certain decisions about
whether a document already had text.
• Fixed file size increase warning to take overhead of small files into account.
• Added instructions for installing on Cygwin.
2.85 v9.8.1
• Fixed an issue where unexpected files in the %PROGRAMFILES%\gs directory (Windows) caused an exception.
• Mark pdfminer.six 20200517 as supported.
• If jbig2enc is missing and optimization is requested, a warning is issued instead of an error, which was the
intended behavior.
• Documentation updates.
2.86 v9.8.0
• Fixed issue where only the first PNG (FlateDecode) image in a file would be considered for optimization. File
sizes should be improved from here on.
• Fixed a startup crash when the chosen language was Japanese (#543).
• Added options to configure polling and log level to watcher.py.
2.87 v9.7.2
2.84. v9.8.2 25
ocrmypdf Documentation, Release 0.1.dev83+g072db75
2.88 v9.7.1
2.89 v9.7.0
2.90 v9.6.1
2.91 v9.6.0
• Fixed a regression with transferring metadata from the input PDF to the output PDF in certain situations.
• pdfminer.six is now supported up to version 2020-01-24.
• Messages are explaining page rotation decisions are now shown at the standard verbosity level again when
--rotate-pages. In some previous version they were set to debug level messages that only appeared with
the parameter -v1.
• Improvements to misc/watcher.py. Thanks to @ianalexander and @svenihoney.
• Documentation improvements.
2.92 v9.5.0
2.93 v9.4.0
2.94 v9.3.0
• Improved native Windows support: we now check in the obvious places in the “Program Files” folders in-
stallations of Tesseract and Ghostscript, rather than relying on the user to edit PATH to specify their location.
The PATH environment variable can still be used to differentiate when multiple installations are present or the
programs are installed to non- standard locations.
• Fixed an exception on parsing Ghostscript error messages.
• Added an improved example demonstrating how to set up a watched folder for automated OCR processing
(thanks to @ianalexander for the contribution).
2.95 v9.2.0
2.92. v9.5.0 27
ocrmypdf Documentation, Release 0.1.dev83+g072db75
• OCRmyPDF no longer depends on the QPDF executable qpdf or libqpdf. It uses pikepdf (which in turn
depends on libqpdf). Package maintainers should adjust dependencies so that OCRmyPDF no longer calls
for libqpdf on its own. For users of Python binary wheels, this change means a separate installation of QPDF is
no longer necessary. This change is mainly to simplify installation on Windows.
• Fixed a rare case where log messages from Tesseract would be discarded.
• Fixed incorrect function signature for pixFindPageForeground, causing exceptions on certain plat-
forms/Leptonica versions.
2.96 v9.1.1
2.97 v9.1.0
• Improved diagnostics when file size increases at output. Now warns if JBIG2 or pngquant were not available.
• pikepdf 1.7.0 is now required, to pick up changes that remove the need for a source install on Linux systems
running Python 3.8.
2.98 v9.0.5
• The Alpine Docker image (jbarlow83/ocrmypdf-alpine) has been dropped due to the difficulties of supporting
Alpine Linux.
• The primary Docker image (jbarlow83/ocrmypdf) has been improved to take on the extra features that used to
be exclusive to the Alpine image.
• No changes to application code.
• pdfminer.six version 20191020 is now supported.
2.99 v9.0.4
• Fixed compatibility with Python 3.8 (but requires source install for the moment).
• Fixed Tesseract settings for --user-words and --user-patterns.
• Changed to pikepdf 1.6.5 (for Python 3.8).
• Changed to Pillow 6.2.0 (to mitigate a security vulnerability in earlier Pillow).
• A debug message now mentions when English is automatically selected if the locale is not English.
2.100 v9.0.3
• Embed an encoded version of the sRGB ICC profile in the intermediate Postscript file (used for PDF/A con-
version). Previously we included the filename, which required Postscript to run with file access enabled. For
security, Ghostscript 9.28 enables -dSAFER and as such, no longer permits access to any file by default. This
fix is necessary for compatibility with Ghostscript 9.28.
• Exclude a test that sometimes times out and fails in continuous integration from the standard test suite.
2.101 v9.0.2
• The image optimizer now skips optimizing flate (PNG) encoded images in some situations where the optimiza-
tion effort was likely wasted.
• The image optimizer now ignores images that specify arbitrary decode arrays, since these are rare.
• Fixed an issue that caused inversion of black and white in monochrome images. We are not certain but the
problem seems to be linked to Leptonica 1.76.0 and older.
• Fixed some cases where the test suite failed if English or German Tesseract language packs were not installed.
• Fixed a runtime error if the Tesseract English language is not installed.
• Improved explicit closing of Pillow images after use.
• Actually fixed of Alpine Docker image build.
• Changed to pikepdf 1.6.3.
2.102 v9.0.1
• Fixed test suite failing when either of optional dependencies unpaper and pngquant were missing.
• Attempted fix of Alpine Docker image build.
• Documented that FreeBSD ports are now available.
• Changed to pikepdf 1.6.1.
2.103 v9.0.0
Breaking changes
• The --mask-barcodes experimental feature has been dropped due to poor reliability and occasional crashes,
both due to the underlying library that implements this feature (Leptonica).
• The -v (verbosity level) parameter now accepts only 0, 1, and 2.
• Dropped support for Tesseract 4.00.00-alpha releases. Tesseract 4.0 beta and later remain supported.
• Dropped the ocrmypdf-polyglot and ocrmypdf-webservice images.
New features
• Added a high level API for applications that want to integrate OCRmyPDF. Special thanks to Martin Wind
(@mawi1988) whose made significant contributions to this effort.
2.100. v9.0.3 29
ocrmypdf Documentation, Release 0.1.dev83+g072db75
2.104 v8.3.2
• Dropped workaround for macOS that allowed it work without pdfminer.six, now a proper sdist release of
pdfminer.six is available.
• pikepdf 1.5.0 is now required.
2.105 v8.3.1
• Fixed an issue where PDFs with malformed metadata would be rendered as blank pages. #398.
2.106 v8.3.0
• Improved the strategy for updating pages when a new image of the page was produced. We now attempt to
preserve more content from the original file, for annotations in particular.
• For PDFs with more than 100 pages and a sequence where one PDF page was replaced and one or more subse-
quent ones were skipped, an intermediate file would be corrupted while grafting OCR text, causing processing
to fail. This is a regression, likely introduced in v8.2.4.
• Previously, we resized the images produced by Ghostscript by a small number of pixels to ensure the output
image size was an exactly what we wanted. Having discovered a way to get Ghostscript to produce the exact
image sizes we require, we eliminated the resizing step.
• Command line completions for bash are now available, in addition to fish, both in misc/completion.
Package maintainers, please install these so users can take advantage.
• Updated requirements.
• pikepdf 1.3.0 is now required.
2.107 v8.2.4
• Fixed a false positive while checking for a certain type of PDF that only Acrobat can read. We now more
accurately detect Acrobat-only PDFs.
• OCRmyPDF holds fewer open file handles and is more prompt about releasing those it no longer needs.
• Minor optimization: we no longer traverse the table of contents to ensure all references in it are resolved, as
changes to libqpdf have made this unnecessary.
• pikepdf 1.2.0 is now required.
2.108 v8.2.3
• Fixed that --mask-barcodes would occasionally leave a unwanted temporary file named junkpixt in the
current working folder.
• Fixed (hopefully) handling of Leptonica errors in an environment where a non-standard sys.stderr is
present.
• Improved help text for --verbose.
2.109 v8.2.2
• Fixed a regression from v8.2.0, an exception that occurred while attempting to report that unpaper or another
optional dependency was unavailable.
• In some cases, ocrmypdf [-c|--clean] failed to exit with an error when unpaper is not installed.
2.110 v8.2.1
2.107. v8.2.4 31
ocrmypdf Documentation, Release 0.1.dev83+g072db75
2.111 v8.2.0
• A major improvement to our Docker image is now available thanks to hard work contributed by @mawi12345.
The new Docker image, ocrmypdf-alpine, is based on Alpine Linux, and includes most of the functionality of
three existed images in a smaller package. This image will replace the main Docker image eventually but for
now all are being built. See documentation for details.
• Documentation reorganized especially around the use of Docker images.
• Fixed a problem with PDF image optimization, where the optimizer would unnecessarily decompress and re-
compress PNG images, in some cases losing the benefits of the quantization it just had just performed. The
optimizer is now capable of embedding PNG images into PDFs without transcoding them.
• Fixed a minor regression with lossy JBIG2 image optimization. All JBIG2 candidates images were incorrectly
placed into a single optimization group for the whole file, instead of grouping pages together. This usually makes
a larger JBIG2Globals dictionary and results in inferior compression, so it worked less well than designed.
However, quality would not be impacted. Lossless JBIG2 was entirely unaffected.
• Updated dependencies, including pikepdf to 1.1.0. This fixes #358.
• The install-time version checks for certain external programs have been removed from setup.py. These tests are
now performed at run-time.
• The non-standard option to override install-time checks (setup.py install --force) is now deprecated
and prints a warning. It will be removed in a future release.
2.112 v8.1.0
• Added a feature, --unpaper-args, which allows passing arbitrary arguments to unpaper when using
--clean or --clean-final. The default, very conservative unpaper settings are suppressed.
• The argument --clean-final now implies --clean. It was possible to issue --clean-final on its
before this, but it would have no useful effect.
• Fixed an exception on traversing corrupt table of contents entries (specifically, those with invalid destination
objects)
• Fixed an issue when using --tesseract-timeout and image processing features on a file with more than
100 pages. #347
• OCRmyPDF now always calls os.nice(5) to signal to operating systems that it is a background process.
2.113 v8.0.1
• Fixed an exception when parsing PDFs that are missing a required field. #325
• pikepdf 1.0.5 is now required, to address some other PDF parsing issues.
2.114 v8.0.0
No major features. The intent of this release is to sever support for older versions of certain dependencies.
Breaking changes
• Dropped support for Tesseract 3.x. Tesseract 4.0 or newer is now required.
2.115 v7.4.0
• --force-ocr may now be used with the new --threshold and --mask-barcodes features
• pikepdf >= 0.9.1 is now required.
• Changed metadata handling to pikepdf 0.9.1. As a result, metadata handling of non-ASCII characters in
Ghostscript 9.25 or later is fixed.
• chardet >= 3.0.4 is temporarily listed as required. pdfminer.six depends on it, but the most recent release does
not specify this requirement. (#326)
• python-xmp-toolkit and libexempi are no longer required.
• A new Docker image is now being provided for users who wish to access OCRmyPDF over a simple HTTP
interface, instead of the command line.
• Increase tolerance of PDFs that overflow or underflow the PDF graphics stack. (#325)
2.116 v7.3.1
• Fixed performance regression from v7.3.0; fast page analysis was not selected when it should be.
• Fixed a few exceptions related to the new --mask-barcodes feature and improved argument checking
• Added missing detection of TrueType fonts that lack a Unicode mapping
2.117 v7.3.0
• Added a new feature --redo-ocr to detect existing OCR in a file, remove it, and redo the OCR. This may
be particularly helpful for anyone who wants to take advantage of OCR quality improvements in Tesseract 4.0.
Note that OCR added by OCRmyPDF before version 3.0 cannot be detected since it was not properly marked
as invisible text in the earliest versions. OCR that constructs a font from visible text, such as Adobe Acrobat’s
ClearScan.
2.115. v7.4.0 33
ocrmypdf Documentation, Release 0.1.dev83+g072db75
• OCRmyPDF’s content detection is generally more sophisticated. It learns more about the contents of each PDF
and makes better recommendations:
– OCRmyPDF can now detect when a PDF contains text that cannot be mapped to Unicode (meaning it is
readable to human eyes but copy-pastes as gibberish). In these cases it recommends --force-ocr to
make the text searchable.
– PDFs containing vector objects are now rendered at more appropriate resolution for OCR.
– We now exit with an error for PDFs that contain Adobe LiveCycle Designer’s dynamic XFA forms. Cur-
rently the open source community does not have tools to work with these files.
– OCRmyPDF now warns when a PDF that contains Adobe AcroForms, since such files probably do not
need OCR. It can work with these files.
• Added three new experimental features to improve OCR quality in certain conditions. The name, syntax and
behavior of these arguments is subject to change. They may also be incompatible with some other features.
– --remove-vectors which strips out vector graphics. This can improve OCR quality since OCR will
not search artwork for readable text; however, it currently removes “text as curves” as well.
– --mask-barcodes to detect and suppress barcodes in files. We have observed that barcodes can inter-
fere with OCR because they are “text-like” but not actually textual.
– --threshold which uses a more sophisticated thresholding algorithm than is currently in use in Tesser-
act OCR. This works around a known issue in Tesseract 4.0 with dark text on bright backgrounds.
• Fixed an issue where an error message was not reported when the installed Ghostscript was very old.
• The PDF optimizer now saves files with object streams enabled when the optimization level is --optimize
1 or higher (the default). This makes files a little bit smaller, but requires PDF 1.5. PDF 1.5 was first released
in 2003 and is broadly supported by PDF viewers, but some rudimentary PDF parsers such as PyPDF2 do not
understand object streams. You can use the command line tool qpdf --object-streams=disable or
pikepdf library to remove them.
• New dependency: pdfminer.six 20181108. Note this is a fork of the Python 2-only pdfminer.
• Deprecation notice: At the end of 2018, we will be ending support for Python 3.5 and Tesseract 3.x. OCRmyPDF
v7 will continue to work with older versions.
2.118 v7.2.1
2.119 v7.2.0
Only users who have reviewed the concerns with JBIG2 in lossy mode should opt-in. As such, lossy mode JBIG2
is only turned on when the new argument --jbig2-lossy is issued. This is independent of the setting for
--optimize.
Users who did not install an optional JBIG2 encoder are unaffected.
(Thanks to user ‘bsdice’ for reporting this issue.)
Other issues
• When the image optimizer quantizes an image to 1 bit per pixel, it will now attempt to further optimize that
image as CCITT or JBIG2, instead of keeping it in the “flate” encoding which is not efficient for 1 bpp images.
(#297)
• Images in PDFs that are used as soft masks (i.e. transparency masks or alpha channels) are now excluded from
optimization.
• Fixed handling of Tesseract 4.0-rc1 which now accepts invalid Tesseract configuration files, which broke the
test suite.
2.120 v7.1.0
• Improve the performance of initial text extraction, which is done to determine if a file contains existing text of
some kind or not. On large files, this initial processing is now about 20x times faster. (#299)
• pikepdf 0.3.3 is now required.
• Fixed #231, a problem with JPEG2000 images where image metadata was only available inside the JPEG2000
file.
• Fixed some additional Ghostscript 9.25 compatibility issues.
• Improved handling of KeyboardInterrupt error messages. (#301)
• README.md is now served in GitHub markdown instead of reStructuredText.
2.121 v7.0.6
• Blacklist Ghostscript 9.24, now that 9.25 is available and fixes many regressions in 9.24.
2.122 v7.0.5
• Improve capability with Ghostscript 9.24, and enable the JPEG passthrough feature when this version in in-
stalled.
• Ghostscript 9.24 lost the ability to set PDF title, author, subject and keyword metadata to Unicode strings.
OCRmyPDF will set ASCII strings and warn when Unicode is suppressed. Other software may be used to
update metadata. This is a short term work around.
• PDFs generated by Kodak Capture Desktop, or generally PDFs that contain indirect references to null objects
in their table of contents, would have an invalid table of contents after processing by OCRmyPDF that might
interfere with other viewers. This has been fixed.
• Detect PDFs generated by Adobe LiveCycle, which can only be displayed in Adobe Acrobat and Reader cur-
rently. When these are encountered, exit with an error instead of performing OCR on the “Please wait” error
message page.
2.120. v7.1.0 35
ocrmypdf Documentation, Release 0.1.dev83+g072db75
2.123 v7.0.4
• Fixed exception thrown when trying to optimize a certain type of PNG embedded in a PDF with the -O2
• Update to pikepdf 0.3.2, to gain support for optimizing some additional image types that were previously ex-
cluded from optimization (CMYK and grayscale). Fixes #285.
2.124 v7.0.3
• Fixed #284, an error when parsing inline images that have are also image masks, by upgrading pikepdf to 0.3.1
2.125 v7.0.2
• Fixed a regression with --rotate-pages on pages that already had rotations applied. (#279)
• Improve quality of page rotation in some cases by rasterizing a higher quality preview image. (#281)
2.126 v7.0.1
• Fixed compatibility with img2pdf >= 0.3.0 by rejecting input images that have an alpha channel
• Add forward compatibility for pikepdf 0.3.0 (unrelated to img2pdf)
• Various documentation updates for v7.0.0 changes
2.127 v7.0.0
• The core algorithm for combining OCR layers with existing PDF pages has been rewritten and improved con-
siderably. PDFs are no longer split into single page PDFs for processing; instead, images are rendered and the
OCR results are grafted onto the input PDF. The new algorithm uses less temporary disk space and is much
more performant especially for large files.
• New dependency: pikepdf. pikepdf is a powerful new Python PDF library driving the latest OCRmyPDF fea-
tures, built on the QPDF C++ library (libqpdf).
• New feature: PDF optimization with -O or --optimize. After OCR, OCRmyPDF will perform image opti-
mizations relevant to OCR PDFs.
– If a JBIG2 encoder is available, then monochrome images will be converted, with the potential for huge
savings on large black and white images, since JBIG2 is far more efficient than any other monochrome
(bi-level) compression. (All known US patents related to JBIG2 have probably expired, but it remains the
responsibility of the user to supply a JBIG2 encoder such as jbig2enc. OCRmyPDF does not implement
JBIG2 encoding.)
– If pngquant is installed, OCRmyPDF will optionally use it to perform lossy quantization and compres-
sion of PNG images.
– The quality of JPEGs can also be lowered, on the assumption that a lower quality image may be suitable
for storage after OCR.
– This image optimization component will eventually be offered as an independent command line utility.
– Optimization ranges from -O0 through -O3, where 0 disables optimization and 3 implements all options.
1, the default, performs only safe and lossless optimizations. (This is similar to GCC’s optimization
parameter.) The exact type of optimizations performed will vary over time.
• Small amounts of text in the margins of a page, such as watermarks, page numbers, or digital stamps, will no
longer prevent the rest of a page from being OCRed when --skip-text is issued. This behavior is based on
a heuristic.
• Removed features
– The deprecated --pdf-renderer tesseract PDF renderer was removed.
– -g, the option to generate debug text pages, was removed because it was a maintenance burden and
only worked in isolated cases. HOCR pages can still be previewed by running the hocrtransform.py with
appropriate settings.
• Removed dependencies
– PyPDF2
– defusedxml
– PyMuPDF
• The sandwich PDF renderer can be used with all supported versions of Tesseract, including that those prior
to v3.05 which don’t support -c textonly. (Tesseract v4.0.0 is recommended and more efficient.)
• --pdf-renderer auto option and the diagnostics used to select a PDF renderer now work better with old
versions, but may make different decisions than past versions.
• If everything succeeds but PDF/A conversion fails, a distinct return code is now returned (ExitCode.
pdfa_conversion_failed (10)) where this situation previously returned ExitCode.
invalid_output_pdf (4). The latter is now returned only if there is some indication that the
output file is invalid.
• Notes for downstream packagers
– There is also a new dependency on python-xmp-toolkit which in turn depends on libexempi3.
– It may be necessary to separately pip install pycparser to avoid another Python 3.7 issue.
2.128 v6.2.5
• Disable a failing test due to Tesseract 4.0rc1 behavior change. Previously, Tesseract would exit with an error
message if its configuration was invalid, and OCRmyPDF would intercept this message. Now Tesseract issues
a warning, which OCRmyPDF v6.2.5 may relay or ignore. (In v7.x, OCRmyPDF will respond to the warning.)
• This release branch no longer supports using the optional PyMuPDF installation, since it was removed in v7.x.
• This release branch no longer supports macOS. macOS users should upgrade to v7.x.
2.129 v6.2.4
• Backport Ghostscript 9.25 compatibility fixes, which removes support for setting Unicode metadata
• Backport blacklisting Ghostscript 9.24
• Older versions of Ghostscript are still supported
2.128. v6.2.5 37
ocrmypdf Documentation, Release 0.1.dev83+g072db75
2.130 v6.2.3
• Fixed compatibility with img2pdf >= 0.3.0 by rejecting input images that have an alpha channel
• This version will be included in Ubuntu 18.10
2.131 v6.2.2
• Backport compatibility fixes for Python 3.7 and ruffus 2.7.0 from v7.0.0
• Backport fix to ignore masks when deciding what colors are on a page
• Backport some minor improvements from v7.0.0: better argument validation and warnings about the Tesseract
4.0.0 --user-words regression
2.132 v6.2.1
• Fixed recent versions of Tesseract (after 4.0.0-beta1) not being detected as supporting the sandwich renderer
(#271).
2.133 v6.2.0
• Docker: The Docker image ocrmypdf-tess4 has been removed. The main Docker images, ocrmypdf and
ocrmypdf-polyglot now use Ubuntu 18.04 as a base image, and as such Tesseract 4.0.0-beta1 is now the
Tesseract version they use. There is no Docker image based on Tesseract 3.05 anymore.
• Creation of PDF/A-3 is now supported. However, there is no ability to attach files to PDF/A-3.
• Lists more reasons why the file size might grow.
• Fixed #262, --remove-background error on PDFs contained colormapped (paletted) images.
• Fixed another XMP metadata validation issue, in cases where the input file’s creation date has no timezone and
the creation date is not overridden.
2.134 v6.1.5
• Fixed #253, a possible division by zero when using the hocr renderer.
• Fixed incorrectly formatted <xmp:ModifyDate> field inside XMP metadata for PDF/As. veraPDF flags this
as a PDF/A validation failure. The error is caused the timezone and final digit of the seconds of modified time
to be omitted, so at worst the modification time stamp is rounded to the nearest 10 seconds.
2.135 v6.1.4
• Fixed #248 --clean argument may remove OCR from left column of text on certain documents. We now set
--layout none to suppress this.
• The test cache was updated to reflect the change above.
• Change test suite to accommodate Ghostscript 9.23’s new ability to insert JPEGs into PDFs without transcoding.
• XMP metadata in PDFs is now examined using defusedxml for safety.
• If an external process exits with a signal when asked to report its version, we now print the system error message
instead of suppressing it. This occurred when the required executable was found but was missing a shared library.
• qpdf 7.0.0 or newer is now required as the test suite can no longer pass without it.
2.135.1 Notes
• An apparent regression in Ghostscript 9.23 will cause some ocrmypdf output files to become invalid in rare
cases; the workaround for the moment is to set --force-ocr.
2.136 v6.1.3
2.137 v6.1.2
2.138 v6.1.1
• Fixed text being reported as found on all pages if PyMuPDF is not installed.
2.139 v6.1.0
• PyMuPDF is now an optional but recommended dependency, to alleviate installation difficulties on platforms
that have less access to PyMuPDF than the author anticipated. (For version 6.x only) install OCRmyPDF with
pip install ocrmypdf[fitz] to use it to its full potential.
• Fixed FileExistsError that could occur if OCR timed out while it was generating the output file. (#218)
• Fixed table of contents/bookmarks all being redirected to page 1 when generating a PDF/A (with PyMuPDF).
(Without PyMuPDF the table of contents is removed in PDF/A mode.)
• Fixed “RuntimeError: invalid key in dict” when table of contents/bookmarks titles contained the character ).
(#239)
• Added a new argument --skip-repair to skip the initial PDF repair step if the PDF is already well-formed
(because another program repaired it).
2.136. v6.1.3 39
ocrmypdf Documentation, Release 0.1.dev83+g072db75
2.140 v6.0.0
• The software license has been changed to GPLv3 [it has since changed again]. Test resource files and some
individual sources may have other licenses.
• OCRmyPDF now depends on PyMuPDF. Including PyMuPDF is the primary reason for the change to GPLv3.
• Other backward incompatible changes
– The OCRMYPDF_TESSERACT, OCRMYPDF_QPDF, OCRMYPDF_GS and OCRMYPDF_UNPAPER envi-
ronment variables are no longer used. Change PATH if you need to override the external programs
OCRmyPDF uses.
– The ocrmypdf package has been moved to src/ocrmypdf to avoid issues with accidental import.
– The function ocrmypdf.exec.get_program was removed.
– The deprecated module ocrmypdf.pageinfo was removed.
– The --pdf-renderer tess4 alias for sandwich was removed.
• Fixed an issue where OCRmyPDF failed to detect existing text on pages, depending on how the text and fonts
were encoded within the PDF. (#233#232)
• Fixed an issue that caused dramatic inflation of file sizes when --skip-text --output-type pdf was
used. OCRmyPDF now removes duplicate resources such as fonts, images and other objects that it generates.
(#237)
• Improved performance of the initial page splitting step. Originally this step was not believed to be expensive
and ran in a process. Large file testing revealed it to be a bottleneck, so it is now parallelized. On a 700 page
file with quad core machine, this change saves about 2 minutes. (#234)
• The test suite now includes a cache that can be used to speed up test runs across platforms. This also does not
require computing checksums, so it’s faster. (#217)
2.141 v5.7.0
• Fixed an issue that caused poor CPU utilization on machines with more than 4 cores when running Tesseract 4.
(Related to #217.)
• The ‘hocr’ renderer has been improved. The ‘sandwich’ and ‘tesseract’ renderers are still better for most use
cases, but ‘hocr’ may be useful for people who work with the PDF.js renderer in English/ASCII languages.
(#225)
– It now formats text in a matter that is easier for certain PDF viewers to select and extract copy and paste
text. This should help macOS Preview and PDF.js in particular.
– The appearance of selected text and behavior of selecting text is improved.
– The PDF content stream now uses relative moves, making it more compact and easier for viewers to
determine when two words on the same line.
– It can now deal with text on a skewed baseline.
– Thanks to @cforcey for the pull request, @jbreiden for many helpful suggestions, @ctbarbour for another
round of improvements, and @acaloiaro for an independent review.
2.142 v5.6.3
2.143 v5.6.2
2.144 v5.6.1
• Fixed #219: change how the final output file is created to avoid triggering permission errors when the output is
a special file such as /dev/null
• Fixed test suite failures due to a qpdf 8.0.0 regression and Python 3.5’s handling of symlink
• The “encrypted PDF” error message was different depending on the type of PDF encryption. Now a single clear
message appears for all types of PDF encryption.
• ocrmypdf is now in Homebrew. Homebrew users are advised to the version of ocrmypdf in the official
homebrew-core formulas rather than the private tap.
• Some linting
2.145 v5.6.0
2.146 v5.5
• Add new argument --max-image-mpixels. Pillow 5.0 now raises an exception when images may be
decompression bombs. This argument can be used to override the limit Pillow sets.
• Fixed output page cropped when using the sandwich renderer and OCR is skipped on a rotated and image-
processed page
• A warning is now issued when old versions of Ghostscript are used in cases known to cause issues with non-
Latin characters
• Fixed a few parameter validation checks for -output-type pdfa-1 and pdfa-2
2.147 v5.4.4
• Fixed #181: fix final merge failure for PDFs with more pages than the system file handle limit (ulimit -n)
• Fixed #200: an uncommon syntax for formatting decimal numbers in a PDF would cause qpdf to issue a warning,
which ocrmypdf treated as an error. Now this the warning is relayed.
2.142. v5.6.3 41
ocrmypdf Documentation, Release 0.1.dev83+g072db75
• Fixed an issue where intermediate PDFs would be created at version 1.3 instead of the version of the original
file. It’s possible but unlikely this had side effects.
• A warning is now issued when older versions of qpdf are used since issues like #200 cause qpdf to infinite-loop
• Address issue #140: if Tesseract outputs invalid UTF-8, escape it and print its message instead of aborting with
a Unicode error
• Adding previously unlisted setup requirement, pytest-runner
• Update documentation: fix an error in the example script for Synology with Docker images, improved security
guidance, advised pip install --user
2.148 v5.4.3
• If a subprocess fails to report its version when queried, exit cleanly with an error instead of throwing an exception
• Added test to confirm that the system locale is Unicode-aware and fail early if it’s not
• Clarified some copyright information
• Updated pinned requirements.txt so the homebrew formula captures more recent versions
2.149 v5.4.2
• Fixed a regression from v5.4.1 that caused sidecar files to be created as empty files
2.150 v5.4.1
• Add workaround for Tesseract v4.00alpha crash when trying to obtain orientation and the latest language packs
are installed
2.151 v5.4
2.152 v5.3.3
• Fixed missing error message that should occur when trying to force --pdf-renderer sandwich on old
versions of Tesseract
• Update copyright information in test files
• Set system LANG to UTF-8 in Dockerfiles to avoid UTF-8 encoding errors
2.153 v5.3.2
2.154 v5.3.1
• Fixed wrong return code given for missing Tesseract language packs
• Fixed “brew audit” crashing on Travis when trying to auto-brew
2.155 v5.3
• Added --user-words and --user-patterns arguments which are forwarded to Tesseract OCR as words
and regular expressions respective to use to guide OCR. Supplying a list of subject-domain words should assist
Tesseract with resolving words. (#165)
• Using a non Latin-1 language with the “hocr” renderer now warns about possible OCR quality and recommends
workarounds (#176)
• Output file path added to error message when that location is not writable (#175)
• Otherwise valid PDFs with leading whitespace at the beginning of the file are now accepted
2.156 v5.2
• When using Tesseract 3.05.01 or newer, OCRmyPDF will select the “sandwich” PDF renderer by default, unless
another PDF renderer is specified with the --pdf-renderer argument. The previous behavior was to select
--pdf-renderer=hocr.
• The “tesseract” PDF renderer is now deprecated, since it can cause problems with Ghostscript on Tesseract
3.05.00
• The “tess4” PDF renderer has been renamed to “sandwich”. “tess4” is now a deprecated alias for “sandwich”.
2.157 v5.1
• Files with pages larger than 200” (5080 mm) in either dimension are now supported with
--output-type=pdf with the page size preserved (in the PDF specification this feature is called UserUnit
scaling). Due to Ghostscript limitations this is not available in conjunction with PDF/A output.
2.158 v5.0.1
• Fixed #169, exception due to failure to create sidecar text files on some versions of Tesseract 3.04, including the
jbarlow83/ocrmypdf Docker image
2.153. v5.3.2 43
ocrmypdf Documentation, Release 0.1.dev83+g072db75
2.159 v5.0
2.160 v4.5.6
• Fixed #156, ‘NoneType’ object has no attribute ‘getObject’ on pages with no optional /Contents record. This
should resolve all issues related to pages with no /Contents record.
• Fixed #158, ocrmypdf now stops and terminates if Ghostscript fails on an intermediate step, as it is not possible
to proceed.
• Fixed #160, exception thrown on certain invalid arguments instead of error message
2.161 v4.5.5
2.162 v4.5.4
• Fixed --skip-big raising an exception if a page contains no images (#152) (thanks to @TomRaz)
• Fixed an issue where pages with no images might trigger “cannot write mode P as JPEG” (#151)
2.163 v4.5.3
• Added a workaround for Ghostscript 9.21 and probably earlier versions would fail with the error message
“VMerror -25”, due to a Ghostscript bug in XMP metadata handling
• High Unicode characters (U+10000 and up) are no longer accepted for setting metadata on the command line,
as Ghostscript may not handle them correctly.
• Fixed an issue where the tess4 renderer would duplicate content onto output pages if tesseract failed or timed
out
• Fixed tess4 renderer not recognized when lossless reconstruction is possible
2.164 v4.5.2
• Fixed #147, --pdf-renderer tess4 --clean will produce an oversized page containing the original
image in the bottom left corner, due to loss DPI information.
• Make “using Tesseract 4.0” warning less ominous
• Set up machinery for homebrew OCRmyPDF tap
2.165 v4.5.1
• Fixed #137, proportions of images with a non-square pixel aspect ratio would be distorted in output for
--force-ocr and some other combinations of flags
2.166 v4.5
• PDFs containing “Form XObjects” are now supported (issue #134; PDF reference manual 8.10), and images
they contain are taken into account when determining the resolution for rasterizing
• The Tesseract 4 Docker image no longer includes all languages, because it took so long to build something
would tend to fail
• OCRmyPDF now warns about using --pdf-renderer tesseract with Tesseract 3.04 or lower due to
issues with Ghostscript corrupting the OCR text in these cases
2.167 v4.4.2
• The Docker images (ocrmypdf, ocrmypdf-polyglot, ocrmypdf-tess4) are now based on Ubuntu 16.10 instead of
Debian stretch
– This makes supporting the Tesseract 4 image easier
– This could be a disruptive change for any Docker users who built customized these images with their own
changes, and made those changes in a way that depends on Debian and not Ubuntu
• OCRmyPDF now prevents running the Tesseract 4 renderer with Tesseract 3.04, which was permitted in v4.4
and v4.4.1 but will not work
2.168 v4.4.1
• To prevent a TIFF output error caused by img2pdf >= 0.2.1 and Pillow <= 3.4.2, dependencies have been
tightened
2.164. v4.5.2 45
ocrmypdf Documentation, Release 0.1.dev83+g072db75
• The Tesseract 4.00 simultaneous process limit was increased from 1 to 2, since it was observed that 1 lowers
performance
• Documentation improvements to describe the --tesseract-config feature
• Added test cases and fixed error handling for --tesseract-config
• Tweaks to setup.py to deal with issues in the v4.4 release
2.169 v4.4
2.170 v4.3.5
• Update documentation to confirm Python 3.6.0 compatibility. No code changes were needed, so many earlier
versions are likely supported.
2.171 v4.3.4
• Fixed “decimal.InvalidOperation: quantize result has too many digits” for high DPI images
2.172 v4.3.3
2.173 v4.3.2
• Fixed a PDF/A creation issue with Ghostscript 9.20 (note: this fix did not actually work)
2.174 v4.3.1
• Fixed an issue where pages produced by the “hocr” renderer after a Tesseract timeout would be rotated incor-
rectly if the input page was rotated with a /Rotate marker
• Fixed a file handle leak in LeptonicaErrorTrap that would cause a “too many open files” error for files around
hundred pages of pages long when --deskew or --remove-background or other Leptonica based image
processing features were in use, depending on the system value of ulimit -n
• Ability to specify multiple languages for multilingual documents is now advertised in documentation
• Reduced the file sizes of some test resources
• Cleaned up debug output
• Tesseract caching in test cases is now more cautious about false cache hits and reproducing exact output, not
that any problems were observed
2.175 v4.3
• New feature --remove-background to detect and erase the background of color and grayscale images
• Better documentation
• Fixed an issue with PDFs that draw images when the raster stack depth is zero
• ocrmypdf can now redirect its output to stdout for use in a shell pipeline
– This does not improve performance since temporary files are still used for buffering
– Some output validation is disabled in this mode
2.176 v4.2.5
• Fixed an issue (#100) with PDFs that omit the optional /BitsPerComponent parameter on images
• Removed non-free file milk.pdf
2.177 v4.2.4
• Fixed an error (#90) caused by PDFs that use stencil masks properly
• Fixed handling of PDFs that try to draw images or stencil masks without properly setting up the graphics state
(such images are now ignored for the purposes of calculating DPI)
2.173. v4.3.2 47
ocrmypdf Documentation, Release 0.1.dev83+g072db75
2.178 v4.2.3
• Fixed an issue with PDFs that store page rotation (/Rotate) in an indirect object
• Integrated a few fixes to simplify downstream packaging (Debian)
– The test suite no longer assumes it is installed
– If running Linux, skip a test that passes Unicode on the command line
• Added a test case to check explicit masks and stencil masks
• Added a test case for indirect objects and linearized PDFs
• Deprecated the OCRmyPDF.sh shell script
2.179 v4.2.2
• Improvements to documentation
2.180 v4.2.1
• Fixed an issue where PDF pages that contained stencil masks would report an incorrect DPI and cause
Ghostscript to abort
• Implemented stdin streaming
2.181 v4.2
• ocrmypdf will now try to convert single image files to PDFs if they are provided as input (#15)
– This is a basic convenience feature. It only supports a single image and always makes the image fill the
whole page.
– For better control over image to PDF conversion, use img2pdf (one of ocrmypdf’s dependencies)
• New argument --output-type {pdf|pdfa} allows disabling Ghostscript PDF/A generation
– pdfa is the default, consistent with past behavior
– pdf provides a workaround for users concerned about the increase in file size from Ghostscript forcing
JBIG2 images to CCITT and transcoding JPEGs
– pdf preserves as much as it can about the original file, including problems that PDF/A conversion fixes
• PDFs containing images with “non-square” pixel aspect ratios, such as 200x100 DPI, are now handled and
converted properly (fixing a bug that caused to be cropped)
• --force-ocr rasterizes pages even if they contain no images
– supports users who want to use OCRmyPDF to reconstruct text information in PDFs with damaged Uni-
code maps (copy and paste text does not match displayed text)
– supports reinterpreting PDFs where text was rendered as curves for printing, and text needs to be recovered
– fixes issue #82
• Fixes an issue where, with certain settings, monochrome images in PDFs would be converted to 8-bit grayscale,
increasing file size (#79)
• Support for Ubuntu 12.04 LTS “precise” has been dropped in favor of (roughly) Ubuntu 14.04 LTS “trusty”
– Some Ubuntu “PPAs” (backports) are needed to make it work
• Support for some older dependencies dropped
– Ghostscript 9.15 or later is now required (available in Ubuntu trusty with backports)
– Tesseract 3.03 or later is now required (available in Ubuntu trusty)
• Ghostscript now runs in “safer” mode where possible
2.182 v4.1.4
• Bug fix: monochrome images with an ICC profile attached were incorrectly converted to full color images if
lossless reconstruction was not possible due to other settings; consequence was increased file size for these
images
2.183 v4.1.3
• More helpful error message for PDFs with version 4 security handler
• Update usage instructions for Windows/Docker users
• Fixed order of operations for matrix multiplication (no effect on most users)
• Add a few leptonica wrapper functions (no effect on most users)
2.184 v4.1.2
• Replace IEC sRGB ICC profile with Debian’s sRGB (from icc-profiles-free) which is more compatible with the
MIT license
• More helpful error message for an error related to certain types of malformed PDFs
2.185 v4.1
• --rotate-pages now only rotates pages when reasonably confidence in the orientation. This behavior can
be adjusted with the new argument --rotate-pages-threshold
• Fixed problems in error checking if unpaper is uninstalled or missing at run-time
• Fixed problems with “RethrownJobError” errors during error handling that suppressed the useful error messages
2.186 v4.0.7
2.182. v4.1.4 49
ocrmypdf Documentation, Release 0.1.dev83+g072db75
2.187 v4.0.6
2.188 v4.0.5
2.189 v4.0.4
Released with verbose debug message turned on. Do not use. Skip to v4.0.5.
2.190 v4.0.3
New features
• Page orientations detected are now reported in a summary comment
Fixes
• Show stack trace if unexpected errors occur
• Treat “too few characters” error message from Tesseract as a reason to skip that page rather than abort the file
• Docker: fix blank JPEG2000 issue by insisting on Ghostscript versions that have this fixed
2.191 v4.0.2
Fixes
• Fixed compatibility with Tesseract 3.04.01 release, particularly its different way of outputting orientation infor-
mation
• Improved handling of Tesseract errors and crashes
• Fixed use of chmod on Docker that broke most test cases
2.192 v4.0.1
Fixes
• Fixed a KeyError if tesseract fails to find page orientation information
2.193 v4.0
New features
• Automatic page rotation (-r) is now available. It uses ignores any prior rotation information on PDFs and
sets rotation based on the dominant orientation of detectable text. This feature is fairly reliable but some false
positives occur especially if there is not much text to work with. (#4)
• Deskewing is now performed using Leptonica instead of unpaper. Leptonica is faster and more reliable at image
deskewing than unpaper.
Fixes
• Fixed an issue where lossless reconstruction could cause some pages to be appear incorrectly if the page was
rotated by the user in Acrobat after being scanned (specifically if it a /Rotate tag)
• Fixed an issue where lossless reconstruction could misalign the graphics layer with respect to text layer if the
page had been cropped such that its origin is not (0, 0) (#49)
Changes
• Logging output is now much easier to read
• --deskew is now performed by Leptonica instead of unpaper (#25)
• libffi is now required
• Some changes were made to the Docker and Travis build environments to support libffi
• --pdf-renderer=tesseract now displays a warning if the Tesseract version is less than 3.04.01, the
planned release that will include fixes to an important OCR text rendering bug in Tesseract 3.04.00. You can
also manually install ./share/sharp2.ttf on top of pdf.ttf in your Tesseract tessdata folder to correct the problem.
2.194 v3.2.1
Changes
• Fixed #47 “convert() got and unexpected keyword argument ‘dpi’” by upgrading to img2pdf 0.2
• Tweaked the Dockerfiles
2.195 v3.2
New features
• Lossless reconstruction: when possible, OCRmyPDF will inject text layers without otherwise manipulating the
content and layout of a PDF page. For example, a PDF containing a mix of vector and raster content would
see the vector content preserved. Images may still be transcoded during PDF/A conversion. (--deskew and
--clean-final disable this mode, necessarily.)
• New argument --tesseract-pagesegmode allows you to pass page segmentation arguments to Tesseract
OCR. This helps for two column text and other situations that confuse Tesseract.
• Added a new “polyglot” version of the Docker image, that generates Tesseract with all languages packs installed,
for the polyglots among us. It is much larger.
Changes
• JPEG transcoding quality is now 95 instead of the default 75. Bigger file sizes for less degradation.
2.193. v4.0 51
ocrmypdf Documentation, Release 0.1.dev83+g072db75
2.196 v3.1.1
Changes
• Fixed bug that caused incorrect page size and DPI calculations on documents with mixed page sizes
2.197 v3.1
Changes
• Default output format is now PDF/A-2b instead of PDF/A-1b
• Python 3.5 and macOS El Capitan are now supported platforms - no changes were needed to implement support
• Improved some error messages related to missing input files
• Fixed #20: uppercase .PDF extension not accepted
• Fixed an issue where OCRmyPDF failed to text that certain pages contained previously OCR’ed text, such as
OCR text produced by Tesseract 3.04
• Inserts /Creator tag into PDFs so that errors can be traced back to this project
• Added new option --pdf-renderer=auto, to let OCRmyPDF pick the best PDF renderer. Currently it
always chooses the ‘hocrtransform’ renderer but that behavior may change.
• Set up Travis CI automatic integration testing
2.198 v3.0
New features
• Easier installation with a Docker container or Python’s pip package manager
• Eliminated many external dependencies, so it’s easier to setup
• Now installs ocrmypdf to /usr/local/bin or equivalent for system-wide access and easier typing
• Improved command line syntax and usage help (--help)
• Tesseract 3.03+ PDF page rendering can be used instead for better positioning of recognized text
(--pdf-renderer tesseract)
• PDF metadata (title, author, keywords) are now transferred to the output PDF
• PDF metadata can also be set from the command line (--title, etc.)
• Automatic repairs malformed input PDFs if possible
• Added test cases to confirm everything is working
• Added option to skip extremely large pages that take too long to OCR and are often not OCRable (e.g. large
scanned maps or diagrams); other pages are still processed (--skip-big)
• Added option to kill Tesseract OCR process if it seems to be taking too long on a page, while still processing
other pages (--tesseract-timeout)
• Less common colorspaces (CMYK, palette) are now supported by conversion to RGB
• Multiple images on the same PDF page are now supported
Changes
• New, robust rewrite in Python 3.4+ with ruffus pipelines
• Now uses Ghostscript 9.14’s improved color conversion model to preserve PDF colors
• OCR text is now rendered in the PDF as invisible text. Previous versions of OCRmyPDF incorrectly rendered
visible text with an image on top.
• All “tasks” in the pipeline can be executed in parallel on any available CPUs, increasing performance
• The -o DPI argument has been phased out, in favor of --oversample DPI, in case we need -o
OUTPUTFILE in the future
• Removed several dependencies, so it’s easier to install. We no longer use:
– GNU parallel
– ImageMagick
– Python 2.7
– Poppler
– MuPDF tools
– shell scripts
– Java and JHOVE
– libxml2
• Some new external dependencies are required or optional, compared to v2.x:
– Ghostscript 9.14+
– qpdf 5.0.0+
– Unpaper 6.1 (optional)
– some automatically managed Python packages
Release candidates^
• rc9:
– Fix #118: report error if ghostscript iccprofiles are missing
– fixed another issue related to #111: PDF rasterized to palette file
– add support image files with a palette
– don’t try to validate PDF file after an exception occurs
• rc8:
– Fix #111: exception thrown if PDF is missing DocumentInfo dictionary
• rc7:
– fix error when installing direct from pip, “no such file ‘requirements.txt’”
• rc6:
– dropped libxml2 (Python lxml) since Python 3’s internal XML parser is sufficient
– set up Docker container
– fix Unicode errors if recognized text contains Unicode characters and system locale is not UTF-8
• rc5:
2.198. v3.0 53
ocrmypdf Documentation, Release 0.1.dev83+g072db75
-l
deu
--author
A. Merkel
--pdf-renderer
tesseract
Fixes
• Handling of filenames containing spaces: fixed
Notes and known issues
• Some dependencies may work with lower versions than tested, so try overriding dependencies if they are “in the
way” to see if they work.
• --pdf-renderer tesseract will output files with an incorrect page size in Tesseract 3.03, due to a bug
in Tesseract.
• PDF files containing “inline images” are not supported and won’t be for the 3.0 release. Scanned images almost
never contain inline images.
OCRmyPDF versions 1 and 2 were implemented as shell scripts. OCRmyPDF 3.0+ is a fork that gradually replaced all
shell scripts with Python while maintaining the existing command line arguments. No one is maintaining old versions.
For details on older versions, see the final version of its release notes.
Installing OCRmyPDF
The easiest way to install OCRmyPDF is to follow the steps for your operating system/platform. This version may be
out of date, however.
These platforms have one-liner installs:
More detailed procedures are outlined below. If you want to do a manual install, or install a more recent version than
your platform provides, read on.
Platform-specific steps
• Installing on Linux
– Debian and Ubuntu 20.04 or newer
– Fedora
– Installing the latest version on Ubuntu 22.04 LTS
– Ubuntu 20.04 LTS
– Arch Linux (AUR)
– Alpine Linux
– Gentoo Linux
57
ocrmypdf Documentation, Release 0.1.dev83+g072db75
As indicated in the table above, Debian and Ubuntu releases may lag behind the latest version. If the version available
for your platform is out of date, you could opt to install the latest version from source. See Installing HEAD revision
from sources.
For full details on version availability for your platform, check the Debian Package Tracker or Ubuntu launchpad.net.
Note: OCRmyPDF for Debian and Ubuntu currently omit the JBIG2 encoder. OCRmyPDF works fine without it but
will produce larger output files. If you build jbig2enc from source, ocrmypdf will automatically detect it (specifically
the jbig2 binary) on the PATH. To add JBIG2 encoding, see Installing the JBIG2 encoder.
3.1.2 Fedora
OCRmyPDF version
For full details on version availability, check the Fedora Package Tracker.
If the version available for your platform is out of date, you could opt to install the latest version from source. See
Installing HEAD revision from sources.
Note: OCRmyPDF for Fedora currently omits the JBIG2 encoder due to patent issues. OCRmyPDF works fine
without it but will produce larger output files. If you build jbig2enc from source, ocrmypdf 7.0.0 and later will
automatically detect it on the PATH. To add JBIG2 encoding, see Installing the JBIG2 encoder.
Ubuntu 22.04 includes ocrmypdf 13.4.0 - you can install that with apt install ocrmypdf. To install a more
recent version for the current user, follow these steps:
If you get the message WARNING: The script ocrmypdf is installed in '/home/$USER/.
local/bin' which is not on PATH., you may need to re-login or open a new shell, or manually adjust
your PATH.
To add JBIG2 encoding, see Installing the JBIG2 encoder.
Ubuntu 20.04 includes ocrmypdf 9.6.0 - you can install that with apt. To install a more recent version, uninstall the
system-provided version of ocrmypdf, and install the following dependencies:
export PATH=$HOME/.local/bin:$PATH
pip3 install --user ocrmypdf
curl -O https://aur.archlinux.org/cgit/aur.git/snapshot/ocrmypdf.tar.gz
tar xvzf ocrmypdf.tar.gz
cd ocrmypdf
makepkg -sri
At this point you will have a working install of OCRmyPDF, but the Tesseract install won’t include any OCR language
data. You can install the tesseract-data package group to add all supported languages, or use that package listing to
identify the appropriate package for your desired language.
As an alternative to this manual procedure, consider using an AUR helper. Such a tool will automatically fetch, build
and install the AUR package, resolve dependencies (including dependencies on AUR packages), and ease the upgrade
procedure.
If you have any difficulties with installation, check the repository package page.
Note: The OCRmyPDF AUR package currently omits the JBIG2 encoder. OCRmyPDF works fine without it but will
produce larger output files. The encoder is available from the jbig2enc-git AUR package and may be installed using
the same series of steps as for the installation OCRmyPDF AUR package. Alternatively, it may be built manually from
source following the instructions in Installing the JBIG2 encoder. If JBIG2 is installed, OCRmyPDF 7.0.0 and later
will automatically detect it.
3.2.1 Homebrew
This will include only the English language pack. If you need other languages you can optionally install them all:
These instructions probably work on all macOS supported by Homebrew, and are for installing a more current version
of OCRmyPDF than is available from Homebrew. Note that the Homebrew versions usually track the release versions
fairly closely.
If it’s not already present, install Homebrew.
Update Homebrew:
brew update
Install or upgrade the required Homebrew packages, if any are missing. To do this, use brew edit ocrmypdf to
obtain a recent list of Homebrew dependencies. You could also check the .workflows/build.yml.
This will include the English, French, German and Spanish language packs. If you need other languages you can
optionally install them all:
You can then install OCRmyPDF from PyPI for the current user:
ocrmypdf --help
Warning: As of early 2021, users have reported problems with the Microsoft Store version of Python and
OCRmyPDF. These issues affect many other third party Python packages. Please download Python from
Python.org or Chocolatey instead, and do not use the Microsoft Store version.
1. Install Ubuntu 22.04 for Windows Subsystem for Linux, if not already installed.
2. Follow the procedure to install OCRmyPDF on Ubuntu 22.04.
3. Open the Windows command prompt and create a symlink:
You can then run OCRmyPDF in the Windows command prompt or Powershell, prefixing wsl, and call it from
Windows programs or batch files.
3.3.3 Cygwin64
First install the the following prerequisite Cygwin packages using setup-x86_64.exe:
gcc-g++
ghostscript
libexempi3
libexempi-devel
libffi6
libffi-devel
pngquant
qpdf
libqpdf-devel
tesseract-ocr
tesseract-ocr-devel
Then open a Cygwin terminal (i.e. mintty), run the following commands. Note that if you are using the version of
pip that was installed with the Cygwin Python package, the command name will be pip3. If you have since updated
pip (with, for instance pip3 install --upgrade pip) the the command is likely just pip instead of pip3:
The optional dependency “unpaper” that is currently not available under Cygwin. Without it, certain options such as
--clean will produce an error message. However, the OCR-to-text-layer functionality is available.
3.3.4 Docker
You can also Install the Docker container on Windows. Ensure that your command prompt can run the docker “hello
world” container.
To install a more recent version, you could attempt to first install the system version with pkg, then use pip
install --user ocrmypdf.
For some users, installing the Docker image will be easier than installing all of OCRmyPDF’s dependencies.
See OCRmyPDF Docker image for more information.
OCRmyPDF is delivered by PyPI because it is a convenient way to install the latest version. However, PyPI and pip
cannot address the fact that ocrmypdf depends on certain non-Python system libraries and programs being installed.
For best results, first install your platform’s version of ocrmypdf, using the instructions elsewhere in this document.
Then you can use pip to get the latest version if your platform version is out of date. Chances are that this will satisfy
most dependencies.
Use ocrmypdf --version to confirm what version was installed.
Then you can install the latest OCRmyPDF from the Python wheels. First try:
You should then be able to run ocrmypdf --version and see that the latest version was located.
OCRmyPDF currently requires these external programs and libraries to be installed, and must be satisfied using the
operating system package manager. pip cannot provide them.
The following versions are required:
If you have git and Python 3.9 or newer installed, you can install from source. When the pip installer runs, it will
alert you if dependencies are missing.
If you prefer to build every from source, you will need to build pikepdf from source. First ensure you can build and
install pikepdf.
To install the HEAD revision from sources in the current Python 3 environment:
Or, to install in development mode, allowing customization of OCRmyPDF, use the -e flag:
You may find it easiest to install in a virtual environment, rather than system-wide:
However, ocrmypdf will only be accessible on the system PATH when you activate the virtual environment.
To run the program:
ocrmypdf --help
If not yet installed, the script will notify you about dependencies that need to be installed. The script requires specific
versions of the dependencies. Older version than the ones mentioned in the release notes are likely not to be compatible
to OCRmyPDF.
Completions for bash and fish are available in the project’s misc/completion folder. The bash completions
are likely zsh compatible but this has not been confirmed. Package maintainers, please install these at the appropriate
locations for your system.
To manually install the bash completion, copy misc/completion/ocrmypdf.bash to /etc/
bash_completion.d/ocrmypdf (rename the file).
To manually install the fish completion, copy misc/completion/ocrmypdf.fish to ~/.config/fish/
completions/ocrmypdf.fish.
Many Python libraries no longer 32-bit binary wheels for Linux. This includes many of the libraries that OCRmyPDF
depends on, such as Pillow. The easiest way to express this to end users is to say we don’t support 32-bit Linux.
However, if your Linux distribution still supports 32-bit binaries, you can still install and use OCRmyPDF. A warning
message will appear. In practice, OCRmyPDF may need more than 32-bit memory space to run when large documents
are processed, so there are practical limitations to what users can accomplish with it. Still, for the common use case
of an 32-bit ARM NAS or Raspberry Pi processing small documents, it should work.
OCRmyPDF uses Tesseract for OCR, and relies on its language packs for all languages. On most platforms, English
is installed with Tesseract by default, but not always.
Tesseract supports most languages. Languages are identified by standardized three-letter codes (called ISO 639-2
Alpha-3). Tesseract’s documentation also lists the three-letter code for your language. Some are anglicized, e.g.
Spanish is spa rather than esp, while others are not, e.g. German is deu and French is fra.
After you have installed a language pack, you can use it with ocrmypdf -l <language>, for example
ocrmypdf -l spa. For multilingual documents, you can specify all languages to be expected, e.g. ocrmypdf
-l eng+fra for English and French. English is assumed by default unless other language(s) are specified.
For Linux users, you can often find packages that provide language packs:
You can then pass the -l LANG argument to OCRmyPDF to give a hint as to what languages it should search for.
Multiple languages can be requested using either -l eng+fra (English and French) or -l eng -l fra.
67
ocrmypdf Documentation, Release 0.1.dev83+g072db75
You can then pass the -l LANG argument to OCRmyPDF to give a hint as to what languages it should search for.
Multiple languages can be requested using either -l eng+fra (English and French) or -l eng -l fra.
# Add global English and German language support (the `l10n_` from equery has to be
˓→omitted)
You can then pass the -l LANG argument to OCRmyPDF to give a hint as to what languages it should search for.
Multiple languages can be requested using either -l eng+fra (English and French) or -l eng -l fra.
You can install additional language packs by installing Tesseract using Homebrew with all language packs.
Users of the OCRmyPDF Docker image should install language packs into a derived Docker image as described in
that section.
The Tesseract installer provided by Chocolatey currently includes only English language. To install other languages,
download the respective language pack (.traineddata file) from https://github.com/tesseract-ocr/tessdata/ and
place it in C:\\Program Files\\Tesseract-OCR\\tessdata (or wherever Tesseract OCR is installed).
Most Linux distributions do not include a JBIG2 encoder since JBIG2 encoding was patented for a long time. All
known JBIG2 US patents have expired as of 2017, but it is possible that unknown patents exist.
JBIG2 encoding is recommended for OCRmyPDF and is used to losslessly create smaller PDFs. If JBIG2 encoding
is not available, lower quality encodings will be used.
JBIG2 decoding is not patented and is performed automatically by most PDF viewers. It is widely supported and has
been part of the PDF specification since 2001.
On macOS, Homebrew packages jbig2enc and OCRmyPDF includes it by default. The Docker image for OCRmyPDF
also builds its own JBIG2 encoder from source.
For all other Linux, you must build a JBIG2 encoder from source:
Dependencies include libtoolize and libleptonica, which on Ubuntu systems are packaged as libtool and libleptonica-
dev. On Fedora (35) they are packaged as libtool and leptonica-devel.
OCRmyPDF provides lossy mode JBIG2 as an advanced feature. Users should review the technical concerns with
JBIG2 in lossy mode and decide if this feature is acceptable for their use case.
JBIG2 lossy mode does achieve higher compression ratios than any other monochrome (bitonal) compression technol-
ogy; for large text documents the savings are considerable. JBIG2 lossless still gives great compression ratios and is a
major improvement over the older CCITT G4 standard. As explained above, there is some risk of substitution errors.
To turn on JBIG2 lossy mode, add the argument --jbig2-lossy. --optimize {1,2,3} are necessary for the
argument to take effect also required. Also, a JBIG2 encoder must be installed as described in the previous section.
69
ocrmypdf Documentation, Release 0.1.dev83+g072db75
Due to an oversight, ocrmypdf v7.0 and v7.1 used lossy mode by default.
Cookbook
6.1.1 Help!
ocrmypdf --help
6.1.4 Create a PDF/A with all color and grayscale images converted to JPEG
71
ocrmypdf Documentation, Release 0.1.dev83+g072db75
OCR will attempt to automatic correct the rotation of each page. This can help fix a scanning job that contains a mix
of landscape and portrait pages.
You can increase (decrease) the parameter --rotate-pages-threshold to make page rotation more (less) ag-
gressive. The threshold number is the ratio of how confidence the OCR engine is that the document image should be
changed, compared to kept the same. The default value is quite conservative; on some files it may not attempt rotations
at all unless it is very confident that the current rotation is wrong. A lower value of 2.0 will produce more rotations,
and more false positives. Run with -v1 to see the confidence level for each page to see if there may be a better value
for your files.
If the page is “just a little off horizontal”, like a crooked picture, then you want --deskew. --rotate-pages is
for when the cardinal angle is wrong.
OCRmyPDF assumes the document is in English unless told otherwise. OCR quality may be poor if the wrong
language is used.
Language packs must be installed for all languages specified. See Installing additional language packs.
Unfortunately, the Tesseract OCR engine has no ability to detect the language when it is unknown.
This produces a file named “output.pdf” and a companion text file named “output.txt”.
Note: The sidecar file contains the OCR text found by OCRmyPDF. If the document contains pages that already
have text, that text will not appear in the sidecar. If the option --pages is used, only those pages on which OCR
was performed will be included in the sidecar. If certain pages were skipped because of options like --skip-big or
--tesseract-timeout, those pages will not be in the sidecar.
If you don’t want to generate the output PDF, use --output-type=none to avoid generating one. Set the output
filename to - (i.e. redirect to stdout).
To extract all text from a PDF, whether generated from OCR or otherwise, use a program like Poppler’s pdftotext
or pdfgrep.
72 Chapter 6. Cookbook
ocrmypdf Documentation, Release 0.1.dev83+g072db75
If you are starting with images, you can just use Tesseract directly to convert images to PDFs:
Tesseract’s PDF output is quite good – OCRmyPDF uses it internally, in some cases. However, OCRmyPDF has many
features not available in Tesseract like image processing, metadata control, and PDF/A generation.
You can also use a program like img2pdf to convert your images to PDFs, and then pipe the results to run ocrmypdf.
The - tells ocrmypdf to read standard input.
img2pdf is recommended because it does an excellent job at generating PDFs without transcoding images.
For convenience, OCRmyPDF can also convert single images to PDFs on its own. If the resolution (dots per inch,
DPI) of an image is not set or is incorrect, it can be overridden with --image-dpi. (As 1 inch is 2.54 cm, 1 dpi =
0.39 dpcm).
If you have multiple images, you must use img2pdf to convert the images to PDF.
Not recommended
We caution against using ImageMagick or Ghostscript to convert images to PDF, since they may transcode images or
produce downsampled images, sometimes without warning.
OCRmyPDF perform some image processing on each page of a PDF, if desired. The same processing is applied to
each page. It is suggested that the user review files after image processing as these commands might remove desirable
content, especially from poor quality scans.
• --rotate-pages attempts to determine the correct orientation for each page and rotates the page if neces-
sary.
• --remove-background attempts to detect and remove a noisy background from grayscale or color images.
Monochrome images are ignored. This should not be used on documents that contain color photos as it may
remove them.
• --deskew will correct pages were scanned at a skewed angle by rotating them back into place.
• --clean uses unpaper to clean up pages before OCR, but does not alter the final output. This makes it less
likely that OCR will try to find text in background noise.
• --clean-final uses unpaper to clean up pages before OCR and inserts the page into the final output. You
will want to review each page to ensure that unpaper did not remove something important.
Note: In many cases image processing will rasterize PDF pages as images, potentially losing quality.
Warning: --clean-final and --remove-background may leave undesirable visual artifacts in some
images where their algorithms have shortcomings. Files should be visually reviewed after using these options.
Deskew:
Image processing commands can be combined. The order in which options are given does not matter. OCRmyPDF
always applies the steps of the image processing pipeline in the same order (rotate, remove background, deskew,
clean).
If you set --tesseract-timeout 0 OCRmyPDF will apply its image processing without performing OCR (by
causing OCR to time out). This works if all you want to is to apply image processing or PDF/A conversion.
Changed in version v14.1.0: Prior to this version, --tesseract-timeout 0 would prevent other uses of Tesser-
act, such as deskewing, from working. This is no longer the case. Use --tesseract-non-ocr-timeout to
control the timeout for non-OCR operations, if needed.
You can also optimize all images without performing any OCR:
You can ask OCRmyPDF to only apply image processing and OCR to certain pages.
74 Chapter 6. Cookbook
ocrmypdf Documentation, Release 0.1.dev83+g072db75
Hyphens denote a range of pages and commas separate page numbers. If you prefer to use spaces, quote all of the
page numbers: --pages '2, 3, 5, 7'.
OCRmyPDF will warn if your list of page numbers contains duplicates or overlapping pages. OCRmyPDF does not
currently account for document page numbers, such as an introduction section of a book that uses Roman numerals.
It simply counts the number of virtual pieces of paper since the start. If your list of pages is out of numerical order,
OCRmyPDF will sort it for you.
Regardless of the argument to --pages, OCRmyPDF will optimize all pages/images in the file and convert it to
PDF/A, unless you disable those options. Both of these steps are “whole file” operations. In this example, we want to
OCR only the title and otherwise change the PDF as little as possible:
To redo OCR on a file OCRed with other OCR software or a previous version of OCRmyPDF and/or Tesseract, you
may use the --redo-ocr argument. (Normally, OCRmyPDF will exit with an error if asked to modify a file with
OCR.)
This may be helpful for users who want to take advantage of accuracy improvements in Tesseract for files they previ-
ously OCRed with an earlier version of Tesseract and OCRmyPDF.
This method will replace OCR without rasterizing, reducing quality or removing vector content. If a file contains a mix
of pure digital text and OCR, digital text will be ignored and OCR will be replaced. As such this mode is incompatible
with image processing options, since they alter the appearance of the file.
In some cases, existing OCR cannot be detected or replaced. Files produced by OCRmyPDF v2.2 or earlier, for
example, are internally represented as having visible text with an opaque image drawn on top. This situation cannot
be detected.
If --redo-ocr does not work, you can use --force-ocr, which will force rasterization of all pages, potentially
reducing quality or losing vector content.
By default OCRmyPDF will attempt to perform lossless optimizations on the images inside PDFs after OCR is com-
plete. Optimization is performed even if no OCR text is found.
The --optimize N (short form -O) argument controls optimization, where N ranges from 0 to 3 inclusive, analo-
gous to the optimization levels in the GCC compiler.
Level Comments
Disables optimization.
--optimize
0
Enables lossless optimizations, such as transcoding images to more efficient formats. Also compress other
--optimize
1 uncompressed objects in the PDF and enables the more efficient “object streams” within the PDF. (If
--jbig2-lossy is issued, then lossy JBIG2 optimization is used. The decision to use lossy JBIG2 is
separate from standard optimization settings.)
All of the above, and enables lossy optimizations and color quantization.
--optimize
2
All of the above, and enables more aggressive optimizations and targets lower image quality.
--optimize
3
Optimization is improved when a JBIG2 encoder is available and when pngquant is installed. If either of these
components are missing, then some types of images cannot be optimized.
The types of optimization available may expand over time. By default, OCRmyPDF compresses data streams inside
PDFs, and will change inefficient compression modes to more modern versions. A program like qpdf can be used to
change encodings, e.g. to inspect the internals for a PDF.
Note: Image processing and PDF/A conversion can also introduce lossy transformations to your PDF images, even
when --optimize 1 is in use.
OCRmyPDF cannot preserve digital signatures in PDFs and also add to OCR to them. By default, it
will refuse to modify a signed PDF regardless of other settings. You can override this behavior with
--invalidate-digital-signatures; as the name suggests, any digital signatures will be invalidated.
OCRmyPDF cannot open documents that are encrypted with a digital certificate.
Versions of OCRmyPDF prior to 14.4.0 would invalidate existing digital signatures without warning.
76 Chapter 6. Cookbook
CHAPTER 7
PDF optimization
OCRmyPDF includes an image-oriented PDF optimizer. By default, the optimizer runs with safe settings with the goal
of improving compression at no loss of quality. At higher optimization levels, lossy optimizations may be applied and
tuned. Optimization occurs after OCR, and only if OCR succeeded. It does not perform other possible optimizations
such as deduplicating resources, consolidating fonts, simplifying vector drawings, or anything of that nature.
The exact type of optimizations performed will vary over time, and depend on the availability of third-party tools.
Despite optimizations, OCRmyPDF might still increase the overall file size, since it must embed information about
the recognized text, and depending on the settings chosen, may not be able to represent the output file as compactly as
the input file.
OCRmyPDF will automatically replace obsolete or inferior compression schemes such as RLE or LZW with superior
schemes such as Deflate and converting monochrome images to CCITT G4. Since this is harmless it always occurs
and there is no way to disable it. Other non-image compressed objects are compressed as well.
OCRmyPDF automatically optimizes PDFs for “fast web view” in Adobe Acrobat’s parlance, or equivalently, lin-
earizes PDFs so that the resources they reference are presented in the order a viewer needs them for sequential display.
This reduces the latency of viewing a PDF both online and from local storage, in exchange for a slight increase in file
size.
To disable this optimization and all others, use ocrmypdf --optimize 0 ... or the shorthand -O0.
Adobe Acrobat might not report the file as being “fast web view”.
77
ocrmypdf Documentation, Release 0.1.dev83+g072db75
At optimization level -O1 (the default), OCRmyPDF will also attempt lossless image optimization.
If a JBIG2 encoder is available, then monochrome images will be converted to JBIG2, with the potential for huge
savings on large black and white images, since JBIG2 is far more efficient than any other monochrome (bi-level)
compression. (All known US patents related to JBIG2 have probably expired, but it remains the responsibility of the
user to supply a JBIG2 encoder such as jbig2enc. OCRmyPDF does not implement JBIG2 encoding on its own.)
OCRmyPDF currently does not attempt to recompress losslessly compressed objects more aggressively.
At optimization level -O2 and -O3, OCRmyPDF will some attempt lossy image optimization.
If pngquant is installed, OCRmyPDF will use it to perform quantize paletted images to reduce their size.
The quality of JPEGs may be lowered, on the assumption that a lower quality image may be suitable for storage after
OCR.
It is not possible to optimize all image types. Uncommon image types may be skipped by the optimizer.
OCRmyPDF provides lossy mode JBIG2 as an advanced feature that additional requires the argument
--jbig2-lossy.
OCRmyPDF is also available in a Docker image that packages recent versions of all dependencies.
For users who already have Docker installed this may be an easy and convenient option. However, it is less performant
than a system installation and may require Docker engine configuration.
OCRmyPDF needs a generous amount of RAM, CPU cores, temporary storage space, whether running in a Docker
container or on its own. It may be necessary to ensure the container is provisioned with additional resources.
If you have Docker installed on your system, you can install a Docker image of the latest release.
If you can run this command successfully, your system is ready to download and execute the image:
OCRmyPDF will use all available CPU cores. By default, the VirtualBox machine instance on Windows and macOS
has only a single CPU core enabled. Use the VirtualBox Manager to determine the name of your Docker engine host,
and then follow these optional steps to enable multiple CPUs:
See the Docker documentation for adjusting memory and CPU on other platforms.
79
ocrmypdf Documentation, Release 0.1.dev83+g072db75
Unlike typical Docker containers, in this section the OCRmyPDF Docker container is ephemeral – it runs for one
OCR job and terminates, just like a command line program. We are using Docker to deliver an application (as opposed
to the more conventional case, where a Docker container runs as a server).
To start a Docker container (instance of the image):
For convenience, create a shell alias to hide the Docker command. It is easier to send the input file as stdin and read
the output from stdout – this avoids the messy permission issues with Docker entirely.
Alternately, you could mount the local current working directory as a Docker volume:
alias docker_ocrmypdf='docker run --rm -i --user "$(id -u):$(id -g)" --workdir /data
˓→-v "$PWD:/data" ocrmypdf'
By default the Docker image includes English, German, Simplified Chinese, French, Portuguese and Spanish, the
most popular languages for OCRmyPDF users based on feedback. You may add other languages by creating a new
Dockerfile based on the public one.
FROM jbarlow83/ocrmypdf
To install language packs (training data) such as the tessdata_best suite or custom data, you first need to determine the
version of Tesseract data files, which may differ from the Tesseract program version. Use this command to determine
the data file version:
FROM jbarlow83/ocrmypdf
Alternately, you can copy training data into a Docker container as follows:
The OCRmyPDF Docker image includes an example, barebones HTTP web service. The webservice may be launched
as follows:
This will configure the machine to listen on port 5000. On Linux machines this is port 5000 of localhost. On macOS
or Windows machines running Docker, this is port 5000 of the virtual machine that runs your Docker images. You can
find its IP address using the command docker-machine ip.
Unlike command line usage this program will open a socket and wait for connections.
Warning: The OCRmyPDF web service wrapper is intended for demonstration or development. It provides no
security, no authentication, no protection against denial of service attacks, and no load balancing. The default Flask
WSGI server is used, which is intended for development only. The server is single-threaded and so can respond to
only one client at a time. While running OCR, it cannot respond to any other clients.
Clients must keep their open connection while waiting for OCR to complete. This may entail setting a long timeout;
this interface is more useful for internal HTTP API calls.
Unlike the rest of OCRmyPDF, this web service is licensed under the Affero GPLv3 (AGPLv3) since Ghostscript is
also licensed in this way.
In addition to the above, please read our general remarks on using OCRmyPDF as a service.
Advanced features
OCRmyPDF uses unpaper to provide the implementation of the --clean and --clean-final arguments.
unpaper provides a variety of image processing filters to improve images.
By default, OCRmyPDF uses only unpaper arguments that were found to be safe to use on almost all files without
having to inspect every page of the file afterwards. This is particularly true when only --clean is used, since that
instructs OCRmyPDF to only clean the image before OCR and not the final image.
However, if you wish to use the more aggressive options in unpaper, you may use --unpaper-args '...'
to override the OCRmyPDF’s defaults and forward other arguments to unpaper. This option will forward arguments
to unpaper without any knowledge of what that program considers to be valid arguments. The string of arguments
must be quoted as shown in the examples below. No filename arguments may be included. OCRmyPDF will assume
it can append input and output filename of intermediate images to the --unpaper-args string.
In this example, we tell unpaper to expect two pages of text on a sheet (image), such as occurs when two facing
pages of a book are scanned. unpaper uses this information to deskew each independently and clean up the margins
of both.
Warning: Some unpaper features will reposition text within the image. --clean-final is recommended
to avoid this issue.
Warning: Some unpaper features cause multiple input or output files to be consumed or produced. OCRmyPDF
requires unpaper to consume one file and produce one file; errors will result if this assumption is not met.
83
ocrmypdf Documentation, Release 0.1.dev83+g072db75
Note: unpaper uses uncompressed PBM/PGM/PPM files for its intermediate files. For large images or documents,
it can take a lot of temporary disk space.
OCRmyPDF provides many features to control the behavior of the OCR engine, Tesseract.
If a page in a PDF seems to have text, by default OCRmyPDF will exit without modifying the PDF. This is to ensure
that PDFs that were previously OCRed or were “born digital” rather than scanned are not processed.
If --skip-text is issued, then no image processing or OCR will be performed on pages that already have text.
The page will be copied to the output. This may be useful for documents that contain both “born digital” and scanned
content, or to use OCRmyPDF to normalize and convert to PDF/A regardless of their contents.
If --redo-ocr is issued, then a detailed text analysis is performed. Text is categorized as either visible or invisible.
Invisible text (OCR) is stripped out. Then an image of each page is created with visible text masked out. The page
image is sent for OCR, and any additional text is inserted as OCR. If a file contains a mix of text and bitmap images
that contain text, OCRmyPDF will locate the additional text in images without disrupting the existing text. Some PDF
OCR solutions render text as technically printable or visible in some way, perhaps by drawing it and then painting
over it. OCRmyPDF cannot distinguish this type of OCR text from real text, so it will not be “redone”.
If --force-ocr is issued, then all pages will be rasterized to images, discarding any hidden OCR text, rasterizing
any printable text, and flattening form fields or interactive objects into their visual representation. This is useful for
redoing OCR, for fixing OCR text with a damaged character map (text is selectable but not searchable), and destroying
redacted information.
By default, OCRmyPDF permits tesseract to run for three minutes (180 seconds) per page. This is usually more than
enough time to find all text on a reasonably sized page with modern hardware.
If a page is skipped, it will be inserted without OCR. If preprocessing was requested, the preprocessed image layer
will be inserted.
If you want to adjust the amount of time spent on OCR, change --tesseract-timeout. You can also automati-
cally skip images that exceed a certain number of megapixels with --skip-big. (A 300 DPI, 8.5×11” page image
is 8.4 megapixels.)
# Allow 300 seconds for OCR; skip any page larger than 50 megapixels
ocrmypdf --tesseract-timeout 300 --skip-big 50 bigfile.pdf output.pdf
Separate from these settings, Tesseract has internal limits on the size of images it will process. If you issue
--tesseract-downsample-large-images, OCRmyPDF will downsample images to fit Tesseract limits.
(The limits are usually entered only for scanned images of oversized media, such as large maps or blueprints ex-
ceeding 110 cm or 43 inches in either dimension, and at high DPI.)
env \
PATH=/home/user/src/tesseract/api:$PATH \
TESSDATA_PREFIX=/home/user/src/tesseract \
ocrmypdf input.pdf output.pdf
In this example TESSDATA_PREFIX is required to redirect Tesseract to an alternate folder for its “tessdata” files.
You can override tesseract’s default control parameters with a configuration file.
As an example, this configuration will disable Tesseract’s dictionary for current language. Normally the dictionary is
helpful for interpolating words that are unclear, but it may interfere with OCR if the document does not contain many
words (for example, a list of part numbers).
Create a file named “no-dict.cfg” with these contents:
load_system_dawg 0
language_model_penalty_non_dict_word 0
language_model_penalty_non_freq_dict_word 0
then run ocrmypdf as follows (along with any other desired arguments):
Warning: Some combinations of control parameters will break Tesseract or break assumptions that OCRmyPDF
makes about Tesseract’s output.
The sandwich renderer uses Tesseract’s new text-only PDF feature, which produces a PDF page that lays out the
OCR in invisible text. This page is then “sandwiched” onto the original PDF page, allowing lossless application of
OCR even to PDF pages that contain other vector objects.
Currently this is the best renderer for most uses, however it is implemented in Tesseract so OCRmyPDF cannot
influence it. Currently some problematic PDF viewers like Mozilla PDF.js and macOS Preview have problems with
segmenting its text output, and mightrunseveralwordstogether.
When image preprocessing features like --deskew are used, the original PDF will be rendered as a full page and the
OCR layer will be placed on top.
The hocr renderer works with older versions of Tesseract. The image layer is copied from the original PDF page if
possible, avoiding potentially lossy transcoding or loss of other PDF information. If preprocessing is specified, then
the image layer is a new PDF. (You may need to disable PDF/A conversion nad optimization to eliminate all lossy
transformations.)
Unlike sandwich this renderer is implemented within OCRmyPDF; anyone looking to customize how OCR is pre-
sented should look here. A major disadvantage of this renderer is it not capable of correctly handling text outside the
Latin alphabet (specifically, it supports the ISO 8859-1 character). Pull requests to improve the situation are welcome.
Currently, this renderer has the best compatibility with Mozilla’s PDF.js viewer.
This works in all versions of Tesseract.
OCRmyPDF writes all messages to stderr. stdout is reserved for piping output files. stdin is reserved for
piping input files.
The return codes generated by the OCRmyPDF are considered part of the stable user interface. They may be imported
from ocrmypdf.exceptions.
OCRmyPDF normally saves its intermediate results to a temporary folder and deletes this folder when it exits, whether
it succeeded or failed.
If the --keep-temporary-files (-k`) argument is issued on the command line, OCRmyPDF will keep the
temporary folder and print the location, whether it succeeded or failed. An example message is:
The organization of this folder is an implementation detail and subject to change between releases. However the
general organization is that working files on a per page basis have the page number as a prefix (starting with page 1),
an infix indicates the processing stage, and a suffix indicates the file type. Some important files include:
• _rasterize.png - what the input page looks like
• _ocr.png - the file that is sent to Tesseract for OCR; depending on arguments this may differ from the pre-
sentation image
• _pp_deskew.png - the image, after deskewing
• _pp_clean.png - the image, after cleaning with unpaper
• _ocr_tess.pdf - the OCR file; appears as a blank page with invisible text embedded
• _ocr_tess.txt - the OCR text (not necessarily all text on the page, if the page is mixed format)
• fix_docinfo.pdf - a temporary file created to fix the PDF DocumentInfo data structure
• graft_layers.pdf - the rendered PDF with OCR layers grafted on
• pdfa.pdf - graft_layers.pdf after conversion to PDF/A
• pdfa.ps - a PostScript file used by Ghostscript for PDF/A conversion
• optimize.pdf - the PDF generated before optimization
• optimize.out.pdf - the PDF generated by optimization
• origin - the input file
• origin.pdf - the input file or the input image converted to PDF
• images/* - images extracted during the optimization process; here the prefix indicates a PDF object ID not a
page number
Batch processing
This article provides information about running OCRmyPDF on multiple files or configuring it as a service triggered
by file system events.
Consider using the excellent GNU Parallel to apply OCRmyPDF to multiple files at once.
Both parallel and ocrmypdf will try to use all available processors. To maximize parallelism without overloading
your system with processes, consider using parallel -j 2 to limit parallel to running two jobs at once.
This command will run ocrmypdf on all files named *.pdf in the current directory and write them to the previously
created output/ folder. It will not search subdirectories.
The --tag argument tells parallel to print the filename as a prefix whenever a message is printed, so that one can
trace any errors to the file that produced them.
OCRmyPDF automatically repairs PDFs before parsing and gathering information from them.
This will walk through a directory tree and run OCR on all files in place, and printing each filename in between runs:
This only runs one ocrmypdf process at a time. This variation uses find to create a directory list and parallel
to parallelize runs of ocrmypdf, again updating files in place.
89
ocrmypdf Documentation, Release 0.1.dev83+g072db75
With a Docker container, you will need to stream through standard input and output:
Listing 1: misc/batch.py
#!/usr/bin/env python3
# SPDX-FileCopyrightText: 2016 findingorder <https://github.com/findingorder>
# SPDX-License-Identifier: MIT
This script will recursively search a directory for PDF files and run OCR on
them. It will log the results. It runs OCR on every file, even if it already
has text. OCRmyPDF will detect files that already have text.
import logging
import sys
from pathlib import Path
import ocrmypdf
# pylint: disable=logging-format-interpolation
# pylint: disable=logging-not-lazy
script_dir = Path(__file__).parent
if len(sys.argv) > 1:
start_dir = Path(sys.argv[1])
else:
start_dir = Path('.')
if len(sys.argv) > 2:
log_file = Path(sys.argv[2])
else:
log_file = script_dir.with_name('ocr-tree.log')
logging.basicConfig(
level=logging.INFO,
format='%(asctime)s %(message)s',
(continues on next page)
ocrmypdf.configure_logging(ocrmypdf.Verbosity.default)
Synology DiskStations (Network Attached Storage devices) can run the Docker image of OCRmyPDF if the Synology
Docker package is installed. Attached is a script to address particular quirks of using OCRmyPDF on one of these
devices.
At the time this script was written, it only worked for x86-based Synology products. It is not known if it will work on
ARM-based Synology products. Further adjustments might be needed to deal with the Synology’s relatively limited
CPU and RAM.
# pylint: disable=logging-format-interpolation
# pylint: disable=logging-not-lazy
script_dir = os.path.dirname(os.path.realpath(__file__))
timestamp = time.strftime("%Y-%m-%d-%H%M_")
log_file = script_dir + '/' + timestamp + 'ocrmypdf.log'
logging.basicConfig(
level=logging.INFO,
format='%(asctime)s %(message)s',
filename=log_file,
filemode='w',
)
(continues on next page)
If you have thousands of files to work with, contact the author. Consulting work related to OCRmyPDF helps fund
this open source project and all inquiries are appreciated.
OCRmyPDF has a folder watcher called watcher.py, which is currently included in source distributions but not part
of the main program. It may be used natively or may run in a Docker container. Native instances tend to give better
performance. watcher.py works on all platforms.
Users may need to customize the script to meet their requirements.
pip3 install ocrmypdf[watcher]
env OCR_INPUT_DIRECTORY=/mnt/input-pdfs \
OCR_OUTPUT_DIRECTORY=/mnt/output-pdfs \
OCR_OUTPUT_DIRECTORY_YEAR_MONTH=1 \
python3 watcher.py
One could configure a networked scanner or scanning computer to drop files in the watched folder.
The watcher service is included in the OCRmyPDF Docker image. To run it:
docker run \
--volume <path to files to convert>:/input \
--volume <path to store results>:/output \
--volume <path to store processed originals>:/archive \
--env OCR_OUTPUT_DIRECTORY_YEAR_MONTH=1 \
--env OCR_ON_SUCCESS_ARCHIVE=1 \
(continues on next page)
This service will watch for a file that matches /input/\*.pdf, convert it to a OCRed PDF in /output/, and
move the processed original to /archive. The parameters to this image are:
This service relies on polling to check for changes to the filesystem. It may not be suitable for some environments,
such as filesystems shared on a slow network.
A configuration manager such as Docker Compose could be used to ensure that the service is always available.
Listing 3: misc/docker-compose.example.yml
# SPDX-FileCopyrightText: 2022 James R. Barlow
# SPDX-License-Identifier: MIT
---
version: "3.3"
services:
ocrmypdf:
restart: always
container_name: ocrmypdf
image: jbarlow83/ocrmypdf
volumes:
- "/media/scan:/input"
- "/mnt/scan:/output"
environment:
- OCR_OUTPUT_DIRECTORY_YEAR_MONTH=0
user: "<SET TO YOUR USER ID>:<SET TO YOUR GROUP ID>"
entrypoint: python3
command: watcher.py
10.3.3 Caveats
• watchmedo may not work properly on a networked file system, depending on the capabilities of the file system
client and server.
• This simple recipe does not filter for the type of file system event, so file copies, deletes and moves, and directory
operations, will all be sent to ocrmypdf, producing errors in several cases. Disable your watched folder if you
are doing anything other than copying files to it.
• If the source and destination directory are the same, watchmedo may create an infinite loop.
• On BSD, FreeBSD and older versions of macOS, you may need to increase the number of file descriptors to
monitor more files, using ulimit -n 1024 to watch a folder of up to 1024 files.
10.3.4 Alternatives
• On Linux, systemd user services can be configured to automatically perform OCR on a collection of files.
• Watchman is a more powerful alternative to watchmedo.
You can use the Automator app with macOS, to create a Workflow or Quick Action. Use a Run Shell Script action in
your workflow. In the context of Automator, the PATH may be set differently your Terminal’s PATH; you may need to
explicitly set the PATH to include ocrmypdf. The following example may serve as a starting point:
You may customize the command sent to ocrmypdf.
Online deployments
OCRmyPDF is designed to be used as a command line tool, but it can be used in a web service. This document
describes some considerations for doing so.
A basic web service implementation is provided in the source code repository, as misc/webservice.py. It is only
demonstration quality and is not intended for production use.
OCRmyPDF is not designed for use as a public web service where a malicious user could upload a chosen PDF. In
particular, it is not necessarily secure against PDF malware or PDFs that cause denial of service. For further discussino
of security, see security.
OCRmyPDF relies on Ghostscript, and therefore, if deployed online one should be prepared to comply with
Ghostscript’s Affero GPL license, and any other licenses.
Setting aside these concerns, a side effect of OCRmyPDF is that it may incidentally sanitize PDFs containing certain
types of malware. It repairs the PDF with pikepdf/libqpdf, which could correct malformed PDF structures that are part
of an attack. When PDF/A output is selected (the default), the input PDF is partially reconstructed by Ghostscript.
When --force-ocr is used, all pages are rasterized and reconverted to PDF, which could remove malware in
embedded images.
OCRmyPDF will attempt to use all available CPUs and storage, so executing nice ocrmypdf or limiting the
number of jobs with the --jobs argument may ensure the server remains responsive. Another option would be to run
OCRmyPDF jobs inside a Docker container, a virtual machine, or a cloud instance, which can impose its own limits
on CPU usage and be terminated “from orbit” if it fails to complete.
OCRmyPDF will use a large amount of temporary storage for its work, proportional to the total number of pix-
els needed to rasterize the PDF. The raster image of a 8.5×11” color page at 300 DPI takes 25 MB uncompressed;
OCRmyPDF saves its intermediates as PNG, but that still means it requires about 9 MB per intermediate based on
97
ocrmypdf Documentation, Release 0.1.dev83+g072db75
average compression ratios. Multiple intermediates per page are also required, depending on the command line given.
A rule of thumb would be to allow 100 MB of temporary storage per page in a file – meaning that a small cloud servers
or small VM partitions should be provisioned with plenty of extra space, if say, a 500 page file might be sent.
To change the temporary directory, see Changing temporary storage location.
On Amazon Web Services or other cloud vendors, consider setting your temporary directory to empheral storage.
11.3 Timeouts
To prevent excessively long OCR jobs consider setting --tesseract-timeout and/or --skip-big arguments.
--skip-big is particularly helpful if your PDFs include documents such as reports on standard page sizes with
large images attached - often large images are not worth OCR’ing anyway.
If you are looking for a full document management system, consider paperless-ngx, which is a web application that
uses OCRmyPDF to automatically OCR and archive documents.
The author also provides professional services that include OCR and building databases around PDFs, and is happy to
provide consultation.
Abbyy Cloud OCR is viable commercial alternative with a web services API. Amazon Textract, Google Cloud Vision,
and Microsoft Azure Computer Vision provide advanced OCR but have less PDF rendering capability.
Performance
Some users have noticed that current versions of OCRmyPDF do not run as quickly as some older versions (specifically
6.x and older). This is because OCRmyPDF added image optimization as a postprocessing step, and it is enabled by
default.
12.1 Speed
If running OCRmyPDF quickly is your main goal, you can use settings such as:
• --optimize 0 to disable file size optimization
• --output-type pdf to disable PDF/A generation
• --fast-web-view 0 to disable fast web view optimization
• --skip-big to skip large images, if some pages have large images
You can also avoid:
• --force-ocr
• Image preprocessing
99
ocrmypdf Documentation, Release 0.1.dev83+g072db75
OCRmyPDF should only be used on PDFs you trust. It is not designed to protect you against malware.
Recognizing that many users have an interest in handling PDFs and applying OCR to PDFs they did not generate
themselves, this article discusses the security implications of PDFs and how users can protect themselves.
The disclaimer applies: this software has no warranties of any kind.
PDF is a rich, complex file format. The official PDF 1.7 specification, ISO 32000:2008, is hundreds of pages long and
references several annexes each of which are similar in length. PDFs can contain video, audio, XML, JavaScript and
other programming, and forms. In some cases, they can open internet connections to pre-selected URLs. All of these
are possible attack vectors.
In short, PDFs may contain viruses.
If you do not trust a PDF or its source, do not open it or use OCRmyPDF on it. Consider using a Docker container or
virtual machine to isolate an untrusted PDF from your system.
OCRmyPDF must open and interpret your PDF in order to insert an OCR layer. First, it runs all PDFs through
pikepdf, a library based on QPDF, a program that repairs PDFs with syntax errors. This is done because, in the
author’s experience, a significant number of PDFs in the wild, especially those created by scanners, are not well-
formed files. QPDF makes it more likely that OCRmyPDF will succeed, but offers no security guarantees. QPDF is
also used to split the PDF into single page PDFs.
Finally, OCRmyPDF rasterizes each page of the PDF using Ghostscript in -dSAFER mode.
Depending on the options specified, OCRmyPDF may graft the OCR layer into the existing PDF or it may essentially
reconstruct (“re-fry”) a visually identical PDF that may be quite different at the binary level. That said, OCRmyPDF
is not a tool designed for sanitizing PDFs.
101
ocrmypdf Documentation, Release 0.1.dev83+g072db75
Password protected PDFs usually have two passwords, and owner and user password. When the user password is set
to empty, PDF readers will open the file automatically and mark it as “(SECURED)”. Password security can also
request certain restrictions on the PDF, but anyone can remove these restrictions if they have either the owner or user
password. Passwords mainly present a barrier for casual users.
OCRmyPDF cannot remove passwords from PDFs. If you want to remove a password from a PDF, you must use other
software, such as qpdf.
If the owner and user password are set, a password is required for qpdf. If only the owner password is set, then the
password can be stripped, even if one does not have the owner password. To remove the password from a using QPDF,
use:
Many programs exist which are capable of inserting an image of someone’s signature. On its own, this offers no
security guarantees. It is trivial to remove the signature image and apply it to other files. This practice offers no real
security.
Important documents can be digitally signed and certified to attest to their authorship, approval or execu-
tion of a legal agreement. OCRmyPDF will detect signed PDFs and will not modify them, unless the
--invalidate-digital-signatures option is used, which will invalidate any signatures. (The signature
may still be present in the PDF if opened, but PDF readers will not validate it.)
A digital signature adds a cryptographic hash of the document to the document, so tamper protection is provided. That
also precludes OCRmyPDF from modifying the document and preserving the signature.
Digital signatures are not the same as a signature image. A digital signature is a cryptographic hash of the document
that is encrypted with the author’s private key. The signature is decrypted with the author’s public key. The public key
is usually distributed by a certificate authority. The signature is then verified by the PDF reader. If the document is
modified, the signature will be invalidated.
PDFs can be encrypted with a certificate. This is a more secure form of encryption than a password. The certificate
is usually issued by a certificate authority. A certificate is used to encrypt the document using the public key for the
benefit of a specific recipient who possesses the private key.
OCRmyPDF cannot open certificate-encrypted PDFs. If you have the certificate, you can use other PDF software,
such as Acrobat, to decrypt the PDF.
You ran ocrmypdf on a file that already contains printable text or a hidden OCR text layer (it can’t quite tell the
difference). You probably don’t want to do this, because the file is already searchable.
As the error message suggests, your options are:
• ocrmypdf --force-ocr to rasterize all vector content and run OCR on the images. This is useful if a
previous OCR program failed, or if the document contains a text watermark.
• ocrmypdf --skip-text to skip OCR and other processing on any pages that contain text. Text pages will
be copied into the output PDF without modification.
• ocrmypdf --redo-ocr to scan the file for any existing OCR (non-printing text), remove it, and do OCR
again. This is one way to take advantage of improvements in OCR accuracy. Printable vector text is excluded
from OCR, so this can be used on files that contain a mix of digital and scanned files.
OCRmyPDF checks files with pikepdf, a library that in turn uses libqpdf to fixes errors in PDFs, before it tries to work
on them. In most cases this happens because the PDF is corrupt and truncated (incomplete file copying) and not much
can be done.
You can try rewriting the file with Ghostscript:
103
ocrmypdf Documentation, Release 0.1.dev83+g072db75
OCRmyPDF originated as a command line program and continues to have this legacy, but parts of it can be imported
and used in other Python applications.
Some applications may want to consider running ocrmypdf from a subprocess call anyway, as this provides isolation
of its activities.
15.1 Example
OCRmyPDF provides one high-level function to run its main engine from an application. The parameters are sym-
metric to the command line arguments and largely have the same functions.
import ocrmypdf
With some exceptions, all of the command line arguments are available and may be passed as equivalent keywords.
A few differences are that verbose and quiet are not available. Instead, output should be managed by configuring
logging.
The ocrmypdf.ocr() function runs OCRmyPDF similar to command line execution. To do this, it will:
• create a monitoring thread
• create worker processes (on Linux, forking itself; on Windows and macOS, by spawning)
• manage the signal flags of its worker processes
• execute other subprocesses (forking and executing other programs)
105
ocrmypdf Documentation, Release 0.1.dev83+g072db75
The Python process that calls ocrmypdf.ocr() must be sufficiently privileged to perform these actions.
There currently is no option to manage how jobs are scheduled other than the argument jobs= which will limit the
number of worker processes.
Creating a child process to call ocrmypdf.ocr() is suggested. That way your application will survive and remain
interactive even if OCRmyPDF fails for any reason.
Programs that call ocrmypdf.ocr() should also install a SIGBUS signal handler (except on Windows), to raise an
exception if access to a memory mapped file fails. OCRmyPDF may use memory mapping.
ocrmypdf.ocr() will take a threading lock to prevent multiple runs of itself in the same Python interpreter process.
This is not thread-safe, because of how OCRmyPDF’s plugins and Python’s library import system work. If you need
to parallelize OCRmyPDF, use processes.
Warning: On Windows and macOS, the script that calls ocrmypdf.ocr() must be protected by an “ifmain”
guard (if __name__ == '__main__'). If you do not take at least one of these steps, process semantics will
prevent OCRmyPDF from working correctly.
15.1.2 Logging
OCRmyPDF will log under loggers named ocrmypdf. In addition, it imports pdfminer and PIL, both of which
post log messages under those logging namespaces.
You can configure the logging as desired for your application or call ocrmypdf.configure_logging() to
configure logging the same way OCRmyPDF itself does. The command line parameters such as --quiet and
--verbose have no equivalents in the API; you must use the provided configuration function or do configuration in
a way that suits your use case.
OCRmyPDF uses the tqdm package to implement its progress bars. ocrmypdf.configure_logging() will set
up logging output to sys.stderr in a way that is compatible with the display of the progress bar. Use ocrmypdf.
ocr(...progress_bar=False) to disable the progress bar.
15.1.4 Exceptions
OCRmyPDF may throw standard Python exceptions, ocrmypdf.exceptions.* exceptions, some exceptions
related to multiprocessing, and KeyboardInterrupt. The parent process should provide an exception handler.
OCRmyPDF will clean up its temporary files and worker processes automatically when an exception occurs.
Programs that call OCRmyPDF should consider trapping KeyboardInterrupt so that they allow OCR to terminate with
the whole program terminating.
When OCRmyPDF succeeds conditionally, it returns an integer exit code.
15.1.5 Reference
Plugins
The key words “MUST”, “MUST NOT”, “REQUIRED”, “SHALL”, “SHALL NOT”, “SHOULD”,
“SHOULD NOT”, “RECOMMENDED”, “MAY”, and “OPTIONAL” in this document are to be inter-
preted as described in RFC 2119.
You can use plugins to customize the behavior of OCRmyPDF at certain points of interest.
Currently, it is possible to:
• add new command line arguments
• override the decision for whether or not to perform OCR on a particular file
• modify the image is about to be sent for OCR
• modify the page image before it is converted to PDF
• replace the Tesseract OCR with another OCR engine that has similar behavior
• replace Ghostscript with another PDF to image converter (rasterizer) or PDF/A generator
OCRmyPDF plugins are based on the Python pluggy package and conform to its conventions. Note that: plugins
installed with as setuptools entrypoints are not checked currently, because OCRmyPDF assumes you may not want to
enable plugins for all files.
Script plugins may be called from the command line, by specifying the name of a file. Script plugins may be convenient
for informal or “one-off” plugins, when a certain batch of files needs a special processing step for example.
Multiple plugins may be installed by issuing the --plugin argument multiple times.
111
ocrmypdf Documentation, Release 0.1.dev83+g072db75
Installed plugins may be installed into the same virtual environment as OCRmyPDF is installed into. They may be
invoked using Python standard module naming. If you are intending to distribute a plugin, please package it.
OCRmyPDF does not automatically import plugins, because the assumption is that plugins affect different files differ-
ently and you may not want them activated all the time. The command line or ocrmypdf.ocr(plugin='...')
must call for them.
Third parties that wish to distribute packages for ocrmypdf should package them as packaged plugins, and these
modules should begin with the name ocrmypdf_ similar to pytest packages such as pytest-cov (the package)
and pytest_cov (the module).
Note: We recommend plugin authors name their plugins with the prefix ocrmypdf- (for the package name on PyPI)
and ocrmypdf_ (for the module), just like pytest plugins. At the same time, please make it clear that your package
is not official.
You can also create a plugin that OCRmyPDF will always automatically load if both are installed in the same virtual
environment, using a setuptools entrypoint.
Your package’s pyproject.toml would need to contain the following, for a plugin named
ocrmypdf-exampleplugin:
[project]
name = "ocrmypdf-exampleplugin"
[project.entry-points."ocrmypdf"]
exampleplugin = "exampleplugin.pluginmodule"
# equivalent setup.cfg
[options.entry_points]
ocrmypdf =
exampleplugin = exampleplugin.pluginmodule
OCRmyPDF generally uses multiple worker processes. When a new worker is started, Python will import all plugins
again, including all plugins that were imported earlier. This means that the global state of a plugin in one worker will
not be shared with other workers. As such, plugin hook implementations should be stateless, relying only on their
inputs. Hook implementations may use their input parameters to to obtain a reference to shared state prepared by
another hook implementation. Plugins must expect that other instances of the plugin will be running simultaneously.
The context object that is passed to many hooks can be used to share information about a file being worked
on. Plugins must write private, plugin-specific data to a subfolder named {options.work_folder}/
ocrmypdf-plugin-name. Plugins MAY read and write files in options.work_folder, but should be aware
that their semantics are subject to change.
OCRmyPDF will delete options.work_folder when it has finished OCRing a file, unless invoked with
--keep-temporary-files.
The documentation for some plugin hooks contain a detailed description of the execution context in which they will
be called.
Plugins should be prepared to work whether executed in worker threads or worker processes. Generally, OCRmyPDF
uses processes, but has a semi-hidden threaded argument that simplifies debugging.
A plugin may provide the following hooks. Hooks must be decorated with ocrmypdf.hookimpl, for example:
@hookimpl
def add_options(parser):
pass
The following is a complete list of hooks that are available, and when they are called.
Note on firstresult hooks
If multiple plugins install implementations for this hook, they will be called in the reverse of the order in which they
are installed (i.e., last plugin wins). When each hook implementation is called in order, the first implementation that
returns a value other than None will “win” and prevent execution of all other hooks. As such, you cannot “chain” a
series of plugin filters together in this way. Instead, a single hook implementation should be responsible for any such
chaining operations.
16.6 Examples
• OCRmyPDF’s test suite contains several plugins that are used to simulate certain test conditions.
• ocrmypdf-papermerge is a production plugin that integrates OCRmyPDF and the Papermerge document man-
agement system.
Note: This hook will be called from the main process, and may modify global state before child worker
processes are forked.
Note: This hook will be called from the main process, and may modify global state before child worker
processes are forked.
Note: This hook will be called from the main process, and may modify global state before child worker
processes are forked.
ocrmypdf.pluginspec.get_executor(progressbar_class) → ocrmypdf._concurrent.Executor
Called to obtain an object that manages parallel execution.
This may be used to replace OCRmyPDF’s default parallel execution system with a third party alternative. For
example, you could make OCRmyPDF run in a distributed environment.
OCRmyPDF’s executors are analogous to the standard Python executors in conconcurrent.futures, but
they do not work the same way. Executors may be reused for different, unrelated batch operations, since all of
the context for a given job are passed to Executor.__call__().
Should be of type Executor or otherwise conforming to the protocol of that call.
Note: This hook will be called from the main process, and may modify global state before child worker
processes are forked.
ocrmypdf.pluginspec.get_progressbar_class()
Called to obtain a class that can be used to monitor progress.
A progress bar is assumed, but this could be used for any type of monitoring.
The class should follow a tqdm-like protocol. Calling the class should return a new progress bar object, which
is activated with __enter__ and terminated __exit__. An update method is called whenever the progress
bar is updated. Progress bar objects will not be reused; a new one will be created for each group of tasks.
The progress bar is held in the main process/thread and not updated by child process/threads. When a child
notifies the parent of completed work, the parent updates the progress bar.
The arguments are the same as tqdm accepts.
Progress bars should never write to sys.stdout, or they will corrupt the output if OCRmyPDF writes a PDF
to standard output.
The type of events that OCRmyPDF reports to a progress bar may change in minor releases.
Here is how OCRmyPDF will use the progress bar:
Example
Note: This hook will be called from the main process, and may modify global state before child worker
processes are forked.
Note: This hook will be called from child processes. Modifying global state will not affect the main process or
other child processes.
Generally speaking, the output image should be a faithful representation of of the input image. You may change
the pixel width and height of the the input image, but you must not change the aspect ratio, and you must
calculate the DPI of the output image based on the new pixel width and height or the OCR text layer will be
misaligned with the visual position.
The built-in Tesseract OCR engine uses this hook itself to downsample very large images to fit its constraints.
Note: This hook will be called from child processes. Modifying global state will not affect the main process or
other child processes.
Note: This hook will be called from child processes. Modifying global state will not affect the main process or
other child processes.
The provider of the hook implementation is responsible for ensuring that the OCR text layer is aligned with the
PDF produced here, or text misalignment will result.
Currently this function must produce a single page PDF or the pipeline will fail. If the intent is to remove the
PDF, then create a single page empty PDF.
Parameters
• page – Context for this page.
• image_filename – Filename of the input image used to create output_pdf, for “refer-
ence” if recreating the output_pdf entirely.
• output_pdf – The previous created output_pdf.
Returns output_pdf
Note: This hook will be called from child processes. Modifying global state will not affect the main process or
other child processes.
ocrmypdf.pluginspec.get_ocr_engine() → ocrmypdf.pluginspec.OcrEngine
Returns an OcrEngine to use for processing this file.
The OcrEngine may be instantiated multiple times, by both the main process and child process.
class ocrmypdf.pluginspec.OcrEngine
A class representing an OCR engine with capabilities similar to Tesseract OCR.
This could be used to create a plugin for another OCR engine instead of Tesseract OCR.
__str__()
Returns name of OCR engine and version.
This is used when OCRmyPDF wants to mention the name of the OCR engine to the user, usually in an
error message.
static creator_tag(options: argparse.Namespace) → str
Returns the creator tag to identify this software’s role in creating the PDF.
This tag will be inserted in the XMP metadata and DocumentInfo dictionary as appropriate. Ideally you
should include the name of the OCR engine and its version. The text should not contain line breaks. This
is to help developers like yourself identify the software that produced this file.
OCRmyPDF will always prepend its name to this value.
static generate_hocr(input_file: pathlib.Path, output_hocr: pathlib.Path, output_text: path-
lib.Path, options: argparse.Namespace) → None
Called to produce a hOCR file and sidecar text file.
• pdf_version – The minimum PDF version that the output file should be. At its own
discretion, the PDF/A generator may raise the version, but should not lower it.
• pdfa_part – The desired PDF/A compliance level, such as '2B'.
• progressbar_class – The class of a progress bar with a tqdm-like API. An in-
stance of this class will be initialized when PDF/A conversion begins, using instance =
progressbar_class(total: int, desc: str, unit:str), defining the
number of work units, a user-visible description, and the name of the work units (“page”).
Then instance.update() will be called when a work unit is completed. If None, no
progress information is reported.
• stop_on_soft_error – If there is an “soft error” such that PDF/A generation can pro-
ceed and produce a valid PDF/A, but output may be invalid or may not visually resemble the
original, the implementer of this hook should raise a detailed exception. If False, continue
processing and report by logging it. If the hook cannot proceed, it should always raise an
exception, regardless of this setting.
Returns Path – If successful, the hook should return output_file.
Note: Before version 15.0.0, the context was not provided and compression was provided instead.
Plugins should now read the context object to determine if compression is requested.
See also:
https://github.com/tqdm/tqdm
ocrmypdf.pluginspec.is_optimization_enabled(context: ocrmypdf._jobcontext.PdfContext)
→ bool
For a given PdfContext, OCRmyPDF asks the plugin if optimization is enabled.
An optimization plugin might be installed and active but could be disabled by user settings.
If this returns False, OCRmyPDF will take certain actions to finalize the PDF.
Returns True if the plugin’s optimization is enabled.
API Reference
This page summarizes the rest of the public API. Generally speaking this should mainly of interest to plugin develop-
ers.
17.1 ocrmypdf
123
ocrmypdf Documentation, Release 0.1.dev83+g072db75
get_page_contexts() → Iterator[ocrmypdf._jobcontext.PageContext]
Get all PageContext for this PDF.
get_path(name: str) → pathlib.Path
Generate a Path for an intermediate file involved in processing.
The path will be in a temporary folder that is common for all processing of this particular PDF.
options = None
The specified options for processing this PDF.
origin = None
The filename of the original input file.
pdfinfo = None
Detailed data for this PDF.
plugin_manager = None
PluginManager for processing the current PDF.
17.2 ocrmypdf.exceptions
OCRmyPDF’s exceptions.
exception ocrmypdf.exceptions.BadArgsError
Invalid arguments on the command line or API.
exit_code = 1
exception ocrmypdf.exceptions.DigitalSignatureError
PDF has a digital signature.
exit_code = 2
message = 'Input PDF has a digital signature. OCR would alter the document,\ninvalidati
exception ocrmypdf.exceptions.DpiError
Missing information about input image DPI.
exit_code = 2
exception ocrmypdf.exceptions.EncryptedPdfError
Input PDF is encrypted.
exit_code = 8
message = "Input PDF is encrypted. The encryption must be removed to\nperform OCR.\n\nF
class ocrmypdf.exceptions.ExitCode
OCRmyPDF’s exit codes.
already_done_ocr = 6
bad_args = 1
child_process_error = 7
ctrl_c = 130
encrypted_pdf = 8
file_access_error = 5
input_file = 2
invalid_config = 9
invalid_output_pdf = 4
missing_dependency = 3
ok = 0
other_error = 15
pdfa_conversion_failed = 10
exception ocrmypdf.exceptions.ExitCodeException
An exception which should return an exit code with sys.exit().
exit_code = 15
message = ''
exception ocrmypdf.exceptions.InputFileError
Something is wrong with the input file.
exit_code = 2
exception ocrmypdf.exceptions.MissingDependencyError
A third-party dependency is missing.
exit_code = 3
exception ocrmypdf.exceptions.OutputFileAccessError
Cannot access the intended output file path.
exit_code = 5
exception ocrmypdf.exceptions.PriorOcrFoundError
This file already has OCR.
exit_code = 6
exception ocrmypdf.exceptions.SubprocessOutputError
A subprocess returned an unexpected error.
exit_code = 7
exception ocrmypdf.exceptions.TesseractConfigError
Tesseract config can’t be parsed.
exit_code = 9
message = 'Error occurred while parsing a Tesseract configuration file'
exception ocrmypdf.exceptions.UnsupportedImageFormatError
The image format is not supported.
exit_code = 2
17.3 ocrmypdf.helpers
Support functions.
exception ocrmypdf.helpers.NeverRaise
An exception that is never raised.
class ocrmypdf.helpers.Resolution(x: T, y: T)
The number of pixels per inch in each 2D direction.
Resolution objects are considered “equal” for == purposes if they are equal to a reasonable tolerance.
flip_axis() → ocrmypdf.helpers.Resolution[~T][T]
Return a new Resolution object with x and y swapped.
is_finite
True if both x and y are finite numbers.
is_square
True if the resolution is square (x == y).
round(ndigits: int) → ocrmypdf.helpers.Resolution
Round to ndigits after the decimal point.
take_max(vals: Iterable[Any], yvals: Iterable[Any] | None = None) → Resolution
Return a new Resolution object with the maximum resolution of inputs.
take_min(vals: Iterable[Any], yvals: Iterable[Any] | None = None) → Resolution
Return a new Resolution object with the minimum resolution of inputs.
to_int() → ocrmypdf.helpers.Resolution[int][int]
Round to nearest integer.
to_scalar() → float
Return the harmonic mean of x and y as a 1D approximation.
In most cases, Resolution is 2D, but typically it is “square” (x == y) and can be approximated as a single
number. When not square, the harmonic mean is used to approximate the 2D resolution as a single number.
ocrmypdf.helpers.available_cpu_count() → int
Returns number of CPUs in the system.
ocrmypdf.helpers.check_pdf(input_file: pathlib.Path) → bool
Check if a PDF complies with the PDF specification.
Checks for proper formatting and proper linearization. Uses pikepdf (which in turn, uses QPDF) to perform the
checks.
ocrmypdf.helpers.clamp(n: T, smallest: T, largest: T) → T
Clamps the value of n to between smallest and largest.
ocrmypdf.helpers.is_file_writable(test_file: os.PathLike) → bool
Intentionally racy test if target is writable.
We intend to write to the output file if and only if we succeed and can replace it atomically. Before doing the
OCR work, make sure the location is writable.
ocrmypdf.helpers.is_iterable_notstr(thing: Any) → bool
Is this is an iterable type, other than a string?
ocrmypdf.helpers.monotonic(seq: Sequence) → bool
Does this sequence increase monotonically?
ocrmypdf.helpers.page_number(input_file: os.PathLike) → int
Get one-based page number implied by filename (000002.pdf -> 2).
ocrmypdf.helpers.pikepdf_enable_mmap() → None
Enable pikepdf mmap.
ocrmypdf.helpers.remove_all_log_handlers(logger: logging.Logger) → None
Remove all log handlers, usually used in a child process.
The child process inherits the log handlers from the parent process when a fork occurs. Typically we want to
remove all log handlers in the child process so that the child process can set up a single queue handler to forward
log messages to the parent process.
ocrmypdf.helpers.safe_symlink(input_file: os.PathLike, soft_link_name: os.PathLike) → None
Create a symbolic link at soft_link_name, which references input_file.
Think of this as copying input_file to soft_link_name with less overhead.
Use symlinks safely. Self-linking loops are prevented. On Windows, file copy is used since symlinks may
require administrator privileges. An existing link at the destination is removed.
ocrmypdf.helpers.samefile(file1: os.PathLike, file2: os.PathLike) → bool
Return True if two files are the same file.
Attempts to account for different relative paths to the same file.
17.4 ocrmypdf.hocrtransform
17.5 ocrmypdf.pdfa
References
17.6 ocrmypdf.quality
17.7 ocrmypdf.subprocess
Design notes
PyTesseract is a Python wrapper around the Tesseract OCR engine. When OCRmyPDF was first written, PyTesseract
used ABI bindings to call the Tesseract library. This was not a good fit for OCRmyPDF because ABI bindings can be
fragile.
PyTesseract has since evolved calling the Tesseract executable, abandoning the ABI approach and using the CLI
instead, just like OCRmyPDF does. If it were written from scratch today, OCRmyPDF might use PyTesseract.
PyTesseract has more features don’t particularly need PDF output, but less features than OCRmyPDF’s API for creat-
ing PDFs.
OCRmyPDF uses a custom concurrent executor which can support either threads or processes with the same inter-
face. This is useful because OCRmyPDF can use either threads or processes to parallelize work, whichever is more
appropriate for the task at hand.
The interface is currently private and subject to change. In particular, if experiments with asyncio and anyio are
successful, the interface will change.
131
ocrmypdf Documentation, Release 0.1.dev83+g072db75
Contributing guidelines
Please open a new issue to discuss or propose a major change. Not only is it fun to discuss big ideas, but we might save
each other’s time too. Perhaps some of the work you’re contemplating is already half-done in a development branch.
We use PEP8, black for code formatting and ruff for everything else. The settings for these programs are in
pyproject.toml and setup.cfg. Pull requests should follow the style guide. One difference we use from
“black” style is that strings shown to the user are always in double quotes (") and strings for internal uses are in single
quotes (').
19.3 Tests
New features should come with tests that confirm their correctness.
If you are proposing a change that will require a new dependency, we prefer dependencies that are already packaged
by Debian or Red Hat. This makes life much easier for our downstream package maintainers. A package that is only
available on PyPI or GitHub, and not more widely packaged, may not be accepted.
We are unlikely to accept a dependency on CUDA or other GPU-based libraries, because these are still difficult to
package and install on many systems. We recommend implementing these changes as plugins.
133
ocrmypdf Documentation, Release 0.1.dev83+g072db75
Python dependencies must also be license-compatible. GPLv3 or AGPLv3 are likely incompatible with the project’s
license, but LGPLv3 is compatible.
OCRmyPDF uses several external programs (Tesseract, Ghostscript and others) for its functionality. In general we
prefer to avoid adding new external programs, and if we are to add external programs, we prefer those that are already
packaged by Debian or Red Hat.
19.6 Plugins
Some new features may be a good fit for a plugin. Plugins are a way to add features to OCRmyPDF without adding
them to the core program. Plugins are installed separately from OCRmyPDF. They are written in Python and can be
installed from PyPI. See the plugin documentation.
We are happy to link users to your plugin from the documentation.
The program/project is OCRmyPDF and the name of the executable or library is ocrmypdf.
For contributions over 10 lines of code, please add your name to list of copyright holders for that file. The core
program is licensed under MPL-2.0, test files and documentation under CC-BY-SA 4.0, and miscellaneous files under
MIT, with a few minor exceptions. Please contribute only content that you own or have the right to contribute under
these licenses.
Maintainer notes
This is for those who package OCRmyPDF for downstream use. (Thank you for your hard work.)
OCRmyPDF has been ported to many platforms already. If you are interesting in porting to a new platform, check
with Repology to see the status of that platform.
pikepdf, created by the same author, is a mixed Python and C++14 package with much stiffer build requirements. If
you want to use OCRmyPDF on some novel platform or distribution, first make sure you can package pikepdf.
Note that we have non-Python dependencies. In particular, OCRmyPDF requires Ghostscript and Tesseract OCR to be
installed and needs to be able to locate their binaries on the system PATH. On Windows, OCRmyPDF will also check
the registry for their locations.
Tesseract OCR relies on SIMD for performance and only has proper support for this on ARM and x86_64. Perfor-
mance may be poor on other processor architectures.
OCRmyPDF uses setuptools-scm for versioning, which derives the version from Git as a single source of truth. This
may be unsuitable for some distributions, e.g. to indicate that your distribution modifies OCRmyPDF in some way.
You can patch the __version__ variable in src/ocrmypdf/_version.py if necessary.
135
ocrmypdf Documentation, Release 0.1.dev83+g072db75
20.1.4 jbig2enc
OCRmyPDF will use jbig2enc, a JBIG2 encoder, if one can be found. Some distributions have shied away from
packaging JBIG2 because it contains patented algorithms, but all patents have expired since 2017. If possible, consider
packaging it too to improve OCRmyPDF’s compression.
Please ensure that command line completions are installed, as described in the installation documentation.
If you maintain a Linux distribution that supports 32-bit x86 or ARM, OCRmyPDF should continue to work as long
as all of its dependencies continue to be available in 32-bit form. Please note we do not test on 32-bit platforms.
• genindex
• modindex
• search
137
ocrmypdf Documentation, Release 0.1.dev83+g072db75
o
ocrmypdf.exceptions, 124
ocrmypdf.hocrtransform, 127
ocrmypdf.pdfa, 128
ocrmypdf.quality, 129
ocrmypdf.subprocess, 129
139
ocrmypdf Documentation, Release 0.1.dev83+g072db75
141
ocrmypdf Documentation, Release 0.1.dev83+g072db75
F invalid_output_pdf
file_access_error (ocrmypdf.exceptions.ExitCode (ocrmypdf.exceptions.ExitCode attribute),
attribute), 124 125
file_claims_pdfa() (in module ocrmypdf.pdfa), is_optimization_enabled() (in module
128 ocrmypdf.pluginspec), 121
filter_ocr_image() (in module
ocrmypdf.pluginspec), 116 L
filter_page_image() (in module languages() (ocrmypdf.pluginspec.OcrEngine static
ocrmypdf.pluginspec), 117 method), 119
filter_pdf_page() (in module
ocrmypdf.pluginspec), 117 M
measure_words_matched()
G (ocrmypdf.quality.OcrQualityDictionary
generate_hocr() (ocrmypdf.pluginspec.OcrEngine method), 129
static method), 118 message (ocrmypdf.exceptions.DigitalSignatureError
generate_pdf() (ocrmypdf.pluginspec.OcrEngine attribute), 124
static method), 118 message (ocrmypdf.exceptions.EncryptedPdfError at-
generate_pdfa() (in module ocrmypdf.pluginspec), tribute), 124
119 message (ocrmypdf.exceptions.ExitCodeException at-
generate_pdfa_ps() (in module ocrmypdf.pdfa), tribute), 125
128 message (ocrmypdf.exceptions.TesseractConfigError
get_deskew() (ocrmypdf.pluginspec.OcrEngine attribute), 125
static method), 119 missing_dependency
get_executor() (in module ocrmypdf.pluginspec), (ocrmypdf.exceptions.ExitCode attribute),
114 125
get_logging_console() (in module MissingDependencyError, 125
ocrmypdf.pluginspec), 114
get_ocr_engine() (in module O
ocrmypdf.pluginspec), 118 ocr() (in module ocrmypdf ), 107
get_orientation() OcrEngine (class in ocrmypdf.pluginspec), 118
(ocrmypdf.pluginspec.OcrEngine static ocrmypdf.exceptions (module), 124
method), 119 ocrmypdf.hocrtransform (module), 127
get_page_contexts() (ocrmypdf.PdfContext ocrmypdf.pdfa (module), 128
method), 123 ocrmypdf.quality (module), 129
get_path() (ocrmypdf.PageContext method), 123 ocrmypdf.subprocess (module), 129
get_path() (ocrmypdf.PdfContext method), 124 OcrQualityDictionary (class in
get_progressbar_class() (in module ocrmypdf.quality), 129
ocrmypdf.pluginspec), 115 ok (ocrmypdf.exceptions.ExitCode attribute), 125
get_version() (in module ocrmypdf.subprocess), optimize_pdf() (in module ocrmypdf.pluginspec),
129 120
options (ocrmypdf.PageContext attribute), 123
H options (ocrmypdf.PdfContext attribute), 124
HocrTransform (class in ocrmypdf.hocrtransform), OrientationConfidence (class in
127 ocrmypdf.pluginspec), 119
HocrTransformError, 128 origin (ocrmypdf.PageContext attribute), 123
origin (ocrmypdf.PdfContext attribute), 124
I other_error (ocrmypdf.exceptions.ExitCode at-
initialize() (in module ocrmypdf.pluginspec), 113 tribute), 125
input_file (ocrmypdf.exceptions.ExitCode attribute), OutputFileAccessError, 125
124
InputFileError, 125
P
invalid_config (ocrmypdf.exceptions.ExitCode at- PageContext (class in ocrmypdf ), 123
tribute), 124 pageinfo (ocrmypdf.PageContext attribute), 123
pageno (ocrmypdf.PageContext attribute), 123
142 Index
ocrmypdf Documentation, Release 0.1.dev83+g072db75
Q
quiet (ocrmypdf.Verbosity attribute), 108
R
rasterize_pdf_page() (in module
ocrmypdf.pluginspec), 116
Rect (class in ocrmypdf.hocrtransform), 128
replace_unsupported_chars()
(ocrmypdf.hocrtransform.HocrTransform
class method), 127
run() (in module ocrmypdf.subprocess), 129
run_polling_stderr() (in module
ocrmypdf.subprocess), 130
S
SubprocessOutputError, 125
T
TesseractConfigError, 125
to_pdf() (ocrmypdf.hocrtransform.HocrTransform
method), 127
U
UnsupportedImageFormatError, 125
V
validate() (in module ocrmypdf.pluginspec), 115
Verbosity (class in ocrmypdf ), 108
version() (ocrmypdf.pluginspec.OcrEngine static
method), 119
X
x1 (ocrmypdf.hocrtransform.Rect attribute), 128
x2 (ocrmypdf.hocrtransform.Rect attribute), 128
Y
y1 (ocrmypdf.hocrtransform.Rect attribute), 128
y2 (ocrmypdf.hocrtransform.Rect attribute), 128
Index 143