Docker: ibmcom/ibm-fhir-schematool supports onboarding and offboarding of a schema in support of the IBM FHIR Server. I am working on a project that runs on OpenShift using CodeReadyContainers, and I needed to test the image with a restricted policy.
Note, these are roughly my notes from testing, and converted to a post.
Recipe
Start up Code Ready Containers or create your OpenShift environment.
The IBM FHIR Server has support for extended operations beyond the standard C-R-U-D. The Extended Operations are supported at the System, Resource, and Instance levels. Operations are packaged as JAR files, and the IBM FHIR Server loads the specific Operation using the Java ServiceLoader framework at startup. More implementation specific details are at FHIROperationFramework.
The IBM FHIR Server repository builds a number of operations, release to Maven Repository, and bundles a subset automatically with the Docker container. Some of the bundles operations include Healthcheck, Terminology, BulkData Import, BulkData Export, Reindex, Everything, Erase, Convert, Document and Validate. The IBM FHIR Server maintains the Capability Statement which advertises the System, Type and Instance operations on the metadata endpoint.
The IBM FHIR Server can take any configured custom extended operation, such as HelloOperation which is part of the fhir-operation-test.
This document outlines how to add a custom extended operation to the IBM FHIR Server.
curl -L -o fhir-server-config.json \
https://raw.githubusercontent.com/IBM/FHIR/main/fhir-server/liberty-config/config/default/fhir-server-config.json
% Total % Received % Xferd Average Speed Time Time Time Current
Dload Upload Total Spent Left Speed
100 8423 100 8423 0 0 35095 0 --:--:-- --:--:-- --:--:-- 34950
curl -L -o fhir-operation-test-4.8.3.jar \
https://repo1.maven.org/maven2/com/ibm/fhir/fhir-operation-test/4.8.3/fhir-operation-test-4.8.3-tests.jar
% Total % Received % Xferd Average Speed Time Time Time Current
Dload Upload Total Spent Left Speed
100 12393 100 12393 0 0 58734 0 --:--:-- --:--:-- --:--:-- 58734
Create a directory called userlib, don’t put the jar in the userlib yet.
mkdir -p userlib
Start the Docker container, and capture the container id. It’s going to take a few moments to start up as it lays down the test database.
docker logs a096978867195ff6e610c36cdba77ff423c31c0ad488a7390f42cef6e89e7fd0
...
[6/16/21, 15:31:34:533 UTC] 0000002a FeatureManage A CWWKF0011I: The defaultServer server is ready to run a smarter planet. The defaultServer server started in 17.665 seconds.
Check the Type and Instance operations, you can check using this curl/jq command.
curl -k --location --request GET 'https://localhost:9443/fhir-server/api/v4/metadata' \
| jq -r '.rest[].resource[].operation[].name' | sort -u
% Total % Received % Xferd Average Speed Time Time Time Current
Dload Upload Total Spent Left Speed
100 1096k 0 1096k 0 0 103k 0 --:--:-- 0:00:10 --:--:-- 259k
apply
closure
convert
document
erase
everything
expand
export
lookup
reindex
subsumes
translate
validate
validate-code
Note, that the hello operation is not displayed. The CapabilityStatement is best looked at selectively, since it’s such a big document, thus the use of jq.
To check the System Level operations, you can check using this curl/jq command.
curl -k --location --request GET 'https://localhost:9443/fhir-server/api/v4/metadata' \
| jq -r '.rest[].operation[].name'
% Total % Received % Xferd Average Speed Time Time Time Current
Dload Upload Total Spent Left Speed
100 1096k 0 1096k 0 0 2592k 0 --:--:-- --:--:-- --:--:-- 2592k
bulkdata-status
closure
convert
export
healthcheck
import
reindex
Note, that the hello operation is not displayed.
Copy over the fhir-operation-test-*-tests.jar to userlib/.
docker logs a096978867195ff6e610c36cdba77ff423c31c0ad488a7390f42cef6e89e7fd0
...
[6/16/21, 15:31:34:533 UTC] 0000002a FeatureManage A CWWKF0011I: The defaultServer server is ready to run a smarter planet. The defaultServer server started in 17.665 seconds.
To check the System Level operations, you can check using this curl/jq command.
curl -k --location --request GET 'https://localhost:9443/fhir-server/api/v4/metadata' \
| jq -r '.rest[].operation[].name'
% Total % Received % Xferd Average Speed Time Time Time Current
Dload Upload Total Spent Left Speed
100 1096k 0 1096k 0 0 2562k 0 --:--:-- --:--:-- --:--:-- 2556k
bulkdata-status
closure
convert
export
healthcheck
hello
import
reindex
I had duplicate Postgres base signatures, and I needed to diagnose the reasons why it was failing to update. The following are helpful for diagnosing functions:
When I wanted to deploy my key to openpgp, I hit the issue where it said No keyserver available:
$ gpg --verbose --keyserver hkps://keys.openpgp.org --send-keys KEYDYID
gpg: Note: RFC4880bis features are enabled.
gpg: sending key KEYDYID to hkps://keys.openpgp.org
gpg: keyserver send failed: No keyserver available
gpg: keyserver send failed: No keyserver available
If you hit this, you can ps -ef dirmgr and then kill -9 the pid for the dirmngr.
Restart the dirmngr --debug-all --daemon --standard-resolver
Check the output for any errors (in my case a TLS issue – TLS connection authentication failed: General error)
09:02:27-homedir@machine:~$ dirmngr --debug-all --daemon --standard-resolver
dirmngr[58503]: reading options from '/Users/homedir/.gnupg/dirmngr.conf'
dirmngr[58503]: reading options from '[cmdline]'
dirmngr[58503]: enabled debug flags: x509 crypto memory cache memstat hashing ipc dns network lookup extprog
dirmngr[58503]: listening on socket '/Users/homedir/.gnupg/S.dirmngr'
DIRMNGR_INFO=/Users/homedir/.gnupg/S.dirmngr:58504:1; export DIRMNGR_INFO;
09:02:52-homedir@machine:~$ dirmngr[58504.0]: permanently loaded certificates: 133
dirmngr[58504.0]: runtime cached certificates: 0
dirmngr[58504.0]: trusted certificates: 133 (132,0,0,1)
dirmngr[58504.4]: handler for fd 4 started
dirmngr[58504.4]: DBG: chan_4 -> # Home: /Users/homedir/.gnupg
dirmngr[58504.4]: DBG: chan_4 -> # Config: /Users/homedir/.gnupg/dirmngr.conf
dirmngr[58504.4]: DBG: chan_4 -> OK Dirmngr 2.3.1 at your service
dirmngr[58504.4]: DBG: END Certificate
dirmngr[58504.4]: DBG: BEGIN Certificate 'server[2]':
dirmngr[58504.4]: DBG: serial: 4001772137D4E942B8EE76AA3C640AB7
dirmngr[58504.4]: DBG: notBefore: 2021-01-20 19:14:03
dirmngr[58504.4]: DBG: notAfter: 2024-09-30 18:14:03
dirmngr[58504.4]: DBG: issuer: CN=DST Root CA X3,O=Digital Signature Trust Co.
dirmngr[58504.4]: DBG: subject: CN=ISRG Root X1,O=Internet Security Research Group,C=US
dirmngr[58504.4]: DBG: hash algo: 1.2.840.113549.1.1.11
dirmngr[58504.4]: DBG: SHA1 fingerprint: 933C6DDEE95C9C41A40F9F50493D82BE03AD87BF
dirmngr[58504.4]: DBG: END Certificate
dirmngr[58504.4]: TLS connection authentication failed: General error
dirmngr[58504.4]: error connecting to 'http://keys.openpgp.org:80': General error
dirmngr[58504.4]: command 'KS_PUT' failed: General error <Unspecified source>
dirmngr[58504.4]: DBG: chan_4 -> ERR 1 General error <Unspecified source>
dirmngr[58504.4]: DBG: chan_4 <- BYE
dirmngr[58504.4]: DBG: chan_4 -> OK closing connection
dirmngr[58504.4]: handler for fd 4 terminated
Create the file ~/.gnupg/dirmngr.conf with the following contents
The IBM FHIR Server supports many different persistence stores – Apache Derby, Postgres, and IBM Db2. The IBM Db2 offering includes a multi-tenancy feature with row-level permissions. Each tenant is assigned a unique internal id for the tenantName and tenantKey. The tenantKey on each JDBC connection is verified using a one-way hash, and supports allocating a tenant, adding a key to an existing tenant, revoking a key and revoking all tenant keys.
Recipe
Once you have created the schema on your database via the --update-schema, you can allocate a tenant.
2021-06-22 11:19:13.226 00000001 INFO .common.JdbcConnectionProvider Opening connection to database: jdbc:db2://db:50000/fhirdb
2021-06-22 11:19:14.346 00000001 WARNING ls.pool.PoolConnectionProvider Get connection took 1.120 seconds
...
2021-06-22 11:22:59.846 00000001 INFO com.ibm.fhir.schema.app.Main tenantId [29] is being pre-populated with lookup table data.
2021-06-22 11:22:59.930 00000001 INFO com.ibm.fhir.schema.app.Main Finished prepopulating the resource type and search parameter code/name tables tables
2021-06-22 11:22:59.948 00000001 INFO com.ibm.fhir.schema.app.Main Allocated tenant: acme32 [key=UsBglbyMJpSi/RjXwrkp0Bj2bAljUI+MixfAikdrcN0=] with Id = 29
2021-06-22 11:22:59.949 00000001 INFO com.ibm.fhir.schema.app.Main The tenantKey JSON follows:
{"tenantKey": "UsBglbyMJpSi/RjXwrkp0Bj2bAljUI+MixfAikdrcN0="}
2021-06-22 11:22:59.949 00000001 INFO com.ibm.fhir.schema.app.Main Processing took: 226.765 s
2021-06-22 11:22:59.950 00000001 INFO com.ibm.fhir.schema.app.Main SCHEMA CHANGE: OK
2021-06-22 11:25:53.254 00000001 INFO .common.JdbcConnectionProvider Opening connection to database: jdbc:db2://db:50000/fhirdb
2021-06-22 11:25:54.197 00000001 INFO com.ibm.fhir.schema.app.Main New tenant key: acme32 [key=UVFuDqD/V3v8d9S/XRjRQNu9eFTniksvxgIBbI6mEkg=]
2021-06-22 11:25:54.201 00000001 INFO com.ibm.fhir.schema.app.Main Processing took: 0.975 s
2021-06-22 11:25:54.202 00000001 INFO com.ibm.fhir.schema.app.Main SCHEMA CHANGE: OK
2021-06-22 11:28:43.566 00000001 INFO .common.JdbcConnectionProvider Opening connection to database: jdbc:db2://db:50000/fhirdb
TenantId Status TenantName Schema
29 ALLOCATED acme32 FHIRDATA
2021-06-22 11:28:44.395 00000001 INFO com.ibm.fhir.schema.app.Main Processing took: 0.854 s
2021-06-22 11:28:44.396 00000001 INFO com.ibm.fhir.schema.app.Main SCHEMA CHANGE: OK
2021-06-22 11:27:17.280 00000001 INFO .common.JdbcConnectionProvider Opening connection to database: jdbc:db2://db:50000/fhirdb
2021-06-22 11:27:18.112 00000001 INFO com.ibm.fhir.schema.app.Main Tenant Key revoked for 'acme32' total removed=[1]
2021-06-22 11:27:18.119 00000001 INFO com.ibm.fhir.schema.app.Main Processing took: 0.867 s
2021-06-22 11:27:18.120 00000001 INFO com.ibm.fhir.schema.app.Main SCHEMA CHANGE: OK
2021-06-22 11:25:53.254 00000001 INFO .common.JdbcConnectionProvider Opening connection to database: jdbc:db2://db:50000/fhirdb
2021-06-22 11:25:54.197 00000001 INFO com.ibm.fhir.schema.app.Main New tenant key: acme32 [key=UVFuDqD/V3v8d9S/XRjRQNu9eFTniksvxgIBbI6mEkg=]
2021-06-22 11:25:54.201 00000001 INFO com.ibm.fhir.schema.app.Main Processing took: 0.975 s
2021-06-22 11:25:54.202 00000001 INFO com.ibm.fhir.schema.app.Main SCHEMA CHANGE: OK
2021-06-22 11:32:34.061 00000001 INFO .common.JdbcConnectionProvider Opening connection to database: jdbc:db2://db:50000/fhirdb
2021-06-22 11:32:35.112 00000001 INFO com.ibm.fhir.schema.app.Main Tenant Key revoked for 'acme32' total removed=[2]
2021-06-22 11:32:35.144 00000001 INFO com.ibm.fhir.schema.app.Main Processing took: 1.116 s
2021-06-22 11:32:35.146 00000001 INFO com.ibm.fhir.schema.app.Main SCHEMA CHANGE: OK
You’ll see that two are removed.
You now know the lifecycle for the IBM FHIR Server tenantKey – allocate, add and revoke.
The IBM FHIR Server supports hard delete using the custom FHIR Operation Framework. The operation is called $erase; you can always read the design document.
Version Specific Hard Delete – the latest version may not be hard deleted, you must execute a soft DELETE, then a hard delete. Otherwise, all versions may be deleted.
By default, the $erase operation is not enabled, and must be set to true, and the roles allowed to execute the operation, can be tweaked to FHIROperationAdmin or FHIRUsers.
Property
Type
Description
fhirServer/operations/erase/enabled
boolean
Enables the $erase operation
fhirServer/operations/erase/allowedRoles
list
The list of allowed roles, allowed entries are: FHIRUsers every authenticated user, FHIROperationAdmin which is authenticated FHIRAdmin users
Properties
The fhir-server-config.json should be amended with a snippet like the following:
As the $erase operation uses stored procedures for Db2 and functions for Postgres, the data schema must be updated to the 4.8.3 level. It’s also worth noting that Derby is also supported as it uses the same DAO as the Db2 and Postgres to run.
A tip, you should search for the references to the Patient’s ResourceId, and delete the referenced resources in a Transaction Bundle. Net – Search, Assemble Bundle, Execute Batch.
docker logs cde61943964464b528eb77d132fd2a4952e0eaf43588da1c9b6bb2fa584f0608
...
[6/16/21, 15:31:34:533 UTC] 0000002a FeatureManage A CWWKF0011I: The defaultServer server is ready to run a smarter planet. The defaultServer server started in 17.665 seconds.
The IBM FHIR Server supports storage providers S3 (aws-s3,ibm-cos), File System (file), and now Azure (azure-blob).
I recently implemented support for the Azure Blob Service in IBM/FHIR Pull Request #2413. The code that supports the Azure client uses the OK Http library and not Netty to communicate with the backend. The storage provider is implemented as AzureProvider, and uses the Java – AppendBlobClient to facilitate export. To facilitate import, the code uses the BlobClient to read from a location in 10Kb blocks, and reassemble when a full block is reached. Importantly, this is done in a way to ensure full resources are returned up to a 2Gb limit.
Overview Video
If you want to use the IBM FHIR Server’s bulk data feature with the IBM FHIR Server, use the following recipe:
5. Enter the appropriate subscription and resource group
6. Give it a name such as fhirintegrationit
7. Click Review+Create
8. Click Create
You’ll see Deployment in Progress for a period of time.
Once it is done, you see Your deployment is complete, proceed to the next steps.
9. Click Go to Resource
10. Click on Access Key
11. Click Show keys and copy the key1’s Connection string. It’ll look like DefaultEndpointsProtocol=https;AccountName=fhirintegrationit;AccountKey=HIDDEN==;EndpointSuffix=core.windows.net
12. Click on Data > Containers
13. Click +Container
14. Fill in the container details and call it bulkdata
You are now ready to use this with the IBM FHIR Server from the main branch. If you need to clone the repo go to IBM/FHIR.
16. Build the Maven Projects and the Docker Build. You should see “[INFO] BUILD SUCCESS” after each Maven build, and “=> naming to docker.io/ibmcom/ibm-fhir-server:latest” when the Docker build is successful.
18. Open the fhir-server-config.json and update fhirServer/bulkdata/storageProviders/default entry to be the connection is updated to your Connection string from above.
17. Start the Docker container, and capture the container id. It’s going to take a few moments to start up as it lays down the test database.
docker logs cde61943964464b528eb77d132fd2a4952e0eaf43588da1c9b6bb2fa584f0608
...
[6/16/21, 15:31:34:533 UTC] 0000002a FeatureManage A CWWKF0011I: The defaultServer server is ready to run a smarter planet. The defaultServer server started in 17.665 seconds.
19. Create an $import request and capture the content-location header.
20. Check and poll until you get response code 200 (you should only get 202 in the interim) until you see (note 99 failed validation which is expected):
I’ve been debugging a drop tablespace issue on Db2 – IBM/FHIR: 2354. The core issue was a timing problem with lots of partitions dettaching. This adds a delay to the dropTablespace so the async operation can complete, and cleanly exit with a specific error code so downstream consumers can work around the issue.
When debugging why a partition hasn’t dettached, I found that it’s worth checking the details when a drop tablespace fails:
[db2inst1@53fe3a4d3ad2 ~]$ db2 list utilities show detail
ID = 18435
Type = ASYNCHRONOUS PARTITION DETACH
Database Name = FHIRDB
Member Number = 0
Description = Finalize the detach for partition '3' of table 'FHIRDATA.PARAMETER_NAMES'
Start Time = 06/08/2021 16:31:05.526513
State = Executing
Invocation Type = Automatic
Progress Monitoring:
Description = Performing detach operation and
making the target table available; new compilations blocked
Start Time = 06/08/2021 16:31:10.836936
I’ve attached a useful partition.sql to demo partitions and check the system catalog.
Shows the contents of the functions in the schema.
SELECT pg_get_functiondef(f.oid)
FROM pg_catalog.pg_proc f
INNER JOIN pg_catalog.pg_namespace n ON (f.pronamespace = n.oid)
WHERE n.nspname = 'fhirdata';
Show all the details of the functions in the schema.
SELECT *
FROM pg_catalog.pg_proc f
INNER JOIN pg_catalog.pg_namespace n ON (f.pronamespace = n.oid)
WHERE n.nspname = 'fhirdata';
Tracing Bulk Data with Cloud Object Storage
I wanted to figure out why my code was failing to connect to the backend S3 bucket.
I used the environment variable in my docker image called TRACE_SPEC. TRACE_SPEC is loaded into the logging as the traceSpecification.
I set this to *=info:com.ibm.cloud.*=FINEST which spits out great detail to S3 using the IBM COS SDK.
The output looks like:
With this level of trace, you can really dive into the connection, and determine what is going on.
Note, if you want the whole picture of what is happening with COS and JavaBatch and BulkData, use the following: