Ecosyste.ms: Awesome

An open API service indexing awesome lists of open source software.

Awesome Lists | Featured Topics | Projects

https://github.com/tompollard/omopetl

Package for creating OMOP ETLs.
https://github.com/tompollard/omopetl

Last synced: about 2 months ago
JSON representation

Package for creating OMOP ETLs.

Awesome Lists containing this project

README

        

# OMOPETL

Package for creating OMOP ETLs.

## Usage

The `omopetl` package has two core tools:

1. `startproject`: Create an ETL project that contains configuration files for mapping between two data structures, and
2. `run`: Run the ETL project on source data to create a set of data in the target format.

### 1. Initializing a transformation project with `startproject`

When a user runs `omopetl startproject myproject`, the following folder structure is created:

```
myproject/
├── config/
│ ├── etl_config.yaml # ETL configuration
│ ├── mappings.yaml # Column mappings and transformations
│ ├── source_schema.yaml # Source schema
│ ├── target_schema.yaml # Target schema (OMOP)
├── data/
│ ├── source/ # Input CSV files
│ ├── target/ # Output CSV files (OMOP)
```

After initialising the project, it is necessary to configure the transformation rules by updating the files in the `config` folder.

### 2. Running a transformation with `run`

Once your project is configured, you can run transform the data in your source folder (`./myproject/data/source/*`) with the following command:

```
omopetl run myproject --dry
```

The `--dry` argument does a dry run only, meaning the code is run but the files are not saved. Remove the `--dry` argument to run the full transformation.

## Demo

For an example project, you can start a new demo project with:

```
omopetl startdemo mydemo
```

This mirrors the structure created with `startproject`. The only difference is that `config` folder is configured to transform the data in the `source` folder to OMOP.

## Transformations

At the core of the `omopetl` project are the transformations. These are a set of rules that allow you to map from the source format to the target format.

1. Direct Column Mapping: Map a column from the source table directly to a column in the target table without modification.

Example: Mapping subject_id in patients to person_id in PERSON.

```
- source_column: subject_id
target_column: person_id
```

2. Value Mapping: Transform specific values in a column to standardized values, such as OMOP concept IDs.

Example: Mapping gender values M and F in patients to OMOP concept IDs 8507 (male) and 8532 (female).

```
- source_column: gender
target_column: gender_concept_id
transformation:
type: map
values:
M: 8507
F: 8532
```

3. Lookups: Map source codes (e.g., ICD-9/ICD-10 codes) to OMOP concept IDs using a vocabulary or lookup table.

Example: Mapping ICD codes in diagnoses_icd to OMOP standard concept IDs.

```
- source_column: icd_code
target_column: condition_concept_id
transformation:
type: lookup
vocabulary: icd_to_snomed
```

4. Date Normalization: Format or extract parts of dates (e.g., year, month, day) from source columns.

Example: Extracting year_of_birth from the dob column in patients.

```
- source_column: dob
target_column: birth_datetime
transformation:
type: normalize_date
format: "%Y-%m-%d"
```

5. Aggregation: Combine multiple rows or columns to calculate summary values (e.g., min, max, sum).

Example: Aggregating multiple labevents rows for the same patient and time window into a single measurement.

```
- source_column: value
target_column: aggregated_value
transformation:
type: aggregate
method: sum
group_by: [subject_id, charttime]
```

6. Row Filtering: Include or exclude rows based on conditions.

Example: Exclude diagnoses_icd rows with icd_version = 10.

```
- source_column: icd_version
target_column: condition_concept_id
transformation:
type: filter
condition: "icd_version == 9"
```

7. Concatenation: Concatenate multiple columns into a single column, often used for generating unique identifiers.

Example: Concatenating subject_id and stay_id to form visit_detail_id.

```
- source_columns: [subject_id, stay_id]
target_column: visit_detail_id
transformation:
type: concatenate
separator: "-"
```

8. Default Values: Assign a default value to a column when the source column is missing or null.

Example: Assigning a default concept ID for missing admission_type.

```
- target_column: visit_concept_id
transformation:
type: default
value: 44818518
```

9. Multi-Table Merging: Combine data from multiple source tables into a single target table.

Example: Combining admissions and transfers into VISIT_OCCURRENCE.

```
- source_columns: [admissions.hadm_id, transfers.hadm_id]
target_column: visit_occurrence_id
transformation:
type: merge
merge_key: hadm_id
```

10. Conditional Transformations: Apply transformations based on conditions in the source data.

Example: Assigning different visit_concept_id values based on admission_type.

```
- source_column: admission_type
target_column: visit_concept_id
transformation:
type: conditional_map
conditions:
- condition: "admission_type == 'EMERGENCY'"
value: 9203
- condition: "admission_type == 'ELECTIVE'"
value: 9201
```

11. Derived Columns: Calculate new columns from existing data (e.g., differences between dates).

Example: Calculating length_of_stay as the difference between dischtime and admittime.

```
- source_columns: [admittime, dischtime]
target_column: length_of_stay
transformation:
type: derive
formula: "dischtime - admittime"
```

12. Splitting Columns: Split a single source column into multiple target columns.

Example: Splitting dob into year_of_birth, month_of_birth, and day_of_birth.

```
- source_column: dob
target_columns:
- year_of_birth
- month_of_birth
- day_of_birth
transformation:
type: split_date
```

13. Multi-Step Transformations: Apply a sequence of transformations to a single column.

Example: Normalize a date and then filter rows based on the normalized value.

```
- source_column: admittime
target_column: visit_start_datetime
transformations:
- type: normalize_date
format: "%Y-%m-%d"
- type: filter
condition: "visit_start_datetime >= '2020-01-01'"
```

## Configuring your transformation

After creating your new project with `omopetl startproject `, you will need to configure the transformation. We recommend the following approach:

1. Update `source_schema.yaml` to match your list of source tables, e.g.:

```
patients:
table_name: patients
columns:
subject_id:
type: Integer
primary_key: true
gender:
type: String

admissions:
table_name: admissions
columns:
hadm_id:
type: Integer
primary_key: true
subject_id:
type: Integer
admittime:
type: DateTime
dischtime:
type: DateTime
```

This file is primarily used for validating your transform (for example, making sure that relationships are maintained between variables).

You can use `omopetl inferschema ` to create a draft schema based on a set of CSV files. Careful though, the `primary_key` fields are likely to be incorrect, so review is needed.

2. Update `target_schema.yaml` to match your list of target tables, e.g.:

```
person:
table_name: person
columns:
person_id:
type: Integer
primary_key: true
gender_concept_id:
type: Integer

visit_occurrence:
table_name: visit_occurrence
columns:
visit_occurrence_id:
type: Integer
primary_key: true
person_id:
type: Integer
visit_start_datetime:
type: DateTime
visit_end_datetime:
type: DateTime
```

This file is primarily used for validating your transform (for example, making sure that relationships are maintained between variables).

3. Add details of your source tables and column mapping rules to `etl_config.yaml`:

```
etl:
source:
type: csv
directory: ./data/source
schema_file: ./config/source_schema.yaml

target:
type: csv
directory: ./data/target
schema_file: ./config/target_schema.yaml

mappings:
- source_tables: # list of tables used for creating the target table
- patients
target_table: person # target table
column_mappings: person_mapping # refers to a ruleset in mappings.yaml

- source_tables:
- admissions
- transfers
target_table: visit_occurrence
column_mappings: visit_occurrence_mapping
```

4. Add your transformations to `mappings.yaml`:

```
person_mapping:
- source_column: subject_id
target_column: person_id
- source_column: gender
target_column: gender_concept_id
transformation:
type: map
values:
M: 8507
F: 8532

visit_occurrence_mapping:
- source_column: hadm_id
target_column: visit_occurrence_id
- source_column: subject_id
target_column: person_id
- source_column: admittime
target_column: visit_start_datetime
- source_column: dischtime
target_column: visit_end_datetime
```

We suggest starting out with the simple transformations first. For example, try creating a first version of your transformation using only direct mappings.

5. Add custom validation rules to `validation.yaml`. For example, if dates of birth should be within the range of `1900-01-01` and `2023-12-31`, you might choose to add the following rule:

```
validation:
person:
- column: birth_datetime
rule: range
min: "1900-01-01"
max: "2023-12-31"
```