Skip to content

Fair evaluation and detailed error analysis of labeled spans

License

Notifications You must be signed in to change notification settings

rubcompling/FairEval

 
 

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 

History

23 Commits
 
 
 
 
 
 
 
 
 
 
 
 

Repository files navigation

FairEval

The traditional evaluation of labeled spans with precision, recall, and F1-score leads to double penalties for close-to-correct annotations. As Manning (2006) argues in an article about named entity recognition, this can lead to undesirable effects when systems are optimized for these traditional metrics.

Building on his ideas, in Ortmann (2022), I developed a new evaluation method that more accurately reflects true annotation quality by ensuring that every error is counted only once. In addition to the traditional categories of true positives (TP), false positives (FP), and false negatives (FN), the new method takes into account the more fine-grained error types suggested by Manning: labeling errors (LE), boundary errors (BE), and labeling-boundary errors (LBE).

In addition, I also distinguish between different types of boundary errors, which enable an even more detailed error analysis:

  • BES: the system's annotation is smaller than the target span
  • BEL: the system's annotation is larger than the target span
  • BEO: the system span overlaps with the target span

This repository contains a Python implementation of the suggested algorithm to identify the new error types in flat and hierarchical spans. The program FairEval.py is located in the /src folder and can be used as a stand-alone tool (see here) or be imported as a Python module (see here).

In the /data folder, I also provide the example data sets from my paper with system and target annotations of named entities, chunks, and topological fields in a custom input format. The config files that I used for the exemplary evaluation are located in the /config folder. The result files can be found in the /eval folder.

For more details on the additional error types, the new algorithm, and the example evaluation, please refer to my paper.

NEW: Hugging Face Version

The FairEval algorithm has been integrated into Hugging Face for free and easy usage by the AI/NLP community. It provides the default behaviour as described in the paper, with FP, FN, LE, BE, and LBE errors and default weights. Thanks to Ignacio Llorca Rodríguez and Florian Borchert (@phlobo) from @hpi-dhc for the implementation!

If you need more customization options, see the documentation in this repository.

Table of content

  1. Usage
  2. Example data
  3. Acknowledgement
  4. References

Usage

The usage of this FairEval implementation requires Python 3 to be installed.

Stand-alone version

To use FairEval as a stand-alone version to evaluate your annotations, call it via the command line:

py FairEval.py --config file.config

The config argument specifies a configuration file with the required parameters. For more information on the config file, see the section on configuration.

When called as described above, the tool will import your target and system data and carry out the evaluation according to the specified options. The results will be written to an output file or the command line.

Configuration

When FairEval is used as a stand-alone version, it requires a config file that specifies the evaluation settings. The following parameters are mandatory:

  • target_in: file or directory containing the target annotation (filename(s) must be identical to the ones in system_in)
  • system_in: file or directory containing the system annotation (filename(s) must be identical to the ones in target_in)

Optionally, you can also specify:

  • labels: comma-separated list of labels to evaluate (e.g., NP, PP, AP, ADVP; default: all)
  • exclude: comma-separated list of labels to exclude (e.g., DUMMY, ROOT if you used dummy labels or don't want to evaluate the ROOT node; default NONE, EMPTY)
  • ignore_punct: whether to ignore punctuation during evaluation (only possible for CoNLL input with POS annotation; default: True)
  • focus: which label to count in the case of labeling and labeling boundary errors (system or target (default))
  • eval_method: comma-separated list of evaluation methods to apply (e.g., traditional, fair, weighted; default: all)
  • weights: formula with weights for different error types for the weighted evaluation (see below)
  • eval_out: where to write the results; if a directory is given, will create a file eval.csv there; if omitted, will output results to the command line (see also here)

Input

The stand-alone version of this implementation accepts two different input formats: span files and CoNLL2000 files.

1. Span files

A span file contains one span per line with sentences separated by an empty line. The lines are expected to contain four tab-separated columns:

Column Content
Label Span label
Begin Index of the first included token (must be convertible to int)
End Index of the last included token (must be convertible to int and equal or greater than begin)
Tokens Comma-separated list of indices of the tokens in the span (must be convertible to int with begin <= i <= end); if no (valid) indices are given, the range begin:end is used

For an example of span files, have a look at the /data folder of this repository. This is a sample from the NER data:

PER	6	7	6, 7        <-- This is the first span of type PER from token 6 to 7 in the first sentence
PER	13	14	13, 14      <-- And the second span of type PER from token 13 to 14 in the first sentence
ORG	20	20	20

PER	4	4	4           <-- The new line indicates a new sentence

OTH	2	5	2, 3, 4, 5  <-- And here is a third sentence
PER	19	20	19, 20
ORG	26	26	26

System and target files are aligned automatically, so make sure that they contain the same number of sentences! Otherwise, annotations from different sentences will be compared with each other. If a sentence does not include any spans, add a dummy span instead like:

EMPTY    999    999    999

2. CoNLL2000 files

CoNLL2000 files contain one token per line with sentences separated by an empty line. Each line consists of three space-separated columns.

Column Content
Form The word form
XPOS POS tag of the word (ideally STTS, UPOS, or PTB)
Annotation Span annotation in BIO format (see below); multiple spans are separated with a pipe symbol

BIO tags consist of the token's position in the span (begin B, inside I, outside O), a dash - and the span label, e.g., B-NP, I-AC, or in the case of stacked annotations I-RELC|B-NP.

The importer accepts O, _ and an empty string as annotations outside of spans.

For stacked spans, the annotations must be sorted by length from left to right. A span on the right side of a pipe sign is included in the span left to the pipe sign. Overlapping spans are not supported.

An example for a stacked span annotation could be (displayed with tabs instead of spaces for better readability):

Ruhr-Universität  NE    B-ORG          
Bochum            NE    I-ORG|B-LOC

It is possible to encode complex hierarchical structures with BIO tags (as long as they are continuous; discontinuous annotations or "crossing edges" cannot be encoded). The following example displays a toy constituency tree in the form of BIO tags.

Das         This        B-S|B-NP
ist         is          I-S|B-VP
ein         a           I-S|I-VP|B-NP
einfacher   simple      I-S|I-VP|I-NP|B-AP
Satz        sentence    I-S|I-VP|I-NP
.           .           I-S|

Output

When run as a stand-alone tool, FairEval outputs:

  1. The evaluation settings
  2. Results for each evaluation method (traditional, fair, and/or weighted evaluation) including per-label and overall counts of true positives and errors and the calculated statistics
  3. A comparison of precision, recall, and F1-score for the different evaluation methods
  4. The confusion matrix with target labels (rows) and system labels (columns). The _ row shows false positives, the _ column contains the false negatives. On the diagonal, boundary errors are displayed. The other cells contain labeling and labeling-boundary errors.
  5. The distribution of labels in the target data in absolute numbers and percent

The /eval folder of this repository contains the result files for the example data from the /data folder.

Weights

In my paper, I suggested that every labeling, boundary and labeling-boundary error should count as half false positive and half false negative. I also gave suggestions for alternative weights for different types of boundary errors. The weights parameter in the config file allows to specify your own weights that should be used for the calculation of precision and recall.

You can also use the weight formula to indicate, which of the available additional error types should be considered.

Weights are parsed from comma-separated input formulas of the form:

error_type = weight1 * TP + weight2 * FP + weight3 * FN

You can leave out zero-weights. For example, in my paper, I suggested the following formula:

weights = LE = 0.5 FP + 0.5 FN, BES = 0.5 TP + 0.5 FN, BEL = 0.5 TP + 0.5 FP, BEO = 0.5 TP + 0.25 FP + 0.25 FN, LBE = 0.5 FP + 0.5 FN

Asterisks and spaces are optional. As long as commas and equal signs are used properly, the formulas will likely be recognized properly. You can have a look at the settings section in the output file to double-check if your weights were parsed correctly.

Import as module

If you want to integrate and use FairEval in your own code, you can import it as a Python module with

import FairEval

Suggested applications include:

  1. Counting the different fine-grained error types in your data for a detailed error analysis
  2. Determine the type of overlap for a (labeling-)boundary error
  3. Calculate fair or customized precision, recall, and F1 scores on your error counts
  4. Get simple statistics about your data: how often do certain labels occur in the data set

Determine fine-grained error types

To count the different fine-grained error types in your data set, according to the algorithm described in Ortmann (2022), you can use the function compare_spans(target_spans, system_spans, focus). The focus argument is optional and can be set to "system" or "target" (default), depending on which label should be counted in the case of labeling and labeling-boundary errors.

The function expects a list of target spans and system spans. Each span is a 4-tuple of

  • label: the span type as string
  • begin: index of the first token; equals end for spans of length 1
  • end: index of the last token; equals begin for spans of length 1
  • tokens: a set of token indices included in the span (this allows the correct evaluation of partially and multiply overlapping spans; you can also use it to exclude punctutation inside of spans from the evaluation)

To allow for changes of the token set, the span tuple is actually implemented as a list, e.g.,

[["NP", 1, 3, {1, 2, 3}], ["PP", 5, 10, {5, 6, 7, 8, 9, 10}], ...]

The function first performs traditional evaluation on these spans to identify true positives, false positives, and false negatives. Then, the additional error types for fair evaluation are determined, following steps 1 to 4:

  1. Count 1:1 mappings (TP, LE)
  2. Count boundary errors (BE = BES + BEL + BEO)
  3. Count labeling-boundary errors (LBE)
  4. Count 1:0 and 0:1 mappings (FN, FP)

The function outputs a dictionary containing

  • the counts of TP, FP, and FN according to traditional evaluation (per label and overall)
  • the counts of TP, FP, LE, BE, BES, BEL, BEO, and FN (per label and overall; BE = BES + BEL + BEO)
  • a confusion matrix

The dictionary has the following structure:

{"overall" : {"traditional" : {"TP" : count, "FP" : count, "FN" : count},
              "fair" :        {"TP" : count, "FP" : count, "FN" : count,
                               "additionalErrorType1" : count, "additionalErrorType2" : count, ...}
             },
 "per_label" : {"traditional" : {"label1" : {"TP" : count, "FP" : count, "FN" : count},
                                 "label2" : ... },
                "fair" :        {"label1" : {"TP" : count, "FP" : count, "FN" : count,
                                             "additionalErrorType1" : count, "additionalErrorType2" : count, ...},
                                 "label2" : ... } 
               },
 "conf" : { see below }
}

The confusion matrix is implemented as a dictionary with the following structure:

{target_label1 : {system_label1 : count,
                  system_label2 : count,
                  ...},
 target_label2 : ... }
 
with an underscore '_' representing an empty label (FN/FP)

Determine span overlap type

If you want to know, if and how two spans overlap, you can use the function overlap_type(span1, span2). As input, it expects two tuples of integers (beginSpan1, endSpan1) and (beginSpan2, endSpan2), where begin and end are the indices of the corresponding tokens. For spans of length one, begin equals end.

The function checks, if and how span1 and span2 overlap. The first span serves as the basis against which the second span is evaluated.

span1 ---XXXX---
span2 ---XXXX--- TP (identical)
span2 ----XXXX-- BEO (overlap)
span2 --XXXX---- BEO (overlap)
span2 ----XX---- BES (smaller)
span2 ---XX----- BES (smaller)
span2 --XXXXXX-- BEL (larger)
span2 --XXXXX--- BEL (larger)
span2 -X-------- False (no overlap)

Possible output values are either one of the following strings

  • "TP" = span1 and span2 are identical, i.e., actually no boundary error here
  • "BES" = span2 is shorter and contained within span1 (with at most one identical boundary)
  • "BEL" = span2 is longer and contains span1 (with at most one identical boundary)
  • "BEO" = span1 and span2 overlap with no identical boundary
  • or False if span1 and span2 do not overlap.

Calculate fair or customized scores

Suppose you already have an evaluation dictionary that contains your error counts in the following form:

{
 "TP"                   : count, 
 "FP"                   : count, 
 "FN"                   : count,
 "additionalErrorType1" : count, 
 "additionalErrorType2" : count, 
 ...
}

Then, you can use the functions precision(eval_dict, version, weights) and recall(eval_dict, version, weights) to calculate different versions of precision and recall.

Precision is calculated as the number of true positives divided by the number of true positives plus false positives plus (optionally) additional error types. Similarly, recall is calculated as the number of true positives divided by the number of true positives plus false negatives plus (optionally) additional error types.

As input, the functions expect a dictionary with error types as keys and counts as values. For 'traditional' evaluation, true positives (key: TP) and false positives (key: FP) or false negatives (key: FN) are required. The 'fair' evaluation is based on true positives (TP), false positives (FP) or false negatives (FN), labeling errors (LE), boundary errors (BE) and labeling-boundary errors (LBE). The 'weighted' evaluation can include any error type that is given as key in the weight dictionary. For missing keys, the count is set to 0.

Which evaluation method is used, depends on the second argument. Options are traditional (default), fair, and weighted. If no weight dictionary is specified, weighted is identical to fair.

The weight dictionary is passed to the functions to specify how much an error type should count as one of the traditional error types (or as true positive). Per default, every traditional error is counted as one error (or true positive) and each error of the additional types is counted as half false positive and half false negative:

{
 "TP" :  {"TP" : 1},
 "FP" :  {"FP" : 1},
 "FN" :  {"FN" : 1},
 "LE" :  {"TP" : 0, "FP" : 0.5, "FN" : 0.5},
 "BE" :  {"TP" : 0, "FP" : 0.5, "FN" : 0.5},
 "LBE" : {"TP" : 0, "FP" : 0.5, "FN" : 0.5}
}

Other suggested weights to count boundary errors as half true positives:

{
 "TP" :  {"TP" : 1},
 "FP" :  {"FP" : 1},
 "FN" :  {"FN" : 1},
 "LE" :  {"TP" : 0, "FP" : 0.5, "FN" : 0.5},
 "BE" :  {"TP" : 0.5, "FP" : 0.25, "FN" : 0.25},
 "LBE" : {"TP" : 0, "FP" : 0.5, "FN" : 0.5}
}

Or to include different types of boundary errors (this corresponds to the formula in the weights section):

{
 "TP" :  {"TP" : 1},
 "FP" :  {"FP" : 1},
 "FN" :  {"FN" : 1},
 "LE" :  {"TP" : 0, "FP" : 0.5, "FN" : 0.5},
 "LBE" : {"TP" : 0, "FP" : 0.5, "FN" : 0.5},
 "BEO" : {"TP" : 0.5, "FP" : 0.25, "FN" : 0.25},
 "BES" : {"TP" : 0.5, "FP" : 0, "FN" : 0.5},
 "BEL" : {"TP" : 0.5, "FP" : 0.5, "FN" : 0}
} 

Finally, the function fscore(eval_dict) takes a dictionary that contains a precision (key: Prec) and recall (key: Rec) value and calculates the harmonic mean, i.e., the F1-score.

In case of a ZeroDivisionError, the score of the respective functions is set to 0.

Get simple data statistics

By calling the function annotation_stats(span_list, result_dict) with your list of (target) spans, you can get an overview of the distribution of labels in your data. The function expects a list of target spans with each span being a 4-tuple (label, begin, end, tokens). It counts the frequency of labels in the list and stores the result in the dictionary under the key data_stats.

For the NER target data, the result would look like this:

{ 'data_stats' : { 'OTH' : 778,
                   'PER' : 1694,
                   'ORG' : 1330,
                   'LOC' : 2376 } }

Example data

In my study, I run an example evaluation on three different types of spans: named entities, chunks, and topological fields. The /data folder contains the target (i.e., gold standard) annotation and the system annotation. For license and copyright reasons, the data is provided in a custom format that only contains the span annotations and does not allow for the reconstruction of the original corpus data.

Annotation Evaluation data set
NER The NER data set corresponds to the test partition of the GermEval 2014 data set (Benikova et al. 2014). Only top-level entities from the four main classes are included.
Chunks The chunk annotations are taken from 10% of the chunked TüBa-D/Z corpus (Telljohann et al. 2017)
Topological fields The topological field annotations are taken from 10% of the TüBa-D/Z treebank (Telljohann et al. 2017)

More details about training and evaluation data sets and the NLP tools used for the annotation can be found in my paper.

Acknowledgement

If you use FairEval in your work, please cite the corresponding paper:

  • Ortmann, Katrin. 2022. Fine-Grained Error Analysis and Fair Evaluation of Labeled Spans. In Proceedings of the Language Resources and Evaluation Conference (LREC), Marseille, France, pages 1400–1407. PDF

References

Benikova, D., Biemann, C., Kisselew, M., and Padó, S. 2014. Germeval 2014 named entity recognition shared task: Companion paper.

Manning, C. 2006. Doing Named Entity Recognition? Don’t optimize for F1. Retreived from https://nlpers.blogspot.com/2006/08/doing-named-entity-recognition-dont.html.

Ortmann, K. 2022. Fine-Grained Error Analysis and Fair Evaluation of Labeled Spans. In Proceedings of the Language Resources and Evaluation Conference (LREC), Marseille, France, pages 1400–1407.

Telljohann, H., Hinrichs, E. W., Kübler, S., Zinsmeister, H., and Beck, K. 2017. Stylebook for the Tübingen Treebank of Written German (TüBa-D/Z). Seminar für Sprachwissenschaft, Universität Tübingen, Germany.

About

Fair evaluation and detailed error analysis of labeled spans

Resources

License

Stars

Watchers

Forks

Releases

No releases published

Packages

No packages published

Languages

  • Python 100.0%