FHIR converter is a Java based library that enables converting HL7v2 messages to FHIR resources.
FHIR converter utilized the open source HAPI Library for parsing Hl7 messages and it also utilizes the HAPI library for FHIR resources to validate the generated FHIR resources.
Maven dependency
<dependency>
<groupId>io.github.linuxforhealth</groupId>
<artifactId>hl7v2-fhir-converter</artifactId>
<version>1.0.4</version>
</dependency>
Also add following repo to your build file to fetch requiired dependency:
jcenter()
maven {
url "https://dl.bintray.com/ibm-watson-health/ibm-fhir-server-releases"
}
In order to convert a Hl7 message to FHIR resource, create a new instance of the class FHIRConverter and invoke the function convert and pass the hl7message data (file contents).
HL7ToFHIRConverter ftv = new HL7ToFHIRConverter();
String output= ftv.convert(hl7message); // generated a FHIR output
- ADT_A01
- ORU_R01
- PPR_PC1
- VXU_V04
What you’ll need
- JDK 8 or later
- Install Gradle
Steps:
git clone git@github.com:LinuxForHealth/hl7v2-fhir-converter.git
cd hl7v2-fhir-converter
gradle build
Converter supports following configuration options using the config.properties file.
- base.path.resource=
- supported.hl7.messages=ADT_A01, ORU_R01, PPR_PC1
- default.zoneid=+08:00 //ZoneOffset. This consists of 'Z' and IDs starting with '+' or '-'. If default zone offset is provided then if HL7 dates do not contain zone details but FHIR conversion requires zone information then default zone provided here will be used. If no default zone is set, then such date value will be ignored during conversion.
- additional.conceptmap.file=/absolute/path/to/additionalconceptmap_yaml_file (to import additional concept map details)
If the base.path.resource is empty then the resources (templates) then default resources(templates) are used during the conversion.
supported.hl7.messages lists the messages that are currently supported for conversion.
The converter looks for the config.properties file in the location specified by System property config.home. Example: -Dconfig.home=/path/to/folder/containing/config/test
HL7v2-FHIR converter converts a given HL7 message to FHIR bundle resource using the message templates. These templates are yaml files. Each message template defines what all FHIR resources needs to be generated from a particular message.
A message template file consists of list of resources that can be generated from that message type. For each resource in a template following attributes needs to be defined:
resourceName: [REQUIRED]
segment: [REQUIRED]
resourcePath: [REQUIRED]
repeats: [DEFAULT false]
isReferenced: [DEFAULT false]
additionalSegments: [DEFAULT empty]
Attribute description:
- resourceName: Name of the resource example: Patient
- segment: Primary segment that this resource depends on. Example patient resource depends on PID segment. If the primary segment is not in the message then the resource will not be generated.
- resourcePath: Path for resource file relative to the directory that has all the templates. The default path for the templates is src/main/resources/hl7. Example: Patient resource :resource/Patient (no need for file extension).
- repeats: HL7 have certain segments that repeat and so the converter needs to generate multiple resources from those segments. Example OBX segment. If this field is set to false then only the first occurrence of that segment will be used for resource generation. If this is set to true then multiple resources will be generated from each of the occurrences of that segment.
- additionalSegments: Any additional segments the resource needs.
- isReferenced : If this resource would be references by other resources then this field should be true.
Note: The order of resource template is important within a message template. The resources are generated in the order in which they are listed in the message template. If a resource needs to reference another resource then that resource should ve generated before this resource.
Example of a message template:
# FHIR Resources to extract from ADT_A01 message
resources:
- resourceName: Patient
segment: PID
resourcePath: resource/Patient
repeats: false
isReferenced: true
additionalSegments:
- resourceName: Encounter
segment: PV1
resourcePath: resource/Encounter
repeats: false
isReferenced: true
additionalSegments:
- PV2
- EVN
- resourceName: Observation
segment: OBX
resourcePath: resource/Observation
repeats: true
isReferenced: true
additionalSegments:
- resourceName: AllergyIntolerance
segment: AL1
resourcePath: resource/AllergyIntolerance
repeats: true
additionalSegments:
Resource template represents a FHIR resource. In order to generate a resource, a resource template for that resource should exist in this location: src/main/resources/hl7/resource. The resource template defines list of fields and a way to extract values for each of these fields.
Sample resource template:
#
# (C) Copyright IBM Corp. 2020
#
# SPDX-License-Identifier: Apache-2.0
#
# Represents data that needs to be extracted for a Patient Resource in FHIR
# reference: https://www.hl7.org/fhir/patient.html
---
resourceType: Patient
id:
type: STRING
valueOf: 'UUID.randomUUID()'
expressionType: JEXL
identifier:
valueOf: datatype/Identifier
generateList: true
expressionType: resource
specs: PID.3
name:
valueOf: datatype/HumanName
generateList: true
expressionType: resource
specs: PID.5
gender:
type: ADMINISTRATIVE_GENDER
valueOf: PID.8
expressionType: HL7Spec
birthDate:
type: DATE
valueOf: PID.7
expressionType: HL7Spec
#
# (C) Copyright IBM Corp. 2020
#
# SPDX-License-Identifier: Apache-2.0
#
---
resourceType: Condition
id:
type: STRING
valueOf: 'UUID.randomUUID()'
expressionType: JEXL
category_x1:
valueOf: datatype/CodeableConcept_var
generateList: true
expressionType: resource
condition: $source NOT_NULL
vars:
code: CONDITION_CATEGORY_CODES, $type
text: $type
source: PRB.3
constants:
type: problem-list-item
category_x2:
valueOf: datatype/CodeableConcept_var
generateList: true
expressionType: resource
condition: $source NULL
vars:
code: CONDITION_CATEGORY_CODES, $type
text: $type
source: PRB.3
constants:
type: encounter-diagnosis
severity:
valueOf: datatype/CodeableConcept
generateList: true
expressionType: resource
specs: PRB.26
vars:
code: PRB.26
code:
valueOf: datatype/CodeableConcept
generateList: true
expressionType: resource
specs: PRB.3
vars:
code: PRB.3
encounter:
valueOf: datatype/Reference
expressionType: resource
specs: $Encounter
subject:
valueOf: datatype/Reference
expressionType: resource
specs: $Patient
onsetDateTime:
type: DATE_TIME
valueOf: PRB.16
expressionType: HL7Spec
stage:
valueOf: secondary/Stage
generateList: true
expressionType: resource
specs: PRB.14
vars:
code: PRB.14
evidence:
valueOf: secondary/evidence
generateList: true
expressionType: resource
specs: $Observation
useGroup: true
The extraction logic for each field can be defined by using expressions. This component supports 4 different type of expressions. All expressions have following attributes:
-
type: DEFAULT - Object
Represents the class type for the final return value extracted for the field. -
specs: DEFAULT - NONE
Represents the base value for a resource, if no spec is provided then parents base value would be used as base value for child resource. Refer to the section on supported formats for Specification. -
default: DEFAULT - NULL
If extraction of the value fails, then the default value can be used. -
required : DEFAULT - false
If a field is required and cannot be extracted then the resource generation will fail even if other fields were extracted. -
vars: DEFAULT - EMPTY
List of variables and there value can be provided which can be used during the extraction process. Refer to the section on supported formats for Variables. -
condition: DEFAULT - true
If a condition is provided then the expression will be resolved only if condition evaluates to true. Refer to the section on supported formats for Condition. -
value: This represents a constant value for the value to be extracted for an expression. This is attribute is only valid for SimpleExpression type.
-
valueOf: This represents the value to be extracted through the evaluation of the expression. This value extraction depends on expression type.
-
expressionType: Based on the expression type a valueOf attribute will get evaluated.
-
generateList: DEFAULT [false] When specified list output will be generated and all values of specs would be used. If this value is false, then first valid value of spec would be used for evaluating the expression.
-
Constants: DEFAULT - EMPTY
List of Constants (string values) which can be used during the extraction process.
type: String
valueOf: CX.1
default: 'abc'
required: true
condition: $var1 != null
vars:
var1: CX.1
var2: CX.2
constants:
code: 'some code'
Specification represents the base value for a expression. There are two types of Specifications -
- SimpleSpecification -- Represents simple specification that can be extracted from context values. Example: specs: $Patient. Note BASE_VALUE is reserved for base value provided to an expression during evaluation. Do not use or name variable as BASE_VALUE.
- HL7Specification -- Represents specification for extracting values from HL7 message.
HL7Specification represents expression that helps to identify the value to extracted from the HL7 message.
The specification expression has the following format :
- Single SPEC - where spec can be
- SEGMENT
- SEGMENT.FIELD
- SEGMENT.FIELD.COMPONENT
- FIELD
- FIELD.COMPONENT
- FIELD.COMPONENT.SUBCOMPONENT
Example:OBX, OBX.1, CWE, CWE.1, OBX.3.1
- Multiple SPEC - Where each single spec is separated by |, where | represents "or'. If the value from first spec is extracted, then the remaining specs are ignored.
Example:OBX.1 |OBX.2|OBX.3
, if OBX.1 is null then only OBX.2 will be extracted. - Multiple value extraction - In HL7 several fields can have repeated values, so extract all repetition for that field the spec string should end with *.
Example:PID.3 *
,OBX.1 |OBX.2 |OBX.3 *
Variables can be used during expression evaluation. This engine supports defining 3 types of variables:
- SimpleVariable : These are variables where value is extracted from simple Specification or another variable from the context values. Example:
var1: CWE.1 |CE.1 |CNE.1
- ExpressionVariable : Value of a variable is extracted by evaluating a java function. Example:
low: OBX.7, GeneralUtils.split(low, "-", 0)
- DataTypeVariable: Value of a variable is extracted from Specification and this value is converted to a particular data type. Example:
var1: STRING, OBX.2
Note: BASE_VALUE is reserved for base value provided to an expression during evaluation. Do not use or name variable as BASE_VALUE.
Conditions evaluate to true or false.
Engine supports the following condition types:
- Null check, example:
condition:$var1 NULL
- Not null check, example:
condition:$var1 NOT_NULL
- Simple condition, example:
condition: $obx2 EQUALS DR
- Conditions with AND, example:
condition: $obx2 EQUALS SN && $obx5.3 EQUALS ':'
- Conditions with OR, example:
condition: $obx2 EQUALS TX || $obx2 EQUALS ST
- ResourceExpression : This type of expression is used when a field is a data type defined in one of the data type templates. These data type templates define different FHIR data types. Example:
identifier:
valueOf: datatype/IdentifierCX
expressionType: resource
specs: PID.3
- ReferenceExpression : This type of expression is used when a field references a FHIR resource which has to be first generated based on provided hl7spec data. Then in thhe current resource this FHIR resource is referenced using Reference data type. Example:
performer:
valueOf: resource/Practitioner
expressionType: reference
specs: OBX.16
- JEXLExpression: This type of expression is used when a field value needs to be extracted by executing a Java method.
type: STRING
valueOf: 'GeneralUtils.generateName( prefix, given,family, suffix)'
expressionType: JEXL
var:
prefix: STRING, XPN.4
given: STRING, XPN.2
family: STRING, XPN.1
suffix: STRING, XPN.5
- Hl7Expression : This type of expression is used when a field value has to be extracted directly from the HL7 segment/field/component.
given:
type: STRING
valueOf: XPN.2
expressionType: HL7Spec
- SimpleExpression : If the field value is constant and no extraction or conversion is required then this expression is used. Example 1: Constant value
text:
value: 'ABX'
Example 2: Value needs to be extracted from a variable.
text:
valueOf: $var
{
"resourceType": "Bundle",
"id": "32744a5d-8c36-4cd1-93bf-7da16a1bedde",
"meta": {
"lastUpdated": "2021-02-20T10:02:45.547+08:00"
},
"type": "collection",
"entry": [ {
"fullUrl": "urn:uuid:Patient/7425c8c2-50e7-433f-a804-79c0eab69a59",
"resource": {
"resourceType": "Patient",
"id": "7425c8c2-50e7-433f-a804-79c0eab69a59",
"identifier": [ {
"type": {
"coding": [ {
"system": "http://terminology.hl7.org/CodeSystem/v2-0203",
"code": "MR",
"display": "Medical record number"
} ],
"text": "MR"
},
"value": "PID1234",
"assigner": {
"reference": "Organization/A"
}
}, {
"type": {
"coding": [ {
"system": "http://terminology.hl7.org/CodeSystem/v2-0203",
"code": "SS",
"display": "Social Security number"
} ],
"text": "SS"
},
"value": "1234568965",
"assigner": {
"reference": "Organization/USA"
}
} ],
"name": [ {
"family": "DOE",
"given": [ "JOHN" ]
} ],
"gender": "female",
"birthDate": "1980-02-02"
}
}, {
"fullUrl": "urn:uuid:Organization/A",
"resource": {
"resourceType": "Organization",
"id": "A",
"name": "Assigning Authority"
}
}, {
"fullUrl": "urn:uuid:Organization/USA",
"resource": {
"resourceType": "Organization",
"id": "USA",
"name": "Assigning Authority"
}
}, {
"fullUrl": "urn:uuid:Encounter/f333437c-9a9e-4981-855c-5d31f4f579a4",
"resource": {
"resourceType": "Encounter",
"id": "f333437c-9a9e-4981-855c-5d31f4f579a4",
"identifier": [ {
"value": "48390"
} ],
"status": "finished",
"class": {
"code": "ff"
},
"type": [ {
"text": "EL"
} ],
"serviceType": {
"coding": [ {
"system": "http://terminology.hl7.org/CodeSystem/v2-0069",
"code": "MED",
"display": "Medical Service"
} ],
"text": "MED"
},
"subject": {
"reference": "Patient/7425c8c2-50e7-433f-a804-79c0eab69a59"
},
"period": {
"start": "2014-09-12T22:00:00+08:00",
"end": "2015-02-06T03:17:26+08:00"
},
"length": {
"value": 210557,
"unit": "Minutes"
},
"hospitalization": {
"preAdmissionIdentifier": {
"value": "ABC"
},
"specialCourtesy": [ {
"text": "E"
} ],
"specialArrangement": [ {
"coding": [ {
"system": "http://terminology.hl7.org/CodeSystem/v2-0009",
"code": "B6",
"display": "Pregnant"
} ],
"text": "B6"
} ]
}
}
}, {
"fullUrl": "urn:uuid:Observation/6be3985b-ade4-47f4-b3b0-3e4e8de355c0",
"resource": {
"resourceType": "Observation",
"id": "6be3985b-ade4-47f4-b3b0-3e4e8de355c0",
"identifier": [ {
"value": "1234_"
} ],
"status": "final",
"code": {
"coding": [ {
"code": "1234"
} ]
},
"subject": {
"reference": "Patient/7425c8c2-50e7-433f-a804-79c0eab69a59"
},
"encounter": {
"reference": "Encounter/f333437c-9a9e-4981-855c-5d31f4f579a4"
},
"issued": "2012-09-12T01:12:30+08:00",
"performer": [ {
"reference": "Practitioner/1fbe4699-7821-4176-92cf-648fe799b4f2"
}, {
"reference": "Practitioner/191b914f-4eb8-4190-b620-b6bde27141e8"
}, {
"reference": "Practitioner/dd9db2fd-ae96-4ca1-8db0-d873568b901a"
}, {
"reference": "Practitioner/8a630e35-52c2-43b5-aca3-1485ba986412"
} ],
"valueString": "ECHOCARDIOGRAPHIC REPORT"
}
}, {
"fullUrl": "urn:uuid:Practitioner/1fbe4699-7821-4176-92cf-648fe799b4f2",
"resource": {
"resourceType": "Practitioner",
"id": "1fbe4699-7821-4176-92cf-648fe799b4f2",
"identifier": [ {
"value": "2740"
} ],
"name": [ {
"text": "Janetary TRDSE",
"family": "TRDSE",
"given": [ "Janetary" ]
} ]
}
}, {
"fullUrl": "urn:uuid:Practitioner/191b914f-4eb8-4190-b620-b6bde27141e8",
"resource": {
"resourceType": "Practitioner",
"id": "191b914f-4eb8-4190-b620-b6bde27141e8",
"identifier": [ {
"value": "2913"
} ],
"name": [ {
"text": "Darren MRTTE",
"family": "MRTTE",
"given": [ "Darren" ]
} ]
}
}, {
"fullUrl": "urn:uuid:Practitioner/dd9db2fd-ae96-4ca1-8db0-d873568b901a",
"resource": {
"resourceType": "Practitioner",
"id": "dd9db2fd-ae96-4ca1-8db0-d873568b901a",
"identifier": [ {
"value": "3065"
} ],
"name": [ {
"text": "Paul MGHOBT",
"family": "MGHOBT",
"given": [ "Paul" ]
} ]
}
}, {
"fullUrl": "urn:uuid:Practitioner/8a630e35-52c2-43b5-aca3-1485ba986412",
"resource": {
"resourceType": "Practitioner",
"id": "8a630e35-52c2-43b5-aca3-1485ba986412",
"identifier": [ {
"value": "4723"
} ],
"name": [ {
"text": "Robert LOTHDEW",
"family": "LOTHDEW",
"given": [ "Robert" ]
} ]
}
}, {
"fullUrl": "urn:uuid:AllergyIntolerance/44ef3e65-650b-4eff-b662-2057112f49f9",
"resource": {
"resourceType": "AllergyIntolerance",
"id": "44ef3e65-650b-4eff-b662-2057112f49f9",
"clinicalStatus": {
"coding": [ {
"system": "http://terminology.hl7.org/CodeSystem/allergyintolerance-clinical",
"code": "active",
"display": "Active"
} ]
},
"verificationStatus": {
"coding": [ {
"system": "http://terminology.hl7.org/CodeSystem/allergyintolerance-verification",
"code": "confirmed",
"display": "Confirmed"
} ]
},
"criticality": "unable-to-assess",
"code": {
"coding": [ {
"code": "00000741",
"display": "OXYCODONE"
} ],
"text": "OXYCODONE"
},
"patient": {
"reference": "Patient/7425c8c2-50e7-433f-a804-79c0eab69a59"
},
"reaction": [ {
"manifestation": [ {
"text": "HYPOTENSION"
} ]
} ]
}
}, {
"fullUrl": "urn:uuid:AllergyIntolerance/38e4e23f-593c-406b-bd4d-36b51d580006",
"resource": {
"resourceType": "AllergyIntolerance",
"id": "38e4e23f-593c-406b-bd4d-36b51d580006",
"clinicalStatus": {
"coding": [ {
"system": "http://terminology.hl7.org/CodeSystem/allergyintolerance-clinical",
"code": "active",
"display": "Active"
} ]
},
"verificationStatus": {
"coding": [ {
"system": "http://terminology.hl7.org/CodeSystem/allergyintolerance-verification",
"code": "confirmed",
"display": "Confirmed"
} ]
},
"criticality": "unable-to-assess",
"code": {
"coding": [ {
"code": "00001433",
"display": "TRAMADOL"
} ],
"text": "TRAMADOL"
},
"patient": {
"reference": "Patient/7425c8c2-50e7-433f-a804-79c0eab69a59"
},
"reaction": [ {
"manifestation": [ {
"text": "SEIZURES"
}, {
"text": "VOMITING"
} ]
} ]
}
} ]
}