sequence-to-sequence translator for noisy channel error correction
Project description
cor-asv-ann
OCR post-correction with encoder-attention-decoder LSTMs
Contents:
Introduction
This is a tool for automatic OCR post-correction (reducing optical character recognition errors) with recurrent neural networks. It uses sequence-to-sequence transduction on the character level with a model architecture akin to neural machine translation, i.e. a stacked encoder-decoder network with attention mechanism.
Architecture
The attention model always applies to full lines (in a local, monotonic configuration), and uses a linear additive alignment model. (This transfers information between the encoder and decoder hidden layer states, and calculates a soft alignment between input and output characters. It is imperative for character-level processing, because with a simple final-initial transfer, models tend to start "forgetting" the input altogether at some point in the line and behave like unconditional LM generators. Local alignment is necessary to prevent snapping back to earlier states during long sequences.)
The network architecture is as follows:
- The input characters are represented as unit vectors (or as a probability distribution in case of uncertainty and ambiguity). These enter a dense projection layer to be picked up by the encoder.
- The bottom hidden layer of the encoder is a bi-directional LSTM.
- The next encoder layers are forward LSTMs stacked on top of each other.
- The outputs of the top layer enter the attention model as constants (both in raw form to be weighted with the decoder state recurrently, and in a pre-calculated dense projection).
- The hidden layers of the decoder are forward LSTMs stacked on top of each other.
- The top hidden layer of the decoder has double width and contains the attention model:
- It reads the attention constants from 3. and uses the alignment as attention state (to be input as initial and output as final state).
- The attention model masks a window around the center of the previous alignment plus 1 character, calculates a new alignment between encoder outputs and current decoder state, and superimposes this with the encoder outputs to yield a context vector.
- The context vector is concatenated to the previous layers output and enters the LSTM.
- The decoder outputs enter a dense projection and get normalized to a probability distribution (softmax) for each character. (The output projection weights are the transpose of the input projection weights in 0. – weight tying.)
- Depending on the decoder mode, the decoder output is fed back directly (greedy) or indirectly (beamed) into the decoder input. (The first position is fed with a start symbol. Decoding ends on receiving a stop symbol.)
- The result is the character sequences corresponding to the argmax probabilities of the decoder outputs.
HL depth and width, as well as many other topology and training options can be configured:
- residual connections between layers in encoder and decoder?
- deep bidirectional encoder (with fw/bw cross-summarization)?
- LM loss/prediction as secondary output (multi-task learning, dual scoring)?
(cf. training options)
Multi-OCR input
not yet!
Decoder feedback
One important empirical finding is that the softmax output (full probability distribution) of the decoder can carry important information for the next state when input directly. This greatly improves the accuracy of both alignments and predictions. (This is in part attributable to exposure bias.) Therefore, instead of following the usual convention of feeding back argmax unit vectors, this implementation feeds back the softmax output directly.
This can even be done for beam search (which normally splits up the full distribution into a few select explicit candidates, represented as unit vectors) by simply resetting maximum outputs for lower-scoring candidates successively.
Decoder modes
While the encoder can always be run in parallel over a batch of lines and by passing the full sequence of characters in one tensor (padded to the longest line in the batch), which is very efficient with Keras backends like Tensorflow, a beam-search decoder requires passing initial/final states character-by-character, with parallelism employed to capture multiple history hypotheses of a single line. However, one can also greedily use the best output only for each position (without beam search). This latter option also allows to run in parallel over lines, which is much faster – consuming up to ten times less CPU time.
Thererfore, the backend function can operate the decoder network in either of the following modes:
fast
Decode greedily, but feeding back the full softmax distribution in batch mode (lines-parallel).
greedy
Decode greedily, but feeding back the full softmax distribution for each line separately.
default
Decode beamed, selecting the best output candidates of the best history hypotheses for each line and feeding back their (successively reset) partial softmax distributions in batch mode (hypotheses-parallel). More specifically:
Start decoder with start-of-sequence, then keep decoding until end-of-sequence is found or output length is way off, repeatedly. Decode by using the best predicted output characters and several next-best alternatives (up to some degradation threshold) as next input. Follow-up on the N best overall candidates (estimated by accumulated score, normalized by length and prospective cost), i.e. do A*-like breadth-first search, with N equal
batch_size
. Pass decoder initial/final states from character to character, for each candidate respectively.
Rejection
During beam search (default decoder mode), whenever the input and output is in good alignment (i.e. the attention model yields an alignment approximately 1 character after their predecessor's alignment on average), it is possible to estimate the current position in the source string. This input character's predicted output score, when smaller than a given (i.e. variable) probability threshold can be clipped to that minimum. This effectively adds a candidate which rejects correction at that position (keeping the input unchanged).
Underspecification and gap
Input characters that have not been seen during training must be well-behaved at inference time: They must be represented by a reserved index, and should behave like neutral/unknown characters instead of spoiling HL states and predictions in a longer follow-up context. This is achieved by dedicated leave-one-out training and regularization to optimize for interpolation of all known characters. At runtime, the encoder merely shows a warning of the previously unseen character.
The same device is useful to fill a known gap in the input (the only difference being that no warning is shown).
Training
Possibilities:
- incremental training and pretraining (on clean-only text)
- scheduled sampling (mixed teacher forcing and decoder feedback)
- LM transfer (initialization of the decoder weights from a language model of the same topology)
- shallow transfer (initialization of encoder/decoder weights from a model of lesser depth)
For existing models, cf. models subrepository.
For tools and datasets, cf. data processing subrepository.
Processing PAGE annotations
When applied on PAGE-XML (as OCR-D workspace processor, cf. usage), this component also allows processing below the TextLine
hierarchy level, i.e. on Word
or Glyph
level.
For that it uses the soft alignment scores to calculate an optimal hard alignment path for characters, and thereby distributes the transduction onto the lower level elements (keeping their coordinates and other meta-data), while changing Word segmentation if necessary (i.e. merging and splitting tokens).
Evaluation
Text lines can be compared (by aligning and computing a distance under some metric) across multiple inputs. (This would typically be GT and OCR vs post-correction.) This can be done both on plain text files (cor-asv-ann-eval
) and PAGE-XML annotations (ocrd-cor-asv-ann-evaluate
).
Distances are accumulated (as micro-averages) as character error rate (CER) mean and stddev, but only on the character level.
There are a number of distance metrics available (all operating on grapheme clusters, not mere codepoints):
Levenshtein
:
simple unweighted edit distance (fastest, standard; GT level 3)NFC
:
likeLevenshtein
, but apply Unicode normal form with canonical composition before (i.e. less than GT level 2)NFKC
:
likeLevenshtein
, but apply Unicode normal form with compatibility composition before (i.e. less than GT level 2, except forſ
, which is already normalized tos
)historic_latin
:
likeLevenshtein
, but decomposing non-vocalic ligatures before and treating as equivalent (i.e. zero distances) confusions of certain semantically close characters often found in historic texts (e.g. umlauts with combining lettere
as inWuͤſte
instead of toWüſte
,ſ
vss
, or quotation/citation marks; GT level 1)
...perplexity measurement...
Installation
Besides OCR-D, this builds on Keras/Tensorflow.
Required Ubuntu packages:
- Python (
python
orpython3
) - pip (
python-pip
orpython3-pip
) - venv (
python-venv
orpython3-venv
)
Create and activate a virtual environment as usual.
To install Python dependencies:
make deps
Which is the equivalent of:
pip install -r requirements.txt
To install this module, then do:
make install
Which is the equivalent of:
pip install .
The module can use CUDA-enabled GPUs (when sufficiently installed), but can also run on CPU only. Models are always interchangable.
Usage
This packages has the following user interfaces:
command line interface cor-asv-ann-train
To be used with string arguments and plain-text files.
Usage: cor-asv-ann-train [OPTIONS] [DATA]...
Train a correction model.
Configure a sequence-to-sequence model with the given parameters.
If given `load_model`, and its configuration matches the current
parameters, then load its weights. If given `init_model`, then transfer
its mapping and matching layer weights. (Also, if its configuration has 1
less hidden layers, then fixate the loaded weights afterwards.) If given
`reset_encoder`, re-initialise the encoder weights afterwards.
Then, regardless, train on the file paths `data` using early stopping. If
no `valdata` were given, split off a random fraction of lines for
validation. Otherwise, use only those files for validation.
If the training has been successful, save the model under `save_model`.
Options:
-m, --save-model FILE model file for saving
--load-model FILE model file for loading (incremental/pre-training)
--init-model FILE model file for initialisation (transfer from LM
or shallower model)
--reset-encoder reset encoder weights after load/init
-w, --width INTEGER RANGE number of nodes per hidden layer
-d, --depth INTEGER RANGE number of stacked hidden layers
-v, --valdata FILE file to use for validation (instead of random
split)
--help Show this message and exit.
command line interface cor-asv-ann-eval
To be used with string arguments and plain-text files.
Usage: cor-asv-ann-eval [OPTIONS] [DATA]...
Evaluate a correction model.
Load a sequence-to-sequence model from the given path.
Then apply on the file paths `data`, comparing predictions (both greedy
and beamed) with GT target, and measuring error rates.
Options:
-m, --load-model FILE model file to load
-f, --fast only decode greedily
-r, --rejection FLOAT RANGE probability of the input characters in all
hypotheses (set 0 to use raw predictions)
-n, --normalization [Levenshtein|NFC|NFKC|historic_latin]
normalize character sequences before
alignment/comparison (set Levenshtein for
none)
-l, --gt-level INTEGER RANGE GT transcription level to use for
historic_latin normlization (1: strongest,
3: none)
-c, --confusion INTEGER RANGE show this number of most frequent (non-
identity) edits (set 0 for none)
--help Show this message and exit.
command line interface cor-asv-ann-repl
This tool provides a Python read-eval-print-loop for interactive usage (including some visualization):
Usage: cor-asv-ann-repl [OPTIONS]
Try a correction model interactively.
Import Sequence2Sequence, instantiate `s2s`, then enter REPL. Also,
provide function `transcode_line` for single line correction.
Options:
--help Show this message and exit.
Here is what you see after starting up the interpreter:
usage example:
>>> s2s.load_config('model')
>>> s2s.configure()
>>> s2s.load_weights('model')
>>> s2s.evaluate(['filename'])
>>> transcode_line('hello world!')
now entering REPL...
Python 3.6.7 (default, Oct 22 2018, 11:32:17)
[GCC 8.2.0] on linux
Type "help", "copyright", "credits" or "license" for more information.
(InteractiveConsole)
OCR-D processor interface ocrd-cor-asv-ann-process
To be used with PAGE-XML documents in an OCR-D annotation workflow.
Input could be anything with a textual annotation (TextEquiv
on the given textequiv_level
).
Pretrained model files are contained in the models subrepository. At runtime, you can use both absolute and relative paths for model files. The latter are searched for in the installation directory, and under the path in the environment variable CORASVANN_DATA
(if given).
"ocrd-cor-asv-ann-process": {
"executable": "ocrd-cor-asv-ann-process",
"categories": [
"Text recognition and optimization"
],
"steps": [
"recognition/post-correction"
],
"description": "Improve text annotation by character-level encoder-attention-decoder ANN model",
"input_file_grp": [
"OCR-D-OCR-TESS",
"OCR-D-OCR-KRAK",
"OCR-D-OCR-OCRO",
"OCR-D-OCR-CALA",
"OCR-D-OCR-ANY"
],
"output_file_grp": [
"OCR-D-COR-ASV"
],
"parameters": {
"model_file": {
"type": "string",
"format": "uri",
"content-type": "application/x-hdf;subtype=bag",
"description": "path of h5py weight/config file for model trained with cor-asv-ann-train",
"required": true,
"cacheable": true
},
"textequiv_level": {
"type": "string",
"enum": ["line", "word", "glyph"],
"default": "glyph",
"description": "PAGE XML hierarchy level to read/write TextEquiv input/output on"
},
"rejection_threshold": {
"type": "number",
"format": "float",
"default": 0.5,
"description": "minimum probability of the candidate corresponding to the input character in each hypothesis during beam search, helps balance precision/recall trade-off; set to 0 to disable rejection (max recall) or 1 to disable correction (max precision)"
},
"relative_beam_width": {
"type": "number",
"format": "float",
"default": 0.2,
"description": "minimum fraction of the best candidate's probability required to enter the beam in each hypothesis; controls the quality/performance trade-off"
},
"fixed_beam_width": {
"type": "number",
"format": "integer",
"default": 15,
"description": "maximum number of candidates allowed to enter the beam in each hypothesis; controls the quality/performance trade-off"
},
"fast_mode": {
"type": "boolean",
"default": false,
"description": "decode greedy instead of beamed, with batches of parallel lines instead of parallel alternatives; also disables rejection and beam parameters; enable if performance is far more important than quality"
}
}
}
...
OCR-D processor interface ocrd-cor-asv-ann-evaluate
To be used with PAGE-XML documents in an OCR-D annotation workflow.
Inputs could be anything with a textual annotation (TextEquiv
on the line level), but at least 2. The first in the list of input file groups will be regarded as reference/GT.
There are various evaluation metrics available.
The tool can also aggregate and show the most frequent character confusions.
"ocrd-cor-asv-ann-evaluate": {
"executable": "ocrd-cor-asv-ann-evaluate",
"categories": [
"Text recognition and optimization"
],
"steps": [
"recognition/evaluation"
],
"description": "Align different textline annotations and compute distance",
"input_file_grp": [
"OCR-D-GT-SEG-LINE",
"OCR-D-OCR-TESS",
"OCR-D-OCR-KRAK",
"OCR-D-OCR-OCRO",
"OCR-D-OCR-CALA",
"OCR-D-OCR-ANY",
"OCR-D-COR-ASV"
],
"parameters": {
"metric": {
"type": "string",
"enum": ["Levenshtein", "NFC", "NFKC", "historic_latin"],
"default": "Levenshtein",
"description": "Distance metric to calculate and aggregate: historic_latin for GT level 1, NFKC for GT level 2 (except ſ-s), Levenshtein for GT level 3"
},
"confusion": {
"type": "number",
"format": "integer",
"minimum": 0,
"default": 0,
"description": "Count edits and show that number of most frequent confusions (non-identity) in the end."
}
}
}
There is no output file group for the evaluation tool: it only uses logging.
Testing
not yet! ...
Project details
Release history Release notifications | RSS feed
Download files
Download the file for your platform. If you're not sure which to choose, learn more about installing packages.
Source Distribution
Built Distribution
File details
Details for the file ocrd_cor_asv_ann-0.1.6.tar.gz
.
File metadata
- Download URL: ocrd_cor_asv_ann-0.1.6.tar.gz
- Upload date:
- Size: 70.6 kB
- Tags: Source
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/3.2.0 pkginfo/1.5.0.1 requests/2.21.0 setuptools/41.0.0 requests-toolbelt/0.9.1 tqdm/4.42.1 CPython/3.6.7
File hashes
Algorithm | Hash digest | |
---|---|---|
SHA256 | be38445d1167611a86c7b1ac84f2da5e9c8a4d4fa94746197f73aa4797eaaf75 |
|
MD5 | 5ab3ebd0eb69a6909091dcbff34ac03e |
|
BLAKE2b-256 | ae219e5c6f654519d13845bfb261483497d53fc7b0720c98f816c92196874ae1 |
File details
Details for the file ocrd_cor_asv_ann-0.1.6-py3-none-any.whl
.
File metadata
- Download URL: ocrd_cor_asv_ann-0.1.6-py3-none-any.whl
- Upload date:
- Size: 76.0 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/3.2.0 pkginfo/1.5.0.1 requests/2.21.0 setuptools/41.0.0 requests-toolbelt/0.9.1 tqdm/4.42.1 CPython/3.6.7
File hashes
Algorithm | Hash digest | |
---|---|---|
SHA256 | 0db0fb146b72a4b66376e7b16022f2dbf9a2943411edb2b2bcb35773353d3639 |
|
MD5 | 244fcdc2f717079af4897d0f7992a461 |
|
BLAKE2b-256 | 3ec4bd6e9f55511084f8a033968b58e1b9e405e3a8f8398fd3457008c53b5e79 |