-
Notifications
You must be signed in to change notification settings - Fork 70
Briefcase Aggregate API
This describes the interfaces ODK Briefcase uses when interacting with ODK Aggregate v1.0.
In general, the server should adhere to the OpenRosa API described here: https://bitbucket.org/javarosa/javarosa/wiki/OpenRosaAPI
Interfaces used during Pull actions.
The APIs used:
- OpenRosa Form Discovery API
- download forms and media files using URLs form discovery.
-
view/submissionList
to obtain a chunk of submission keys. -
view/downloadSubmission
to download an individual submission.
Interfaces used during Push actions.
- 'formUpload' to upload a form and its media files to ODK Aggregate.
-
view/submissionList
to obtain a chunk of submission keys. - OpenRosa Form Submission/Overwite API (with extensions)
Vanilla conformance.
The server does special processing to look for an instanceID
or a submissionDate
attribute on the top-level element of the submission.
If the OpenRosa Metadata block is not present, it uses the instanceID
attribute value as the instance ID for this submission.
If the submissionDate
is present, it uses that value as the submission date for this submission (rather than the current datetime). SubmissionDate should be in ISO8601 format (see below).
The response from the server looks like:
<OpenRosaResponse xmlns="http://openrosa.org/http/response">
<message>...</message>
<submissionMetadata xmlns="http://www.opendatakit.org/xforms"
id="formid" version="..." encrypted="yes" instanceID="..."
submissionDate="..."
isComplete="true" markedAsCompleteDate="..."/>
Note the additional <submissionMetadata/>
tag. This contains all the metadata for this submission that the server has assigned it.
The form id and version are given (version is omitted if null).
If the form is an encrypted form, the encrypted
attribute is present, otherwise it is omitted. Note that if present, it has a
value of "yes" (vs. true
).
The instanceID
is either the server-assigned instance id or the instanceID from the OpenRosa Metadata block.
The submissionDate
is the date of first submission of this record on the server.
The isComplete
flag is true
if the server has all attachments associated with this filled-in form. Otherwise, it will be false
. If it is true
, the markedAsCompleteDate
will be the date when the submission became complete.
The date fields are formatted using:
import org.javarosa.core.model.utils.DateUtils;
DateUtils.formatDateTime(d, DateUtils.FORMAT_ISO8601)
The only 2 fields ODK Briefcase currently uses are the instanceID
and submissionDate
fields. It may eventually use the isComplete
flag.
After a submission has been pushed to the server, it updates the top-level element, inserting the instanceID
and submissionDate
fields if they are not already present.
If they are present, it compares the fields and warns if there is any discrepancy between them.
Nothing special here -- same as the API ODK Collect uses.
ODK Briefcase downloads the forms so that it has the form model available for CSV exports and to ensure that a push uses matching form definitions.
This is FormUploadServlet
in ODK Aggregate.
It is a multipart POST. The POST always contains the form definition file, and has as many media files as would fit in under 10MB. This follows the behavior of the OpenRosa Form Submission/Overwrite API.
Because of the limitations of HTML file input tags, all media files must be in a single folder (no sub-directories). The server assumes this is the case.
Here is a sample web page fragment to post to this servlet:
<form id="ie_backward_compatible_form" action="/formUpload"
enctype="multipart/form-data" encoding="multipart/form-data"
method="POST" accept-charset="UTF-8">
<table id="uploadTable">
<tbody>
<tr>
<td>
<label for="form_def_file">Form definition (xml file):</label>
</td>
<td>
<input id="form_def_file" class="gwt-Button" type="file"
name="form_def_file" size="80">
</td>
</tr>
<tr>
<td>
<label for="mediaFiles">Optional Media file(s):</label>
</td>
<td>
<input id="mediaFiles" class="gwt-Button" type="file"
multiple="" name="datafile" size="80,20">
<input id="clear_media_files" class="gwt-Button"
type="button"
onclick="clearMediaInputField('mediaFiles')"
value="Clear">
</td>
</tr>
<tr>
<td>
<input class="gwt-Button" type="submit"
value="Upload Form" name="button">
</td>
</tr>
</tbody>
</table>
</form>
Download the list of completed submissions for a given form. Incomplete or partial submissions must not be returned. You may wish to report only 'approved' submissions if you have a QA review step.
3 Query arguments:
-
formId
-- identifies the form. Just theid
from the form definition (e.g.,geo_tagger_v2
). -
cursor
-- an opaque string used by the server to as a query resume point. Omit on the first call. -
numEntries
-- the number of entries to return.
The returned XML document is of the form:
<idChunk xmlns="http://opendatakit.org/submissions">
<idList>
<id>uuid:e5aa4247-cfb8-4cc5-87cd-52fbfe491b13</id>
<id>uuid:8ca4433a-fed2-4d1c-b333-c0378203f68f</id>
</idList>
<resumptionCursor>opaquedata</resumptionCursor>
</idChunk>
Where the idList
contains a series of id
elements containing strings that can be used to construct the needed string to pass to the view/downloadSubmission
API.
The resumptionCursor
holds opaque data that is used by the server to track the location at which to resume the list of ids.
ODK Briefcase repeatedly calls this API, passing in the previous response's resumptionCursor
value until the returned resumptionCursor
value matches that given in the request. Once it no longer changes, ODK Briefcase assumes that all id
data has been downloaded from the server.
Download an individual submission.
One Query argument: formId
-- a multipart XPath-like string query that identifies the individual submission.
The formId is a string of the form:
formid[@version=null and @uiVersion=null]/topElement[@key=idvalue]
Where formid
is the form id, as would be provided to the submissionList
API. version
is either null if not used or the version number of the form (ODK Aggregate only supports integer version strings at this time). topElement
is the name of the top-level element in the submission. This is the element nested just within the <instance>
element of the <model>
(it also has the id
attribute, which should be equal to the formid
). And idvalue
is the value of the individual submission id
returned by the view/submissionList
API, usually a UUID.
An example, where no version
was attached to the form definition uploaded to Aggregate:
GET /view/downloadSubmission?formId=my_odk_form[@version=null and @uiVersion=null]/data[@key=uuid:38fd4ef4-28b9-441e-a818-dd8cbe514b2c]
While it is desirable for the idvalue
to be the instanceID
of the submission, it is not required. For ODK Aggregate v1.x, it will be the instance ID; for ODK Aggregate 0.9.x, it is not the instance ID.
Note that formid
may be a URL (since it might be an xmlns), so when parsing this query parameter, it is safest to find the last instance of [@version
and split the string at that location. In any case, it is your string to parse and interpret for your server configuration.
The response is of the form:
<submission xmlns="http://opendatakit.org/submissions"
xmlns:orx="http://openrosa.org/xforms" >
<data>
...reconstructed submission XML...
</data>
<mediaFile>
<fileName>...</fileName>
<hash>md5:...</hash>
<downloadUrl>...</downloadUrl>
</mediaFile>
...repeat as needed...
</submission>
The <mediaFile>
tag has the same interpretation as it does in the OpenRosa Form Listing API's manifest XML.
The reconstructed submission XML generally does not respect the namespaces of the original form definition. As a special case, if it finds a form group that could be interpreted as the OpenRosa Metadata block, it does use the orx
namespace for that.
Like the <submissionMetadata>
tag on the Form Submission/Overwrite API response, the top-level element in the submission XML contains all the metadata fields supplied by the server.