Skip to content

Java solution integrating DHIS2 and WhatsApp using the dhis2-rapidpro connector

License

Notifications You must be signed in to change notification settings

dev-otta/integration-dhis2-whatsapp

 
 

Repository files navigation

DHIS-to-RapidPro

Build Status

Table of Contents

Introduction

DHIS-to-RapidPro is a stand-alone Java solution that integrates DHIS2 with RapidPro. DHIS2 is an open-source information system primarily used in the health domain while RapidPro is an open-source workflow engine for running mobile-based services.

DHIS-to-RapidPro provides:

  • Routine synchronisation of RapidPro contacts with DHIS2 users
  • Aggregate report transfer from RapidPro to DHIS2 via polling or webhook messaging
  • Automated reminders to RapidPro contacts when their aggregate reports are overdue

Requirements

  • Java 11
  • RapidPro v7.4
  • DHIS >= v2.37.8

Getting Started

Shell

*inux

The JAR distribution of DHIS-to-RapidPro allows you to run the application as a stand-alone process. On *nix operating systems, you can execute DHIS-to-RapidPro from your terminal like so:

./dhis2rapidpro.jar

The above command will give an error since no parameters are provided. The next commands are common DHIS-to-RapidPro *nix usage examples:

Accept webhook reports
export DHIS2_API_PAT=d2pat_apheulkR1x7ac8vr9vcxrFkXlgeRiFc94200032556
export RAPIDPRO_API_TOKEN=3048a3b9a04c1948aa5a7fd06e7592ba5a17d3d0

./dhis2rapidpro.jar --dhis2.api.url=https://play.dhis2.org/2.38.1/api \
--rapidpro.api.url=https://rapidpro.dhis2.org/api/v2 \
--rapidpro.webhook.enabled=true
Auto-reminders
export DHIS2_API_PAT=d2pat_apheulkR1x7ac8vr9vcxrFkXlgeRiFc94200032556
export RAPIDPRO_API_TOKEN=3048a3b9a04c1948aa5a7fd06e7592ba5a17d3d0

./dhis2rapidpro.jar --dhis2.api.url=https://play.dhis2.org/2.38.1/api \
--rapidpro.api.url=https://rapidpro.dhis2.org/api/v2 \
--reminder.data.set.codes=DS_359414,DS_543073,HIV_CARE
Activate contact synchronisation
export DHIS2_API_PAT=d2pat_apheulkR1x7ac8vr9vcxrFkXlgeRiFc94200032556
export RAPIDPRO_API_TOKEN=3048a3b9a04c1948aa5a7fd06e7592ba5a17d3d0

./dhis2rapidpro.jar --dhis2.api.url=https://play.dhis2.org/2.38.1/api \
--rapidpro.api.url=https://rapidpro.dhis2.org/api/v2 \
--sync.rapidpro.contacts=true
Poll reports
export DHIS2_API_PAT=d2pat_apheulkR1x7ac8vr9vcxrFkXlgeRiFc94200032556
export RAPIDPRO_API_TOKEN=3048a3b9a04c1948aa5a7fd06e7592ba5a17d3d0

./dhis2rapidpro.jar --dhis2.api.url=https://play.dhis2.org/2.38.1/api \
--rapidpro.api.url=https://rapidpro.dhis2.org/api/v2 \
--rapidpro.flow.uuids=21a055c2-f0a7-4ec3-9e5e-bc05504b8967,1baa7dd3-9ccf-4ee8-b7a4-8779ba22b933,a6fd08af-4757-46a0-b4a7-c9a210b425db

Windows

To execute DHIS-to-RapidPro from Windows, enter the following terminal command:

java -jar dhis2-to-rapidpro.jar

The above command will give an error since no parameters are provided. The next commands are common DHIS-to-RapidPro Windows usage examples:

Accept webhook reports
set DHIS2_API_PAT=d2pat_apheulkR1x7ac8vr9vcxrFkXlgeRiFc94200032556
set RAPIDPRO_API_TOKEN=3048a3b9a04c1948aa5a7fd06e7592ba5a17d3d0

java -jar dhis2rapidpro.jar \
--dhis2.api.url=https://play.dhis2.org/2.38.1/api \
--rapidpro.api.url=https://rapidpro.dhis2.org/api/v2 \
--rapidpro.webhook.enabled=true
Auto-reminders
set DHIS2_API_PAT=d2pat_apheulkR1x7ac8vr9vcxrFkXlgeRiFc94200032556
set RAPIDPRO_API_TOKEN=3048a3b9a04c1948aa5a7fd06e7592ba5a17d3d0

java -jar dhis2rapidpro.jar \ 
--dhis2.api.url=https://play.dhis2.org/2.38.1/api \
--rapidpro.api.url=https://rapidpro.dhis2.org/api/v2 \
--reminder.data.set.codes=DS_359414,DS_543073,HIV_CARE
Activate contact synchronisation
set DHIS2_API_PAT=d2pat_apheulkR1x7ac8vr9vcxrFkXlgeRiFc94200032556
set RAPIDPRO_API_TOKEN=3048a3b9a04c1948aa5a7fd06e7592ba5a17d3d0

java -jar dhis2rapidpro.jar \
--dhis2.api.url=https://play.dhis2.org/2.38.1/api \
--rapidpro.api.url=https://rapidpro.dhis2.org/api/v2 \
--sync.rapidpro.contacts=true
Poll reports
set DHIS2_API_PAT=d2pat_apheulkR1x7ac8vr9vcxrFkXlgeRiFc94200032556
set RAPIDPRO_API_TOKEN=3048a3b9a04c1948aa5a7fd06e7592ba5a17d3d0

java -jar dhis2rapidpro.jar \
--dhis2.api.url=https://play.dhis2.org/2.38.1/api \
--rapidpro.api.url=https://rapidpro.dhis2.org/api/v2 \
--rapidpro.flow.uuids=21a055c2-f0a7-4ec3-9e5e-bc05504b8967,1baa7dd3-9ccf-4ee8-b7a4-8779ba22b933,a6fd08af-4757-46a0-b4a7-c9a210b425db

WAR

To run DHIS-to-RapidPro as a web application inside a web container like Tomcat, download the latest WAR distribution and drop it in the web container's applications directory. Configuration properties for WAR deployment can be expressed as:

  • OS environments variables
  • Key/value pairs in a file named application.properties. Create a directory called config within the web container's working directory and place application.properties in this new directory.
  • YAML in a file named application.yml. Create a directory called config within the web container's working directory and place application.yml in this new directory.

Features

Contact Synchronisation

SECURITY: contact synchronisation copies personal data from DHIS to RapidPro. Ensure that the data provider agrees to sharing DHIS2 user details with the data receiver before activating synchronisation.

During contact synchronisation, DHIS-to-RapidPro fetches the users from your DHIS2 server to either:

  • create RapidPro contacts containing the DHIS2 user's ID, organisation unit ID, name, mobile phone number, Telegram identifier, WhatsApp phone number, Facebook messenger username, and Twitter handle, or
  • update existing RapidPro contacts to match any changes in the corresponding DHIS2 users.

Prior to synchronisation, DHIS-to-RapidPro automatically creates in RapidPro:

  • the contact group DHIS2, and
  • two contact fields named dhis2_organisation_unit_id and dhis2_user_id

DHIS-to-RapidPro will re-create this group and these fields should they be deleted. During synchronisation, each contact is assigned to the DHIS2 group and has its fields populated accordingly. Application errors during the syncing of a contact will lead to warnings in the log but the error will not abort the synchronisation process. In other words, synchronisation may be partially successful.

Contact synchronisation is disabled by default. Setting sync.rapidpro.contacts to true enables synchronisation. The interval rate at which contacts are synchronised is expressed as a cron expression with the config key sync.schedule.expression. Alternatively, from your web browser, enter the DHIS-to-RapidPro's URL (e.g., https://localhost:8443/dhis2rapidpro) together with the path /services/tasks/sync in the address bar to kick off syncing.

Aggregate Report Transfer

Follow the subsequent DHIS2 and RapidPro setup instructions to be able to transfer aggregate reports from RapidPro to DHIS2.

DHIS2 Instructions

  1. Configure codes for the data sets that the reports transmitted from RapidPro to DHIS-to-RapidPro will target. To configure the data set code:

    1. Go to the maintenance app
    2. Open the data sets page
    3. Search for the data set
    4. Enter a suitable code in the Code field as shown next: Data set form

      IMPORTANT: you need to enter a code that starts with a letter, a hyphen, an underscore, or a whitespace to achieve successful interoperability between DHIS2 and RapidPro. Special characters that are not permitted in a RapidPro result name should NOT be part of the code. Hyphens, underscores, and whitespaces are typically permitted.

  2. Configure a code in each data element that will capture an aggregate value from RapidPro. To configure the data element code:

    1. Go to the maintenance app
    2. Open the data elements page
    3. Search for the data element
    4. Enter a suitable code in the Code field as shown next: Data element form

      IMPORTANT: you need to enter a code that starts with a letter, a hyphen, an underscore, or a whitespace to achieve successful interoperability between DHIS2 and RapidPro. Special characters that are not permitted in a RapidPro result name should NOT be part of the code. Hyphens, underscores, and whitespaces are typically permitted.

  3. Configure a code in each category option combination that will be used to disaggregate captured values. To configure the category option combination code:

    1. Go to the maintenance app
    2. Open the category option combination page
    3. Search for the category option combination
    4. Enter a suitable code in the Code field as shown next: Category option combination form

      IMPORTANT: you need to enter a code that starts with a letter, a hyphen, an underscore, or a whitespace to achieve successful interoperability between DHIS2 and RapidPro. Special characters that are not permitted in a RapidPro result name should NOT be part of the code. Hyphens, underscores, and whitespaces are typically permitted.

RapidPro Instructions

DHIS-to-RapidPro can ingest aggregate reports from RapidPro as:

  • Completed flow executions that are retrieved while polling the RapidPro API, or
  • RapidPro webhook messages

Each ingestion mode comes with its own set of trade-offs. For instance, webhook messaging scales better than polling but reports can be lost due to consecutive network failures. In contrast, having DHIS-to-RapidPro routinely scan flow executions leads to more load on the RapidPro server, however, polling is more reliable than webhook messaging since network failures during polling will only interrupt DHIS-to-RapidPro from ingesting the report rather than losing the report itself. Generally speaking, report polling is recommended over webhook messaging but your requirements will dictate which ingestion mode to employ. The next sections describe the configuration steps for the respective ingestion modes.

Polling
  1. Open a RapidPro flow definition that processes the contact's report or create a new flow definition.

  2. Identify the root of each happy flow path, that is, the root of each successful execution path. You should apply the proceeding steps to these root paths.

  3. Save a result containing the DHIS2 code of the data set representing the report:

    Type the result name data_set_code and give it as a value the code of the data set as retrieved from DHIS2's maintenance app.

  4. Save each incoming report value to a result as per the example shown next:

    The result name must match the code of the corresponding data element in DHIS2. Upper case letters in the data element code can be entered as lower case letters in the result name field while whitespaces and hyphens can be entered as underscores If a category option combination is required, suffix the result name with two underscores and append the category option combination code to the suffix:

  5. Optionally, save a result which contains the report period offset:

    Type the result name report_period_offset and give it as a value the relative period to add or subtract from the current reporting period sent to DHIS2. If omitted, the report period offset defaults to -1.

  6. Another optional result is org_unit_id. This result overrides the value set in the contact's DHIS2 Organisation Unit ID field:

  7. If contact synchronisation is disabled (see sync.rapidpro.contacts in Configuration), then create a custom contact field named DHIS2 Organisation Unit ID:

    Unless the org_unit_id result is set, you must populate this field, either manually or automatically, for each contact belonging to a DHIS2 organisation unit. The field should hold the contact's DHIS2 organisation unit identifier. By default, DHIS-to-RapidPro expects the organisation unit identifier to be the ID (see org.unit.id.scheme in Configuration).

  8. Copy the UUID of the flow definition from your web browser's address bar: browser address bar

  9. Paste the copied flow definition UUID into DHIS-to-RapidPro's rapidpro.flow.uuids config property. For example:

    java -jar dhis2rapidpro.jar \ 
    --dhis2.api.url=https://play.dhis2.org/2.38.1/api \ 
    --rapidpro.api.url=https://rapidpro.dhis2.org/api/v2 \
    --rapidpro.flow.uuids=21a055c2-f0a7-4ec3-9e5e-bc05504b8967

    You can poll multiple flows by having the flow UUIDs comma separated:

     java -jar dhis2rapidpro.jar \ 
    --dhis2.api.url=https://play.dhis2.org/2.38.1/api \ 
    --rapidpro.api.url=https://rapidpro.dhis2.org/api/v2 \
    --rapidpro.flow.uuids=21a055c2-f0a7-4ec3-9e5e-bc05504b8967,1baa7dd3-9ccf-4ee8-b7a4-8779ba22b933,a6fd08af-4757-46a0-b4a7-c9a210b425db

    NOTE: scan.reports.schedule.expression config property determines how often flow executions are polled. Consult the configuration section for further information.

While DHIS-to-RapidPro is running, to manually kick off the scanning of flow runs:

  1. Open your web browser
  2. Type the DHIS-to-RapidPro URL together with the path /services/tasks/scan inside the browser address bar
  3. Press enter
Webhook
  1. Open a RapidPro flow definition that processes the contact's report or create a new flow definition.

  2. Identify the root of each happy flow path, that is, the root of each successful execution path. You should apply the proceeding steps to these root paths.

  3. Save each incoming aggregate value in the RapidPro flow to a result like what is shown next:

    The result name must match the code of the corresponding data element in DHIS2. Upper case letters in the data element code can be entered as lower case letters in the result name field while whitespaces and hyphens can be entered as underscores. If a category option combination is required, suffix the result name with two underscores and append the category option combination code to the suffix:

  4. Create a webhook call node in the RapidPro flow to dispatch the results to DHIS-to-RapidPro:

    The webhook call node must be configured as follows:

    • Select the HTTP method to be POST:

    • Set the URL field to the HTTP(S) address that DHIS-to-RapidPro is listening on. The default HTTPS port number is 8443 (see server.port in Configuration): the path in the URL field is required to end with /dhis2rapidpro/services/webhook: URL webhook

    • Append to the URL the dataSetCode query parameter which identifies by code the data set that the contact is reporting. You need to look up the data set from the DHIS2 maintenance app and hard-code its code as shown below: Data set ID code parameter

    • You can optionally append the reportPeriodOffset query parameter which is the relative period to add or subtract from the current reporting period sent to DHIS2. If omitted, the reportPeriodOffset parameter defaults to -1. Report period offset query parameter

    • Another optional query parameter you can append is orgUnitId. This parameter overrides the value set in the contact's DHIS2 Organisation Unit ID field.

    • If you have set the config property webhook.security.auth in DHIS-to-RapidPro to token in order to protect the webhook endpoint from unauthorised access, switch to the HTTP Headers tab and enter a new header named Authorization having as value the authentication scheme Token alongside the token generated at startup from DHIS-to-RapidPro:

      SECURITY: inside the Authorization header value text field, you should reference a global holding the secret token instead of directly entering the token so that the token is not accidentally compromised when exporting the flow definition.

  5. If contact synchronisation is disabled (see sync.rapidpro.contacts in Configuration), then create a custom contact field named DHIS2 Organisation Unit ID:

    Unless the orgUnitId webhook query parameter is set, you must populate this field, either manually or automatically, for each contact belonging to a DHIS2 organisation unit. The field should hold the contact's DHIS2 organisation unit identifier. By default, DHIS-to-RapidPro expects the organisation unit identifier to be the ID (see org.unit.id.scheme in Configuration).

  6. Enable the rapidpro.webhook.enabled config property when starting DHIS-to-RapidPro. For example:

    java -jar dhis2rapidpro.jar \ 
    --dhis2.api.url=https://play.dhis2.org/2.38.1/api \ 
    --rapidpro.api.url=https://rapidpro.dhis2.org/api/v2 \
    --rapidpro.webhook.enabled=true

Auto-Reminders

Reminders for overdue reports are sent for each DHIS2 data set specified in the config property reminder.data.set.codes. In this property, you enter the data set codes separated by comma. Reminders are sent to contacts that are within the DHIS2 group. This group is automatically created and contacts assigned to it as part of the contact synchronisation process but you can also manually create the group in RapidPro as shown below:

CAUTION: do not forget to assign auto-reminder contacts to the DHIS2 group

The interval rate at which contacts are reminded is expressed as a cron expression with the config key reminder.schedule.expression. Alternatively, open the web browser and enter DHIS-to-RapidPro's URL followed by the path /services/tasks/reminders to instantly broadcast the reminders for overdue reports.

Configuration

By order of precedence, a config property can be specified:

  1. as a command-line argument (e.g., --dhis2.api.url=https://play.dhis2.org/2.38.1/api)
  2. as an OS environment variable (e.g., export DHIS2_API_URL=https://play.dhis2.org/2.38.1/api)
  3. in a key/value property file called application.properties or a YAML file named application.yml

SECURITY: the application rejects secrets like passwords set from command-line arguments.

Config name Description Default value Example value
dhis2.api.url DHIS2 server Web API URL. https://play.dhis2.org/2.38.1/api
dhis2.api.pat Personal access token to authenticate with on DHIS2. This property is mutually exclusive to dhis2.api.username and dhis2.api.password. d2pat_apheulkR1x7ac8vr9vcxrFkXlgeRiFc94200032556
dhis2.api.username Username of the DHIS2 user to operate as. admin
dhis2.api.password Password of the DHIS2 user to operate as. district
rapidpro.api.url RapidPro server Web API URL. https://rapidpro.dhis2.org/api/v2
rapidpro.api.token API token to authenticate with on RapidPro. 3048a3b9a04c1948aa5a7fd06e7592ba5a17d3d0
server.port The TCP port number the application will bind to for accepting HTTP requests. 8443 443
sync.schedule.expression Cron expression for synchronising RapidPro contacts with DHIS2 users. By default, synchronisation occurs every half hour. 0 0/30 * * * ? 0 0 0 * * ?
reminder.schedule.expression Cron expression for broadcasting reminders of overdue reports to RapidPro contacts. By default, overdue report reminders are sent at 9 a.m. every day. 0 0 9 ? * * 0 0 0 * * ?
scan.reports.schedule.expression Cron expression specifying how often RapidPro is queried for flow executions. By default, RapidPro is queried every thirty minutes. 0 0/30 * * * ? 0 0 0 * * ?
report.delivery.schedule.expression Cron expression specifying when queued reports are delivered to DHIS2. 0 0 0 * * ?
sync.rapidpro.contacts Whether to routinely create and update RapidPro contacts from DHIS2 users. false true
rapidpro.webhook.enabled Whether to accept webhook requests from RapidPro. false true
reminder.data.set.codes Comma-delimited list of DHIS2 data set codes for which overdue report reminders are sent. DS_359414,HIV_CARE
rapidpro.flow.uuids Comma-delimited list of RapidPro flow definition UUIDs to scan for completed flow executions. 2db0f7fa-be5d-486f-bda5-096d0f68db3e,51d660b5-5137-4d92-b874-0a6b7cf5c02c,ceef94f4-e0ae-4e10-9dd5-9afe51c110c5
org.unit.id.scheme By which field organisation units are identified. ID CODE
webhook.security.auth Authentication scheme protecting the webhook HTTP(S) endpoint. Supported values are none and token. none token
server.ssl.enabled Whether to enable TLS support. true false
test.connection.startup Test connectivity with DHIS2 and RapidPro during start-up. In case of connection failure, the application wil print an error and terminate. true false
spring.security.user.name Login username for non-webhook services like the Hawtio and H2 web consoles. dhis2rapidpro admin
spring.security.user.password Login password for non-webhook services like the Hawtio and H2 web consoles. dhis2rapidpro secret
spring.h2.console.enabled Whether to enable the H2 web console. true false
spring.h2.console.settings.web-allow-others Whether to enable remote access to the H2 web console. false true
spring.jmx.enabled Whether to expose the JMX metrics. true false
management.endpoints.web.exposure.include Management endpoint IDs that should be included or '*' for all. *

Database

DHIS-to-RapidPro requires a relational database to store:

  • delivered as well as undelivered reports

  • the context between successive flow polls

  • the security token generated at start-up for webhook authentication

    SECURITY: the token is a 32-byte key that is hashed using SHA-256 before it is written to the database. A data breach could result the hashed token being leaked though it would be very hard to recover the clear token from the hash. Nonetheless, the hashed token is still vulnerable to bruteforce attacks, therefore, it is imperative that the table TOKEN is truncated after a suspected data breach in order to generate a new security token.

H2 is the embedded database that DHIS-to-RapidPro offers out-of-the-box. H2 is production-ready but may be lacking features that are available from your favourite database. You might even want to persist DHIS-to-RapidPro's state in your organisation's central database.

The following configuration properties should be considered when persisting to a different database:

Config name Description Default value Example value
spring.sql.init.platform Database platform to use in the default schema or the DML statements. h2 psotgresql
spring.datasource.url JDBC URL for persisting the application state. jdbc:h2:./dhis2rapidpro;AUTO_SERVER=TRUE jdbc:postgresql://localhost:5432/dhis2rapidpro
spring.datasource.username Username to access the JDBC data source. dhis2rapidpro postgres
spring.datasource.password Password to access the JDBC data source. dhis2rapidpro postgres
spring.datasource.driver-class-name Class name of the JDBC driver used to connect to the database. org.h2.Driver org.postgresql.Driver
spring.sql.init.schema-locations Locations of the schema (DDL) scripts to apply to the database. classpath:/schema.sql file:db/schema.postgres.sql
sql.data-location Location of the properties file containing the DML statements to run on the database. classpath:/sql.properties file:db/sql.properties
spring.sql.init.mode Mode to apply when determining whether database initialisation should be performed. always never

SECURITY: create a dedicated database user for DHIS-to-RapidPro when using another database. The database user should only have read and write privileges to the database objects created by DHIS-to-RapdPro.

Switching databases requires that you add the database vendor's JDBC driver to the Java classpath. When running the DHIS-to-RapidPro executable, third-party libraries should reside in the lib directory relative to DHIS-to-RapidPro's working directory.

NOTE: DHIS-to-RapidPro's working directory is relative to the current directory when DHIS-to-RapidPro is executed as a JAR (e.g., java -jar dhis2-to-rapidpro.jar). On the other hand, the working directory is relative to the JAR binary when DHIS-to-RapidPro is executed as a shell command (e.g., ./dhis2-to-rapidpro.jar).

Apart from H2, PostgreSQL is supported as well. To configure the application's connection to PostgreSQL:

  1. Set the spring.sql.init.platform configuration property to postgresql
  2. Set the spring.datasource.url configuration property to the required JDBC address
  3. Set the spring.datasource.username and spring.datasource.password configuration properties to the database username and password, respectively.
  4. Set the spring.datasource.driver-class-name configuration property to org.postgresql.Driver
  5. Download the PostgreSQL JDBC driver and place it in the lib directory as explained earlier.

For databases other than H2 and PostgreSQL, you might need to tweak the application's DDL and DML statements to be compatible with your database. Modified DDL statements should reside in a file that spring.sql.init.schema-locations is referencing. Modified DML statements should reside in a file that sql.data-location is referencing. The bundled PostgreSQL schema and queries are a useful point of reference when writing these SQL statements.

Management & Monitoring

DHIS-to-RapidPro exposes its metrics through JMX. A JMX client like VisualVM can be used to observe these metrics, however, DHIS-to-RapidPro comes bundled with Hawtio so that the system operator can easily monitor and manage the application's runtime operations without prior setup.

From the Hawtio web console, apart from browsing application logs, the system operator can manage queues and endpoints, observe the application health status and queued RapidPro webhook messages, collect CPU and memory diagnostics, as well as view application settings:

Hawtio Management Console

You can log into the Hawtio console locally from https://localhost:8443/dhis2rapidpro/management/hawtio using the username and password dhis2rapidpro. Set the parameter management.endpoints.web.exposure.include to an empty value (i.e., --management.endpoints.web.exposure.include=) to deny HTTP access to the Hawtio web console.

SECURITY: immediately change the login credentials during setup (see spring.security.user.name and spring.security.user.password in Configuration).

Stopping Routes

Individual integration points, or routes, can be shut down from Hawtio while the application is running. This is especially useful for maintenance reasons. For example, you may want to suspend the processing of reports while DHIS2 is down to undergo scheduled maintenance. To stop a route, from the Hawtio console:

  1. Click the Camel tab on the left-hand side menu
  2. Expand Camel Contexts from the navigation tree
  3. Expand camel-1
  4. Expand routes
  5. Select the route you want to stop
  6. Move the cursor over to the Started button on the far right-hand side corner of the page and click on it to reveal the drop-down menu
  7. Click on Stop

Stop route

You should see a console notification saying Route stopped successfully and the route marked as Stopped. To restart the route, click on the Stopped button and select Start.

Recovering Reports

A report that fails to be delivered to DHIS2, perhaps because of an invalid webhook payload or an HTTP timeout error, has its associated RapidPro webhook JSON payload pushed to a relational dead letter channel for manual inspection. The dead letter channel table schema is as follows:

Column name Column type Description Column value example
ID INTEGER An auto-increment number identifying the row uniquely. 6
PAYLOAD VARCHAR RapidPro webhook message or flow run JSON document. {"contact":{"name":"John Doe","urn":"tel:+12065551213","uuid":"fb3787ab-2eda-48a0-a2bc-e2ddadec1286"},"flow":{"name":"APT","uuid":"cb0360e3-d82a-4521-aad3-15afd704ec26"},"results":{"msg":{"value":"APT.2.4.6"},"gen_ext_fund":{"value":"2"},"mal_pop_total":{"value":"10"},"mal_llin_distr_pw":{"value":"3"},"gen_domestic_fund":{"value":"5"}}}
DATA_SET_CODE VARCHAR Code of the DHIS2 data set that the report belongs to. HIV_CARE
REPORT_PERIOD_OFFSET INTEGER Relative period to add or subtract from the current reporting period. -1
ORGANISATION_UNIT_ID VARCHAR Identifier of the DHIS2 organisation unit that the contact belongs to. Vth0fbpFcsO
ERROR_MESSAGE VARCHAR Message describing the root cause of the error. Response{protocol=http/1.1, code=500, message=, url=https://play.dhis2.org/2.38.1/api//dataValueSets?dataElementIdScheme=CODE&orgUnitIdScheme=bar}
STATUS ENUM Specifies the row's state which determines how the application processes the row. The user sets the status to RETRY for payloads that need to be retried. DHIS-to-RapidPro sets the status to ERROR for payloads that could not be processed successfully. Alternatively, payloads that are processed are marked as PROCESSED. ERROR
CREATED_AT TIMESTAMP WITH TIME ZONE Denotes the time the row was created. 2022-07-20 11:09:57.992 +0200
LAST_PROCESSED_AT TIMESTAMP WITH TIME ZONE Denotes the last time the row was processed. 2022-07-20 11:09:57.992 +0200

You can re-process a failed report by setting its corresponding row status column to RETRY using an ANSI SQL UPDATE command issued from an SQL client connected to the data store. For instance:

UPDATE DEAD_LETTER_CHANNEL SET status = 'RETRY' WHERE status = 'ERROR' 

H2 is the default relational data store that manages the dead letter channel. H2 has an in-built web console which allows you to issue SQL commands in order to view, edit, and retry failed reports:

H2 Web Console

The H2 console is pre-configured to be available locally at https://localhost:8443/dhis2rapidpro/management/h2-console. The console's relative URL path can be changed with the config property spring.h2.console.path. You will be greeted by the database's login page after logging into the monitoring & management system using the default login username and password dhis2rapidpro. Both the default database login username and password are dhis2rapidpro.

SECURITY: immediately change the management and database credentials during setup (see spring.security.user.name and spring.security.user.password together with spring.datasource.username and spring.datasource.password in Configuration).

For security reasons, the console only permits local access but this behaviour can be overridden by setting spring.h2.console.settings.web-allow-others to true. To completely disable access to the web console, set the parameter spring.h2.console.enabled to false though you still can connect to the data store with an SQL client.

The H2 DBMS is embedded with DHIS-to-RapidPro but the DBMS can be easily substituted with a more scalable JDBC-compliant DBMS such as PostgreSQL. You would need to change spring.datasource.url to a JDBC URL that references the new data store. Note: for a non-H2 data store, the data store vendor's JDBC driver needs to be added to the DHIS-to-RapidPro's Java classpath.

Success Log

Apart from the DEAD_LETTER_CHANNEL table, DHIS-to-RapidPro saves reports that were successfully delivered to DHIS2 in another table named SUCCESS_LOG. This table allows you to audit the transmitted reports. Its schema is as follows:

Column name Column type Description Column value example
ID INTEGER An auto-increment number identifying the row uniquely. 6
DHIS_REQUEST VARCHAR DHIS2 request sent to create the data value set. {{"completedDate":"2022-11-17","orgUnit":"HBqizVkKthQ","dataSet":"MAL_YEARLY","period":"2021","dataValues":[{"dataElement":"GEN_EXT_FUND","value":"2","comment":"RapidPro contact details: \"{\\n \\\"name\\\": \\\"John Doe\\\",\\n \\\"urn\\\": \\\"tel:+12065551212\\\",\\n \\\"uuid\\\": \\\"0008a629-c330-4664-ae28-689f051d79bc\\\"\\n}\""},{"dataElement":"MAL_POP_TOTAL","value":"10","comment":"RapidPro contact details: \"{\\n \\\"name\\\": \\\"John Doe\\\",\\n \\\"urn\\\": \\\"tel:+12065551212\\\",\\n \\\"uuid\\\": \\\"0008a629-c330-4664-ae28-689f051d79bc\\\"\\n}\"","categoryOptionCombo":"MAL-0514Y"},{"dataElement":"MAL_LLIN_DISTR_PW","value":"3","comment":"RapidPro contact details: \"{\\n \\\"name\\\": \\\"John Doe\\\",\\n \\\"urn\\\": \\\"tel:+12065551212\\\",\\n \\\"uuid\\\": \\\"0008a629-c330-4664-ae28-689f051d79bc\\\"\\n}\""},{"dataElement":"GEN_DOMESTIC_FUND","value":"5","comment":"RapidPro contact details: \"{\\n \\\"name\\\": \\\"John Doe\\\",\\n \\\"urn\\\": \\\"tel:+12065551212\\\",\\n \\\"uuid\\\": \\\"0008a629-c330-4664-ae28-689f051d79bc\\\"\\n}\""}]}
DHIS_RESPONSE VARCHAR DHIS2 reply acknowledging the created the data value set. {"responseType":"ImportSummary","status":"SUCCESS","importOptions":{"idSchemes":{},"dryRun":false,"async":false,"importStrategy":"CREATE_AND_UPDATE","mergeMode":"REPLACE","reportMode":"FULL","skipExistingCheck":false,"sharing":false,"skipNotifications":false,"skipAudit":false,"datasetAllowsPeriods":false,"strictPeriods":false,"strictDataElements":false,"strictCategoryOptionCombos":false,"strictAttributeOptionCombos":false,"strictOrganisationUnits":false,"requireCategoryOptionCombo":false,"requireAttributeOptionCombo":false,"skipPatternValidation":false,"ignoreEmptyCollection":false,"force":false,"firstRowIsHeader":true,"skipLastUpdated":false,"mergeDataValues":false,"skipCache":false},"description":"Import process completed successfully","importCount":{"imported":4,"updated":0,"ignored":0,"deleted":0},"conflicts":[],"dataSetComplete":"false"}
RAPIDPRO_PAYLOAD VARCHAR RapidPro webhook message or flow run JSON document. {"contact":{"name":"John Doe","urn":"tel:+12065551212","uuid": "0008a629-c330-4664-ae28-689f051d79bc" },"flow":{"name": "APT", "uuid": "cb0360e3-d82a-4521-aad3-15afd704ec26" }, "results": { "msg": { "value": "APT.2.4.6" },"gen_ext_fund":{"value":"2"},"mal_pop_total__mal-0514y":{"value":"10"},"mal_llin_distr_pw":{"value":"3"},"gen_domestic_fund":{"value":"5"}}}
DATA_SET_CODE VARCHAR Code of the DHIS2 data set that the report belongs to. HIV_CARE
REPORT_PERIOD_OFFSET INTEGER Relative period to add or subtract from the current reporting period. -1
ORGANISATION_UNIT_ID VARCHAR Identifier of the DHIS2 organisation unit that the contact belongs to. Vth0fbpFcsO
CREATED_AT TIMESTAMP WITH TIME ZONE Denotes the time the row was created. 2022-07-20 11:09:57.992 +0200

In addition to auditing, you can modify and re-transmit reports to DHIS2 thanks to this table. The sequence of steps for re-transmitting reports is:

  1. Copying the RAPIDPRO_PAYLOAD column values from the relevant rows in SUCCESS_LOG (i.e., SELECT rapidpro_payload FROM SUCCESS_LOG WHERE ...)
  2. Updating the retrieved RAPIDPRO_PAYLOAD column values accordingly, and
  3. Inserting rows into DEAD_LETTER_CHANNEL where PAYLOAD is equal to the updated RAPIDPRO_PAYLOAD column values and STATUS is equal to RETRY

Extending DHIS-to-RapidPro

Besides being highly configurable, just about any piece of DHIS-to-RapidPro's functionality can be extended during configuration to suit your particular needs. A prerequisite to extending the behaviour is having knowledge of Apache Camel: the routing engine powering DHIS-to-RapidPro. In particular, you should be knowledgeable in Apache Camel's YAML or XML DSL in order to be able to define integration flows that override or complement the existing flows.

Integration flows in DHIS-to-RapidPro, known as routes in Apache Camel, are named according to their purpose. You can override any route if you know its name. The following is a list of the important routes that you may want to override:

Route name Description
RapidPro Webhook Accepts and queues RapidPro webhook messages
Consume Report De-queues the report for delivery to DHIS2
Transform Report Maps and enriches the report as received by RapidPro prior to transmitting it to DHIS2
Transmit Report Transmits the report to DHIS2
Retry Reports Re-queues reports marked for replay
Scan RapidPro Flows Polls RapidPro for flow runs and queues them
Broadcast Reminders Queries DHIS2 for overdue reports and sends any reminders to RapidPro
Set up RapidPro Configures RapidPro for integration with DHIS2
Create RapidPro Fields Creates contact fields on RapidPro
Create RapidPro Group Creates contact group on RapidPro
Sync RapidPro Contacts Synchronises RapidPro contacts with DHIS2 users

You should place the file or files containing the custom routes in a directory named routes within DHIS-to-RapidPro's current directory. The custom route will override the inbuilt route if the routes match by name. DHIS-to-RapidPro can reload the routes while its running therefore you have the option to extend the application at runtime.

IMPORTANT: Hot reloading is only recommended for non-production environments.

What follows is an example of a custom YAML route that overrides the inbuilt Transmit Report route:

- route:
    id: "Transmit Report"
    from:
      uri: "direct:transmitReport"
      steps:
        - setProperty:
            name: msisdn
            jsonpath:
              headerName: originalPayload
              expression: "$.contact.urn"
        - setProperty:
            name: raw_msg
            jsonpath:
              headerName: originalPayload
              expression: "$.results.msg.value"
        - setProperty:
            name: report_type
            jsonpath:
              headerName: originalPayload
              expression: "$.flow.name"
        - toD:
            uri: "https://legacy.example/dhis2?authenticationPreemptive=true&authMethod=Basic&authUsername=alice&authPassword=secret&httpMethod=POST&msisdn=${exchangeProperty.msisdn}&raw_msg=${exchangeProperty.raw_msg}&facility=${header.orgUnitId}&report_type=${exchangeProperty.report_type}&aParam=${header.aParam}"

The above custom route overrides the original route such that aggregate reports are delivered to a non-DHIS2 system. It extracts a number of values from the report payload with the setProperty key and adds them to destination URL as HTTP query parameters. Consult the Set Property and JSONPath Apache Camel documentation for further information about setting properties and extracting values from within a route.

Besides adding query parameters, the route also configures the HTTP client for basic authentication using the reserved query parameters authenticationPreemptive, authMethod, authUsername, and authPassword. Consult the HTTP component Apache Camel documentation for further information about configuring the HTTP client.

Troubleshooting Guide

Unexpected behaviour in DHIS-to-RapidPro typically manifests itself as:

  • errors in the applications logs, or
  • incorrect data (e.g., wrong organisation unit ID in the data value sets).

The first step to determine the root cause of unexpected behaviour is to search for recent errors in the dead letter channel:

-- SQL is compatible with H2
SELECT * FROM DEAD_LETTER_CHANNEL 
WHERE status = 'ERROR' AND created_at > DATEADD('DAY', -1, CURRENT_TIMESTAMP())	

The above SQL returns the reports that failed to be saved in DHIS2 within the last 24 hours. Zoom in the ERROR_MESSAGE column to read the technical error message that was given by the application. Should the error message describe an ephemeral failure like a network timeout, the rule of thumb is for the system operator to update the STATUS column to RETRY in order for DHIS-to-RapidPro to re-processes the failed reports:

-- SQL is compatible with H2
UPDATE DEAD_LETTER_CHANNEL 
SET status = 'RETRY' 
WHERE status = 'ERROR' AND created_at > DATEADD('DAY', -1, CURRENT_TIMESTAMP())	

After issuing the above SQL, DHIS-to-RapidPro will poll for the RETRY rows from the data store and re-process the reports. Processed rows, whether successful or not, are updated as PROCESSED and have their LAST_PROCESSED_AT column updated to the current time. If a retry fails, DHIS-to-RapidPro will go on to insert a corresponding new ERROR row in the DEAD_LETTER_CHANNEL table.

Non-transient failures such as validation errors require human intervention which might mean that you have to update the payload column value so that it conforms with the expected structure or data type:

UPDATE DEAD_LETTER_CHANNEL 
SET status = 'RETRY', payload = '{"contact":{"name":"John Doe","urn":"tel:+12065551213","uuid":"fb3787ab-2eda-48a0-a2bc-e2ddadec1286"},"flow":{"name":"APT","uuid":"cb0360e3-d82a-4521-aad3-15afd704ec26"},"results":{"msg":{"value":"APT.2.4.6"},"gen_ext_fund":{"value":"2"},"mal_pop_total":{"value":"10"},"mal_llin_distr_pw":{"value":"3"},"gen_domestic_fund":{"value":"5"}}}' 
WHERE id = '1023'

Deeper technical problems might not manifest themselves up as failed reports but as exceptions in the application logs. The logs can be analysed from the Hawtio web console or directly from the log file dhis2rapidpro.log, situated in DHIS-to-RapidPro's working directory. Keep an eye out for exceptions while combing through the logs. Any exception messages, including their stack traces, should be collected from the logs and further analysed. You may want to reach out to the DHIS2 Community of Practice for troubleshooting support. If all else fails, you can try increasing the log verbosity to zone in on the root cause. Setting the config property logging.level.org.hisp.dhis.integration.rapidpro to DEBUG will lead to the application printing more detail in the logs. As a last resort, though not recommended, you can have the application print even more detail by setting logging.level.root to DEBUG.

CAUTION: be careful about increasing log verbosity since it may quickly eat up the server's disk space if the application is logging to a file, the default behaviour.

Acknowledgments

This project is funded by UNICEF and developed by HISP Centre in collaboration with HISP Uganda and ITINORDIC.

About

Java solution integrating DHIS2 and WhatsApp using the dhis2-rapidpro connector

Resources

License

Stars

Watchers

Forks

Releases

No releases published

Packages

No packages published

Languages

  • Java 95.7%
  • Python 2.6%
  • Dockerfile 1.1%
  • Other 0.6%