Skip to content

ETUDE (Evaluation Tool for Unstructured Data and Extractions) is a Python-based tool that provides consistent evaluation options across a range of annotation schemata and corpus formats

License

Notifications You must be signed in to change notification settings

MUSC-TBIC/etude-engine

Repository files navigation

pytest Status Documentation Status

Documentation

The latest documentation (compiled from the contents of the docs folder) can be viewed on-line: ETUDE Engine's documentation

Documentation for the ETUDE engine is managed via reStructuredText files and Sphinx. If you don't have Sphinx installed, you should check out a quick primer (First Steps with Sphinx) or install it as below:

## If you don't have Sphinx installed already
pip install Sphinx

## Generate a locally viewable HTML version
cd docs
make html

The latest version of the documentation can be generated as locally viewable HTML: file:///path/to/git/repository/docs/_build/html/index.html

Sample Runs

Basic Run

The simplest test run requires that we specify a reference directory and a test directory. The default file matching assumes that our reference and test files match names exactly and both end in .xml. With just the two directory arguments, we get micro-average scores for the default metrics across the full directory.

python $ETUDE_DIR/etude.py \
    --reference-input $ETUDE_DIR/tests/data/i2b2_2016_track-1_reference \
    --test-input $ETUDE_DIR/tests/data/i2b2_2016_track-1_test
exact TP FP TN FN
micro-average 374.0 8.0 0.0 108.0

Note

You may get a warning if you run the previous command from a directory other than $ETUDE_DIR:

ERROR: Config file is missing or unreadable: config/i2b2_2016_track-1.conf ERROR: No reference patterns extracted from config. Bailing out now.

This warning is because the default configuration files use relative paths. See the section below

In the next sample runs, you can see how to include a per-file score breakdown and a per-annotation-type score breakdown.

python $ETUDE_DIR/etude.py \
    --reference-input $ETUDE_DIR/tests/data/i2b2_2016_track-1_reference \
    --test-input $ETUDE_DIR/tests/data/i2b2_2016_track-1_test \
    --by-file
exact TP FP TN FN
micro-average 374.0 8.0 0.0 108.0
0005_gs.xml 36.0 0.0 0.0 0.0
0016_gs.xml 23.0 0.0 0.0 31.0
0267_gs.xml 29.0 0.0 0.0 34.0
0273_gs.xml 0.0 0.0 0.0 35.0
0389_gs.xml 32.0 8.0 0.0 8.0
0475_gs.xml 46.0 0.0 0.0 0.0
0617_gs.xml 38.0 0.0 0.0 0.0
0709_gs.xml 45.0 0.0 0.0 0.0
0982_gs.xml 100.0 0.0 0.0 0.0
0992_gs.xml 25.0 0.0 0.0 0.0
macro-average by file 374.0 8.0 0.0 108.0
python $ETUDE_DIR/etude.py \
    --reference-input $ETUDE_DIR/tests/data/i2b2_2016_track-1_reference \
    --test-input $ETUDE_DIR/tests/data/i2b2_2016_track-1_test \
    --by-type
exact TP FP TN FN
micro-average 374.0 8.0 0.0 108.0
Age 63.0 2.0 0.0 29.0
DateTime 91.0 2.0 0.0 33.0
HCUnit 61.0 4.0 0.0 15.0
OtherGeo 1.0 0.0 0.0 4.0
OtherID 7.0 0.0 0.0 0.0
OtherOrg 18.0 0.0 0.0 3.0
Patient 16.0 0.0 0.0 3.0
PhoneFax 5.0 0.0 0.0 1.0
Provider 54.0 0.0 0.0 10.0
SSN 0.0 0.0 0.0 0.0
StateCountry 24.0 0.0 0.0 9.0
StreetCity 28.0 0.0 0.0 1.0
Zip 4.0 0.0 0.0 0.0
eAddress 2.0 0.0 0.0 0.0
macro-average by type 374.0 8.0 0.0 108.0

Specifying Annotation Configs

We can use the same reference corpus to analyze annotations generated by UIMA's DateTime tutorial (see link below). A minimal run requires creating a matching dataset for the default configurations. Process the I2B2 dev set using the DateTime tutorial provided with UIMA. Then, because the output files for the I2B2 dev-annotations end in .xml but the UIMA tutorial files end in .txt, you need to specify a file suffix translation rule. Also, the annotations are encoded slightly differently by the tutorial descriptor than by the I2B2 reference. As such, you will need to load a different configuration for the test directory to tell ETUDE how to find and extract the annotations.

Link: http://uima.apache.org/downloads/releaseDocs/2.2.2-incubating/docs/html/tutorials_and_users_guides/tutorials_and_users_guides.html#ugr.tug.aae.building_aggregates

export I2B2_CORPUS="/path/to/Corpora and annotations/2016 NGRID challenge (deid)/2016_track_1-deidentification"

export I2B2_OUTPUT="/tmp/datetime-out"
mkdir $I2B2_OUTPUT

$UIMA_HOME/bin/runAE.sh \
  $UIMA_HOME/examples/descriptors/tutorial/ex3/TutorialDateTime.xml \
  $I2B2_CORPUS/dev-text \
  $I2B2_OUTPUT

python $ETUDE_DIR/etude.py \
    --reference-input $ETUDE_DIR/tests/data/i2b2_2016_track-1_reference \
    --test-input $I2B2_OUTPUT \
    --by-type \
    --file-suffix ".xml" ".txt" \
    --test-config config/CAS_XMI.conf
exact TP FP TN FN
micro-average 0.0 39.0 0.0 124.0
DateTime 0.0 39.0 0.0 124.0
macro-average 0.0 39.0 0.0 124.0

You may be surprised that the UIMA tutorial doesn't seem to get a single DateTime annotation correct. When you look at the annotations, they seem correct. This discrepancy arises because the default matching style is "exact" matching. This means that the character offset spans needs to be identical. You can select a different matching style with the --fuzzy-match-flags option. Using partial matching instead shows True Positives (TPs) for all spans that at least partially overlap. See the more thorough coverage on the various options under Evaluating Matches in our documentation.

python $ETUDE_DIR/etude.py \
    --reference-input $ETUDE_DIR/tests/data/i2b2_2016_track-1_reference \
    --test-input $I2B2_OUTPUT \
    --by-type \
    --file-suffix ".xml" ".txt" \
    --test-config config/CAS_XMI.conf \
    --fuzzy-match-flags partial
exact TP FP TN FN
micro-average 39.0 0.0 0.0 85.0
DateTime 39.0 0.0 0.0 85.0
macro-average 39.0 0.0 0.0 85.0

If you run this example without the -–test-config argument, you should see all FN matches because nothing can be extracted from the test corpus.

python $ETUDE_DIR/etude.py \
    --reference-input $ETUDE_DIR/tests/data/i2b2_2016_track-1_reference \
    --test-input $I2B2_OUTPUT \
    --file-suffix ".xml" ".txt"
exact TP FP TN FN
micro-average 0.0 0.0 0.0 482.0

Scoring on Different Fields

The above examples show scoring based on the default key in the configuration file used for matching the reference to the test configuration. You may wish to group annotations on different fields, such as the parent class or long description. See the more thorough coverage on the various options under Configuration Files in our documentation.

## You can see output for this command above
python $ETUDE_DIR/etude.py \
    --reference-input $ETUDE_DIR/tests/data/i2b2_2016_track-1_reference \
    --test-input $ETUDE_DIR/tests/data/i2b2_2016_track-1_test \
    --by-type

python $ETUDE_DIR/etude.py \
    --reference-input $ETUDE_DIR/tests/data/i2b2_2016_track-1_reference \
    --test-input $ETUDE_DIR/tests/data/i2b2_2016_track-1_test \
    --by-type \
    --score-key "Parent"

python $ETUDE_DIR/etude.py \
    --reference-input $ETUDE_DIR/tests/data/i2b2_2016_track-1_reference \
    --test-input $ETUDE_DIR/tests/data/i2b2_2016_track-1_test \
    --by-type \
    --score-key "i2b2 14/16"
exact TP FP TN FN
micro-average 375.0 7.0 0.0 107.0
Address 56.0 0.0 0.0 10.0
Contact Information 7.0 0.0 0.0 1.0
Identifiers 7.0 0.0 0.0 0.0
Locations 80.0 4.0 0.0 22.0
Names 70.0 0.0 0.0 13.0
Time 155.0 3.0 0.0 61.0
macro-average by type 375.0 7.0 0.0 107.0
exact TP FP TN FN
micro-average 374.0 8.0 0.0 108.0
ACCOUNT 0.0 0.0 0.0 0.0
AGE 63.0 2.0 0.0 29.0
BIOID 0.0 0.0 0.0 0.0
CITY 24.0 0.0 0.0 1.0
COUNTRY 14.0 0.0 0.0 7.0
DATE 91.0 2.0 0.0 33.0
DEVICE 0.0 0.0 0.0 0.0
DOCTOR 54.0 0.0 0.0 10.0
EMAIL 0.0 0.0 0.0 0.0
FAX 0.0 0.0 0.0 0.0
HEALTHPLAN 0.0 0.0 0.0 0.0
HOSPITAL 61.0 4.0 0.0 15.0
IDNUM 0.0 0.0 0.0 0.0
IPADDRESS 0.0 0.0 0.0 0.0
LICENSE 7.0 0.0 0.0 0.0
LOCATION-OTHER 1.0 0.0 0.0 4.0
MEDICALRECORD 0.0 0.0 0.0 0.0
ORGANIZATION 18.0 0.0 0.0 3.0
PATIENT 16.0 0.0 0.0 3.0
PHONE 5.0 0.0 0.0 1.0
SSN 0.0 0.0 0.0 0.0
STATE 10.0 0.0 0.0 2.0
STREET 4.0 0.0 0.0 0.0
URL 2.0 0.0 0.0 0.0
USERNAME 0.0 0.0 0.0 0.0
VEHICLE 0.0 0.0 0.0 0.0
ZIP 4.0 0.0 0.0 0.0
macro-average by type 374.0 8.0 0.0 108.0

Custom Evaluation Print-Outs

The majority of you evaluation output customization can be handled by the above command-line arguments. However, sometimes you'll need to generate output that exactly matches some very specific formatting requirements. For these instances, ETUDE supports custom print functions. Currently, those print functions must be hard-coded into scoring_metrics.py. Our roadmap includes the ability to load and trigger these print functions from a standard folder to make the system much more modular. Until that point, you can see an example custom print-out that targets the 2018 n2c2 Track 1 output format. The configurations for this sample are in our sister repository: ETUDE Engine Configs for n2c2 The original evaluation script for the competition, used as a point of reference, can be found on github: Evaluation scripts for the 2018 N2C2 shared tasks on clinical NLP See the more thorough coverage on the various Output Formats in our documentation.

export ETUDE_DIR=etude-engine
export ETUDE_CONFIGS_DIR=etude-engine-configs

export N2C2_DATA=/tmp/n2c2

python ${ETUDE_DIR}/etude.py \
  --reference-input ${N2C2_DATA}/train_annotations \
   --reference-config ${ETUDE_CONFIGS_DIR}/n2c2/2018_n2c2_track-1.conf \
   --test-input ${N2C2_DATA}/train_annotations \
   --test-config ${ETUDE_CONFIGS_DIR}/n2c2/2018_n2c2_track-1.conf \
   --no-metrics \
   --print-custom "2018 n2c2 track 1" \
   --fuzzy-match-flag exact \
   --file-suffix ".xml" \
   --empty-value 0.0


******************************************* TRACK 1 ********************************************
                      ------------ met -------------    ------ not met -------    -- overall ---
                      Prec.   Rec.    Speci.  F(b=1)    Prec.   Rec.    F(b=1)    F(b=1)  AUC
           Abdominal  1.0000  1.0000  1.0000  1.0000    1.0000  1.0000  1.0000    1.0000  1.0000
        Advanced-cad  1.0000  1.0000  0.0000  1.0000    0.0000  0.0000  0.0000    0.5000  0.5000
       Alcohol-abuse  0.0000  0.0000  1.0000  0.0000    1.0000  1.0000  1.0000    0.5000  0.5000
          Asp-for-mi  1.0000  1.0000  0.0000  1.0000    0.0000  0.0000  0.0000    0.5000  0.5000
          Creatinine  1.0000  1.0000  1.0000  1.0000    1.0000  1.0000  1.0000    1.0000  1.0000
       Dietsupp-2mos  1.0000  1.0000  1.0000  1.0000    1.0000  1.0000  1.0000    1.0000  1.0000
          Drug-abuse  0.0000  0.0000  1.0000  0.0000    1.0000  1.0000  1.0000    0.5000  0.5000
             English  1.0000  1.0000  0.0000  1.0000    0.0000  0.0000  0.0000    0.5000  0.5000
               Hba1c  1.0000  1.0000  1.0000  1.0000    1.0000  1.0000  1.0000    1.0000  1.0000
            Keto-1yr  0.0000  0.0000  1.0000  0.0000    1.0000  1.0000  1.0000    0.5000  0.5000
      Major-diabetes  1.0000  1.0000  1.0000  1.0000    1.0000  1.0000  1.0000    1.0000  1.0000
     Makes-decisions  1.0000  1.0000  0.0000  1.0000    0.0000  0.0000  0.0000    0.5000  0.5000
             Mi-6mos  1.0000  1.0000  1.0000  1.0000    1.0000  1.0000  1.0000    1.0000  1.0000
                      ------------------------------    ----------------------    --------------
     Overall (micro)  1.0000  1.0000  1.0000  1.0000    1.0000  1.0000  1.0000    1.0000  1.0000
     Overall (macro)  0.7692  0.7692  0.6923  0.7692    0.6923  0.6923  0.6923    0.7308  0.7308

                                                    10 files found

Contextually-Grounded Annotation Examples

A second class of custom outputs is to generate listings of real annotations with left- and right-margins of context. Most often, you will want to use this type of output to generate a listing of all the FP annotations your system generated or all the FN annotations your system failed to find.

The generation of this output is dependent on a score card having been written to disk during a normal evaluation run. You'll also want to make sure to have generated a system output directory. Both flags are show in examples below. Additional flags let you determine how much of a context window (in characters) you want to see on the left and right of the annotation.

If we focus solely on the partial matches, then we're guaranteed to get FP and FN annotations that don't overlap. We don't distinguish between span mismatches and type mismatches.

export ETUDE_DIR=etude-engine

python3 ${ETUDE_DIR}/etude.py \
  --reference-input ${ETUDE_DIR}/tests/data/i2b2_2016_track-1_reference \
  --reference-config ${ETUDE_DIR}/config/i2b2_2016_track-1.conf \
  --test-input ${ETUDE_DIR}/tests/data/i2b2_2016_track-1_test \
  --test-config ${ETUDE_DIR}/config/i2b2_2016_track-1.conf \
  --file-suffix "xml" \
  --by-type \
  -m FP FN \
  --fuzzy-match-flags partial \
  --pretty-print \
  --test-out /tmp/system \
  --write-score-cards

## Use standard settings
python3 ${ETUDE_DIR}/extract_samples.py \
  --score-card /tmp/system/metrics_partial_score_card.csv \
  --annotation-out /tmp/system

## Show a larger left margin than right margin
python3 ${ETUDE_DIR}/extract_samples.py \
  --score-card /tmp/system/metrics_partial_score_card.csv \
  --annotation-out /tmp/system \
  --left-margin 25 \
  --right-margin 10

## Only print the FP annotations
python3 ${ETUDE_DIR}/extract_samples.py \
  --score-card /tmp/system/metrics_partial_score_card.csv \
  --annotation-out /tmp/system \
  --metrics FP

## The system output filenames differ from the reference
## filenames in that they end in '.txt.xmi' rather than
## just '.txt'
python3 ${ETUDE_DIR}/extract_samples.py \
  --score-card /tmp/system/metrics_partial_score_card.csv \
  --annotation-out /tmp/system \
  --file-suffix ".txt" ".txt.xmi"

Configuring Annotation Extraction

Several sample configurations are provided in the config/ folder. Each long name for an annotation description should be unique due to how Python's configuration parser works. XPath's should also be unique within a config file but do not programmitically need to be. The begin and end attribute are required for a pattern to be scorable.

[ Long Name or Description ]
Parent:         (optional; useful for merging multiple child types together for scoring)
Short Name:     (optional; useful for displaying as column output name and merging
                   multiple XPaths into a single scoring category)
XPath:          (required for XML; pattern used by XPath to find annotation)
Begin Attr:     (required; beginning or start offset attribute name)
End Attr:       (required; end offset attribute name)
Text Attr:      (optional; not used by anything currently)

Additional interesting or useful configuration files can be found in our sister repository: ETUDE Engine Configs

Dependencies

Python module requirements for running ETUDE are included in the requirements.txt file. You should be able to install all non-default packages using pip:

pip install -r requirements

Building with PyInstaller

After installing all required dependencies (as above), you can opt to create a stand-alone version of the ETUDE engine with PyInstaller.

The vanilla creation is .. code:: bash

cd $ETUDE_ENGINE_DIR

pyinstaller --onefile --distpath=dist/linux etude.py pyinstaller --onefile --distpath=dist/osx etude.py pyinstaller --onefile --distpath=dist/windows etude.py

Testing

Unit testing is done with the pytest module. Because of a bug in how tests are processed in Python, you should run pytest indirectly rather than directly:

python -m pytest tests/

## You can also generate a coverate report in html format
python3.7 -m pytest --cov-report html:cov_html_py3.7 --cov=./ tests/

## The junit file is helpful for automated systems or CI pipelines
python -m pytest --junitxml=junit.xml tests

About

ETUDE (Evaluation Tool for Unstructured Data and Extractions) is a Python-based tool that provides consistent evaluation options across a range of annotation schemata and corpus formats

Topics

Resources

License

Stars

Watchers

Forks

Packages

No packages published