API to interact with CHT Applications

RESTful Application Programming Interfaces for integrating with CHT applications

This page covers the endpoints to use when integrating with the CHT server. If there isn’t an endpoint that provides the function or data you need, direct access to the database is possible via the CouchDB API. Access to the PostgreSQL database may also prove useful for data analysis. If additional endpoints would be helpful, please make suggestions via a GitHub issue.

Settings

Get and update the app settings.

GET /api/v1/settings

Returns the settings in JSON format.

PUT /api/v1/settings

Query Parameters

VariableDescription
overwriteWhether to replace settings document with input document. If both replace and overwrite are set, then it overwrites only. Defaults to replace.
replaceWhether to replace existing settings for the given properties or to merge. Defaults to false (merging).

Returns a JSON object with two fields:

{
  "success": true,
  "upgraded": false
}
VariableTypeDescription
successBooleanAlways true
upgradedBooleanWhether the settings doc was updated or not

Export

Request different types of data in various formats.

Each of the export endpoints except contacts, feedback, and user-devices supports a parameter which returns date formatted in human readable form (ISO 8601). Setting this parameter to false or leaving it out will return dates formatted as an epoch timestamp.

To set this parameter for a GET request use:

http://admin:pass@localhost:5988/api/v2/export/messages?options[humanReadable]=true

To set this parameter for a POST request submit this as the request body:

{
  "options": {
    "humanReadable": true
  }
}

GET /api/v2/export/dhis

Exports target data formatted as a DHIS2 dataValueSet. The data can be filtered to a specific section of the contact hierarchy or for a given time interval.

ParameterDescription
dataSetA DHIS2 dataSet ID. Targets associated with this dataSet will have their data aggregated. (required)
date.fromFilter the target data to be aggregated to be within the month of this timestamp. (required)
orgUnitFilter the target data to only that associated with contacts with attribute { dhis: { orgUnit } }. (optional)
{
  "filters": {
    "dataSet": "VMuFODsyWaO",
    "date": {
      "from": 949392000000,
    },
    "orgUnit": "KbY9DJ8mBkx"
  }
}

GET /api/v2/export/reports

It uses the search shared library to ensure identical results in the export and the front-end. It also only supports exporting CSV so we can efficiently stream infinitely large exports.

Query parameters

These are identical to the JS objects passed to the shared library, as if you were using it directly in Javascript.

You may either pass JSON in the request body using POST:

POST /api/v2/export/reports
{
  "filters": {
    "forms": {
      "selected": [
        {
          "code": "immunization_visit"
        }
      ]
    }
  }
}

Or using form-style parameters as GET:

GET /api/v2/export/reports?filters[search]=&filters[forms][selected][0][code]=immunization_visit

NB: this API is bound directly to this library. For more information on what queries you can perform with the search library, see its documentation.

GET /api/v2/export/messages

Download messages.

Output

ColumnDescription
Record UUIDThe unique ID for the message in the database.
Patient IDThe generated short patient ID for use in SMS.
Reported DateThe date the message was received or generated.
FromThis phone number the message is or will be sent from.
Contact NameThe name of the user this message is assigned to.
Message TypeThe type of the message
Message StateThe state of the message at the time this export was generated
Received TimestampThe datetime the message was received. Only applies to incoming messages.
Other TimestampsThe datetime the message transitioned to each state.
Sent ByThe phone number the message was sent from. Only applies to incoming messages.
To PhoneThe phone number the message is or will be sent to. Only applies to outgoing messages.
Message BodyThe content of the message.

Examples

/api/v2/export/messages

GET /api/v2/export/feedback

Export a file containing the user feedback.

Query Parameters

VariableDescription
formatThe format of the returned file, either ‘csv’ or ‘xml’. Defaults to ‘csv’.
localeLocale for translatable data. Defaults to ’en’.
tzThe timezone to show date values in, as an offset in minutes from GMT, for example ‘-120’.
skip_header_row’true’ to omit the column headings. Defaults to ‘false’.

GET /api/v2/export/contacts

Returns a JSON array of contacts.

Output

ColumnDescription
idThe unique ID for the contact in the database.
revThe current CouchDb revision of contact in the database.
nameThe name of the user this message is assigned to.
patient_idThe generated short patient ID for use in SMS.
typeThe contact type. For configurable hierarchies, this will always be contact.
contact_typeThe configurable contact type. Will be empty if using the default hierarchy.
place_idThe generated short place ID for use in SMS.

Query parameters

These are identical to the JS objects passed to the shared library, as if you were using it directly in Javascript.

You may either pass JSON in the request body using POST:

POST /api/v2/export/contacts
{
  "filters": {
    "search": "jim"
  }
}

Or using form-style parameters as GET:

GET /api/v2/export/contacts?filters[search]=jim

GET /api/v2/export/user-devices

Added in 4.7.0

Returns a JSON array of CHT-related software versions for each user device. This information is derived from the latest telemetry entry for each user device. If a particular user has used multiple devices, an entry will be included for each device. You can reference the date value to determine which devices have been recently used. If multiple users used the same physical device (e.g. they were logged into the same phone at different times), an entry will be included for each user.

Output

ColumnDescription
userThe user’s name.
deviceIdThe unique key for the user’s device.
dateThe date the telemetry entry was taken in YYYY-MM-DD, see relevant docs.
browser.nameThe name of the browser used.
browser.versionThe version of the browser used.
apkThe Internal version code of the Android app.
androidThe version of Android OS.
chtThe version of CHT the user was on at time the telemetry entry was generated.
settingsThe revision of the App Settings document stored in CouchDB.

Forms

GET /api/v1/forms

Returns a list of currently installed forms (in all available formats) in JSON format.

Headers

KeyValueDescription
X-OpenRosa-Version1.0If this header is specified returns XML formatted forms list. See OpenRosa FormListAPI.

Examples

Get list of forms currently installed.

GET /api/v1/forms
HTTP/1.1 200
Content-Type: application/json; charset=utf-8

["anc_visit.xml","anc_registration.xml","off.xml", "off.json"]

Get OpenRosa XForms compatible forms installed in XML format.

GET /api/v1/forms
Host: medic.local
X-OpenRosa-Version: 1.0
HTTP/1.1 200 OK
Content-Type: text/xml; charset=utf-8
X-OpenRosa-Version: 1.0

<?xml version="1.0" encoding="UTF-8"?>
<xforms xmlns="http://openrosa.org/xforms/xformsList">
  <xform>
    <name>Visit</name>
    <formID>ANCVisit</formID>
    <hash>md5:1f0f096602ed794a264ab67224608cf4</hash>
    <downloadUrl>http://medic.local/api/v1/forms/anc_visit.xml</downloadUrl>
  </xform>
  <xform>
    <name>Registration with LMP</name>
    <formID>PregnancyRegistration</formID>
    <hash>md5:1f0f096602ed794a264ab67224608cf4</hash>
    <downloadUrl>http://medic.local/api/v1/forms/anc_registration.xml</downloadUrl>
  </xform>
  <xform>
    <name>Stop</name>
    <formID>Stop</formID>
    <hash>md5:1f0f096602ed794a264ab67224608cf4</hash>
    <downloadUrl>http://medic.local/api/v1/forms/off.xml</downloadUrl>
  </xform>
</xforms>

GET /api/v1/forms/{{id}}.{{format}}

Return form definition for a given form ID and format.

Parameters

VariableDescription
idForm identifier
formatFormat string or file extension. e.g. xml, json

Examples

Get latest version of the PregnancyRegistration form in xml (XForms) format.

GET /api/v1/forms/pregnancyregistration.xml

Get the latest version of the NPYY form in JSON format.

GET /api/v1/forms/NPYY.json

POST /api/v1/forms/validate

Added in 3.12.0

Validate the XForm passed. Require the can_configure permission.

Headers

KeyValueDescription
Content-Typeapplication/xmlThe form is sent in XML format
AuthorizationBasic KEYKEY is the “basic” token

Examples

POST /api/v1/forms/validate HTTP/1.1
Content-Type: application/xml
Authorization: Basic XXXXXXXXXX

<?xml version="1.0" encoding="UTF-8"?>
<xforms xmlns="http://openrosa.org/xforms/xformsList">
  <xform>
    <name>Visit</name>
    <formID>ANCVisit</formID>
    <hash>md5:1f0f096602ed794a264ab67224608cf4</hash>
    <downloadUrl>http://medic.local/api/v1/forms/anc_visit.xml</downloadUrl>
  </xform>
  <xform>
    <name>Registration with LMP</name>
    <formID>PregnancyRegistration</formID>
    <hash>md5:1f0f096602ed794a264ab67224608cf4</hash>
    <downloadUrl>http://medic.local/api/v1/forms/anc_registration.xml</downloadUrl>
  </xform>
  <xform>
    <name>Stop</name>
    <formID>Stop</formID>
    <hash>md5:1f0f096602ed794a264ab67224608cf4</hash>
    <downloadUrl>http://medic.local/api/v1/forms/off.xml</downloadUrl>
  </xform>
</xforms>

Example response when the form passed the validations:

HTTP/1.1 200 OK
Content-Type: application/json

{ok: true}

Example response when the form failed the validations:

HTTP/1.1 400 Bad Request
Content-Type: application/json

{error: "Error transforming xml. xsltproc return ..."}

Records

POST /api/v2/records

Create a new record based on a JSON form that has been configured.

Records can be created one of two ways, parsing the form data yourself and submitting a JSON object or by submitting the raw message string.

Headers

KeyValueDescription
Content-Typeapplication/x-www-form-urlencodedProcesses form parameters.
Content-Typeapplication/jsonProcesses form data in request body as JSON.

Only one variant of the Content-Type header may be provided; RFC 2616 does not allow multiple content types to appear in a single Content-Type header.

Form Parameters
VariableDescription
messageMessage string in a supported format like Muvuku or Textforms. Depending if your CHT instance is configured in forms-only mode or not you might receive an error if the form is not found.
fromReporting phone number.
sent_timestampTimestamp in MS since Unix Epoch of when the message was received on the gateway. Defaults to now.
JSON Properties

Special values reside in the property _meta, so you can’t have a form field named _meta. Only strings and numbers are currently support as field values.

All property names will be lowercased.

KeyDescription
_meta.formThe form code.
_meta.fromReporting phone number. Optional.
_meta.reported_dateTimestamp in MS since Unix Epoch of when the message was received on the gateway. Defaults to now.
_meta.localeOptional locale string. Example: ‘fr’

Examples

Creating new record using message field.

POST /api/v1/records
Content-Type: application/x-www-form-urlencoded

message=1!YYYZ!Sam#23#2015#ANC&from=+5511943348031&sent_timestamp=1352399720000
HTTP/1.1 200 OK
Content-Type: application/json; charset=utf-8

{
  "success": true,
  "id": "364c796a843fbe0a73476f9153012733"
}

Creating new record with JSON.

POST /api/v1/records
Content-Type: application/json

{
  "nurse": "Sam",
  "week": 23,
  "year": 2015,
  "visit": "ANC",
  "_meta": {
    "form": "YYYZ",
    "reported_date": 1352399720000
  }
}
HTTP/1.1 200 OK
Content-Type: application/json; charset=utf-8

{
  "success": true,
  "id": "364c796a843fbe0a73476f9153012733"
}

Errors

If required fields are not found return 500.

If invalid JSON return error response 500.

If submitting JSON and corresponding form is not found on the server you will receive an error.

SMS

POST /api/sms

Endpoint used by cht-gateway to send sms messages. More documentation in the cht-gateway repo.

POST /api/v1/sms/{aggregator}/{endpoint}

Endpoint for integration with SMS aggregators. More details on the RapidPro and Africa’s Talking pages.

Person

GET /api/v1/person/{{uuid}}

Added in 4.9.0

Returns a person’s data in JSON format.

Permissions

can_view_contacts

Query parameters

NameRequiredDescription
with_lineagefalseIf “true”, the person’s parent lineage will be included in the returned data. Default is “false”.

Examples

Get a person by uuid:

GET /api/v1/person/f512e1d8-841b-4bc1-8154-b6794755f45b
HTTP/1.1 200 OK
Content-Type: application/json; charset=utf-8

{
	"_id": "f512e1d8-841b-4bc1-8154-b6794755f45b",
	"_rev": "3-9dbc362b262f88d63f270fe06a94dfe8",
	"type": "person",
	"name": "Example CHW",
	"date_of_birth": "2002-02-20",
	"phone": "+254712345679",
	"sex": "female",
	"role": "chw",
	"reported_date": 1708453778059,
	"parent": {
		"_id": "d9153705-4574-43c3-b945-71aa2164d1d6",
		"parent": {
			"_id": "b935ef10-0339-4263-99fc-34d4f8d72891"
		}
	},
	"patient_id": "74615"
}

Get a person by uuid with lineage:

GET /api/v1/person/f512e1d8-841b-4bc1-8154-b6794755f45b?with_lineage=true
HTTP/1.1 200 OK
Content-Type: application/json; charset=utf-8

{
	"_id": "f512e1d8-841b-4bc1-8154-b6794755f45b",
	"_rev": "3-9dbc362b262f88d63f270fe06a94dfe8",
	"type": "person",
	"name": "Example CHW",
	"date_of_birth": "2002-02-20",
	"phone": "+254712345679",
	"sex": "female",
	"role": "chw",
	"reported_date": 1708453778059,
	"parent": {
		"_id": "d9153705-4574-43c3-b945-71aa2164d1d6",
		"_rev": "2-5f5fde4a8def0f40f89bd164d93bed4f",
		"parent": {
			"_id": "b935ef10-0339-4263-99fc-34d4f8d72891",
			"_rev": "2-bdea703bfec184085c31a6bab022764f",
			"type": "district_hospital",
			"name": "Example Health Facility",
			"contact": {
				"_id": "e5237f20-2d28-4272-8006-c4903e032ab4",
				"_rev": "3-5a0a8e95cef8bafc186a9494c75afb3c",
				"type": "person",
				"name": "Example Supervisor",
				"date_of_birth": "2002-02-20",
				"phone": "+254712345678",
				"sex": "female",
				"role": "chw_supervisor",
				"reported_date": 1708453756441,
				"parent": {
					"_id": "b935ef10-0339-4263-99fc-34d4f8d72891"
				},
				"patient_id": "20424"
			},
			"reported_date": 1708453756440,
			"place_id": "54380"
		},
		"type": "health_center",
		"name": "Example Health Center",
		"contact": {
			"_id": "f512e1d8-841b-4bc1-8154-b6794755f45b",
			"_rev": "3-9dbc362b262f88d63f270fe06a94dfe8",
			"type": "person",
			"name": "Example CHW",
			"date_of_birth": "2002-02-20",
			"phone": "+254712345679",
			"sex": "female",
			"role": "chw",
			"reported_date": 1708453778059,
			"parent": {
				"_id": "d9153705-4574-43c3-b945-71aa2164d1d6",
				"parent": {
					"_id": "b935ef10-0339-4263-99fc-34d4f8d72891"
				}
			},
			"patient_id": "74615"
		},
		"reported_date": 1708453778059,
		"place_id": "17437"
	},
	"patient_id": "74615"
}

GET /api/v1/person

Added in 4.11.0

Returns a JSON array of people based on the specified page parameters.

Permissions

can_view_contacts

Query Parameters

NameRequiredDescription
typetrueID of the contact_type for the people to fetch.
cursorfalseThe token identifying which page to retrieve. A null value indicates the first page should be returned. Subsequent pages can be retrieved by providing the cursor returned with the previous page. Defaults to null.
limitfalseThe total number of people to fetch. Defaults to 100.

Examples

  1. Get people of type person.
GET /api/v1/person?type=person
HTTP/1.1 200 OK
Content-Type: application/json; charset=utf-8

{
    "data": [
        {
            "_id": "0c9cc77d-7858-44dd-bf44-a25b14334801",
            "_rev": "1-5fd1e08353c3d102e9b6710eed98dd65",
            "type": "person",
            "name": "A",
            "short_name": "",
            "date_of_birth": "1961-09-15",
            "date_of_birth_method": "",
            "ephemeral_dob": {
                "dob_calendar": "1961-09-15",
                "dob_method": "",
                "dob_approx": "2024-08-05T00:00:00.000+06:45",
                "dob_raw": "1961-09-15",
                "dob_iso": "1961-09-15"
            },
            "phone": "",
            "phone_alternate": "",
            "sex": "male",
            "role": "manager",
            "external_id": "",
            "notes": "",
            "user_for_contact": {
                "create": "false"
            },
            "meta": {
                "created_by": "medic",
                "created_by_person_uuid": "",
                "created_by_place_uuid": ""
            },
            "reported_date": 1722840232495,
            "parent": {
                "_id": "8f98977e-8475-4bef-a064-2ba305b93f3f",
                "parent": {
                    "_id": "a7922ef9-ea7a-47d9-b3eb-b78b13d31460"
                }
            },
            "form_version": {
                "time": 1720158606681,
                "sha256": "b76e6304218d7e5d7f63c044da514784f0db4b293c1b35ac301ac7ec99876824"
            }
        },
        {
            "_id": "6fbd5d57-28aa-4a2a-93aa-e9b9f4a3dde3",
            "_rev": "1-3824f633dda834f676d7d27622ed9371",
            "type": "person",
            "name": "B",
            "short_name": "",
            "date_of_birth": "1989-02-10",
            "date_of_birth_method": "",
            "ephemeral_dob": {
                "dob_calendar": "1989-02-10",
                "dob_method": "",
                "dob_approx": "2024-07-26T00:00:00.000+06:45",
                "dob_raw": "1989-02-10",
                "dob_iso": "1989-02-10"
            },
            "phone": "",
            "phone_alternate": "",
            "sex": "male",
            "role": "manager",
            "external_id": "",
            "notes": "",
            "user_for_contact": {
                "create": "false"
            },
            "meta": {
                "created_by": "medic",
                "created_by_person_uuid": "",
                "created_by_place_uuid": ""
            },
            "reported_date": 1722004033697,
            "parent": {
                "_id": "a7922ef9-ea7a-47d9-b3eb-b78b13d31460"
            },
            "form_version": {
                "time": 1720158606680,
                "sha256": "a2bce7debe191964d6f0a26e0d9cce2ca4437c87a59753688cefac6d2edc9873"
            }
        },
        {
            "_id": "99ea9cf6-8bef-475c-a769-3db0d1888f22",
            "_rev": "1-2eea14e985aeffd40b33552b33db55a9",
            "type": "person",
            "name": "C",
            "short_name": "",
            "date_of_birth": "1989-02-11",
            "date_of_birth_method": "",
            "ephemeral_dob": {
                "dob_calendar": "1989-02-11",
                "dob_method": "",
                "dob_approx": "2024-07-26T00:00:00.000+06:45",
                "dob_raw": "1989-02-11",
                "dob_iso": "1989-02-11"
            },
            "phone": "",
            "phone_alternate": "",
            "sex": "male",
            "role": "patient",
            "external_id": "",
            "notes": "",
            "user_for_contact": {
                "create": "false"
            },
            "meta": {
                "created_by": "medic",
                "created_by_person_uuid": "",
                "created_by_place_uuid": ""
            },
            "reported_date": 1722004087809,
            "parent": {
                "_id": "be9eb905-1dd3-4957-89ed-b74d080a8246",
                "parent": {
                    "_id": "b132829c-6267-4d4f-aa8e-11862828e65f",
                    "parent": {
                        "_id": "a7922ef9-ea7a-47d9-b3eb-b78b13d31460"
                    }
                }
            },
            "form_version": {
                "time": 1720158606678,
                "sha256": "c533208263cb5892a50ede65acd31553863a1b9ac84d2b19e993ddc6f9690e1b"
            }
        },
        {
            "_id": "a4e075ce-66cb-47bf-afe8-4c72d1674852",
            "_rev": "1-4a5920ecc73521905fa6661b9b142346",
            "type": "person",
            "name": "D",
            "short_name": "",
            "date_of_birth": "1991-03-05",
            "date_of_birth_method": "",
            "ephemeral_dob": {
                "dob_calendar": "1991-03-05",
                "dob_method": "",
                "dob_approx": "2024-08-05T00:00:00.000+06:45",
                "dob_raw": "1991-03-05",
                "dob_iso": "1991-03-05"
            },
            "phone": "",
            "phone_alternate": "",
            "sex": "male",
            "role": "patient",
            "external_id": "",
            "notes": "",
            "user_for_contact": {
                "create": "false"
            },
            "meta": {
                "created_by": "medic",
                "created_by_person_uuid": "",
                "created_by_place_uuid": ""
            },
            "reported_date": 1722842989644,
            "parent": {
                "_id": "1c5c9538-4832-47f4-9398-e113ec0bc856",
                "parent": {
                    "_id": "8f98977e-8475-4bef-a064-2ba305b93f3f",
                    "parent": {
                        "_id": "a7922ef9-ea7a-47d9-b3eb-b78b13d31460"
                    }
                }
            },
            "form_version": {
                "time": 1720158606678,
                "sha256": "c533208263cb5892a50ede65acd31553863a1b9ac84d2b19e993ddc6f9690e1b"
            }
        },
        {
            "_id": "dc2506e6-55c0-4874-a4a1-334fabef3044",
            "_rev": "1-a9fa456db5b1aff444819ca91cd07c5b",
            "type": "person",
            "name": "E",
            "short_name": "",
            "date_of_birth": "1916-06-14",
            "date_of_birth_method": "",
            "ephemeral_dob": {
                "dob_calendar": "1916-06-14",
                "dob_method": "",
                "dob_approx": "2024-07-26T00:00:00.000+06:45",
                "dob_raw": "1916-06-14",
                "dob_iso": "1916-06-14"
            },
            "phone": "",
            "phone_alternate": "",
            "sex": "male",
            "role": "manager",
            "external_id": "",
            "notes": "",
            "user_for_contact": {
                "create": "false"
            },
            "meta": {
                "created_by": "medic",
                "created_by_person_uuid": "",
                "created_by_place_uuid": ""
            },
            "reported_date": 1722004065958,
            "parent": {
                "_id": "b132829c-6267-4d4f-aa8e-11862828e65f",
                "parent": {
                    "_id": "a7922ef9-ea7a-47d9-b3eb-b78b13d31460"
                }
            },
            "form_version": {
                "time": 1720158606681,
                "sha256": "b76e6304218d7e5d7f63c044da514784f0db4b293c1b35ac301ac7ec99876824"
            }
        },
        ... // 95 more records or less if database has less than 100 records
    ],
    "cursor": null
}
  1. Get 3 people of type person by using cursor 1.
GET /api/v1/person?type=person&cursor=1&limit=3
HTTP/1.1 200 OK
Content-Type: application/json; charset=utf-8

{
    "data": [
        {
            "_id": "6fbd5d57-28aa-4a2a-93aa-e9b9f4a3dde3",
            "_rev": "1-3824f633dda834f676d7d27622ed9371",
            "type": "person",
            "name": "X",
            "short_name": "x",
            "date_of_birth": "1989-02-10",
            "date_of_birth_method": "",
            "ephemeral_dob": {
                "dob_calendar": "1989-02-10",
                "dob_method": "",
                "dob_approx": "2024-07-26T00:00:00.000+06:45",
                "dob_raw": "1989-02-10",
                "dob_iso": "1989-02-10"
            },
            "phone": "",
            "phone_alternate": "",
            "sex": "male",
            "role": "manager",
            "external_id": "",
            "notes": "",
            "user_for_contact": {
                "create": "false"
            },
            "meta": {
                "created_by": "medic",
                "created_by_person_uuid": "",
                "created_by_place_uuid": ""
            },
            "reported_date": 1722004033697,
            "parent": {
                "_id": "a7922ef9-ea7a-47d9-b3eb-b78b13d31460"
            },
            "form_version": {
                "time": 1720158606680,
                "sha256": "a2bce7debe191964d6f0a26e0d9cce2ca4437c87a59753688cefac6d2edc9873"
            }
        },
        {
            "_id": "99ea9cf6-8bef-475c-a769-3db0d1888f22",
            "_rev": "1-2eea14e985aeffd40b33552b33db55a9",
            "type": "person",
            "name": "Y",
            "short_name": "y",
            "date_of_birth": "1989-02-11",
            "date_of_birth_method": "",
            "ephemeral_dob": {
                "dob_calendar": "1989-02-11",
                "dob_method": "",
                "dob_approx": "2024-07-26T00:00:00.000+06:45",
                "dob_raw": "1989-02-11",
                "dob_iso": "1989-02-11"
            },
            "phone": "",
            "phone_alternate": "",
            "sex": "male",
            "role": "patient",
            "external_id": "",
            "notes": "",
            "user_for_contact": {
                "create": "false"
            },
            "meta": {
                "created_by": "medic",
                "created_by_person_uuid": "",
                "created_by_place_uuid": ""
            },
            "reported_date": 1722004087809,
            "parent": {
                "_id": "be9eb905-1dd3-4957-89ed-b74d080a8246",
                "parent": {
                    "_id": "b132829c-6267-4d4f-aa8e-11862828e65f",
                    "parent": {
                        "_id": "a7922ef9-ea7a-47d9-b3eb-b78b13d31460"
                    }
                }
            },
            "form_version": {
                "time": 1720158606678,
                "sha256": "c533208263cb5892a50ede65acd31553863a1b9ac84d2b19e993ddc6f9690e1b"
            }
        },
        {
            "_id": "a4e075ce-66cb-47bf-afe8-4c72d1674852",
            "_rev": "1-4a5920ecc73521905fa6661b9b142346",
            "type": "person",
            "name": "Z",
            "short_name": "z",
            "date_of_birth": "1991-03-05",
            "date_of_birth_method": "",
            "ephemeral_dob": {
                "dob_calendar": "1991-03-05",
                "dob_method": "",
                "dob_approx": "2024-08-05T00:00:00.000+06:45",
                "dob_raw": "1991-03-05",
                "dob_iso": "1991-03-05"
            },
            "phone": "",
            "phone_alternate": "",
            "sex": "male",
            "role": "patient",
            "external_id": "",
            "notes": "",
            "user_for_contact": {
                "create": "false"
            },
            "meta": {
                "created_by": "medic",
                "created_by_person_uuid": "",
                "created_by_place_uuid": ""
            },
            "reported_date": 1722842989644,
            "parent": {
                "_id": "1c5c9538-4832-47f4-9398-e113ec0bc856",
                "parent": {
                    "_id": "8f98977e-8475-4bef-a064-2ba305b93f3f",
                    "parent": {
                        "_id": "a7922ef9-ea7a-47d9-b3eb-b78b13d31460"
                    }
                }
            },
            "form_version": {
                "time": 1720158606678,
                "sha256": "c533208263cb5892a50ede65acd31553863a1b9ac84d2b19e993ddc6f9690e1b"
            }
        }
    ],
    "cursor": "4"
}

People

Supported Properties

Use JSON in the request body to specify a person’s details.

Note: this does not accommodate having a place field on your form and will likely be revised soon.

Required

KeyDescription
nameString used to describe the person.
typeID of the contact_type for the new person. Defaults to ‘person’ for backwards compatibility.

Optional

KeyDescription
placeString that references a place or object that defines a new place.
reported_dateTimestamp of when the record was reported or created. Defaults to now.

POST /api/v1/people

Create new people.

Permissions

By default any user can create or modify a place. Use these permissions to restrict access:

can_create_people, can_create_places

Examples

Create new person and place hierarchy.

POST /api/v1/people
Content-Type: application/json

{
  "name": "Hannah",
  "phone": "+2548277210095",
  "type": "contact",
  "contact_type": "patient",
  "place": {
    "name": "CHP Area One",
    "type": "health_center",
    "parent": {
      "name": "CHP Branch One",
      "type": "district_hospital"
    }
  }
}

Create new person and assign existing place.

POST /api/v1/people
Content-Type: application/json

{
 "name": "Samuel",
 "place": "1d83f2b4a27eceb40df9e9f9ad06d137",
 "type": "contact",
 "contact_type": "chp"
}

Example response:

HTTP/1.1 200 OK
Content-Type: application/json

{
  "id": "71df9d25ed6732ea3b4435862510d115",
  "rev": "1-a4060843d78f46a60a6f41051e40e3b5"
}

Place

GET /api/v1/place/{{uuid}}

Added in 4.10.0

Returns a place’s data in JSON format.

Permissions

can_view_contacts

Query parameters

NameRequiredDescription
with_lineagefalseIf “true”, the place’s parent lineage will be included in the returned data. Default is “false”.

Examples

Get a place by uuid:

GET /api/v1/place/d9153705-4574-43c3-b945-71aa2164d1d6
HTTP/1.1 200 OK
Content-Type: application/json; charset=utf-8

{
	"_id": "d9153705-4574-43c3-b945-71aa2164d1d6",
	"_rev": "2-5f5fde4a8def0f40f89bd164d93bed4f",
	"parent": {
		"_id": "b935ef10-0339-4263-99fc-34d4f8d72891"
	},
	"type": "health_center",
	"name": "Example Health Center",
	"contact": {
		"_id": "f512e1d8-841b-4bc1-8154-b6794755f45b",
		"parent": {
			"_id": "d9153705-4574-43c3-b945-71aa2164d1d6",
			"parent": {
				"_id": "b935ef10-0339-4263-99fc-34d4f8d72891"
			}
		}
	},
	"reported_date": 1708453778059,
	"place_id": "17437"
}

Get a place by uuid with lineage:

GET /api/v1/place/d9153705-4574-43c3-b945-71aa2164d1d6?with_lineage=true
HTTP/1.1 200 OK
Content-Type: application/json; charset=utf-8

{
	"_id": "d9153705-4574-43c3-b945-71aa2164d1d6",
	"_rev": "2-5f5fde4a8def0f40f89bd164d93bed4f",
	"parent": {
		"_id": "b935ef10-0339-4263-99fc-34d4f8d72891",
		"_rev": "2-bdea703bfec184085c31a6bab022764f",
		"parent": "",
		"type": "district_hospital",
		"name": "Example Health Facility",
		"contact": {
			"_id": "e5237f20-2d28-4272-8006-c4903e032ab4",
			"_rev": "3-5a0a8e95cef8bafc186a9494c75afb3c",
			"type": "person",
			"name": "Example Supervisor",
			"date_of_birth": "2002-02-20",
			"phone": "+254712345678",
			"sex": "female",
			"role": "chw_supervisor",
			"reported_date": 1708453756441,
			"parent": {
				"_id": "b935ef10-0339-4263-99fc-34d4f8d72891"
			},
			"patient_id": "20424"
		},
		"reported_date": 1708453756440,
		"place_id": "54380"
	},
	"type": "health_center",
	"name": "Example Health Center",
	"contact": {
		"_id": "f512e1d8-841b-4bc1-8154-b6794755f45b",
		"_rev": "3-9dbc362b262f88d63f270fe06a94dfe8",
		"type": "person",
		"name": "Example CHW",
		"date_of_birth": "2002-02-20",
		"phone": "+254712345679",
		"sex": "female",
		"role": "chw",
		"reported_date": 1708453778059,
		"parent": {
			"_id": "d9153705-4574-43c3-b945-71aa2164d1d6",
			"parent": {
				"_id": "b935ef10-0339-4263-99fc-34d4f8d72891"
			}
		},
		"patient_id": "74615"
	},
	"reported_date": 1708453778059,
	"place_id": "17437"
}

GET /api/v1/place

Added in 4.12.0

Returns a JSON array of places based on the specified page parameters.

Query Parameters

NameRequiredDescription
typetrueID of the contact_type for the places to fetch.
cursorfalseThe token identifying which page to retrieve. A null value indicates the first page should be returned. Subsequent pages can be retrieved by providing the cursor returned with the previous page. Defaults to null.
limitfalseThe total number of places to fetch. Defaults to 100.

Examples

  1. Get places of type clinic.
GET /api/v1/place?type=clinic
HTTP/1.1 200 OK
Content-Type: application/json; charset=utf-8

{
    "data": [
        {
            "_id": "1c5c9538-4832-47f4-9398-e113ec0bc856",
            "_rev": "1-ec14166e5a1b0014d923a6c0aef85362",
            "parent": {
                "_id": "8f98977e-8475-4bef-a064-2ba305b93f3f",
                "parent": {
                    "_id": "a7922ef9-ea7a-47d9-b3eb-b78b13d31460"
                }
            },
            "type": "clinic",
            "is_name_generated": "true",
            "name": "A clinic",
            "external_id": "",
            "notes": "",
            "contact": {
                "_id": "a4e075ce-66cb-47bf-afe8-4c72d1674852"
            },
            "geolocation": "",
            "meta": {
                "created_by": "medic",
                "created_by_person_uuid": "",
                "created_by_place_uuid": ""
            },
            "reported_date": 1722842989644,
            "form_version": {
                "time": 1720158606678,
                "sha256": "c533208263cb5892a50ede65acd31553863a1b9ac84d2b19e993ddc6f9690e1b"
            }
        },
        {
            "_id": "be9eb905-1dd3-4957-89ed-b74d080a8246",
            "_rev": "1-d1444e92bd1973e9d8366c0a55346c0a",
            "parent": {
                "_id": "b132829c-6267-4d4f-aa8e-11862828e65f",
                "parent": {
                    "_id": "a7922ef9-ea7a-47d9-b3eb-b78b13d31460"
                }
            },
            "type": "clinic",
            "is_name_generated": "true",
            "name": "B clinic",
            "external_id": "",
            "notes": "",
            "contact": {
                "_id": "99ea9cf6-8bef-475c-a769-3db0d1888f22"
            },
            "geolocation": "",
            "meta": {
                "created_by": "medic",
                "created_by_person_uuid": "",
                "created_by_place_uuid": ""
            },
            "reported_date": 1722004087809,
            "form_version": {
                "time": 1720158606678,
                "sha256": "c533208263cb5892a50ede65acd31553863a1b9ac84d2b19e993ddc6f9690e1b"
            }
        },
        ... // 98 more records or less if database has less than 100 records
    ],
    "cursor": null
}
  1. Get 2 places of type clinic by using cursor 1.
GET /api/v1/place?type=clinic&cursor=1&limit=2
HTTP/1.1 200 OK
Content-Type: application/json; charset=utf-8

{
    "data": [
        {
            "_id": "1c5c9538-4832-47f4-9398-e113ec0bc856",
            "_rev": "1-ec14166e5a1b0014d923a6c0aef85362",
            "parent": {
                "_id": "8f98977e-8475-4bef-a064-2ba305b93f3f",
                "parent": {
                    "_id": "a7922ef9-ea7a-47d9-b3eb-b78b13d31460"
                }
            },
            "type": "clinic",
            "is_name_generated": "true",
            "name": "A clinic",
            "external_id": "",
            "notes": "",
            "contact": {
                "_id": "a4e075ce-66cb-47bf-afe8-4c72d1674852"
            },
            "geolocation": "",
            "meta": {
                "created_by": "medic",
                "created_by_person_uuid": "",
                "created_by_place_uuid": ""
            },
            "reported_date": 1722842989644,
            "form_version": {
                "time": 1720158606678,
                "sha256": "c533208263cb5892a50ede65acd31553863a1b9ac84d2b19e993ddc6f9690e1b"
            }
        },
        {
            "_id": "be9eb905-1dd3-4957-89ed-b74d080a8246",
            "_rev": "1-d1444e92bd1973e9d8366c0a55346c0a",
            "parent": {
                "_id": "b132829c-6267-4d4f-aa8e-11862828e65f",
                "parent": {
                    "_id": "a7922ef9-ea7a-47d9-b3eb-b78b13d31460"
                }
            },
            "type": "clinic",
            "is_name_generated": "true",
            "name": "B clinic",
            "external_id": "",
            "notes": "",
            "contact": {
                "_id": "99ea9cf6-8bef-475c-a769-3db0d1888f22"
            },
            "geolocation": "",
            "meta": {
                "created_by": "medic",
                "created_by_person_uuid": "",
                "created_by_place_uuid": ""
            },
            "reported_date": 1722004087809,
            "form_version": {
                "time": 1720158606678,
                "sha256": "c533208263cb5892a50ede65acd31553863a1b9ac84d2b19e993ddc6f9690e1b"
            }
        }
    ],
    "cursor": "3"
}

Places

By default any user can create or modify a place.

Supported Properties

Use JSON in the request body to specify a place’s details.

Required Properties

KeyDescription
nameString used to describe the place.
typePlace type
parentString that references a place or object that defines a new place. Optional for District Hospital and National Office types.

Optional Properties

KeyDescription
contactString identifier for a person or object that defines a new person.
reported_dateTimestamp of when the record was reported or created. Defaults to now.

Place Types

KeyDescription
clinicClinic
health_centerHealth Center
district_hospitalDistrict Hospital
national_officeNational Office

POST /api/v1/places

Create a new place and optionally a contact.

Permissions

By default any user can create new places. Use these permissions to restrict access:

can_create_places, can_create_people

Examples

Create new place referencing existing parent.

POST /api/v1/places
Content-Type: application/json

{
 "name": "Busia Clinic",
 "type": "clinic",
 "parent": "1d83f2b4a27eceb40df9e9f9ad06d137"
}

Create child and parent places.

POST /api/v1/places
Content-Type: application/json

{
  "name": "CHP Area One",
  "type": "health_center",
  "parent": {
    "name": "CHP Branch One",
    "type": "district_hospital"
  }
}

Also creates contact (person).

POST /api/v1/places
Content-Type: application/json

{
  "name": "CHP Area One",
  "type": "health_center",
  "parent": {
    "name": "CHP Branch One",
    "type": "district_hospital"
  },
  "contact": {
    "name": "Paul",
    "phone": "+254883720611"
  }
}

Or assigns them.

POST /api/v1/places
Content-Type: application/json

{
  "name": "CHP Area One",
  "type": "health_center",
  "parent": {
    "name": "CHP Branch One",
    "type": "district_hospital"
  },
  "contact": "71df9d25ed6732ea3b4435862510ef8e"
}

Example success response:

HTTP/1.1 200 OK
Content-Type: application/json

{
  "id": "71df9d25ed6732ea3b4435862510d115",
  "rev": "1-a4060843d78f46a60a6f41051e40e3b5"
}

Error response if facility structure is not correct:

HTTP/1.1 400 Bad Request
Content-Type: text/plain

Health Centers should have "district_hospital" parent type.

POST /api/v1/places/{{id}}

Update a place and optionally its contact.

Permissions

By default any user can update a place. Use these permissions to restrict access:

can_update_places

Examples

Update a place’s contact.

POST /api/v1/places/1d83f2b4a27eceb40df9e9f9ad06d137
Content-Type: application/json

{
 "contact": "71df9d25ed6732ea3b4435862505f7a9"
}
HTTP/1.1 200 OK
Content-Type: application/json

{
  "id": "1d83f2b4a27eceb40df9e9f9ad06d137",
  "rev": "12-a4060843d78f46a60a6f41051e40e3b5"
}

Users

All user related requests are limited to users with admin privileges by default.

Supported Properties

Use JSON in the request body to specify user details. Any properties submitted that are not on the list below will be ignored. Any properties not included will be undefined.

NameRequiredTypeDescriptionVersion
usernameyesStringidentifier used for authentication
rolesyesArray
placeyes, if the roles contain an offline rolestring or objectPlace identifier string (UUID) or object this user resides in.
contactyes, if the roles contain an offline rolestring or objectA person identifier string (UUID) or object based on the form configured in the app.
passwordyes, if token_login is not enabled for the userStringPassword string used for authentication. Only allowed to be set, not retrieved.
phoneyes, if token_login is enabled for the userStringValid phone number
token_loginnoBooleanA boolean representing whether or not the Login by SMS should be enabled for this user.3.10.0
fullnamenoStringFull name
emailnoStringEmail address
knownnoBooleanBoolean to define if the user has logged in before.

Login by SMS

When creating or updating a user, sending a truthy value for the field token_login will enable Login by SMS for this user. This action resets the user’s password to an unknown string and generates a complex 64 character token, that is used to generate a token-login URL. The URL is sent to the user’s phone number by SMS, along with another (configurable) SMS that can contain additional information. Accessing this link, before its expiration time, will log the user in directly - without the need of any other credentials. The link can only be accessed once, the token becomes invalid after being used for one login. The token expires in 24 hours, after which logging in is only possible by either generating a new token, or disabling token_login and manually setting a password.

The SMS messages are stored in a doc of a new type login_token. These docs cannot be viewed as reports from the webapp, and can only be edited by admins, but their messages are visible in the Admin Message Queue page.

To disable login by SMS for a user, update the user sending token_login with a false value. To regenerate the token, update the user sending token_login with a true value.

token_loginuser stateaction
undefinednewNone
undefinedexistent, no tokenNone
undefinedexistent, with tokenNone. Login by SMS remains enabled. Token is unchanged.
truenewLogin by SMS enabled. Token is generated and SMS is sent.
trueexistent, no tokenPassword is reset. Login by SMS enabled. Token is generated and SMS is sent. Existent sessions are invalidated.
trueexistent, with tokenPassword is reset. Login by SMS enabled. New token is generated and SMS is sent. Old token is invalid. Existent sessions are invalidated.
falsenewNone.
falseexistent, no tokenNone.
falseexistent, with tokenRequest requires a password. Login by SMS is disabled. Old token is invalidated. Existent sessions are invalidated.

This feature uses app_settings.app_url and app_settings.token_login to be defined and enabled. If app_settings.app_url is not defined, the generated token-login URL will use the Host request header, which may not always be correct.

See Also: Accessing CHT Apps

GET /api/v1/users

DEPRECATED: use /api/v2/users

Returns a list of users and their profile data in JSON format.

Permissions

can_view_users

Examples

Get list of users:

GET /api/v1/users
HTTP/1.1 200 OK
Content-Type: application/json; charset=utf-8

[
  {
    "id": "org.couchdb.user:admin",
    "rev": "10-6486428924d11781c107ea74de6b63b6",
    "type": "admin",
    "username": "admin"
  },
  {
    "id": "org.couchdb.user:demo",
    "rev": "14-8758c8493edcc6dac50366173fc3e24a",
    "type": "district-manager",
    "fullname": "Example User",
    "username": "demo",
    "place": {
      "_id": "eeb17d6d-5dde-c2c0-62c4a1a0ca17d38b",
      "type": "district_hospital",
      "name": "Sample District",
      "contact": {
        "_id": "eeb17d6d-5dde-c2c0-62c4a1a0ca17fd17",
        "type": "person",
        "name": "Paul",
        "phone": "+2868917046"
      }
    },
    "contact": {
      "_id": "eeb17d6d-5dde-c2c0-62c4a1a0ca17fd17",
      "type": "person",
      "name": "Paul",
      "phone": "+2868917046"
    }
  }
]

GET /api/v2/users

Added in 4.1.0

Returns a list of users and their profile data in JSON format.

Permissions

can_view_users

Query Parameters

VariableDescription
facility_idAdded in 4.7.0. String identifier representing the uuid of the user’s facility.
contact_idAdded in 4.7.0. String identifier representing the uuid of the user’s associated contact.

Examples

Get list of users:

GET /api/v2/users
HTTP/1.1 200 OK
Content-Type: application/json; charset=utf-8

[
  {
    "id": "org.couchdb.user:admin",
    "rev": "10-6486428924d11781c107ea74de6b63b6",
    "roles": [ "admin" ],
    "username": "admin"
  },
  {
    "id": "org.couchdb.user:demo",
    "rev": "14-8758c8493edcc6dac50366173fc3e24a",
    "roles": [ "district_admin", "data_user" ],
    "fullname": "Example User",
    "username": "demo",
    "place": {
      "_id": "eeb17d6d-5dde-c2c0-62c4a1a0ca17d38b",
      "type": "district_hospital",
      "name": "Sample District",
      "contact": {
        "_id": "eeb17d6d-5dde-c2c0-62c4a1a0ca17fd17",
        "type": "person",
        "name": "Paul",
        "phone": "+2868917046"
      }
    },
    "contact": {
      "_id": "eeb17d6d-5dde-c2c0-62c4a1a0ca17fd17",
      "type": "person",
      "name": "Paul",
      "phone": "+2868917046"
    }
  }
]

Get users with a specific facility_id:

GET /api/v2/users?facility_id=eeb17d6d-5dde-c2c0-62c4a1a0ca17d38b
HTTP/1.1 200 OK
Content-Type: application/json; charset=utf-8

[
  {
    "id": "org.couchdb.user:demo",
    "rev": "14-8758c8493edcc6dac50366173fc3e24a",
    "roles": [ "district_admin", "data_user" ],
    "fullname": "Example User",
    "username": "demo",
    "place": {
      "_id": "eeb17d6d-5dde-c2c0-62c4a1a0ca17d38b",
      "type": "district_hospital",
      "name": "Sample District",
      "contact": {
        "_id": "eeb17d6d-5dde-c2c0-62c4a1a0ca17fd17",
        "type": "person",
        "name": "Paul",
        "phone": "+2868917046"
      }
    },
    "contact": {
      "_id": "eeb17d6d-5dde-c2c0-62c4a1a0ca17fd17",
      "type": "person",
      "name": "Paul",
      "phone": "+2868917046"
    }
  }
]

Get users with a specific contact_id:

GET /api/v2/users?contact_id=eeb17d6d-5dde-c2c0-62c4a1a0ca17fd17
HTTP/1.1 200 OK
Content-Type: application/json; charset=utf-8

[
  {
    "id": "org.couchdb.user:demo",
    "rev": "14-8758c8493edcc6dac50366173fc3e24a",
    "roles": [ "district_admin", "data_user" ],
    "fullname": "Example User",
    "username": "demo",
    "place": {
      "_id": "eeb17d6d-5dde-c2c0-62c4a1a0ca17d38b",
      "type": "district_hospital",
      "name": "Sample District",
      "contact": {
        "_id": "eeb17d6d-5dde-c2c0-62c4a1a0ca17fd17",
        "type": "person",
        "name": "Paul",
        "phone": "+2868917046"
      }
    },
    "contact": {
      "_id": "eeb17d6d-5dde-c2c0-62c4a1a0ca17fd17",
      "type": "person",
      "name": "Paul",
      "phone": "+2868917046"
    }
  }
]

GET /api/v2/users/{{username}}

Added in 4.7.0

Returns a user’s profile data in JSON format.

Permissions

can_view_users

Examples

Get a user by username:

GET /api/v2/users/demo
HTTP/1.1 200 OK
Content-Type: application/json; charset=utf-8

{
  "id": "org.couchdb.user:demo",
  "rev": "14-8758c8493edcc6dac50366173fc3e24a",
  "type": "district-manager",
  "fullname": "Example User",
  "username": "demo",
  "place": {
    "_id": "eeb17d6d-5dde-c2c0-62c4a1a0ca17d38b",
    "type": "district_hospital",
    "name": "Sample District",
    "contact": {
      "_id": "eeb17d6d-5dde-c2c0-62c4a1a0ca17fd17",
      "type": "person",
      "name": "Paul",
      "phone": "+2868917046"
    }
  },
  "contact": {
    "_id": "eeb17d6d-5dde-c2c0-62c4a1a0ca17fd17",
    "type": "person",
    "name": "Paul",
    "phone": "+2868917046"
  }
}

POST /api/v1/users

Create new users with a place and a contact.

Creating multiple users at once by passing an array of users was introduced in version 3.15. All users need to meet the following requirements before any of them are created:

  • All required fields are filled in
  • The password is at least 8 characters long and difficult to guess
  • The phone number is valid when token_login is enabled

Users are created in parallel and the creation is not aborted even if one of the users fails to be created.

Passing a single user in the request’s body will return a single object whereas passing an array of users will return an array of objects as shown in the examples below.

Permissions

can_create_users

Examples

Create one user

Create a new user that can authenticate with a username of “mary” and password of “secret” that can submit reports and view or modify records associated to their place. The place is created in the background and automatically linked to the contact.

POST /api/v1/users
Content-Type: application/json

{
  "password": "secret",
  "username": "mary",
  "type": "district-manager",
  "place": {
    "name": "Mary's Area",
    "type": "health_center",
    "parent": "d14e1c3d557761320b13a77e7806e8f8"
  },
  "contact": {
    "name": "Mary Anyango",
    "phone": "+2868917046"
  }
}
HTTP/1.1 200
Content-Type: application/json

{
  "contact": {
    "id": "65416b8ceb53ff88ac1847654501aeb3",
    "rev": "1-0b74d219ae13137c1a06f03a0a52e187"
  },
  "user-settings": {
    "id": "org.couchdb.user:mary",
    "rev": "1-6ac1d36b775143835f4af53f9895d7ae"
  },
  "user": {
    "id": "org.couchdb.user:mary",
    "rev": "1-c3b82a0b47cfe68edd9284c89bebbae4"
  }
}
Create many users

Create two new users that can authenticate with a username and a password that can submit reports and view or modify records associated to their place. The place is created in the background and automatically linked to the contact.

POST /api/v1/users
Content-Type: application/json

[
  {
    "password": "secret",
    "username": "mary",
    "type": "district-manager",
    "place": {
      "name": "Mary's Area",
      "type": "health_center",
      "parent": "d14e1c3d557761320b13a77e7806e8f8"
    },
    "contact": {
      "name": "Mary Anyango",
      "phone": "+2868917046"
    }
  },
  {
    "password": "secret",
    "username": "bob",
    "type": "district-manager",
    "place": {
      "name": "Bob's Area",
      "type": "health_center",
      "parent": "d14e1c3d557761320b13a77e7806e8f8"
    },
    "contact": {
      "name": "Bob Johnson",
      "phone": "+2868194607"
    }
  }
]
HTTP/1.1 200
Content-Type: application/json

[
  {
    "contact": {
      "id": "65416b8ceb53ff88ac1847654501aeb3",
      "rev": "1-0b74d219ae13137c1a06f03a0a52e187"
    },
    "user-settings": {
      "id": "org.couchdb.user:mary",
      "rev": "1-6ac1d36b775143835f4af53f9895d7ae"
    },
    "user": {
      "id": "org.couchdb.user:mary",
      "rev": "1-c3b82a0b47cfe68edd9284c89bebbae4"
    }
  },
  {
    "contact": {
      "id": "8d8a741c1cb441058e29a60ab7596bf2",
      "rev": "1-acbc31712fd105eae3cd0806cd20a8f4"
    },
    "user-settings": {
      "id": "org.couchdb.user:bob",
      "rev": "1-d8629838127accd531043f845c416ef6"
    },
    "user": {
      "id": "org.couchdb.user:bob",
      "rev": "1-5eac2542c801ba6b518728f53d9276a0"
    }
  }
]

Errors

Response when the username already exists
HTTP/1.1 400 Bad Request
Content-Type: application/json

{
  "code": 400,
  "error": {
    "message": "Username \"mary\" already taken.",
    "translationKey": "username.taken",
    "translationParams": {
      "username": "mary"
    }
  }
}
Response when users are missing required fields
HTTP/1.1 400 Bad Request
Content-Type: application/json

{
  "code": 400,
  "error": "Missing required fields:\n\Missing fields username, password, type or roles for user at index 0\nMissing fields password, type or roles for user at index 1",
  "details": {
    "failingIndexes": [
      {
        "fields": [
          "username",
          "password",
          "type or roles"
        ],
        "index": 0
      },
      {
        "fields": [
          "password",
          "type or roles"
        ],
        "index": 1
      }
    ]
  }
}
Response when some users failed to be created
HTTP/1.1 200
Content-Type: application/json

[
  {
    "contact": {
      "id": "65416b8ceb53ff88ac1847654501aeb3",
      "rev": "1-0b74d219ae13137c1a06f03a0a52e187"
    },
    "user-settings": {
      "id": "org.couchdb.user:mary",
      "rev": "1-6ac1d36b775143835f4af53f9895d7ae"
    },
    "user": {
      "id": "org.couchdb.user:mary",
      "rev": "1-c3b82a0b47cfe68edd9284c89bebbae4"
    }
  },
  {
    "error": "Failed to find place."
  }
]

POST /api/v2/users

Added in 3.16.0

Create new users with a place and a contact from a CSV file.

Creating users from a CSV file behaves the same as passing a JSON array of users into the POST /api/v1/users where a row represents a user object and a column represents a user object property. Columns with a :excluded suffix will be ignored, this allows providing a more user-friendly experience with autocompletion on fields or dealing with names instead of long, unreadable ids.

In order to facilitate this process, we have made available a spreadsheet compatible with the default configuration of the CHT. Click here to make a copy of the spreadsheet in Google Sheets. A guide on how to import users with this spreadsheet from within the Admin Console is available in case you don’t want to interact with this API yourself.

Logging

A log entry is created with each bulk import that contains the import status for each user and the import progress status that gets updated throughout the import and finalized upon completion. These entries are saved in the medic-logs database and you can access them by querying documents with a key that starts with bulk-user-upload-.

Headers

KeyValueDescription
Content-Typetext/csvProcesses form data in request body as JSON.

Permissions

can_create_users

Example

Create two new users that can authenticate with a username and a password that can submit reports and view or modify records associated to their place. Along with a new user, a new contact and new place are created as well. The new place will be a child of place.parent (see the UUID fe4da0f9-7d65-4834-bb42-88a5239bbd3b below) and must already exist or else the new place will be an orphan record in the hierarchy and not show up in the GUI.

POST /api/v2/users
Content-Type: text/csv

contact.first_name,contact.last_name,contact.sex,contact.phone,email,contact.meta.created_by,token_login,contact.role,contact.type,contact.contact_type,contact.name,username,password,phone,place.parent,place.type,place.name,place.contact_type,type,fullname
Mary,Anyango,female,+2868917046,,,FALSE,chw,contact,person,Mary Anyango,mary,WrAGyGD9805,2868917046,fe4da0f9-7d65-4834-bb42-88a5239bbd3b,health_center,Mary Anyango's Health center,clinic,chw,Mary Anyango
Bob,Johnson,male,+2868194607,,,FALSE,chw,contact,person,Bob Johnson,bob,JzAEQzY2614,2868194607,fe4da0f9-7d65-4834-bb42-88a5239bbd3b,health_center,Bob Johnson's Health center,clinic,chw,Bob Johnson
HTTP/1.1 200
Content-Type: application/json

[
  {
    "contact": {
      "id": "65416b8ceb53ff88ac1847654501aeb3",
      "rev": "1-0b74d219ae13137c1a06f03a0a52e187"
    },
    "user-settings": {
      "id": "org.couchdb.user:mary",
      "rev": "1-6ac1d36b775143835f4af53f9895d7ae"
    },
    "user": {
      "id": "org.couchdb.user:mary",
      "rev": "1-c3b82a0b47cfe68edd9284c89bebbae4"
    }
  },
  {
    "contact": {
      "id": "8d8a741c1cb441058e29a60ab7596bf2",
      "rev": "1-acbc31712fd105eae3cd0806cd20a8f4"
    },
    "user-settings": {
      "id": "org.couchdb.user:bob",
      "rev": "1-d8629838127accd531043f845c416ef6"
    },
    "user": {
      "id": "org.couchdb.user:bob",
      "rev": "1-5eac2542c801ba6b518728f53d9276a0"
    }
  }
]

Errors

Response when the username already exists
HTTP/1.1 400 Bad Request
Content-Type: application/json

[
  {
    "error": "Missing required fields: username"
  }
]
Response when users are missing required fields
HTTP/1.1 400 Bad Request
Content-Type: application/json

[
  {
    "error": "Missing required fields: username"
  },
  {
    "error": "Missing required fields: password"
  }
]
Response when some users failed to be created
HTTP/1.1 200
Content-Type: application/json

[
  {
    "contact": {
      "id": "65416b8ceb53ff88ac1847654501aeb3",
      "rev": "1-0b74d219ae13137c1a06f03a0a52e187"
    },
    "user-settings": {
      "id": "org.couchdb.user:mary",
      "rev": "1-6ac1d36b775143835f4af53f9895d7ae"
    },
    "user": {
      "id": "org.couchdb.user:mary",
      "rev": "1-c3b82a0b47cfe68edd9284c89bebbae4"
    }
  },
  {
    "error": "Failed to find place."
  }
]

POST /api/v1/users/{{username}}

Allows you to change property values on a user account. Properties listed above are supported except for contact.parent. Creating or modifying people through the user is not supported, see People section.

Permissions

can_update_users, can_update_places, can_update_people

Updating yourself

You do not need any of the above permissions if the user you are modifying is yourself. However, you are not allowed to modify your type, roles, contact or place.

Further more, if you’re updating your password you must be authenticating via Basic Auth (either the header or in the URL). This is to ensure the password is known at time of request, and no one is hijacking a cookie.

URL Parameters

VariableDescription
usernameString identifier used for authentication.

Examples

POST /api/v1/users/mary
Content-Type: application/json

{
  "password": "secret",
  "place": "eeb17d6d-5dde-c2c0-62c4a1a0ca17e342"
}
HTTP/1.1 200 OK
Content-Type: application/json

{
  "user": {
    "id": "org.couchdb.user:mary",
    "rev": "23-858e01fafdfa0d367d798fe5b44751ff"
  },
  "user-settings": {
    "id": "org.couchdb.user:mary",
    "rev": "17-c6d03b86d2d5d70f7270c85e67fea96d"
  }
}

DELETE /api/v1/users/{{username}}

Delete a user. Does not affect a person or place associated to a user.

Permissions

can_delete_users

URL Parameters

VariableDescription
usernameString identifier used for authentication.

Examples

DELETE /api/v1/users/mary
HTTP/1.1 200 OK

GET /api/v1/users-info

Returns the total number of documents an offline user would replicate (total_docs), the number of docs excluding tasks the user would replicate (warn_docs), along with a warn flag if this number exceeds the recommended limit (now set at 10 000).

When the authenticated requester has an offline role, it returns the requester doc count.

Example

GET /api/v1/users-info -H 'Cookie: AuthSession=OFFLINE_USER_SESSION;'
HTTP/1.1 200 OK
Content-Type: application/json

{
  "total_docs": 5678,
  "warn_docs": 4852,
  "warn": false,
  "limit: 10000
}

When the requester has an online role, the following query parameters are accepted:

Query Parameters
VariableDescriptionRequired
facility_idString identifier representing the uuid of the user’s facilitytrue
roleString identifier representing the user role - must be configured as an offline role. Accepts valid UTF-8 JSON array for multiple of roles.true
contact_idString identifier representing the uuid of the user’s associated contactfalse

When requested as an online user, the number of tasks are never counted and never returned, so warn_docs is always equal to total_docs.

Example
GET /api/v1/users-info?facility_id={{facility_uuid}}&role={{role}}&contact_id={{contact_uuid}} -H 'Cookie: AuthSession=OFFLINE_USER_SESSION;'
HTTP/1.1 200 OK
Content-Type: application/json

{
  "total_docs": 10265,
  "warn_docs": 10265,
  "warn": true,
  "limit: 10000
}

In case any of the required query parameters are omitted or the requested role is not configured as an offline role, the request will result in an error:

GET /api/v1/users-info?role={{online_role}} -H 'Cookie: AuthSession=OFFLINE_USER_SESSION;'
HTTP/1.1 400 Bad Request
Content-Type: application/json

{
  "code": 400,
  "error": "Missing required query params: role and/or facility_id"
}

Bulk Operations

POST /api/v1/bulk-delete

Bulk delete endpoint for deleting large numbers of documents. Docs will be batched into groups of 100 and will be sent sequentially to couch (new batch sent after the previous one has returned). The response will be chunked JSON (one batch at a time), so if you wish to get an indication of progress you will need to parse incomplete JSON (with a library such as partial-json-parser.

Permissions

Only available to online users.

Parameters

ParameterDescription
docsArray of JSON objects with _id properties

An array of objects each with an _id property is required (rather than an array of strings representing ids) to ensure forwards compatibility if we choose to require that any additional document information (such as _rev) also be passed in to this endpoint.

Errors

If an error is encountered part-way through the response (eg on the third batch), it’s impossible to send new headers to indicate a 5xx error, so the connection will simply be terminated (as recommended here https://github.com/expressjs/express/issues/2700).

Examples

POST /api/v1/bulk-delete
Content-Type: application/json

{
  "docs": [
    { "_id": "id1" },
    { "_id": "id2" },
    ...
    { "_id": "id150" }
  ]
}
HTTP/1.1 200 OK
Content-Type: application/json

[
  [
    { "ok": true, "id": "id1", "rev": "1-rev1" },
    { "ok": true, "id": "id2", "rev": "1-rev2" },
    ...
    { "ok": true, "id": "id100", "rev": "1-rev100" }
  ],
  [
    { "ok": true, "id": "id101", "rev": "1-rev101" },
    { "ok": true, "id": "id102", "rev": "1-rev102" },
    ...
    { "ok": true, "id": "id150", "rev": "1-rev150" }
  ]
]

Monitoring

See the Monitoring and alerting on the CHT page for how to use this API in production.

GET /api/v1/monitoring

DEPRECATED: use /api/v2/monitoring

Used to retrieve a range of metrics about the instance. While the output is human-readable this is intended for automated monitoring allowing for tracking trends over time and alerting about potential issues.

Permissions

No permissions required.

Examples

JSON format
curl http://localhost:5988/api/v1/monitoring

{"version":{"app":"3.9.0","node":"v10.16.0","couchdb":"2.3.1"},"couchdb":{"medic":{"name":"medic","update_sequence":5733,"doc_count":278,"doc_del_count":32,"fragmentation":1.0283517758420173}...

Response content

JSON pathTypeDescription
version.appStringThe version of the webapp.
version.nodeStringThe version of NodeJS.
version.couchdbStringThe version of CouchDB.
couchdb.<dbname>.nameStringThe name of the db, usually one of “medic”, “medic-sentinel”, “medic-users-meta”, “_users”.
couchdb.<dbname>.update_sequenceNumberThe number of changes in the db.
couchdb.<dbname>.doc_countNumberThe number of docs in the db.
couchdb.<dbname>.doc_del_countNumberThe number of deleted docs in the db.
couchdb.<dbname>.fragmentationNumberThe fragmentation of the db, lower is better, “1” is no fragmentation.
date.currentNumberThe current server date in millis since the epoch, useful for ensuring the server time is correct.
date.uptimeNumberHow long API has been running in seconds.
sentinel.backlogNumberNumber of changes yet to be processed by Sentinel.
messaging.outgoing.state.dueNumberThe number of messages due to be sent.
messaging.outgoing.state.scheduledNumberThe number of messages scheduled to be sent in the future.
messaging.outgoing.state.mutedNumberThe number of messages that are muted and therefore will not be sent.
messaging.outgoing.state.deliveredNumberThe number of messages that have been delivered or sent. As of 3.12.x.
messaging.outgoing.state.failedNumberThe number of messages have failed to be delivered. As of 3.12.x
outbound_push.backlogNumberNumber of changes yet to be processed by Outbound Push.
feedback.countNumberNumber of feedback docs created usually indicative of client side errors.
conflict.countNumberNumber of doc conflicts which need to be resolved manually.
replication_limit.countNumberNumber of users that exceeded the replication limit of documents.
connected_users.countNumberNumber of users that have connected to the api in a given number of days. The period defaults to 7 days but this can be changed by adding connected_user_interval as a query parameter e.g. http://localhost:5988/api/v1/monitoring?connected_user_interval=14

Errors

  • A metric of "" (for string values) or -1 (for numeric values) indicates an error occurred while querying the metric - check the API logs for details.
  • If no response or an error response is received the instance is unreachable. Thus, this API can be used as an uptime monitoring endpoint.

GET /api/v2/monitoring

Available as of 3.12.x. Used to retrieve a range of metrics about the instance. While the output is human-readable this is intended for automated monitoring allowing for tracking trends over time and alerting about potential issues.

Permissions

No permissions required.

Examples

JSON format
curl http://localhost:5988/api/v2/monitoring

{"version":{"app":"3.12.0","node":"v10.16.0","couchdb":"2.3.1"},"couchdb":{"medic":{"name":"medic","update_sequence":5733,"doc_count":278,"doc_del_count":32,"fragmentation":1.0283517758420173}...

Response content

JSON pathTypeDescription
version.appStringThe version of the webapp.
version.nodeStringThe version of NodeJS.
version.couchdbStringThe version of CouchDB.
couchdb.<dbname>.nameStringThe name of the db, usually one of “medic”, “medic-sentinel”, “medic-users-meta”, “_users”.
couchdb.<dbname>.update_sequenceNumberThe number of changes in the db.
couchdb.<dbname>.doc_countNumberThe number of docs in the db.
couchdb.<dbname>.doc_del_countNumberThe number of deleted docs in the db.
couchdb.<dbname>.fragmentationNumberThe fragmentation of the db, lower is better, “1” is no fragmentation.
date.currentNumberThe current server date in millis since the epoch, useful for ensuring the server time is correct.
date.uptimeNumberHow long API has been running.
sentinel.backlogNumberNumber of changes yet to be processed by Sentinel.
messaging.outgoing.total.dueNumberThe number of messages due to be sent.
messaging.outgoing.total.scheduledNumberThe number of messages scheduled to be sent in the future.
messaging.outgoing.total.mutedNumberThe number of messages that are muted and therefore will not be sent.
messaging.outgoing.total.deliveredNumberThe number of messages that have been delivered or sent.
messaging.outgoing.total.failedNumberThe number of messages have failed to be delivered.
messaging.outgoing.seven_days.dueNumberThe number of messages due to be sent in the last seven days.
messaging.outgoing.seven_days.scheduledNumberThe number of messages that were scheduled to be sent in the last seven days.
messaging.outgoing.seven_days.mutedNumberThe number of messages that were due in the last seven days and are muted.
messaging.outgoing.seven_days.deliveredNumberThe number of messages that were due in the last seven days and have been delivered or sent.
messaging.outgoing.seven_days.failedNumberThe number of messages that were due in the last seven days and have failed to be delivered.
messaging.outgoing.last_hundred.pendingObjectCounts within last 100 messages that have received status updates, and are one of the “pending” group statuses
messaging.outgoing.last_hundred.pending.pendingNumberNumber of messages that are pending
messaging.outgoing.last_hundred.pending.forwarded-to-gatewayNumberNumber of messages that are forwarded-to-gateway
messaging.outgoing.last_hundred.pending.received-by-gatewayNumberNumber of messages that are received-by-gateway
messaging.outgoing.last_hundred.pending.forwarded-by-gatewayNumberNumber of messages that are forwarded-by-gateway
messaging.outgoing.last_hundred.finalObjectCounts within last 100 messages that have received status updates, and are in one of the “final” group statuses
messaging.outgoing.last_hundred.final.sentNumberNumber of messages that are sent
messaging.outgoing.last_hundred.final.deliveredNumberNumber of messages that are delivered
messaging.outgoing.last_hundred.final.failedNumberNumber of messages that are failed
messaging.outgoing.last_hundred.mutedObjectCounts within last 100 messages that have received status updates, and are in one of the “muted” group statuses
messaging.outgoing.last_hundred.final.deniedNumberNumber of messages that are denied
messaging.outgoing.last_hundred.final.clearedNumberNumber of messages that are cleared
messaging.outgoing.last_hundred.final.mutedNumberNumber of messages that are muted
messaging.outgoing.last_hundred.final.duplicateNumberNumber of messages that are duplicate
outbound_push.backlogNumberNumber of changes yet to be processed by Outbound Push.
feedback.countNumberNumber of feedback docs created usually indicative of client side errors.
conflict.countNumberNumber of doc conflicts which need to be resolved manually.
replication_limit.countNumberNumber of users that exceeded the replication limit of documents.
connected_users.countNumberNumber of users that have connected to the api in a given number of days. The period defaults to 7 days but this can be changed by adding connected_user_interval as a query parameter e.g. http://localhost:5988/api/v2/monitoring?connected_user_interval=14

Errors

  • A metric of "" (for string values) or -1 (for numeric values) indicates an error occurred while querying the metric - check the API logs for details.
  • If no response or an error response is received the instance is unreachable. Thus, this API can be used as an uptime monitoring endpoint.

GET /api/v1/express-metrics

Added in 4.3.0

Used to retrieve a range of metrics for monitoring CHT API’s performance and internals. This API is used by CHT Watchdog.

The response is formatted for the Prometheus Data Model. The metrics exposed are defined by the prometheus-api-metrics package and include optional default metrics and garbage collection metrics.

Permissions

No permissions required.

Upgrades

All of these endpoints require the can_configure permission.

All of these endpoints are asynchronous. Progress can be followed by watching the horti-upgrade document and looking at the log property.

POST /api/v1/upgrade

Performs a complete upgrade to the provided version. This is equivalent of calling /api/v1/upgrade/stage and then /api/v1/upgrade/complete once staging has finished.

Example

POST /api/v1/upgrade
{
  "build": {
    "namespace": "medic",
    "application": "medic",
    "version": "3.0.0-beta.1"
  }
}

For potential forwards compatibility, you must pass the namespace and application as medic.

The version should correspond to a release, pre-release or branch that has been pushed to our builds server, which is currently hard-coded to https://staging.dev.medicmobile.org/builds. This happens automatically upon a successful Continuous Integration run.

Calling this endpoint will eventually cause api and sentinel to restart.

It is expected that the caller ensures forwards or backwards compatibility is maintained between deployed versions. This endpoint does not stop you from “upgrading” to an earlier version, or a branch that is incompatible with your current state.

POST /api/v1/upgrade/stage

Stages an upgrade to the provided version. Does as much of the upgrade as possible without actually swapping versions over and restarting.

Parameters are the same as /api/v1/upgrade.

You know that an upgrade has been staged when the horti-upgrade document in CouchDB has "action": "stage" and "staging_complete": true.

POST /api/v1/upgrade/complete

Completes a staged upgrade. Throws a 404 if there is no upgrade in the staged position.

Hydrate

Accepts a JSON array of document uuids and returns fully hydrated documents, in the same order in which they were requested. When documents are not found, an entry with the missing uuid and an error is added instead. Supports both GET and POST. Only allowed for users with “online” roles.

GET /api/v1/hydrate

Query parameters

NameRequiredDescription
doc_idstrueA JSON array of document uuids

Example

GET /api/v1/hydrate?doc_ids=["id1","missingId","id3"]
HTTP/1.1 200 OK
Content-Type: application/json

[
  { "id": "id1", "doc": { <...the hydrated document...> } },
  { "id": "missingId1", "error": "not_found" },
  { "id": "id3", "doc": { <...the hydrated document...> } },
]

POST /api/v1/hydrate

Parameters

NameRequiredDescription
doc_idstrueA JSON array of document uuids

Example

POST /api/v1/hydrate
Content-Type: application/json

{
  "doc_ids": ["id1","missingId","id3"]
}
HTTP/1.1 200 OK
Content-Type: application/json

[
  { "id": "id1", "doc": { <...the hydrated document...> } },
  { "id": "missingId", "error": "not_found" },
  { "id": "id3", "doc": { <...the hydrated document...> } },
]

Contacts by phone

Added in 3.10.0

Accepts a phone number parameter and returns fully hydrated contacts that match the requested phone number. If multiple contacts are found, all are returned. When no matches are found, a 404 error is returned. Supports both GET and POST. Only allowed for users with “online” roles.

GET /api/v1/contacts-by-phone

Query parameters

NameRequiredDescription
phonetrueA URL encoded string representing a phone number

Example

GET /api/v1/contacts-by-phone?phone="%2B40-(744)-999-999"
HTTP/1.1 200 OK
Content-Type: application/json

{
  "ok": true,
  "docs": {
    { <... first hydrated matching contact found ...> },
    { <... second hydrated matching contact found ...> },
    ...
  }
}

POST /api/v1/contacts-by-phone

Parameters

NameRequiredDescription
phonetrueA string representing a phone number

Example

POST /api/v1/contacts-by-phone
Content-Type: application/json

{
  "phone": "+40 (21) 222-3333"
}
HTTP/1.1 200 OK
Content-Type: application/json

{
  "ok": true,
  "docs": {
    { <... first hydrated matching contact found ...> },
    { <... second hydrated matching contact found ...> },
    ...
  }
}

Replication Limit

Added in 3.11.0

Returns the quantity of documents that were replicated by each user. Accepts filtering by user name, when not provided, it returns all users. Supports GET. Only allowed for users with “_admin” role.

GET /api/v1/users-doc-count

Query parameters

NameRequiredDescription
userfalseUser’s name

Example

GET /api/v1/users-doc-count?user=mary"
HTTP/1.1 200 OK
Content-Type: application/json

{
    "limit": 10000,
    "users": {
        "_id": "replication-count-mary",
        "_rev": "5-cd3252e852ae075da216c3c3fe461291",
        "user": "mary",
        "date": 1595328973273,
        "count": 58
    }
}

Credentials

Securely store credentials for authentication with third party systems such as SMS aggregators and HMIS. Certain CHT services rely on these credentials when enabled.

PUT /api/v1/credentials

Added in 4.0.0

Provide the credential key as a URL parameter and the password in the request body, for example, to set a credential with key “mykey” and password “my pass” use the following command.

curl -X PUT -H "Content-Type: text/plain" https://<user>:<pass>@<domain>/api/v1/credentials/mykey -d 'my pass'

3.x API

For 3.x deployments credentials are stored in CouchDB configuration in a custom medic-credentials section.

To add the credential to the admin config you need to PUT the value using curl, for example, to set a credential with key “mykey” and password “my pass” use the following command.

curl -X PUT https://<user>:<pass>@<domain>/_node/_local/_config/medic-credentials/mykey -d '"my pass"'

You can also add it via Fauxton:

  • Navigate to the Config screen at https://<domain>/_utils/#/_config
  • Click Add Option
  • The Section should be medic-credentials, the Name should be (in this example) mykey and the value should be the password
  • Click Create
  • You should then be able to see your credential in the list of configuration shown