FHIRcast® is an HL7 specification designed to provide a lightweight, inexpensive and http-based application context synchronization standard. Find out more at fhircast.org.
FHIRcast sandboxes provide tools to simulate the workflow of the FHIRcast standard. They also prototype proposals from the Imaging Integration Workgroup (WG20) to provide early feedback on implementability.
This sandbox (sandbox.js) implements the standard using JavaScript and Node.js. If you are a C#/.net developer, you might prefer to use the other FHIRcast sandbox.
The first communication channel proposed by FHIRcast is the W3C WebSub RFC. This model defines a "hub" that receives subscribtion requests from clients (subscribers) for specific events. With WebSub, 'clients' are actually application servers that manage communication between the central hub and their clients. With this concept, each application can choose whatever is most convenient to relay the information to their clients, possibly using an existing proprietary channel.
Clients subscribe to events by sending the hub the location where they want to receive the events (hub.callback) and the endpoint where they want to send events (hub.topic). The hub then performs a validation by asking the client about a common secret. If this step succeeds, the hub will respond to the original POST with code 202-Accepted and start forwarding events to the client. The following diagram describes the subscription sequence.
To publish events between the apps and the hub, POST transactions are used with JSON payload containing the FHIR® context description and the event name. The content of the message is validated by an HMAC signature in the header that is created using the common secret. The following is an example message for an open-patient-chart event sent to a hub with a client using topic 'v7tfwuk17a':
POST https://hub-fhircast.somedomain.org/api/hub/v7tfwuk17a HTTP/1.1
Host: subscriber
X-Hub-Signature: sha256=dce85dc8dfde2426079063ad413268ac72dcf845f9f923193285e693be6ff3ae
Content-Type: application/json
{
"timestamp": "2018-12-13T01:37:05.14",
"id": "q9v3jubddqt63n1",
"event": {
"hub.topic": "v7tfwuk17a",
"hub.event": "open-patient-chart",
"context": [
{
"key": "patient",
"resource": {
"resourceType": "Patient",
"id": "ewUbXT9RWEbSj5wPEdgRaBw3",
"identifier": [
{
"system": "urn:oid:1.2.840.114350",
"value": "185444"
},
{
"system": "urn:oid:1.2.840.114350.1.13.861.1.7.5.737384.27000",
"value": "2667"
}
]
}
}
]
}
}
The following instances are available online (Microsoft Azure - Frankfurt):
Also without TLS (Google Cloud Platform - Montréal):
To open the links in a different tab, use right-click or ctrl-click on MacOs.
Start with the combined hub/client in the cloud.
- Select the hub:
Leave the defaults URLs. - Subscribe to an event:
Send a subscription request with the send button. The hub response code should be 202 - Acccepted. - Publish an event:
Publish the same type of event with the context defined in the text area with the 'send' button. The hub response code should be 200 - OK. - Monitor the endpoints:
Log entries starting with '📡HUB:' and '🖥️CLIENT:' describe backend messages relevant to the standard. Frontend messages can be seen in the browser console using the browser developer tools. The log entries starting with '🔧UI:' and '🚀WEBSOCKET:' are not relevant to the standard. They provide information about internal operations. '🔥SMART_ON_FHIR®:' log entries are posted when receiving a request for launch.html from a SMART launch application.
To retrieve context after start-up, perform a GET request on the hub's notification endpoint with the session id in the query string by clicking on the "context" button in the 'Publish' section. If the hub has not received a notification for this session-id yet, it will return an empty response with code status 200 - 'success'. The context will be shown in an prompt window as pictured below. Notice that there is only context information in the response and no event name.
The FHIR® subscription resource specifies a number of communication channels and there is interest in following that model in FHIRcast starting with the addition of a websocket channel. With this channel, the end clients communicate directly with the hub and the applications servers are not involved. A possible implementation would be to add 'hub.channel.type' and 'hub.channel.endpoint' to the current subscription request and to provide a wss://fhircast-hub/bind/:[endpoint] service on the hub that binds the websocket created by the client to the subscription.
Keep the websub client and open the websocket client in parallel to subscribe to the same topic with a websocket connection.
You should see notifications propagating to both clients: the websub and the websocket. In the browser console log, you will see the browser sending the subscription and immediately after open the websocket connection. The hub acknowleges the binding with the following response:
{
"timestamp": "2018-01-08T01:37:05.14",
"bound": "jhwer87skrfh345drhxf"
}
where jhwer87skrfh345drhxf is the endpoint uid that the client provided in the subscription.
This prototype has the client posting JSON to the hub to publish events. It may be preferable to publish to the hub through the websocket connection. These topics have not been debated at the moment.
When an application is being launched by an EMR, the SMART on FHIR® launch scenario can provide context on start-up.
Test the online SMART App launcher with a preselected patient, provider and 'App Launch URL'. Click the green 'Launch App!' button in the lower right corner to initiate the launch.
This launches the FHIRcast client within the Simulated EHR. Notice that the 'hub.topic' input textbox has been populated with the SMART session-id and the context text area with the correct patient FHIR resource (both highlighted in light yellow).
Alternatively, test the SMART launch by navigating to the SMART launch sandbox: http://launch.smarthealthit.org and selecting a patient, provider and the app url which can be a local instance in debug mode.
SMART Web Messaging proposes HTML5 Web Messaging for communication between the EMR and Clinical Decision Support (CDS) applications. This webmsg endpoint explores how Web Messaging could work in other scenarios without yet implementing the SMART authorization.
Click the 'Send' buttons to see 'postMessage()' actions across iframes. The 'Launch AI in another window' button may produce a 'pop-up' warning in your browser.
Use the PACS client to subscripe and receive events from the hub. Check the hub.callback input box value of this client. How does it differ from the hub?
Next, add the EMR, AI and reporting clients to simulate a complete workflow.
Finally, open two reporting client browser sessions. Subscribe to an event from one of the browsers and generate an event in the hub. Do both reporting browsers get a message?
This section provides guidance on testing the two sandboxes (.Net core and JavaScript) against each other. The procedure assumes that both sandboxes are running on the same machine; one on port 3000 and the other on port 5000(hub)/5001(client). Since both sandboxes have hub and client components, we can test both client/hub scenarios.
- Start the .Net hub with the following command (lastest .Net Core SDK installed)
$ dotnet run --project Hub
- Start the JS sandbox in client mode
MODE=client node sandbox.js
- Submit a subscription
Use Internet Explorer. Chrome or Safari will not work properly because of CORS header restriction(Allow-access origin) on the hub. Somehow IE sees localhost on different port as same domain.
Navigate to http://localhost:3000 and change the hub URL port from 3000 to 5000 in section 1.
Click the send button in section 2. The console window where the hub is running will show the subscription being accepted.
- Send an event
Click the send button in section 3. The message should be seen in the console window and the browser should show a event being received.
- Start the JS sandbox
node sandbox.js
- Start the .Net WebSub client with the following command
$ dotnet run --project WebSubClient
- Submit a subscription
Navigate the browser to http://localhost:5001/client.
Fill in the entries as shown in the picture above. Note that 'Hub url' is pointing to port 3000.
Click the 'Subscribe' button.
-
Send an event
In the 'User session info' section, make sure that 'Topic' has the chosen session id (subscription topic) at the end. Click the 'Update' button.
- The log text area does not display any messages: Possibly the websocket connection between your browser and the hub is not working. There could be a proxy server in your route that needs a software update or configuration change to support the websocket 'upgrade' http header. Another possibility is that you are using more websockets than your deployment allows. For example, the smallest Azure deployment specifies a maximum of 5 sockets. In any case, the lack of a websocket does not prevent operation. You should still see the responses to the messages in the small text areas next to the send buttons but you will not see events being received by the clients.
- The buttons do not work: Using the browser developper tool, check in the console why the http messages are not going out. If you are testing with another instance, you may have to enable 'send data across domains' in your browser security settings. Another possibility is that the receiving endpoint does not have the 'Access-Control-Allow-Origin' header.
- Response error 404-Not found: The configured hub url node may be set to run as a client-only and therefore the hub endpoints are not available.
- The log emojis are black and white: Color emojis require Office 2016 on Windows 7.
Installing your own sandbox allows you to learn, fix bugs, propose new features or simply provide a private test environment for your project or institution. Whether deploying locally or in the cloud, environment variable settings may be needed.
Environment variables can be used to control the mode of operation, default endpoints and appearance of the instance. If no environment variables are set, the instance will run as a combined hub and client.
- MODE: Specifies if the instance is a 'hub' with a client (subscriber/publisher) or only a 'client'. Default is 'hub'.
- PORT: Specifies the listening port. Default is 3000.
- HUB_URL: Specifies the address where the subscriber and publisher will connect to. Default is http://localhost:3000.
- HUB_ENDPOINT: Specifies the endpoint where the subscriber and publisher will connect to. Default is /api/hub/.
- CLIENT_URL: Specifies the address where the client node will receive published events. Default is http://localhost:3000/client.
- TITLE: Sets the title. Default is 'FHIRcast JavaScript Sandbox - Hub and Client'.
- BACKGROUND_COLOR: Sets the background color. Default is 'darkgray'.
- DEFAULT_CONTEXT: Sets the content of the context text area of the browser on page load. Value must be a valid JSON string. Default is a sample patient context with id 185444.
- Install node at http://nodejs.org.
- Install npm, the node package manager, at http://npmjs.org.
- Clone or download the github and run "npm install" in its directory. This will install the modules defined in package.json.
- Run with "node sandbox.js". This starts the endpoints for the hub and the client.
- Navigate your browser to http://localhost:3000/ to access the UI.
Different port settings are required when running multiple sandboxes locally. They can be set on the command-line when starting the instance:
MODE=client PORT=3001 node sandbox.js
Or in launch.json:
"configurations": [
{
"type": "node",
"request": "launch",
"name": "Launch Program",
"program": "${workspaceFolder}/sandbox.js",
"env":
{
"MODE":"hub",
"HUB_URL":"http://localhost:3000",
"CLIENT_URL":"http://localhost:3000/client",
"TITLE":"my FHIRcast tester",
"BACKGROUND_COLOR":"blue"
}
}
]
Official Node.js binary distributions are provided by NodeSource.
sudo apt install git-all
git clone https://github.com/fhircast/sandbox.js.git
cd sandbox.js # change directory to where the githib was downloaded
curl -sL https://deb.nodesource.com/setup_11.x | sudo -E bash - # Download binaries
sudo apt-get install -y nodejs # install
sudo npm install npm --global # update npm
npm install # install the modules
sudo PORT=80 HUB_URL=http://35.185.207.170 CLIENT_URL=http://35.185.207.170/client TITLE="Ubuntu FHIRcast Sandbox - Hub and Client" npm start
The Visual Studio Code Azure App Service extension can be used for deployments. The environment variables can be added to the App Service instance by creating 'Application Settings' as shown in the following picture. Do not set the port environment variable when deploying in the Azure cloud; it will define the required port for its environment.
In the Google Cloud Platform, you can deploy with the Compute-App Engine option. The platform uses the app.yaml file for deployment:
env: flex
runtime: nodejs
env_variables:
MODE: hub
HUB_URL: https://atomic-airship-221013.appspot.com
CLIENT_URL: https://atomic-airship-221013.appspot.com/client
TITLE: GCP FHIRcast Sandbox - Hub and client
network:
forwarded_ports:
- 65080
instance_tag: websocket
# This sample incurs costs to run on the App Engine flexible environment.
# The settings below are to reduce costs during testing and are not appropriate
# for production use.
manual_scaling:
instances: 1
resources:
cpu: 1
memory_gb: 0.5
disk_size_gb: 10
Create a Node.js project in 'App Engine-Services' and run the following commands in the Google Cloud Shell:
git clone https://github.com/fhircast/sandbox.js.git
cd sandbox.js
gcloud app create
gcloud app deploy
gcloud compute firewall-rules create default-allow-websockets \
--allow tcp:65080 \
--target-tags websocket \
--description "Allow websocket traffic on port 65080"
To update:
git pull https://github.com/fhircast/sandbox.js.git
gcloud app deploy
This article provides an overview of the process.
In AWS, you can use the 'Elastic Beanstalk' deployment to create a WebApp. Deployment is done by uploading a zip file containing the four following files:
- sandbox.js
- sandbox.html
- package.json
- fhir.ico
When creating the application in AWS, choose 'Configure more options' and select 'high'availibility'. SSL cannot be configured without the load balancer provided with the HA deployment. You need to have certificates to configure Amazon cloud. Environment variables are set in the AWS console:
Thanks to https://github.com/janeasystems/nodejs-mobile, running Node.js applications and therefore the FHIRcast sandbox is possible. The description from Janea Systems has been used by a contributor to run sandbox.js on iOS.
There are two files: sandbox.js and sandbox.html.
-
The sandbox.js file is the backend that runs on Node.js. It provides all listening (client and hub) endpoints using the express module.
-
The sandbox.html file is the frontend and runs in the browser. It triggers the client subscription requests and performs client event notifications to the hub. It is served to the browser by the sandbox.js app service.
These two endpoints are not active when the MODE environment variable is set to 'emr','pacs','reporting' or 'ai'.
-
"/api/hub": POST with form query string to receive subscription requests from the clients.
-
"/api/hub/:topic": POST with JSON payload to receive events from the clients.
-
"/api/hub/:topic": GET with query string to provide context in JSON.
-
"/client": POST with JSON payload to receive events and subscribtion cancelations from the hub.
-
"/client": GET with standard query string to receive callback check from the hub.
-
"/": GET with HTML/JavaScript to provide the web page to subscribe and post events to the hub. Default is to return the hub.html file. If the MODE environment variable is set to 'emr','pacs','reporting' or 'ai'; the file emr.html, pacs.html,reporting.html or ai.html will be returned.
-
"/log": (on ws not http), Websocket to broadcast the endpoints logs to the browser.
The following two endpoints are not active when the MODE environment variable is set to 'emr','pacs','reporting' or 'ai'.
-
"/status": POST without content will trigger a hub status message to be broadcasted to the connected websockets.
-
"/api/hub": DELETE without content will delete all subscriptions.
The authorization sequence is described here.
sandbox.html: The four sections of the web page are each contained in their own division or 'div': selectHub, subscribe, publish and monitor.
SMARTlaunch.html: THis is the launch.html file from the SMART App launcher example.
The three FHIRcast-relevant functions are sendEvent(), sendSubscription() and getContext(). All are using 'XMLHttpRequest' instead of the newer 'fetch' function in order to support Internet Explorer.
- sendSubscription(): This function builds a query string using the data from the input fields of section 2 and POSTs it to the hub with 'Content-type' header set to 'application/x-www-form-urlencoded'.
- sendEvent(): This function builds a JSON string using the data from the input fields of section 3 and POSTs it to the hub with 'Content-type' header set to 'application/json'.
- getContext(): This function requests the current context from the hub and displays it in a message box.
Other functions are specific to the sandbox:
- getHubStatus(): Makes a POST to the hub to trigger the display of active subscriptions when the button is clicked.
- deleteSubscriptions(): Makes a POST to the hub to clear the subscriptions list when the button is clicked.
We welcome any contributions to help improve this tool for the FHIRcast community !
To contribute to this project, please issue a pull request on the fhircast/sandbox.js repository with your changes for review.
HL7®, FHIR® and the FHIR logo 🔥® are the registered trademarks of Health Level Seven International.