This document details some basic configuration settings to use when setting up CDA Exchange Plus.
Before implementing, it is strongly recommended the planning document for CDA is reviewed to ensure the necessary considerations have been made for a smooth implementation and testing/validating of the CDA Exchange Plus.
✔ Install SmileCDR
✔ add licence to enable CDA Exchange Plus
✔ configure CDA Exchange Plus Module with the licence - if already configured, skip this section
✔ if using Helm charts, go to: method 1, skip other methods
✔ if using Docker or Properties mode, go to: method 2, skip other methods.
✔ if using WebAdmin Console, go to: method 3, skip other methods
✔ configure Message Broker - if already configured, skip this section
✔ if using Channel Import, go to: Channel Import
✔ if using Camel, go to: Camel
✔ configure camel routes
✔ Troubleshooting CDA Exchange Plus
SmileCDR can be installed via a couple of ways, via tarball, using Docker, or using helm chart.
It is recommended the latest version of smileCDR is used with CDA Exchange Plus, at minimum 2025.02.R01 or later.
CDA Exchange Plus is a premium module, as such a license is required in order for it to work. The licence can be added via helm chart, properties file or via the webadmin console.
Via Helm chart The SmileCDR licence is a regular JWT token and can be stored in an AWS Secrets Manager Secret under the jwt
key in the JSON object. It can then be added to the values file via the following as defined by the secretSpec schema:
license:
type: sscsi
provider: aws
secretArn: arn:aws:secretsmanager:us-east-1:111111111111:secret:my-smile-license
Properties file The properties file can take either a location to the file containing the JWT token (JWT_file), or the JWT token directly (JWT_token).
module.license.type=LICENSE
module.license.config.jwt_file=
module.license.config.jwt_text={JWT_token}
Webadmin console In the webadmin console, navigate to: Configuration >> Module Config >> license.
Under this module there are two settings: License JWT File will allow you to point to the file containing the JWT token, while License JWT Text will allow you to paste the JWT token directly into the module.
Once a selection has been made, save the config and restart the module. If all has gone correctly, the UI should update and list which modules the license allows access to.
The CDA Exchange Plus module can be configured in a few different ways:
Method 1: Via helm chart SmileCDR’s helm chart implementation allows for a simplified installation and configuration of SmileCDR on Kubernetes using a number of best practices. This allows you to not just configure SmileCDR but the environment around it as well.
The following code must be added to the values.yaml
file in order for the helm chart to properly configure the CDA Exchange Plus module, the properties must fall under the modules:
header:
{module_id}:
name: CDA EXCHANGE PLUS
enabled: true
type: CDA_EXCHANGE_PLUS
requires:
PERSISTENCE_ALL: {module_id_of_persistence_module}
config:
target_ig: {IG_type}
ecmascript_module: false
prefer_supplied_translations: false
store_original_ccd: true
cda_processing_script.file: {processing_script_file_location}
interceptor_bean_types: {interceptor_bean}
Where:
{module_id}
is the name you gave the CDA Exchange Module
{module_id_of_persistence_module}
is the name of your persistence module{IG_type}
is the IG the CDA Module will map against, there are currently three IGs available: US_CORE_IG
, FHIR_CORE_PROFILE
and USCDI_V2
.{processing_script_file_location}
is the location of the JavaScript processing script for CDA{interceptor_bean}
is the name of the interceptor class for the CDA Exchange Plus Java InterceptorOnce all has been configured the helmchart can be spun up as usual.
Method 2: Via the Properties file If you are installing SmileCDR via Docker or tarball using [Properties Mode](/docs/guide_installation/configuring_smile_cdr.html#module-property-source). The following properties in the cdr-Master-config.properties
file must be set:
Module.license.config.jwt_text
- should contain the SmileCDR licence for CDA Exchange Plus (and any subsequent other licenced modules you may be using){module_id}
is the name you gave the CDA Exchange Module{module_id_of_persistence_module}
is the name of your persistence module{IG_type}
is the IG the CDA Module will map against, there are currently three IGs available: US_CORE_IG
, FHIR_CORE_PROFILE
and USCDI_V2
.{processing_script_file_location}
is the location of the JavaScript processing script for CDA{interceptor_bean}
is the name of the interceptor class for the CDA Exchange Plus Java Interceptormodule.{module_id}.type=CDA_EXCHANGE_PLUS
module.{module_id}.config.store_original_ccd=true
module.{module_id}.requires.PERSISTENCE_ALL={module_id_of_persistence_module}
module.{module_id}.config.target_ig={IG_type}
module.{module_id}.config.cda_processing_script.file={processing_script_file_location}
module.{module_id}.config.interceptor_bean_types={interceptor_bean}\
Once the properties have been configured if configured via tarball run the following command to spin up SmileCDR: bin/smilecdr start
If spinning up via Docker, run the docker command to start up the pods.
Method 3: Via the webadmin console This method requires SmileCDR to already be running in the environment.
bin/smilecdr start
http://localhost:9100
or the domain URL used for it.CDA Exchange Plus by default uses REST API to ingest CCDs via the $import-cda
operation. However, it is possible to have CDA documents ingested via a Message broker.
SmileCDR currently has two modules that can be used for message brokers, Channel Import and Camel.
Pre-requisites: Kafka must be set up prior to configuring Channel Import or Camel.
Using the same methods as mentioned above, Channel Import can be added via the following methods:
Yaml/helm chart:
When using the helm chart method of Smile implementation, the kafka message broker can be configured within the helm chart, please consult the SmileCDR Helm Chart docs for more information on configuring Kafka within helm.
The module itself is added to the helm chart using the following structure:
{module_id}:
name: Channel CDA Exchange Plus
enabled: true
type: CHANNEL_IMPORT
requires:
PERSISTENCE_ALL: {persistence_module_id}
CDA_EXCHANGE: {cda_exchange_plus_module_id}
Properties file:
module.{module_id}.type=CHANNEL_IMPORT
module.{module_id}.requires.CDA_EXCHANGE={cda_exchange_plus_module_id}
module.{module_id}.requires.PERSISTENCE_ALL={persistence_module_id}
module.{module_id}.config.channel.retry.delay_milliseconds=2000
module.{module_id}.config.channel.retry.maximum_delay_milliseconds=2000
module.{module_id}.config.channel.retry.maximum_attempts=5
module.{module_id}.config.channel.concurrent_consumers=8
module.{module_id}.config.default_mediatype=application/fhir+json
module.{module_id}.config.channel.name={channel_name}
module.{module_id}.config.channel.failed.name={failure_channel_name}
module.{module_id}.config.channel.retry.name={retry_channel_name}
module.{module_id}.config.interceptor_bean_secure_context=false
{channel_name}, {failure_channel_name} and {retry_channel_name} must be filled in for the module to start correctly.
The requires.CDA_EXCHANGE
field must be populated with the ID of the CDA Exchange Plus module for Channel Import to be correctly hooked up to the module.
Method 3: Webadmin Console
channel_import
though it’s recommended you highlight it’s the CDA Channel Import module. Ie. cda_channel_import or cda_import) 3. Fill in the properties Channel Name, Retry Channel Name, Failure Channel Name and under Dependencies choose the CDA Exchange+ module under CDA Exchange+ module (optional).Yaml/helm chart:
{module_id}:
name: Channel CDA Exchange Plus
enabled: true
type: CAMEL
requires:
PERSISTENCE_ALL: {persistence_module_id}
config:
routes.script.file: {file_location}
functions.script.file: {functions_script_location}
Configuration file
module.camel.type=CAMEL
module.camel.requires.PERSISTENCE_ALL=persistence
module.camel.config.routes.script.text=
module.camel.config.interceptor_bean_secure_context=false
module.camel.config.routes.script.file={file_location}
Webadmin console Under Configuration >> Module Config >> Add Module, select Camel and click Add.
Under Dependencies select the persistence module of choice.
Save module config and start the module.
Configuring the Camel routes
Under Camel Routes (Text) (routes.script.text) or Camel Routes (File) (routes.script.file). The appropriate routes need to be set for Camel to correctly identify which broker topic it needs to be listening to and where to direct the messages from the broker towards.
Example Camel route:
<route>
<from uri="smile:clustermgr/broker?topic=cda"/>
<to uri="smile:camel/script?function=processKafkaMessage" />
<to uri="smile:cda_exchange_plus/cdaToFhirPreConvertScriptProcessor"/>
<to uri="smile:cda_exchange_plus/cdaToFhirPreConvertInterceptorProcessor"/>
<to uri="smile:cda_exchange_plus/cdaToFhirProcessor"/>
<to uri="smile:cda_exchange_plus/cdaToFhirPostConvertInterceptorProcessor"/>
<to uri="smile:cda_exchange_plus/cdaToFhirPostConvertScriptProcessor"/>
<choice>
<when>
<spel>#{body.isDoProcess()}</spel>
<setBody>
<spel>#{body.bundle}</spel>
</setBody>
<to uri="smile:persistence/bundleProcessor"/>
</when>
</choice>
</route>
For this given Camel route, the data from Kafka topic cda
(clustermgr/broker) is passed through the CDA Exchange flow and stored in the persistence module with the specified partition (persistence/bundleProcessor).
Between these two events the data is passed through a Camel script camel/script?function=processKafkaMessage
the CDA Exchange Plus processors JavaScript and Java Interceptors (cda_exchange_plus/*
, cdaToFhirPreConvertInterceptorProcessor, cdaToFhirProcessor, cdaToFhirPostConvertInterceptorProcessor, cdaToFhirPostConvertScriptProcessor). After all conversions have taken place the data is in the format cdaToFhirConversionResultJson
which is not accepted by the bundleProcessor
therefore, the last part of the routes converts the body into the IBaseBundle
format for use in bundleProcessor
.
Note that all pre- and post-processing processors, both JavaScript and Java are optional! If no processors are necessary these processors can be removed.
The processKafkaMessage
processor extracts the XML string from the kafka message via the following function:
function processKafkaMessage(string) {
return (JSON.parse(string)).payload.payload;
}
Where the incoming data is assumed to be of the following format:
{
"payload": {
"transactionId": "a0c42c2b-43e9-4e09-8cf2-be3a7b95fb19",
"mediaType": "text/xml+cda",
"payload": "<?xml-stylesheet type=\"text/xsl\" href=\"CCDA.xsl\" alternate=\"no\" title=\"Allscripts Default\" ?><?xml-stylesheet type=\"text/xsl\" href=\"ccda.xsl\"?><ClinicalDocument xmlns=\"urn:hl7-org:v3\" xmlns:sdtc=\"urn:hl7-org:sdtc\" xmlns:xsi=\"http://www.w3.org/2001/XMLSchema-instance\"><realmCode code=\"US\"/><typeId root=\"2.16.840.1.113883.1.3\" extension=\"POCD_HD000040\"/><templateId root=\"2.16.840.1.113883.10.20.22.1.8\" extension=\"2015-08-01\"/><templateId root=\"2.16.840.1.113883.10.20.22.1.8\"/><templateId root=\"2.16.840.1.113883.3.3251.1.1\"/><templateId root=\"2.16.840.1.113883.10.20.22.1.1\" extension=\"2015-08-01\"/><templateId root=\"2.16.840.1.113883.10.20.22.1.1\"/><templateId root=\"1.3.6.1.4.1.19376.1.5.3.1.5.5.1\"/><templateId root=\"1.3.6.1.4.1.19376.1.5.3.1.5.5.2\"/><templateId root=\"2.16.840.1.113883.10.20.22.1.2\"/></ClinicalDocument>"
}
}
CDA Exchange Plus has its own troubleshooting log that can be enabled via the customerlib/logback-smile-custom.xml
The following line should be uncommented
<!-- <logger name="ca.cdr.log.cda_troubleshooting" level="DEBUG"/> -->