Here are some short cut APIs for Open Liberty API and IBM FHIR Server’s batch feature.
Blog
-
IBM Digital Developer Conference: Hybrid Cloud – Integrating Healthcare Data in a Serverless World
My lab is now live and available on the IBM Digital Developer Conference. In my session, developers integrate a healthcare data application using IBM FHIR Server with OpenShift serverless, to create and respond to actual healthcare scenarios.
The lab materials Link to the lab material https://prb112.github.io/healthcare-serverless/ and you need to sign up for the lab using the following instructions.
Signing up for the Lab
To get access to the lab environment, follow the following instructions:
1. Get added to the IBM Cloud account “DEGCLOUD” using the following app:– https://account-invite.mybluemix.net/
– Enter Lab key: welcome
– Enter IBM ID: the email you used to sign up for the Digital Developer Conference
2. You will then get an invite message in your email which you must accept to continue. It doesn’t matter if you see an “oops” message here.
3. Then, you can use the following app to get access to a cluster.
– Open Link https://ddc-healthcare-lab.mybluemix.net
– Enter Lab key: oslab
– Enter IBM ID: The email you used to create your IBM Cloud account
To jump right to the session you can go right to the IBM website https://developer.ibm.com/conferences/digital-developer-conference-hybrid-cloud/track-5-labs/s2-healthcare-data-serverless/
Video of the Walk Through
-
Digital Developer Conference – Hybrid Cloud: Integrating Healthcare Data in a Serverless World
Recently I developed and presented this lab… which gets released in late September 2021.
In this lab, developers integrate a healthcare data application using IBM FHIR Server with Red Hat OpenShift Serverless to create and respond to a healthcare scenario.
This lab is a companion to the session Integrating Healthcare Data in a Serverless World at Digital Developer Conference – Hybrid Cloud.
The content for this lab can be found at https://ibm.biz/ibm-fhir-server-healthcare-serverless.
Have fun! Enjoy… Ask Questions… I’m here to help.
-
Playing with buildah and ubi-micro: Part 1
buildah is an intriguing open source tool to build of Open Container Initiative (OCI) container images using a scripted approach versus a traditional Dockerfile. It’s fascinating and I’ve started to use
podman
andbuildah
to build my project’s images.I picked
ubi-micro
as my startingn point. Per Red Hat,ubi-micro
is the smallest possible image excludinng the package manager and all of its dependencies which are normally included in a container image. This approach is an alternative to the current release of the IBM FHIR Server image. The following only documents my first stages with Java testing.- On Fedora, install the prerequisites.
# sudo dnf install buildah -y Last metadata expiration check: 0:23:36 ago on Thu 02 Sep 2021 10:06:55 AM EDT. Dependencies resolved. ===================================================================================================================================================================== Package Architecture Version Repository Size ===================================================================================================================================================================== Installing: buildah x86_64 1.21.4-5.fc33 updates 7.9 M Transaction Summary ===================================================================================================================================================================== Install 1 Package Total download size: 7.9 M Installed size: 29 M Downloading Packages: buildah-1.21.4-5.fc33.x86_64.rpm 7.2 MB/s | 7.9 MB 00:01 --------------------------------------------------------------------------------------------------------------------------------------------------------------------- Total 6.2 MB/s | 7.9 MB 00:01 Running transaction check Transaction check succeeded. Running transaction test Transaction test succeeded. Running transaction Preparing : 1/1 Installing : buildah-1.21.4-5.fc33.x86_64 1/1 Running scriptlet: buildah-1.21.4-5.fc33.x86_64 1/1 Verifying : buildah-1.21.4-5.fc33.x86_64 1/1 Installed: buildah-1.21.4-5.fc33.x86_64 Complete!
- Start the new image
# microcontainer=$(buildah from registry.access.redhat.com/ubi8/ubi-micro) Trying to pull registry.access.redhat.com/ubi8/ubi-micro:latest... Getting image source signatures Copying blob 4f4fb700ef54 done Copying blob 098a109c8679 done Copying config c5ba898d36 done Writing manifest to image destination Storing signatures
- Confirm the container name.
# echo $microcontainer ubi-micro-working-container
- Mount the layer locally and display the path.
# micromount=$(buildah mount $microcontainer) # echo $micromount /var/lib/containers/storage/overlay/14c524d6a5ef0e94887bc52685dbe911b40a5a9e39a6df00dc3b02e5f5ad7796/merged
- Setup the AdoptOpennJdk repository.
cat <<'EOF' > $micromount/etc/yum.repos.d/adoptopenjdk.repo [AdoptOpenJDK] name=AdoptOpenJDK baseurl=http://adoptopenjdk.jfrog.io/adoptopenjdk/rpm/rhel/8/$basearch enabled=1 gpgcheck=1 gpgkey=https://adoptopenjdk.jfrog.io/adoptopenjdk/api/gpg/key/public EOF
- Install to micromount without any ancillary dependencies.
yum install \ --installroot $micromount \ --releasever 8 \ --setopt install_weak_deps=false \ --nodocs -y \ adoptopenjdk-11-openj9xl.x86_64
Results in:
------------------------------------------------------------------------------------------------------------------------------------ Total 8.9 MB/s | 193 MB 00:21 warning: Found bdb Packages database while attempting sqlite backend: using bdb backend. warning: /var/lib/containers/storage/overlay/14c524d6a5ef0e94887bc52685dbe911b40a5a9e39a6df00dc3b02e5f5ad7796/merged/var/cache/dnf/AdoptOpenJDK-096a01411439d076/packages/adoptopenjdk-11-openj9xl-11.0.10+9.openj9-0.24.0-3.x86_64.rpm: Header V4 RSA/SHA1 Signature, key ID 74885c03: NOKEY AdoptOpenJDK 13 kB/s | 3.1 kB 00:00 warning: Found bdb Packages database while attempting sqlite backend: using bdb backend. Importing GPG key 0x74885C03: Userid : "AdoptOpenJDK (used for publishing RPM and DEB files) <adoptopenjdk@gmail.com>" Fingerprint: 8ED1 7AF5 D7E6 75EB 3EE3 BCE9 8AC3 B291 7488 5C03 From : https://adoptopenjdk.jfrog.io/adoptopenjdk/api/gpg/key/public
- Clean up the dependencies
# yum clean all \ --installroot $micromount warning: Found bdb Packages database while attempting sqlite backend: using bdb backend. 61 files removed
- Unmount the container
buildah umount $microcontainer
- Coommit the image
buildah commit $microcontainer ubi-micro-java
- Confirm the image
# buildah images REPOSITORY TAG IMAGE ID CREATED SIZE localhost/ubi-micro-java latest 334404b8ebf2 22 seconds ago 43 MB
It’s about 40M smaller than the ubi-minimal as it has no docs and ancillary dependencies.
Tip: Starting with the IBM FHIR Server
To start with the IBM FHIR Server image, you can use:
buildah from --pull docker.io/ibmcom/ibm-fhir-server:latest [root@localhost ~]# buildah from --pull docker.io/ibmcom/ibm-fhir-server:latest Trying to pull docker.io/ibmcom/ibm-fhir-server:latest... Getting image source signatures Copying blob e2bef77118c7 done Copying blob 45cc8b7f2b43 done Copying blob 5627e846e80f done Copying blob 5f6bf015319e done Copying blob 87212cfd39ea done Copying blob b89ea354ae59 done Copying blob 4a939b72e1c6 done Copying blob d3cbf41efb4e done Copying blob 4feff1abc28e done Copying blob 9ff4465d271b done Copying blob 5e41012b4001 done Copying blob 410af8b678f6 done Copying blob 2f26dc40d01f done Copying blob 1415c9c2e161 done Copying blob e374de62001e done Copying blob 94d978ce0b1f done Copying blob 1fabae8675b6 done Copying blob 7b088cbebf16 done Copying blob 4167c1ebbd85 done Copying config 637552c186 done Writing manifest to image destination Storing signatures ibm-fhir-server-working-container
Tip: Pullinng Fedora
If you need to use Fedora, you can use fedora-minimal.
# buildah from --pull registry.fedoraproject.org/fedora-minimal
To remove the image
$ podman image rm registry.fedoraproject.org/fedora-minimal:34
Tip: Runnning with SELINUX
If you are running with SELINUX, you should set specific selinux permissions.
- set the permission
$ setsebool -P container_manage_cgroup 1
- Confirm the permission
$ getsebool container_manage_cgroup container_manage_cgroup --> on
References
-
Recipe: Azure Postgres with IBM FHIR Server Bulk Data
One of the prerequisites for setting up IBM FHIR Server Bulk Data is setting up max_prepared_transactions since the IBM FHIR Server leverages Open Liberty Java Batch which uses an XA Transaction.
If you are using Azure, here are the steps for updating your Postgres resource.
Navigate to the Azure Portal
Find your Postgres resource
Update your Server Parameters max_prepared_transactions to 200 (anything non-zero is recommended to enable XA)
Click Save
Click Overview
Click Restart
Click On Activity Log
Wait until Postgres is restarted
Restart your IBM FHIR Server, and you are ready to use the Bulk Data feature.
If you don’t do the setup, you’ll see a log like the following:
[9/2/21, 1:49:38:257 UTC] [step1 partition0] com.ibm.fhir.bulkdata.jbatch.listener.StepChunkListener StepChunkListener: job[bulkexportfastjob/8/15] --- javax.transaction.RollbackException com.ibm.jbatch.container.exception.TransactionManagementException: javax.transaction.RollbackException at com.ibm.jbatch.container.transaction.impl.JTAUserTransactionAdapter.commit(JTAUserTransactionAdapter.java:108) at com.ibm.jbatch.container.controller.impl.ChunkStepControllerImpl.invokeChunk(ChunkStepControllerImpl.java:656) at com.ibm.jbatch.container.controller.impl.ChunkStepControllerImpl.invokeCoreStep(ChunkStepControllerImpl.java:795) at com.ibm.jbatch.container.controller.impl.BaseStepControllerImpl.execute(BaseStepControllerImpl.java:295) at com.ibm.jbatch.container.controller.impl.ExecutionTransitioner.doExecutionLoop(ExecutionTransitioner.java:118) at com.ibm.jbatch.container.controller.impl.WorkUnitThreadControllerImpl.executeCoreTransitionLoop(WorkUnitThreadControllerImpl.java:96) at com.ibm.jbatch.container.controller.impl.WorkUnitThreadControllerImpl.executeWorkUnit(WorkUnitThreadControllerImpl.java:178) at com.ibm.jbatch.container.controller.impl.WorkUnitThreadControllerImpl$AbstractControllerHelper.runExecutionOnThread(WorkUnitThreadControllerImpl.java:503) at com.ibm.jbatch.container.controller.impl.WorkUnitThreadControllerImpl.runExecutionOnThread(WorkUnitThreadControllerImpl.java:92) at com.ibm.jbatch.container.util.BatchWorkUnit.run(BatchWorkUnit.java:113) at com.ibm.ws.context.service.serializable.ContextualRunnable.run(ContextualRunnable.java:79) at com.ibm.ws.threading.internal.ExecutorServiceImpl$RunnableWrapper.run(ExecutorServiceImpl.java:238) at java.base/java.util.concurrent.Executors$RunnableAdapter.call(Executors.java:515) at java.base/java.util.concurrent.FutureTask.run(FutureTask.java:264) at java.base/java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1128) at java.base/java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:628) at java.base/java.lang.Thread.run(Thread.java:866) Caused by: javax.transaction.RollbackException at com.ibm.tx.jta.impl.TransactionImpl.stage3CommitProcessing(TransactionImpl.java:978) at com.ibm.tx.jta.impl.TransactionImpl.processCommit(TransactionImpl.java:778) at com.ibm.tx.jta.impl.TransactionImpl.commit(TransactionImpl.java:711) at com.ibm.tx.jta.impl.TranManagerImpl.commit(TranManagerImpl.java:165) at com.ibm.tx.jta.impl.TranManagerSet.commit(TranManagerSet.java:113) at com.ibm.tx.jta.impl.UserTransactionImpl.commit(UserTransactionImpl.java:162) at com.ibm.tx.jta.embeddable.impl.EmbeddableUserTransactionImpl.commit(EmbeddableUserTransactionImpl.java:101) at com.ibm.ws.transaction.services.UserTransactionService.commit(UserTransactionService.java:72) at com.ibm.jbatch.container.transaction.impl.JTAUserTransactionAdapter.commit(JTAUserTransactionAdapter.java:101)
Go back and enable max_prepared_transactions
References
-
Demonstrating the FHIR Extended Operation Composition/$document
Per the specification, a client can ask a server to generate a fully bundled document from a composition resource. I’ve pulled together a Postman to help demonstrate this feature on the IBM FHIR Server.
- Download the Postman
2. Update the SERVER_HOSTNAME
3. Update the Authorization for your username and password
4. Click Tests > Run
5. Click Run
6. You’ll see the tests run.
7. Click on the Tests of Interest, and then check the curl you are interested, such as:
curl --location --request GET 'https://localhost:9443/fhir-server/api/v4/Composition/17b83b99f91-3c0d6274-0498-4fe4-999e-ba8574f85b09/$document?persist=true' \ --header 'Content-Type: application/fhir+json' \ --header 'Authorization: Basic .....'
Good Luck with Composition/$document.
-
Recipe: Building a custom IBM FHIR Server container with Bulk Data Parquet Support
Note: Parquet Support is now obsolete.
The IBM FHIR Server has early support for Bulk Data export to the Apache Parquet format using the Apache Spark libraries. New as of version 4.4.0, the export to parquet feature requires:
- Apache Spark v3.0 and the IBM Stocator adapter (version 1.1)
- the configuration
/fhirServer/bulkdata/storageProviders/(source)/enableParquet
set to true
The Parquet Bulk Data export is activated using a custom
_outputFormat
in the export request.{ "name": "_outputFormat", "valueString": "application/fhir+parquet" },
Let me show you how to build a custom IBM FHIR Server container with parquet support Docker: ibmcom/ibm-fhir-server. It is recommended to use 4.9.0 or higher.
Recipe
- Prior to 4.9.0, build the Maven Projects and the Docker Build. You should see
[INFO] BUILD SUCCESS
after each Maven build, anddocker.io/ibmcom/ibm-fhir-server:latest
when the Docker build is successful.
mvn clean install -f fhir-examples -B -DskipTests -ntp mvn clean install -f fhir-parent -B -DskipTests -ntp docker build -t ibmcom/ibm-fhir-server:latest fhir-install
- Download the dependency files for parquet and stocator.
export WORKSPACE=~/git/wffh/2021/fhir bash ${WORKSPACE}/fhir-bulkdata-webapp/src/main/sh/cache-parquet-deps.sh
- Download the fhir-server-config.json
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 40495 0 --:--:-- --:--:-- --:--:-- 40301
- Update the fhir-server-config.json to use a IBM COS storage provider with parquet support. You’ll need to update with your HMAC id, internal and external URLs and parquet enabled.
"storageProviders": { "default" : { "type": "ibm-cos", "bucketName": "fhir-performance", "location": "us-east", "endpointInternal": "https://s3.us-east.cloud-object-storage.appdomain.cloud", "endpointExternal": "https://s3.us-east.cloud-object-storage.appdomain.cloud", "auth" : { "type": "hmac", "accessKeyId": "key", "secretAccessKey": "secret" }, "enableParquet": true, "disableOperationOutcomes": true, "duplicationCheck": false, "validateResources": false, "create": false } }
- 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 run -d -p 9443:9443 -e BOOTSTRAP_DB=true \ -v $(pwd)/fhir-server-config.json:/config/config/default/fhir-server-config.json \ -v $(pwd)/deps:/config/userlib/ \ ibmcom/ibm-fhir-server 3f8e90f20cd42129adc58df8a0295efc3fb2a0f4507350589f71939a072999ae
- Check the logs until you see:
docker logs 3f8e90f20cd42129adc58df8a0295efc3fb2a0f4507350589f71939a072999ae ... [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.
- Download the Sample Data
curl -L https://raw.githubusercontent.com/IBM/FHIR/main/fhir-server-test/src/test/resources/testdata/everything-operation/Antonia30_Acosta403.json \ -o Antonia30_Acosta403.json
- Load the Sample Data bundle to the IBM FHIR Server
curl -k --location --request POST 'https://localhost:9443/fhir-server/api/v4' \ --header 'Content-Type: application/fhir+json' \ --user "fhiruser:${DUMMY_PASSWORD}" \ --data-binary "@Antonia30_Acosta403.json" -o response.json
Note, DUMMY_PASSWORD should be previously set.
- Scan the response.json for any status that is not "status": "201". For example, the status is in the family of User Request Error or Server Side Error.
cat response.json | jq -r '.entry[].response.status' | sort -u 201
- Call the export to Parquet operation, and grab the content-location.
curl --location --request GET 'https://localhost:9443/fhir-server/api/v4/$export?_outputFormat=application/fhir%2Bparquet&_type=Patient' \ --header 'X-FHIR-TENANT-ID: default' \ --user "fhiruser:${DUMMY_PASSWORD}" \ --header 'Content-Type: application/json' -k -v
< content-location: https://localhost:9443/fhir-server/api/v4/$bulkdata-status?job=LqzauvqtHSmkpChVHo%2B1MQ
- Check the exprot status using the previous URL, and once you see a 200 response, you can go out and use your exported Parquet data.
curl --location --request GET 'https://localhost:9443/fhir-server/api/v4/$bulkdata-status?job=LqzauvqtHSmkpChVHo%2B1MQ' \ --header 'X-FHIR-TENANT-ID: default' \ --user "fhiruser:${DUMMY_PASSWORD}" \ --header 'Content-Type: application/json' -k
{ "transactionTime": "2021-08-09T00:34:11.594Z", "request": "https://localhost:9443/fhir-server/api/v4/$export?_outputFormat=application/fhir%2Bparquet&_type=Patient", "requiresAccessToken": false, "output": [ { "type": "Patient", "url": "https://s3.us-east.cloud-object-storage.appdomain.cloud/fhir-performance/AZ0gsQS05_RqZnHPhj57AfhYSIHU8VzwmnWjDCQdi2I/Patient_1.parquet?X-Amz-Algorithm=AWS4-HMAC-SHA256&X-Amz-Credential=fc85bf9cc1ac49e99e40085f9ba00f77%2F20210809%2Fus-east%2Fs3%2Faws4_request&X-Amz-Date=20210809T003601Z&X-Amz-Expires=86400&X-Amz-SignedHeaders=host&X-Amz-Signature=6d54f677b91d92304caf889eb0a1efbc2b3ebe3d24cefd9c17169b21816d1cdf", "count": 1 } ] }
List of the Parquet Files View of the Parquet File You now have a working Parquet output.
References
-
Recipe: Reindexing with fhir-bucket and the IBM FHIR Server
The IBM FHIR Server enables user defined and implementation guide defined SearchParameter definitions and is constantly improving Specification conformance. Given these dynamic changes, the IBM FHIR Server Search parameter values may require refresh to optimize the Search and Retrieval of the data in the operational data store.
A quick tool to aid in this is
fhir-bucket
Let me show you how to run reindexing with the IBM FHIR Server container Docker: ibmcom/ibm-fhir-server. This feature requires 4.9.0 or higher.
Recipe
- Prior to 4.9.0, build the Maven Projects and the Docker Build. You should see
[INFO] BUILD SUCCESS
after each Maven build, anddocker.io/ibmcom/ibm-fhir-server:latest
when the Docker build is successful.
mvn clean install -f fhir-examples -B -DskipTests -ntp mvn clean install -f fhir-parent -B -DskipTests -ntp docker build -t ibmcom/ibm-fhir-server:latest fhir-install
- Download the fhir-server-config.json
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 40495 0 --:--:-- --:--:-- --:--:-- 40301
- Download the extension-search-parameters.json
curl -L -o extension-search-parameters.json \ https://raw.githubusercontent.com/IBM/FHIR/main/fhir-server/liberty-config/config/default/extension-search-parameters.json % Total % Received % Xferd Average Speed Time Time Time Current Dload Upload Total Spent Left Speed 100 154 100 154 0 0 611 0 --:--:-- --:--:-- --:--:-- 611
- 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 run -d -p 9443:9443 -e BOOTSTRAP_DB=true \ -v $(pwd)/fhir-server-config.json:/config/config/default/fhir-server-config.json \ -v $(pwd)/extension-search-parameters.json:/config/config/default/extension-search-parameters.json \ ibmcom/ibm-fhir-server 9c44ad49784a6ca6456799ab63aea460fc564cc4a9bd5f4fef1aeae4b82225bd
- Check the logs until you see:
docker logs 9c44ad49784a6ca6456799ab63aea460fc564cc4a9bd5f4fef1aeae4b82225bd ... [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.
- Download the Sample Data
curl -L https://raw.githubusercontent.com/IBM/FHIR/main/fhir-server-test/src/test/resources/testdata/everything-operation/Antonia30_Acosta403.json \ -o Antonia30_Acosta403.json
- Load the Sample Data bundle to the IBM FHIR Server
curl -k --location --request POST 'https://localhost:9443/fhir-server/api/v4' \ --header 'Content-Type: application/fhir+json' \ --user "fhiruser:${DUMMY_PASSWORD}" \ --data-binary "@Antonia30_Acosta403.json" -o response.json
Note, DUMMY_PASSWORD should be previously set.
- Scan the response.json for any status that is not "status": "201". For example, the status is in the family of User Request Error or Server Side Error.
cat response.json | jq -r '.entry[].response.status' | sort -u 201
- Search maiden-name = Graciela518 Badillo851, and you’ll see the reindexing completes and the search succeeds.
curl --location --request GET 'https://localhost:9443/fhir-server/api/v4/Patient?maiden-name=Graciela518 Badillo851' \ --header 'X-FHIR-TENANT-ID: default' \ --user "fhiruser:${DUMMY_PASSWORD}" \ --header 'Content-Type: application/json' -k
{ "resourceType":"OperationOutcome", "id":"ac-11-0-3-14c966ad-8c8b-438d-86ad-de4b2fdb27b0", "issue":[ { "severity":"fatal", "code":"invalid", "details": { "text":"Search parameter 'maiden-name' for resource type 'Patient' was not found." }, "expression":["<empty>"] } ] }
- Download the extension-search-parameters.json with an example expression.
curl -L -o extension-search-parameters.json \ https://gist.githubusercontent.com/prb112/03f0d77f72685180868d2f31a1070ebe/raw/14fda1cb86dda74095faa5e32fec6f9226ffb587/extension-search-parameters.json % Total % Received % Xferd Average Speed Time Time Time Current Dload Upload Total Spent Left Speed 100 752 100 752 0 0 2231 0 --:--:-- --:--:-- --:--:-- 2231
- Post 4.9.0’s release, download the fhir-bucket-cli
curl -L -o fhir-bucket-4.9.0-cli.jar \ https://repo1.maven.org/maven2/com/ibm/fhir/fhir-bucket/4.9.0/fhir-bucket-4.9.0-cli.jar % Total % Received % Xferd Average Speed Time Time Time Current Dload Upload Total Spent Left Speed 100 46.7M 100 46.7M 0 0 12.2M 0 0:00:03 0:00:03 --:--:-- 12.2M
- Pre 4.9.0’s release, copy the
fhir-bucket-cli
from the .m2 repository.
cp ~/.m2/repository/com/ibm/fhir/fhir-bucket/4.9.0-SNAPSHOT/fhir-bucket-4.9.0-SNAPSHOT-cli.jar fhir-bucket-4.9.0-cli.jar
- Download the client truststore
curl -L -o fhirClientTrustStore.p12 \ https://github.com/IBM/FHIR/raw/main/fhir-server-test/src/test/resources/fhirClientTrustStore.p12 % Total % Received % Xferd Average Speed Time Time Time Current Dload Upload Total Spent Left Speed 100 174 100 174 0 0 661 0 --:--:-- --:--:-- --:--:-- 661 100 5946 100 5946 0 0 12651 0 --:--:-- --:--:-- --:--:-- 12651
- Download the fhir-bucket logging.properties
curl -L -o logging.properties \ https://raw.githubusercontent.com/IBM/FHIR/main/fhir-bucket/logging.properties % Total % Received % Xferd Average Speed Time Time Time Current Dload Upload Total Spent Left Speed 100 688 100 688 0 0 3245 0 --:--:-- --:--:-- --:--:-- 3245
- Create the bucket.properties file
cat << EOF > bucket.properties fhir.server.host=localhost fhir.server.port=9443 fhir.server.user=fhiruser fhir.server.pass=change-password fhir.server.endpoint=/fhir-server/api/v4/ truststore=fhirClientTrustStore.p12 truststore.pass=change-password read.timeout=125000 connect.timeout=20000 pool.connections.max=400 disable.hostname.verification=true EOF
- Execute the fhir-bucket reindex operation with the current date-time (e.g. now is
2021-12-01T00:00:00Z
)
export JAR="fhir-bucket-4.9.0-cli.jar" java -Djava.util.logging.config.file=logging.properties -jar "${JAR}" \ --fhir-properties bucket.properties \ --tenant-name default \ --max-concurrent-fhir-requests 100 \ --no-scan \ --reindex-tstamp 2021-12-01T00:00:00Z \ --reindex-resource-count 50 \ --reindex-concurrent-requests 20 \ --reindex-client-side-driven
- Wait until the reindexing threads complete.
2021-08-05 15:28:05.629 00000001 INFO x.ClientDrivenReindexOperation Starting monitor thread 2021-08-05 15:28:07.864 0000001b INFO x.ClientDrivenReindexOperation called $retrieve-index: 200 OK [took 0.851 s] 2021-08-05 15:28:07.872 0000001b INFO x.ClientDrivenReindexOperation Index IDs available for processing - filling worker pool 2021-08-05 15:28:09.145 0000001d INFO x.ClientDrivenReindexOperation called $reindex (indexIds=2,...): 200 OK [took 1.268 s] ... 2021-08-05 15:28:27.946 0000001b INFO x.ClientDrivenReindexOperation called $retrieve-index (afterIndexId=1802): 200 OK [took 0.053 s] 2021-08-05 15:28:27.949 0000001b INFO x.ClientDrivenReindexOperation No more index IDs to retrieve 2021-08-05 15:28:27.950 0000001b INFO x.ClientDrivenReindexOperation Nothing left to do, so tell all the worker threads to exit 2021-08-05 15:28:27.951 0000001b INFO x.ClientDrivenReindexOperation Waiting for 20 threads to complete before exiting 2021-08-05 15:28:32.952 0000001b INFO x.ClientDrivenReindexOperation Reindexing was completed 2021-08-05 15:29:32.891 0000001a INFO com.ibm.fhir.bucket.app.Main Stopping all services
- Search maiden-name = Graciela518 Badillo851, and you’ll see the reindexing completes and the search succeeds.
curl --location --request GET 'https://localhost:9443/fhir-server/api/v4/Patient?maiden-name=Graciela518 Badillo851' \ --header 'X-FHIR-TENANT-ID: default' \ --user "fhiruser:${DUMMY_PASSWORD}" \ --header 'Content-Type: application/json' -k
{ "id": "17b17befcde-8fb6aca7-9008-4682-ae7f-4c3c5a91c7e2", "fullUrl": "https://localhost:9443/fhir-server/api/v4/Patient/17b17befcde-8fb6aca7-9008-4682-ae7f-4c3c5a91c7e2", "resource": { "resourceType": "Patient", "id": "17b17befcde-8fb6aca7-9008-4682-ae7f-4c3c5a91c7e2", "meta": { "versionId": "1", "lastUpdated": "2021-08-05T19:16:37.89885Z" }, "text": { "status": "generated", "div": "<div xmlns=\"http://www.w3.org/1999/xhtml\">Generated by <a href=\"https://github.com/synthetichealth/synthea\">Synthea</a>.Version identifier: master-branch-latest-2-g8e7e92c\n . Person seed: 7105486291024734541 Population seed: 0</div>" }, "extension": [ { "url": "http://hl7.org/fhir/StructureDefinition/patient-mothersMaidenName", "valueString": "Graciela518 Badillo851" }, { "url": "http://hl7.org/fhir/StructureDefinition/patient-birthPlace", "valueAddress": { "city": "Caguas", "state": "Puerto Rico", "country": "PR" } }, { "url": "http://synthetichealth.github.io/synthea/disability-adjusted-life-years", "valueDecimal": 1.6165934938317164 }, { "url": "http://synthetichealth.github.io/synthea/quality-adjusted-life-years", "valueDecimal": 47.383406506168288 }, { "url": "http://hl7.org/fhir/us/core/StructureDefinition/us-core-birthsex", "valueCode": "F" }, { "url": "http://hl7.org/fhir/us/core/StructureDefinition/us-core-birthsex", "valueCode": "M" } ], "identifier": [ { "system": "https://github.com/synthetichealth/synthea", "value": "bd958c64-56b6-f206-b03d-8f4b8a417215" }, { "type": { "coding": [ { "system": "http://terminology.hl7.org/CodeSystem/v2-0203", "code": "MR", "display": "Medical Record Number" } ], "text": "Medical Record Number" }, "system": "http://hospital.smarthealthit.org", "value": "bd958c64-56b6-f206-b03d-8f4b8a417215" }, { "type": { "coding": [ { "system": "http://terminology.hl7.org/CodeSystem/v2-0203", "code": "SS", "display": "Social Security Number" } ], "text": "Social Security Number" }, "system": "http://hl7.org/fhir/sid/us-ssn", "value": "999-93-6090" }, } ]
References
- Prior to 4.9.0, build the Maven Projects and the Docker Build. You should see
-
Recipe: Running the IBM FHIR Server behind a Reverse Proxy
A common deployment pattern for the IBM FHIR Server is to run the Application Server behind a reverse proxy (e.g. OpenShift Route, Kubernetes Ingress, NGINX or API Gateway). By default, the IBM FHIR Server runs under the
https://localhost:9443/fhir-server/api/v4
context root and URI. With a modest configuration change, one can change to a context root (baseUrl) or use theX-FHIR-FORWARDED-URL
to forward the incoming url to the IBM FHIR Server (expected to be from a trusted reverse proxy).In #1965, the
fhirServer/core/externalBaseUrl
is a tenant aware configuration that sets the context root and base URL. Note, thefhirServer/core/externalBaseUrl
overrides the incomingUrl fromX-FHIR-FORWARDED-URL
.This document outlines how to set the
externalBaseUrl
for the IBM FHIR Server.Let me show you how to add a set the
externalBaseUrl
to an IBM FHIR Server container Docker: ibmcom/ibm-fhir-server. This feature requires 4.9.0 or higher.Recipe
- Prior to 4.9.0, build the Maven Projects and the Docker Build. You should see
[INFO] BUILD SUCCESS
after each Maven build, anddocker.io/ibmcom/ibm-fhir-server:latest
when the Docker build is successful.
mvn clean install -f fhir-examples -B -DskipTests -ntp mvn clean install -f fhir-parent -B -DskipTests -ntp docker build -t ibmcom/ibm-fhir-server:latest fhir-install</code></pre>
- Download the fhir-server-config.json
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
- Update the fhir-server-config.json at path
fhirServer/core/externalBaseUrl
tohttps://chocolate.fudge
.
"externalBaseUrl": "https://chocolate.fudge"
- 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 run -d -p 9443:9443 -e BOOTSTRAP_DB=true \ -v $(pwd)/fhir-server-config.json:/config/config/default/fhir-server-config.json \ ibmcom/ibm-fhir-server a096978867195ff6e610c36cdba77ff423c31c0ad488a7390f42cef6e89e7fd0
- Check the logs until you see:
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.
- Download the Sample Data
curl -L https://raw.githubusercontent.com/IBM/FHIR/main/fhir-server-test/src/test/resources/testdata/everything-operation/Antonia30_Acosta403.json -o Antonia30_Acosta403.json
- Load the Sample Data bundle to the IBM FHIR Server
curl -k --location --request POST 'https://localhost:9443/fhir-server/api/v4' \ --header 'Content-Type: application/fhir+json' \ --user "fhiruser:${DUMMY_PASSWORD}" \ --data-binary "@Antonia30_Acosta403.json" -o response.json
-
Scan the response.json for any status that is not "status": "201". For example, the status is in the family of User Request Error or Server Side Error.
-
Check the patient and you’ll see the self and next relation links include
https://chocolate.fudge
curl -k --location --request GET 'https://localhost:9443/fhir-server/api/v4/Patient' \ --header 'Content-Type: application/fhir+json' \ --user "fhiruser:${DUMMY_PASSWORD}" \
{ "resourceType": "Bundle", "id": "37c5abc7-d3e7-4506-b596-9725c59f9b6b", "type": "searchset", "total": 23, "link": [ { "relation": "self", "url": "https://chocolate.fudge/Patient?_count=10&_page=1" }, { "relation": "next", "url": "https://chocolate.fudge/Patient?_count=10&_page=2" } ], "entry": [ { "id": "17b123f9a79-bd2011c1-6606-4617-90ed-3187790955b8", "fullUrl": "https://chocolate.fudge/Patient/17b123f9a79-bd2011c1-6606-4617-90ed-3187790955b8", "resource": { "resourceType": "Patient", "id": "17b123f9a79-bd2011c1-6606-4617-90ed-3187790955b8", "meta": { "versionId": "1", "lastUpdated": "2021-08-04T17:39:23.385314Z", "tag": [ { "system": "http://terminology.hl7.org/CodeSystem/v3-ActReason", "code": "HTEST", "display": "test health data" } ] } } } ] }
References
- Prior to 4.9.0, build the Maven Projects and the Docker Build. You should see
-
Postgres and Vacuum with the IBM FHIR Server: Debugging Details
The IBM FHIR Server stores resources in the PostgreSQL database and uses a relational model to store historical FHIR Resource and enable search on the latest FHIR resources. The resource data is spread in a relational model that is occasionally tweaked in order to improve search or optimize the retrieval using the relational model.
In the IBM FHIR Server Performance Guide, the guide outlines some important alterations to the tables that facilitate an optimized Postgres instance. The guide suggests altering, per your providers recommendation, autovacuum_vacuum_cost_limit, autovacuum_vacuum_scale_factor and autovacuum_vacuum_threshold in order to optimize the VACUUM process. With the IBM FHIR Server fhir-persistence-schema-cli, autovacuum_vacuum_scale_factor is not automatically configured, and not recommended on Databases for Postgres on IBM Cloud.
As Postgres uses "multi-version concurrency control (MVCC) to ensure that data remains consistent and accessible in high-concurrency environments", each transaction runs on a snapshot, and needs to be reconciled so dead_rows are removed – vacuumed. The VACUUM process manages dead rows and disk usage (reuse). The VACUUM process (autovacuum) frequently runs – gathering statistics and reconciling the maintenance of the table statitstics and data.
To check for tables that need vacuuming:
SELECT relname AS "table_name", n_tup_ins AS "inserts", n_tup_upd AS "updates", n_tup_del AS "deletes", n_live_tup AS "live_tuples", n_dead_tup AS "dead_tuples" FROM pg_stat_user_tables WHERE schemaname = 'fhirdata' AND (relname = 'logical_resources' OR relname LIKE '%_values') AND n_dead_tup > 0;
Then a database administrator runs –
VACUUM FULL FHIRDATA.PROCEDURE_RESOURCE_TOKEN_REFS;
to execute a vacuum, which runs in the background.While the VACUUM process is running, the
pg_stat_progress_vacuum
view can be queried to see worker process.SELECT * FROM pg_stat_progress_vacuum;
If you need to update a specific tables settings, you can run with
--vacuum-table-name
.java -jar ./fhir-persistence-schema-${VERSION}-cli.jar \ --db-type postgresql --prop db.host=localhost --prop db.port=5432 \ --prop db.database=fhirdb --schema-name fhirdata \ --prop user=fhiradmin --prop password=passw0rd \ --update-vacuum --vacuum-cost-limit 2000 --vacuum-threshold 1000 \ --vacuum-scale-factor 0.01 --vacuum-table-name LOGICAL_RESOURCES
To update all tables in a schema, you can run without the table parameter. If you omit any value, the defaults are picked as described in the Performance guide.
If you hit a lock (ShareUpdateExclusiveLock), the VACUUM worker process is currently churning on the table, and the ALTER statement is waiting.
- wait_type = Lock relation Waiting to acquire a lock on a relation.
- wait_lock_type – ShareUpdateExclusiveLock Acquired by VACUUM and conflicts with ALTER
CHeck for the Blocking PID, and grab the blocking_pid.
-- list bad connections SELECT blocked_locks.pid AS blocked_pid, blocked_activity.usename AS blocked_user, blocking_locks.pid AS blocking_pid, blocking_activity.usename AS blocking_user, blocked_activity.query AS blocked_statement, blocking_activity.query AS current_statement_in_blocking_process, blocked_activity.application_name AS blocked_application, blocking_activity.application_name AS blocking_application FROM pg_catalog.pg_locks blocked_locks JOIN pg_catalog.pg_stat_activity blocked_activity ON blocked_activity.pid = blocked_locks.pid JOIN pg_catalog.pg_locks blocking_locks ON blocking_locks.locktype = blocked_locks.locktype AND blocking_locks.DATABASE IS NOT DISTINCT FROM blocked_locks.DATABASE AND blocking_locks.relation IS NOT DISTINCT FROM blocked_locks.relation AND blocking_locks.page IS NOT DISTINCT FROM blocked_locks.page AND blocking_locks.tuple IS NOT DISTINCT FROM blocked_locks.tuple AND blocking_locks.virtualxid IS NOT DISTINCT FROM blocked_locks.virtualxid AND blocking_locks.transactionid IS NOT DISTINCT FROM blocked_locks.transactionid AND blocking_locks.classid IS NOT DISTINCT FROM blocked_locks.classid AND blocking_locks.objid IS NOT DISTINCT FROM blocked_locks.objid AND blocking_locks.objsubid IS NOT DISTINCT FROM blocked_locks.objsubid AND blocking_locks.pid != blocked_locks.pid JOIN pg_catalog.pg_stat_activity blocking_activity ON blocking_activity.pid = blocking_locks.pid WHERE NOT blocked_locks.GRANTED and blocked_activity.usename = 'fhirserver'
Try canceling the PID,
SELECT pg_cancel_backend(205384);
Else, cancel the current Transaction the blocked pid:
-- cancel the blocking trannsaction/pid (hard stop) SELECT pg_cancel_backend(blocked_locks.pid) AS blocked_pid, blocked_activity.usename AS blocked_user, blocking_locks.pid AS blocking_pid, blocking_activity.usename AS blocking_user, blocked_activity.query AS blocked_statement, blocking_activity.query AS current_statement_in_blocking_process, blocked_activity.application_name AS blocked_application, blocking_activity.application_name AS blocking_application FROM pg_catalog.pg_locks blocked_locks JOIN pg_catalog.pg_stat_activity blocked_activity ON blocked_activity.pid = blocked_locks.pid JOIN pg_catalog.pg_locks blocking_locks ON blocking_locks.locktype = blocked_locks.locktype AND blocking_locks.DATABASE IS NOT DISTINCT FROM blocked_locks.DATABASE AND blocking_locks.relation IS NOT DISTINCT FROM blocked_locks.relation AND blocking_locks.page IS NOT DISTINCT FROM blocked_locks.page AND blocking_locks.tuple IS NOT DISTINCT FROM blocked_locks.tuple AND blocking_locks.virtualxid IS NOT DISTINCT FROM blocked_locks.virtualxid AND blocking_locks.transactionid IS NOT DISTINCT FROM blocked_locks.transactionid AND blocking_locks.classid IS NOT DISTINCT FROM blocked_locks.classid AND blocking_locks.objid IS NOT DISTINCT FROM blocked_locks.objid AND blocking_locks.objsubid IS NOT DISTINCT FROM blocked_locks.objsubid AND blocking_locks.pid != blocked_locks.pid JOIN pg_catalog.pg_stat_activity blocking_activity ON blocking_activity.pid = blocking_locks.pid WHERE NOT blocked_locks.GRANTED and blocked_activity.usename = 'fhirserver'
Now, wait until the VACUUM finishes, and then execute a new ALTER.
You should be all set at this point.
Check wait_type
-- Check Wait Type SELECT waiting.locktype AS waiting_locktype, waiting.relation::regclass AS waiting_table, waiting_stm.query AS waiting_query, waiting.mode AS waiting_mode, waiting.pid AS waiting_pid, other.locktype AS other_locktype, other.relation::regclass AS other_table, other_stm.query AS other_query, other.mode AS other_mode, other.pid AS other_pid, other.granted AS other_granted FROM pg_catalog.pg_locks AS waiting JOIN pg_catalog.pg_stat_activity AS waiting_stm ON ( waiting_stm.pid = waiting.pid ) JOIN pg_catalog.pg_locks AS other ON ( ( waiting."database" = other."database" AND waiting.relation = other.relation ) OR waiting.transactionid = other.transactionid ) JOIN pg_catalog.pg_stat_activity AS other_stm ON ( other_stm.pid = other.pid ) WHERE NOT waiting.granted AND waiting.pid <> other.pid SELECT schemaname, relname, last_vacuum, last_autovacuum, vacuum_count, autovacuum_count, * FROM pg_stat_user_tables WHERE schemaname = 'fhirdata' AND relname = 'observation_date_values';
Check with locks
SELECT now()::time, query, backend_start, xact_start, query_start, state_change, state, now()::time - state_change::time AS locked_since, pid, wait_event_type, wait_event FROM pg_stat_activity WHERE wait_event_type IS NOT NULL and wait_event_type = 'Lock' ORDER BY locked_since DESC;
Check a PID
SELECT a.usename, a.application_name, a.datname, a.query, l.granted, l.mode, transactionid FROM pg_locks l JOIN pg_stat_activity a ON a.pid = l.pid WHERE granted = false AND a.pid = 327589;
Check a tansaction
SELECT a.usename, a.application_name, a.datname, a.query, l.granted, l.mode, transactionid, now()::time - a.state_change::time AS acquired_since, a.pid FROM pg_locks l JOIN pg_stat_activity a ON a.pid = l.pid WHERE granted = true AND transactionid = 3031;
Reference
- IBM FHIR Server: Adjust the Vacuum Settings for PostgreSQL Tables only
- IBM FHIR Server: 4.1.4. Vacuum Monitoring
- Postgres: pg_locks
- Postgres: Lock dependency information
- Postgres: 9.6.1. Table-level locks
- Postgres: 28.4.1. VACUUM Progress Reporting
- StackOverflow: Remove locks without a pid in postgres
- StackOverflow: How to detect query which holds the lock in Postgres?
- Monitoring PostgreSQL VACUUM processes
- Exploring Query Locks in Postgres
- Find transactions that are locked (and who to blame for!)
- PostgreSQL: Decoding Deadlocks
- Locks in PostgreSQL: 1. Relation-level locks
- IBM Cloud – Postgres Settings