From acbf61be0598988e0ea33bc467e83a0f3fd7ec64 Mon Sep 17 00:00:00 2001
From: Patrick Jentsch
Date: Mon, 15 Mar 2021 12:45:05 +0100
Subject: [PATCH] Cleanup and make use of globbing for input files for
binarization and ocr
---
Dockerfile | 75 +++++-----
README.md | 73 +++++-----
hocrtotei | 55 +++----
ocr | 404 +++++++++++++++++++---------------------------------
wrapper/ocr | 40 +++---
5 files changed, 273 insertions(+), 374 deletions(-)
diff --git a/Dockerfile b/Dockerfile
index 08b10ca..b04f4d0 100644
--- a/Dockerfile
+++ b/Dockerfile
@@ -7,41 +7,47 @@ LABEL authors="Patrick Jentsch , Stephan Porada /dev/null \
+ && rm -r "pyflow-${PYFLOW_VERSION}" "pyflow-${PYFLOW_VERSION}.tar.gz"
## Install ocropy ##
-ENV OCROPY_RELEASE=1.3.3
-ADD "https://github.com/tmbdev/ocropy/archive/v${OCROPY_RELEASE}.tar.gz" .
-RUN tar -xzf "v${OCROPY_RELEASE}.tar.gz" \
- && cd "ocropy-${OCROPY_RELEASE}" \
+ENV OCROPY_VERSION=1.3.3
+RUN wget --no-check-certificate --quiet \
+ "https://github.com/tmbdev/ocropy/archive/v${OCROPY_VERSION}.tar.gz" \
+ && tar -xzf "v${OCROPY_VERSION}.tar.gz" \
+ && cd "ocropy-${OCROPY_VERSION}" \
&& apt-get install --no-install-recommends --yes \
+ python2.7 \
python-pil \
python-tk \
$(cat PACKAGES) \
&& python2.7 setup.py install \
- && cd .. \
- && rm -r "ocropy-${OCROPY_RELEASE}" "v${OCROPY_RELEASE}.tar.gz"
+ && cd - > /dev/null \
+ && rm -r "ocropy-${OCROPY_VERSION}" "v${OCROPY_VERSION}.tar.gz"
## Install Tesseract OCR ##
-ENV TESSERACT_RELEASE=4.1.1
-ADD "https://github.com/tesseract-ocr/tesseract/archive/${TESSERACT_RELEASE}.tar.gz" .
-RUN tar -xzf "${TESSERACT_RELEASE}.tar.gz" \
- && cd "tesseract-${TESSERACT_RELEASE}" \
+ENV TESSERACT_VERSION=4.1.1
+RUN wget --no-check-certificate --quiet \
+ "https://github.com/tesseract-ocr/tesseract/archive/${TESSERACT_VERSION}.tar.gz" \
+ && tar -xzf "${TESSERACT_VERSION}.tar.gz" \
+ && cd "tesseract-${TESSERACT_VERSION}" \
&& apt-get install --no-install-recommends --yes \
autoconf \
automake \
@@ -60,35 +66,24 @@ RUN tar -xzf "${TESSERACT_RELEASE}.tar.gz" \
&& make install \
&& ldconfig \
&& cd - > /dev/null \
- && rm -r "tesseract-${TESSERACT_RELEASE}" "${TESSERACT_RELEASE}.tar.gz"
+ && rm -r "tesseract-${TESSERACT_VERSION}" "${TESSERACT_VERSION}.tar.gz"
-ENV TESSDATA_BEST_RELEASE=4.1.0
-ADD "https://github.com/tesseract-ocr/tessdata_best/archive/${TESSDATA_BEST_RELEASE}.tar.gz" .
-RUN tar -xzf "${TESSDATA_BEST_RELEASE}.tar.gz" \
- && mv "tessdata_best-${TESSDATA_BEST_RELEASE}/ara.traineddata" "/usr/local/share/tessdata/" \
- && mv "tessdata_best-${TESSDATA_BEST_RELEASE}/chi_tra.traineddata" "/usr/local/share/tessdata/" \
- && mv "tessdata_best-${TESSDATA_BEST_RELEASE}/dan.traineddata" "/usr/local/share/tessdata/" \
- && mv "tessdata_best-${TESSDATA_BEST_RELEASE}/deu.traineddata" "/usr/local/share/tessdata/" \
- && mv "tessdata_best-${TESSDATA_BEST_RELEASE}/ell.traineddata" "/usr/local/share/tessdata/" \
- && mv "tessdata_best-${TESSDATA_BEST_RELEASE}/eng.traineddata" "/usr/local/share/tessdata/" \
- && mv "tessdata_best-${TESSDATA_BEST_RELEASE}/enm.traineddata" "/usr/local/share/tessdata/" \
- && mv "tessdata_best-${TESSDATA_BEST_RELEASE}/fra.traineddata" "/usr/local/share/tessdata/" \
- && mv "tessdata_best-${TESSDATA_BEST_RELEASE}/frk.traineddata" "/usr/local/share/tessdata/" \
- && mv "tessdata_best-${TESSDATA_BEST_RELEASE}/frm.traineddata" "/usr/local/share/tessdata/" \
- && mv "tessdata_best-${TESSDATA_BEST_RELEASE}/ita.traineddata" "/usr/local/share/tessdata/" \
- && mv "tessdata_best-${TESSDATA_BEST_RELEASE}/por.traineddata" "/usr/local/share/tessdata/" \
- && mv "tessdata_best-${TESSDATA_BEST_RELEASE}/rus.traineddata" "/usr/local/share/tessdata/" \
- && mv "tessdata_best-${TESSDATA_BEST_RELEASE}/spa.traineddata" "/usr/local/share/tessdata/" \
- && rm -r "tessdata_best-${TESSDATA_BEST_RELEASE}" "${TESSDATA_BEST_RELEASE}.tar.gz"
+ENV TESSERACT_MODELS="ara,chi_tra,dan,ell,eng,enm,fra,frk,frm,ita,por,rus,spa"
+ENV TESSDATA_BEST_VERSION=4.1.0
+RUN wget --no-check-certificate --quiet \
+ "https://github.com/tesseract-ocr/tessdata_best/archive/${TESSDATA_BEST_VERSION}.tar.gz" \
+ && tar -xzf "${TESSDATA_BEST_VERSION}.tar.gz" \
+ && for tesseract_model in $(echo ${TESSERACT_MODELS} | tr "," "\n"); do mv "tessdata_best-${TESSDATA_BEST_VERSION}/${tesseract_model}.traineddata" "/usr/local/share/tessdata/"; done \
+ && rm -r "tessdata_best-${TESSDATA_BEST_VERSION}" "${TESSDATA_BEST_VERSION}.tar.gz"
## Further dependencies ##
RUN apt-get install --no-install-recommends --yes \
+ procps \
ghostscript \
- python-pip \
python3.7 \
- zip \
- && pip install natsort
+ rename \
+ zip
## Install Pipeline ##
diff --git a/README.md b/README.md
index d4ba53b..e59b8b8 100644
--- a/README.md
+++ b/README.md
@@ -3,13 +3,14 @@
This software implements a heavily parallelized pipeline to recognize text in PDF files. It is used for nopaque's OCR service but you can also use it standalone, for that purpose a convenient wrapper script is provided.
## Software used in this pipeline implementation
-- Official Debian Docker image (buster-slim) and programs from its free repositories: https://hub.docker.com/_/debian
+
+- Official Debian Docker image (buster-slim): https://hub.docker.com/_/debian
+ - Software from Debian Buster's free repositories
- ocropy (1.3.3): https://github.com/ocropus/ocropy/releases/tag/v1.3.3
- pyFlow (1.1.20): https://github.com/Illumina/pyflow/releases/tag/v1.1.20
- Tesseract OCR (4.1.1): https://github.com/tesseract-ocr/tesseract/releases/tag/4.1.1
- tessdata_best (4.1.0): https://github.com/tesseract-ocr/tessdata_best/releases/tag/4.1.0
-
## Use this image
1. Create input and output directories for the pipeline.
@@ -22,7 +23,7 @@ mkdir -p //input //output
3. Start the pipeline process. Check the [Pipeline arguments](#pipeline-arguments) section for more details.
```
# Option one: Use the wrapper script
-## Install the wrapper script (only on first run). Get it from https://gitlab.ub.uni-bielefeld.de/sfb1288inf/ocr/-/raw/1.0.0/wrapper/ocr, make it executeable and add it to your ${PATH}
+## Install the wrapper script (only on first run). Get it from https://gitlab.ub.uni-bielefeld.de/sfb1288inf/ocr/-/raw/development/wrapper/ocr, make it executeable and add it to your ${PATH}
cd /
ocr -i input -l -o output
@@ -33,37 +34,44 @@ docker run \
-u $(id -u $USER):$(id -g $USER) \
-v //input:/input \
-v //output:/output \
- gitlab.ub.uni-bielefeld.de:4567/sfb1288inf/ocr:1.0.0 \
- -i /input \
- -l
- -o /output \
+ gitlab.ub.uni-bielefeld.de:4567/sfb1288inf/ocr:development \
+ -i /ocr_pipeline/input \
+ -l \
+ -o /ocr_pipeline/output \
```
4. Check your results in the `//output` directory.
-```
### Pipeline arguments
-`-l languagecode`
-* Tells tesseract which language will be used.
-* options = ara (Arabic), chi_tra (Chinese - Traditional), dan (Danish), deu (German), ell (Greek, Modern (1453-)), eng (English), enm (Middle englisch), fra (French), frk (German Fraktur), frm (Middle french), ita (Italian), por (Portuguese), rus (Russian), spa (Spanish)
-* required = True
+#### Mandatory arguments
-`--keep-intermediates`
-* If set, all intermediate files created during the OCR process will be
-kept.
-* default = False
-* required = False
+`-i, --input-dir INPUT_DIR`
+* Input directory
-`--nCores corenumber`
-* Sets the number of CPU cores being used during the OCR process.
-* default = min(4, multiprocessing.cpu_count())
-* required = False
+`-o, --output-dir OUTPUT_DIR`
+* Output directory
-`--skip-binarisation`
-* Used to skip binarization with ocropus. If skipped, only the tesseract binarization is used.
-* default = False
+`-l, --language {spa,fra,dan,deu,eng,frm,chi_tra,ara,enm,ita,ell,frk,rus,por}`
+* Language of the input (3-character ISO 639-2 language codes)
+
+#### Optional arguments
+
+`--binarize`
+* Add binarization as a preprocessing step
+
+`--log-dir`
+* Logging directory
+
+`--mem-mb`
+* Amount of system memory to be used (Default: min(--n-cores * 2048, available system memory))
+
+`--n-cores`
+* Number of CPU threads to be used (Default: min(4, available CPU cores))
+
+`-v, --version`
+* Returns the current version of the OCR pipeline
``` bash
# Example with all arguments used
@@ -71,13 +79,14 @@ docker run \
--rm \
-it \
-u $(id -u $USER):$(id -g $USER) \
- -v "$HOME"/ocr/input:/input \
- -v "$HOME"/ocr/output:/output \
- gitlab.ub.uni-bielefeld.de:4567/sfb1288inf/ocr:1.0.0 \
- -i /input \
+ -v //input:/ocr_pipeline/input \
+ -v //output:/ocr_pipeline/output \
+ -v //logs:/ocr_pipeline/logs \
+ gitlab.ub.uni-bielefeld.de:4567/sfb1288inf/ocr:development \
+ -i /ocr_pipeline/input \
-l eng \
- -o /output \
- --keep_intermediates \
- --nCores 8 \
- --skip-binarisation
+ -o /ocr_pipeline/output \
+ --binarize \
+ --log-dir /ocr_pipeline/logs \
+ --n-cores 8 \
```
diff --git a/hocrtotei b/hocrtotei
index 142f4f5..19aefe2 100755
--- a/hocrtotei
+++ b/hocrtotei
@@ -5,45 +5,50 @@
from xml.sax.saxutils import escape
from argparse import ArgumentParser
+import re
import xml.etree.ElementTree as ET
parser = ArgumentParser(description='Merges hOCR files into a TEI file.')
-parser.add_argument('i', metavar='hOCR-sourcefile', nargs='+')
-parser.add_argument('o', metavar='TEI-destfile',)
+parser.add_argument('i', metavar='hOCR-sourcefile')
+parser.add_argument('o', metavar='TEI-destfile')
args = parser.parse_args()
output_file = open(args.o, 'w')
output_file.write(
'\n'
- + '\n'
- + ' \n'
- + ' \n'
- + ' \n'
- + ' \n'
- + ' \n'
- + ' \n'
- + ' \n'
- + ' \n'
- + ' \n'
- + ' \n'
- + ' \n'
+ + '\n'
+ + ' \n'
+ + ' \n'
+ + ' \n'
+ + ' \n'
+ + ' \n'
+ + ' \n'
+ + ' \n'
+ + ' \n'
+ + ' \n'
+ + ' \n'
+ + ' \n'
)
-for index, input_file in enumerate(args.i):
- tree = ET.parse(input_file)
- output_file.write(' \n' % (index + 1))
- for para in tree.findall('.//*[@class="ocr_par"]'):
+tree = ET.parse(args.i)
+for page in tree.findall('.//*[@class="ocr_page"]'):
+ page_properties = page.attrib.get('title')
+ facsimile = re.search(r'image \"(.*?)\"', page_properties).group(1)
+ page_number = re.search(r'ppageno (\d+)', page_properties).group(1)
+ output_file.write(' \n' % (facsimile, page_number)) # noqa
+ for para in page.findall('.//*[@class="ocr_par"]'):
output_file.write(' \n')
for line in para.findall('.//*[@class="ocr_line"]'):
- first_word_in_line = True
+ output_file.write('
')
+ indent = ''
for word in line.findall('.//*[@class="ocrx_word"]'):
if word.text is not None:
- output_file.write((' ' if first_word_in_line else ' ') + escape(word.text.strip()))
- first_word_in_line = False
- if not first_word_in_line:
- output_file.write('
\n')
+ output_file.write(indent + escape(word.text.strip()))
+ indent = ' '
+ output_file.write('\n')
output_file.write('
\n')
output_file.write(
' \n'
- + ' \n'
- + '')
+ + ' \n'
+ + ''
+)
output_file.close()
diff --git a/ocr b/ocr
index f17ae78..1a2bbc2 100755
--- a/ocr
+++ b/ocr
@@ -8,48 +8,10 @@ __author__ = 'Patrick Jentsch ,' \
__version__ = '1.0.0'
from argparse import ArgumentParser
-from natsort import natsorted
from pyflow import WorkflowRunner
import multiprocessing
import os
import sys
-import tempfile
-
-
-TESSERACT_MODELS = ['deu', 'eng', 'enm', 'fra', 'frk', 'frm', 'ita', 'por', 'spa'] # noqa
-
-
-def parse_args():
- parser = ArgumentParser(
- description='An OCR pipeline for PDF file processing.',
- prog='OCR pipeline'
- )
- parser.add_argument('-i', '--input-directory',
- help='Input directory (only PDF files get processed)',
- required=True)
- parser.add_argument('-o', '--output-directory',
- help='Output directory',
- required=True)
- parser.add_argument('-l', '--language',
- choices=TESSERACT_MODELS,
- required=True)
- parser.add_argument('--binarize',
- action='store_true',
- help='Use ocropy binarisation as preprocessing step.')
- parser.add_argument('--log-dir')
- parser.add_argument('--n-cores',
- default=min(4, multiprocessing.cpu_count()),
- help='Total number of cores available.',
- type=int)
- parser.add_argument('--intermediate-directory')
- parser.add_argument('--zip',
- help='Zips all results in different archives depending'
- ' on result types. Also zips everything into one '
- 'archive.')
- parser.add_argument('-v', '--version',
- action='version',
- version='%(prog)s {}'.format(__version__))
- return parser.parse_args()
class OCRPipelineJob:
@@ -61,41 +23,23 @@ class OCRPipelineJob:
Arguments:
file -- Path to the file
output_dir -- Path to a directory, where job results a stored
- intermediate_dir -- Path to a directory, where intermediate files are
- stored.
"""
- def __init__(self, file, output_dir, intermediate_dir):
+ def __init__(self, file, output_dir):
self.file = file
- self.intermediate_dir = intermediate_dir
self.name = os.path.basename(file).rsplit('.', 1)[0]
self.output_dir = output_dir
+ self.page_dir = os.path.join(output_dir, 'pages')
class OCRPipeline(WorkflowRunner):
- def __init__(self, input_dir, lang, output_dir, binarize, intermediate_dir,
- n_cores, zip):
+ def __init__(self, input_dir, lang, output_dir, binarize, zip):
self.input_dir = input_dir
self.lang = lang
self.output_dir = output_dir
self.binarize = binarize
- if intermediate_dir is None:
- self.intermediate_dir = os.path.join(output_dir, 'tmp')
- else:
- self.intermediate_dir = tempfile.mkdtemp(dir=intermediate_dir)
- self.n_cores = n_cores
- if zip is None:
- self.zip = zip
- else:
- if zip.lower().endswith('.zip'):
- # Remove .zip file extension if provided
- self.zip = zip[:-4]
- self.zip = self.zip if self.zip else 'output'
- else:
- self.zip = zip
- self.jobs = collect_jobs(self.input_dir,
- self.output_dir,
- self.intermediate_dir)
+ self.zip = zip
+ self.jobs = collect_jobs(self.input_dir, self.output_dir)
def workflow(self):
if not self.jobs:
@@ -108,10 +52,7 @@ class OCRPipeline(WorkflowRunner):
'''
setup_output_directory_tasks = []
for i, job in enumerate(self.jobs):
- cmd = 'mkdir'
- cmd += ' -p'
- cmd += ' "{}"'.format(job.intermediate_dir)
- cmd += ' "{}"'.format(os.path.join(job.output_dir, 'poco'))
+ cmd = 'mkdir -p "{}"'.format(job.page_dir)
lbl = 'setup_output_directory_-_{}'.format(i)
task = self.addTask(command=cmd, label=lbl)
setup_output_directory_tasks.append(task)
@@ -122,10 +63,10 @@ class OCRPipeline(WorkflowRunner):
' ##################################################
'''
split_input_tasks = []
- n_cores = min(self.n_cores, max(1, int(self.n_cores / len(self.jobs))))
+ n_cores = max(1, int(self.getNCores() / len(self.jobs)))
for i, job in enumerate(self.jobs):
input_file = job.file
- output_file = '{}/page-%d.tif'.format(job.intermediate_dir)
+ output_file = '{}/page-%d.tif'.format(job.page_dir)
cmd = 'gs'
cmd += ' -dBATCH'
cmd += ' -dNOPAUSE'
@@ -138,15 +79,24 @@ class OCRPipeline(WorkflowRunner):
cmd += ' "{}"'.format(input_file)
deps = 'setup_output_directory_-_{}'.format(i)
lbl = 'split_input_-_{}'.format(i)
- task = self.addTask(command=cmd, dependencies=deps, label=lbl, nCores=n_cores) # noqa
+ task = self.addTask(command=cmd, dependencies=deps, label=lbl,
+ nCores=n_cores)
split_input_tasks.append(task)
if self.binarize:
'''
- ' The binarization_tasks list is created based on the output files
- ' of the split_tasks. So wait until they are finished.
+ ' ##################################################
+ ' # pre binarization #
+ ' ##################################################
'''
- self.waitForTasks()
+ pre_binarization_tasks = []
+ for i, job in enumerate(self.jobs):
+ input_file = os.path.join(job.output_dir, 'binarization_input_files.txt') # noqa
+ cmd = 'ls -dv "{}/"* >> "{}"'.format(job.page_dir, input_file)
+ deps = 'split_input_-_{}'.format(i)
+ lbl = 'pre_binarization_-_{}'.format(i)
+ task = self.addTask(command=cmd, dependencies=deps, label=lbl)
+ pre_binarization_tasks.append(task)
'''
' ##################################################
@@ -154,52 +104,55 @@ class OCRPipeline(WorkflowRunner):
' ##################################################
'''
binarization_tasks = []
- '''
- ' We run ocropus-nlbin with either four or, if there are less then
- ' four cores available for this workflow, the available core
- ' number.
- '''
- n_cores = min(4, self.n_cores)
+ n_cores = self.getNCores()
+ mem_mb = self.getMemMb()
for i, job in enumerate(self.jobs):
- input_dir = job.intermediate_dir
- output_dir = job.intermediate_dir
- files = filter(lambda x: x.endswith('.tif'), os.listdir(input_dir)) # noqa
- files = natsorted(files)
- files = map(lambda x: os.path.join(input_dir, x), files)
- cmd = 'ocropus-nlbin "{}"'.format('" "'.join(files))
+ input_file = os.path.join(job.output_dir, 'binarization_input_files.txt') # noqa
+ cmd = 'ocropus-nlbin "@{}"'.format(input_file)
cmd += ' --nocheck'
- cmd += ' --output "{}"'.format(output_dir)
+ cmd += ' --output "{}"'.format(job.page_dir)
cmd += ' --parallel "{}"'.format(n_cores)
- print(cmd)
- deps = 'split_input_-_{}'.format(i)
+ deps = 'pre_binarization_-_{}'.format(i)
lbl = 'binarization_-_{}'.format(i)
- task = self.addTask(command=cmd, dependencies=deps, label=lbl, nCores=n_cores) # noqa
+ task = self.addTask(command=cmd, dependencies=deps, label=lbl,
+ memMb=mem_mb, nCores=n_cores)
binarization_tasks.append(task)
- self.waitForTasks()
-
'''
' ##################################################
- ' # Renaming of binarization output files #
+ ' # post binarization #
' ##################################################
'''
+ post_binarization_tasks = []
for i, job in enumerate(self.jobs):
- input_dir = job.intermediate_dir
- output_dir = job.intermediate_dir
- files = filter(lambda x: x.endswith('.bin.png'), os.listdir(input_dir)) # noqa
- for file in files:
- # int conversion is done in order to trim leading zeros
- page_number = int(file.split('.', 1)[0])
- output_file = 'page-{}.bin.png'.format(page_number)
- os.rename(os.path.join(output_dir, file),
- os.path.join(output_dir, output_file))
+ input_file = os.path.join(job.output_dir, 'binarization_input_files.txt') # noqa
+ cmd = 'rm "{}"'.format(input_file)
+ cmd += ' && '
+ cmd += 'cd "{}"'.format(job.page_dir)
+ cmd += ' && '
+ cmd += 'rm *.{nrm.png,tif}'
+ cmd += ' && '
+ cmd += 'rename \'s/^0*/page-/\' *'
+ cmd += ' && '
+ cmd += 'cd -'
+ deps = 'binarization_-_{}'.format(i)
+ lbl = 'post_binarization_-_{}'.format(i)
+ task = self.addTask(command=cmd, dependencies=deps, label=lbl)
+ post_binarization_tasks.append(task)
'''
- ' The ocr_tasks are created based of the output files of either the
- ' split_tasks or binarization_tasks. So wait until they are
- ' finished.
+ ' ##################################################
+ ' # pre ocr #
+ ' ##################################################
'''
- self.waitForTasks()
+ pre_ocr_tasks = []
+ for i, job in enumerate(self.jobs):
+ input_file = os.path.join(job.output_dir, 'ocr_input_files.txt')
+ cmd = 'ls -dv "{}/"* >> "{}"'.format(job.page_dir, input_file)
+ deps = 'post_binarization_-_{}'.format(i) if self.binarize else 'split_input_-_{}'.format(i) # noqa
+ lbl = 'pre_ocr_-_{}'.format(i)
+ task = self.addTask(command=cmd, dependencies=deps, label=lbl)
+ pre_ocr_tasks.append(task)
'''
' ##################################################
@@ -207,157 +160,51 @@ class OCRPipeline(WorkflowRunner):
' ##################################################
'''
ocr_tasks = []
+ n_cores = min(4, self.getNCores())
+ mem_mb = min(n_cores * 2048, self.getMemMb())
for i, job in enumerate(self.jobs):
- input_dir = job.intermediate_dir
- output_dir = job.intermediate_dir
- files = os.listdir(input_dir)
- if self.binarize:
- deps = 'binarization_-_{}'.format(i)
- files = filter(lambda x: x.endswith('.bin.png'), files)
- else:
- deps = 'split_input_-_{}'.format(i)
- files = filter(lambda x: x.endswith('.tif'), files)
- files = natsorted(files)
- files = map(lambda x: os.path.join(input_dir, x), files)
- for j, file in enumerate(files):
- if self.binarize:
- output_file_base = os.path.join(output_dir, file.rsplit('.', 2)[0]) # noqa
- else:
- output_file_base = os.path.join(output_dir, file.rsplit('.', 1)[0]) # noqa
- cmd = 'tesseract "{}" "{}"'.format(file, output_file_base)
- cmd += ' -l "{}"'.format(self.lang)
- cmd += ' hocr pdf txt'
- cmd += ' && '
- cmd += 'sed -i \'s+{}/++g\' "{}".hocr'.format(input_dir, output_file_base) # noqa
- lbl = 'ocr_-_{}-{}'.format(i, j)
- task = self.addTask(command=cmd, dependencies=deps, label=lbl, env={"OMP_THREAD_LIMIT": "1"}) # noqa
- ocr_tasks.append(task)
-
- '''
- ' The following jobs are created based of the output files of the
- ' ocr_tasks. So wait until they are finished.
- '''
- self.waitForTasks()
+ input_file = os.path.join(job.output_dir, 'ocr_input_files.txt')
+ output_file_base = os.path.join(job.output_dir, job.name)
+ cmd = 'tesseract "{}" "{}"'.format(input_file, output_file_base)
+ cmd += ' -l "{}"'.format(self.lang)
+ cmd += ' hocr pdf txt'
+ deps = 'pre_ocr_-_{}'.format(i)
+ lbl = 'ocr_-_{}'.format(i)
+ task = self.addTask(command=cmd, dependencies=deps,
+ env={'OMP_THREAD_LIMIT': '{}'.format(n_cores)},
+ label=lbl, memMb=mem_mb, nCores=n_cores)
+ ocr_tasks.append(task)
'''
' ##################################################
- ' # combined pdf creation #
+ ' # post ocr #
' ##################################################
'''
- combined_pdf_creation_tasks = []
- n_cores = min(self.n_cores, max(1, int(self.n_cores / len(self.jobs))))
+ post_ocr_tasks = []
for i, job in enumerate(self.jobs):
- input_dir = job.intermediate_dir
- output_file = os.path.join(job.output_dir, '{}.pdf'.format(job.name)) # noqa
- files = filter(lambda x: x.endswith('.pdf'), os.listdir(input_dir))
- files = natsorted(files)
- files = map(lambda x: os.path.join(input_dir, x), files)
- cmd = 'gs'
- cmd += ' -dBATCH'
- cmd += ' -dNOPAUSE'
- cmd += ' -dNumRenderingThreads={}'.format(n_cores)
- cmd += ' -dPDFSETTINGS=/ebook'
- cmd += ' -dQUIET'
- cmd += ' -sDEVICE=pdfwrite'
- cmd += ' "-sOutputFile={}"'.format(output_file)
- cmd += ' "{}"'.format('" "'.join(files))
- deps = filter(lambda x: x.startswith('ocr_-_{}'.format(i)), ocr_tasks) # noqa
- lbl = 'combined_pdf_creation_-_{}'.format(i)
- task = self.addTask(command=cmd, dependencies=deps, label=lbl, nCores=n_cores) # noqa
- combined_pdf_creation_tasks.append(task)
-
- '''
- ' ##################################################
- ' # combined txt creation #
- ' ##################################################
- '''
- combined_txt_creation_tasks = []
- for i, job in enumerate(self.jobs):
- input_dir = job.intermediate_dir
- output_file = os.path.join(job.output_dir, '{}.txt'.format(job.name)) # noqa
- files = filter(lambda x: x.endswith('.txt'), os.listdir(input_dir))
- files = natsorted(files)
- files = map(lambda x: os.path.join(input_dir, x), files)
- cmd = 'cat "{}" > "{}"'.format('" "'.join(files), output_file)
- deps = filter(lambda x: x.startswith('ocr_-_{}'.format(i)), ocr_tasks) # noqa
- lbl = 'combined_txt_creation_-_{}'.format(i)
+ input_file = os.path.join(job.output_dir, 'ocr_input_files.txt')
+ output_file_base = os.path.join(job.output_dir, job.name)
+ cmd = 'rm "{}"'.format(input_file)
+ cmd += ' && '
+ cmd += 'sed -i \'s+{}+pages+g\' "{}.hocr"'.format(job.page_dir, output_file_base) # noqa
+ deps = 'ocr_-_{}'.format(i)
+ lbl = 'post_ocr_-_{}'.format(i)
task = self.addTask(command=cmd, dependencies=deps, label=lbl)
- combined_txt_creation_tasks.append(task)
+ post_ocr_tasks.append(task)
'''
' ##################################################
- ' # tei p5 creation #
+ ' # hocr to tei #
' ##################################################
'''
- tei_p5_creation_tasks = []
+ hocr_to_tei_tasks = []
for i, job in enumerate(self.jobs):
- input_dir = job.intermediate_dir
- output_file = os.path.join(job.output_dir, '{}.xml'.format(job.name)) # noqa
- files = filter(lambda x: x.endswith('.hocr'),
- os.listdir(input_dir))
- files = natsorted(files)
- files = map(lambda x: os.path.join(input_dir, x), files)
- cmd = 'hocrtotei "{}" "{}"'.format('" "'.join(files),
- output_file)
- deps = filter(lambda x: x.startswith('ocr_-_{}'.format(i)), ocr_tasks) # noqa
- lbl = 'tei_p5_creation_-_{}'.format(i)
+ output_file_base = os.path.join(job.output_dir, job.name)
+ cmd = 'hocrtotei "{}.hocr" "{}.xml"'.format(output_file_base, output_file_base) # noqa
+ deps = 'post_ocr_-_{}'.format(i)
+ lbl = 'hocr_to_tei_-_{}'.format(i)
task = self.addTask(command=cmd, dependencies=deps, label=lbl)
- tei_p5_creation_tasks.append(task)
-
- '''
- ' ##################################################
- ' # poco bundle creation #
- ' ##################################################
- '''
- poco_bundle_creation_tasks = []
- for i, job in enumerate(self.jobs):
- input_dir = job.intermediate_dir
- output_dir = os.path.join(job.output_dir, 'poco')
- files = os.listdir(input_dir)
- if self.binarize:
- files = filter(lambda x: x.endswith(('.bin.png', '.hocr')), files) # noqa
- else:
- files = filter(lambda x: x.endswith(('.tif', '.hocr')), files)
- files = natsorted(files)
- files = map(lambda x: os.path.join(input_dir, x), files)
- cmd = 'mv "{}" "{}"'.format('" "'.join(files), output_dir)
- deps = filter(lambda x: x.startswith('ocr_-_{}'.format(i)), ocr_tasks) # noqa
- deps.append('tei_p5_creation_-_{}'.format(i))
- lbl = 'poco_bundle_creation_-_{}'.format(i)
- task = self.addTask(command=cmd, dependencies=deps, label=lbl)
- poco_bundle_creation_tasks.append(task)
-
- '''
- ' The following jobs are created based of the output files of the
- ' combined_pdf_creation_tasks. So wait until they are finished.
- '''
- self.waitForTasks()
-
- '''
- ' ##################################################
- ' # cleanup #
- ' ##################################################
- '''
- cleanup_tasks = []
- for i, job in enumerate(self.jobs):
- input_dir = job.intermediate_dir
- cmd = 'rm -r "{}"'.format(input_dir)
- deps = ['combined_pdf_creation_-_{}'.format(i),
- 'combined_txt_creation_-_{}'.format(i),
- 'poco_bundle_creation_-_{}'.format(i),
- 'tei_p5_creation_-_{}'.format(i)]
- lbl = 'job_cleanup_-_{}'.format(i)
- task = self.addTask(command=cmd, dependencies=deps, label=lbl)
- cleanup_tasks.append(task)
-
- input_dir = self.intermediate_dir
- cmd = 'rm -r "{}"'.format(input_dir)
- deps = cleanup_tasks
- lbl = 'pipeline_cleanup'
- task = self.addTask(command=cmd, dependencies=deps, label=lbl)
- cleanup_tasks.append(task)
-
- self.waitForTasks()
+ hocr_to_tei_tasks.append(task)
'''
' ##################################################
@@ -376,7 +223,7 @@ class OCRPipeline(WorkflowRunner):
cmd += ' -i "*.pdf" "*.txt" "*.xml" "*.hocr" "*.{}"'.format('bin.png' if self.binarize else 'tif') # noqa
cmd += ' && '
cmd += 'cd -'
- deps = combined_pdf_creation_tasks + combined_txt_creation_tasks + poco_bundle_creation_tasks # noqa
+ deps = hocr_to_tei_tasks
lbl = 'zip_creation_-_all'
task = self.addTask(command=cmd, dependencies=deps, label=lbl)
zip_creation_tasks.append(task)
@@ -390,7 +237,7 @@ class OCRPipeline(WorkflowRunner):
cmd += ' -i "*.pdf"'
cmd += ' && '
cmd += 'cd -'
- deps = combined_pdf_creation_tasks
+ deps = ocr_tasks
lbl = 'zip_creation_-_pdf'
task = self.addTask(command=cmd, dependencies=deps, label=lbl)
zip_creation_tasks.append(task)
@@ -404,7 +251,7 @@ class OCRPipeline(WorkflowRunner):
cmd += ' -i "*.txt"'
cmd += ' && '
cmd += 'cd -'
- deps = combined_txt_creation_tasks
+ deps = ocr_tasks
lbl = 'zip_creation_-_txt'
task = self.addTask(command=cmd, dependencies=deps, label=lbl)
zip_creation_tasks.append(task)
@@ -418,7 +265,7 @@ class OCRPipeline(WorkflowRunner):
cmd += ' -i "*.xml"'
cmd += ' && '
cmd += 'cd -'
- deps = tei_p5_creation_tasks
+ deps = hocr_to_tei_tasks
lbl = 'zip_creation_-_xml'
task = self.addTask(command=cmd, dependencies=deps, label=lbl)
zip_creation_tasks.append(task)
@@ -432,37 +279,80 @@ class OCRPipeline(WorkflowRunner):
cmd += ' -i "*.hocr" "*.{}"'.format('bin.png' if self.binarize else 'tif') # noqa
cmd += ' && '
cmd += 'cd -'
- deps = poco_bundle_creation_tasks
+ deps = post_ocr_tasks
lbl = 'zip_creation_-_poco'
task = self.addTask(command=cmd, dependencies=deps, label=lbl)
zip_creation_tasks.append(task)
-def collect_jobs(input_dir, output_dir, intermediate_dir):
+def collect_jobs(input_dir, output_dir):
jobs = []
for file in os.listdir(input_dir):
if os.path.isdir(os.path.join(input_dir, file)):
jobs += collect_jobs(os.path.join(input_dir, file),
- os.path.join(output_dir, file),
- os.path.join(intermediate_dir, file))
+ os.path.join(output_dir, file))
elif file.lower().endswith('.pdf'):
job = OCRPipelineJob(os.path.join(input_dir, file),
- os.path.join(output_dir, file),
- os.path.join(intermediate_dir, file))
+ os.path.join(output_dir, file))
jobs.append(job)
return jobs
+def parse_args():
+ parser = ArgumentParser(description='OCR pipeline for PDF file processing',
+ prog='OCR pipeline')
+ parser.add_argument('-i', '--input-dir',
+ help='Input directory',
+ required=True)
+ parser.add_argument('-o', '--output-dir',
+ help='Output directory',
+ required=True)
+ parser.add_argument('-l', '--language',
+ choices=list(map(lambda x: x[:-12], filter(lambda x: x.endswith('.traineddata'), os.listdir('/usr/local/share/tessdata')))), # noqa
+ help='Language of the input '
+ '(3-character ISO 639-2 language codes)',
+ required=True)
+ parser.add_argument('--binarize',
+ action='store_true',
+ help='Add binarization as a preprocessing step')
+ parser.add_argument('--log-dir',
+ help='Logging directory')
+ parser.add_argument('--mem-mb',
+ help='Amount of system memory to be used (Default: min(--n-cores * 2048, available system memory))', # noqa
+ type=int)
+ parser.add_argument('--n-cores',
+ default=min(4, multiprocessing.cpu_count()),
+ help='Number of CPU threads to be used', # noqa
+ type=int)
+ parser.add_argument('--zip',
+ help='Create one zip file per filetype')
+ parser.add_argument('-v', '--version',
+ action='version',
+ help='Returns the current version of the OCR pipeline',
+ version='%(prog)s {}'.format(__version__))
+ args = parser.parse_args()
+
+ # Set some tricky default values and check for insufficient input
+ if args.log_dir is None:
+ args.log_dir = args.output_dir
+ if args.n_cores < 1:
+ raise Exception('--n-cores must be greater or equal 1')
+ if args.mem_mb is None:
+ max_mem_mb = int(os.popen('free -t -m').readlines()[-1].split()[1:][0])
+ args.mem_mb = min(args.n_cores * 2048, max_mem_mb)
+ if args.mem_mb < 2048:
+ raise Exception('--mem-mb must be greater or equal 2048')
+ if args.zip is not None and args.zip.lower().endswith('.zip'):
+ # Remove .zip file extension if provided
+ args.zip = args.zip[:-4]
+ args.zip = args.zip if args.zip else 'output'
+ return args
+
+
def main():
args = parse_args()
- ocr_pipeline = OCRPipeline(args.input_directory, args.language,
- args.output_directory, args.binarize,
- args.intermediate_directory, args.n_cores,
- args.zip)
- retval = ocr_pipeline.run(
- dataDirRoot=(args.log_dir or args.output_directory),
- nCores=args.n_cores
- )
+ ocr_pipeline = OCRPipeline(args.input_dir, args.language, args.output_dir, args.binarize, args.zip) # noqa
+ retval = ocr_pipeline.run(dataDirRoot=args.log_dir, memMb=args.mem_mb, nCores=args.n_cores) # noqa
sys.exit(retval)
diff --git a/wrapper/ocr b/wrapper/ocr
index 3ed3e18..c6e5003 100755
--- a/wrapper/ocr
+++ b/wrapper/ocr
@@ -1,43 +1,43 @@
#!/usr/bin/env python3
# coding=utf-8
-"""A wrapper to execute the OCR pipeline in a Docker container"""
+"""A wrapper to execute the OCR pipeline in a Docker container."""
from argparse import ArgumentParser
import os
import subprocess
+import sys
-CONTAINER_IMAGE_TAG = '1.0.0'
-CONTAINER_IMAGE = 'gitlab.ub.uni-bielefeld.de:4567/sfb1288inf/ocr:{}'.format(CONTAINER_IMAGE_TAG) # noqa
+CONTAINER_IMAGE = 'gitlab.ub.uni-bielefeld.de:4567/sfb1288inf/ocr:1.0.0'
CONTAINER_INPUT_DIR = '/input'
-CONTAINER_INTERMEDIATE_DIR = '/intermediate'
+CONTAINER_LOG_DIR = '/logs'
CONTAINER_OUTPUT_DIR = '/output'
UID = str(os.getuid())
GID = str(os.getgid())
parser = ArgumentParser(add_help=False)
-parser.add_argument('-i', '--input-directory')
-parser.add_argument('-o', '--output-directory')
-parser.add_argument('--intermediate-directory')
+parser.add_argument('-i', '--input-dir')
+parser.add_argument('-o', '--output-dir')
+parser.add_argument('--log-dir')
args, remaining_args = parser.parse_known_args()
cmd = ['docker', 'run', '--rm', '-it', '-u', '{}:{}'.format(UID, GID)]
-if args.intermediate_directory is not None:
- cmd += ['-v', '{}:{}'.format(os.path.abspath(args.intermediate_directory),
- CONTAINER_INTERMEDIATE_DIR)]
- remaining_args.insert(0, CONTAINER_INTERMEDIATE_DIR)
- remaining_args.insert(0, '--intermediate-directory')
-if args.output_directory is not None:
- cmd += ['-v', '{}:{}'.format(os.path.abspath(args.output_directory),
- CONTAINER_OUTPUT_DIR)]
- remaining_args.insert(0, CONTAINER_OUTPUT_DIR)
- remaining_args.insert(0, '-o')
-if args.input_directory is not None:
- cmd += ['-v', '{}:{}'.format(os.path.abspath(args.input_directory),
+if args.log_dir is not None:
+ cmd += ['-v', '{}:{}'.format(os.path.abspath(args.log_dir),
+ CONTAINER_LOG_DIR)]
+ remaining_args.insert(0, CONTAINER_LOG_DIR)
+ remaining_args.insert(0, '--log-dir')
+if args.input_dir is not None:
+ cmd += ['-v', '{}:{}'.format(os.path.abspath(args.input_dir),
CONTAINER_INPUT_DIR)]
remaining_args.insert(0, CONTAINER_INPUT_DIR)
remaining_args.insert(0, '-i')
+if args.output_dir is not None:
+ cmd += ['-v', '{}:{}'.format(os.path.abspath(args.output_dir),
+ CONTAINER_OUTPUT_DIR)]
+ remaining_args.insert(0, CONTAINER_OUTPUT_DIR)
+ remaining_args.insert(0, '-o')
cmd.append(CONTAINER_IMAGE)
cmd += remaining_args
-subprocess.run(cmd)
+sys.exit(subprocess.run(cmd).returncode)