Accounts and Users¶
Today, there are two types of accounts: Users
, which are the administrative accounts held by staff members managing the data collection process, and App Users
, which are restricted access keys granted per Form within a Project to data collection clients in the field. Although both of these entities are backed by Actor
s as we explain in the Authentication section above, there is not yet any way to directly create or manipulate an Actor. Today, you can only create, manage, and delete Users and App Users.
Actors (and thus Users) may be granted rights via Roles. The /roles
Roles API is open for all to access, which describes all defined roles on the server. Getting information for an individual role from that same API will reveal which verbs are associated with each role: some role might allow only submission.create
and submission.update
, for example.
Right now, there are four predefined system roles: Administrator (admin
), Project Manager (manager
), Data Collector (formfill
), and App User (app-user
). Administrators are allowed to perform any action upon the server, while Project Managers are allowed to perform any action upon the projects they are assigned to manage.
Data Collectors can see all Forms in a Project and submit to them, but cannot see Submissions and cannot edit Form settings. Similarly, App Users are granted minimal rights: they can read Form data and create new Submissions on those Forms. While Data Collectors can perform these actions directly on the Central administration website by logging in, App Users can only do these things through Collect or a similar data collection client device.
The Roles API alone does not, however, tell you which Actors have been assigned with Roles upon which system objects. For that, you will need to consult the various Assignments resources. There are two, one under the API root (/v1/assignments
), which manages assignments to the entire system, and another nested under each Project (/v1/projects/…/assignments
) which manage assignments to that Project.
Users¶
Presently, it is possible to create and list User
s in the system, as well as to perform password reset operations. In future versions of this API it will be possible to manage existing user information and delete accounts as well.
Listing all Users¶
GET /v1/users
Currently, there are no paging or filtering options, so listing User
s will get you every User in the system, every time.
Optionally, a q
querystring parameter may be provided to filter the returned users by any given string. The search is performed via a trigram similarity index over both the Email and Display Name fields, and results are ordered by match score, best matches first. Note that short search terms (less than 4 or 5 characters) may not return any results. Try a longer search if nothing is appearing.
If a q
parameter is given, and it exactly matches an email address that exists in the system, that user's details will always be returned, even for actors who cannot user.list
. The request must still authenticate as a valid Actor. This allows non-Administrators to choose a user for an action (eg grant rights) without allowing full search.
Actors who cannot user.list
will always receive []
with a 200 OK
response.
Request
Parameters
q (query) |
An optional search parameter. Example: |
Response
HTTP Status: 200
Content Type: application/json
[
{
"createdAt": "2018-04-18T23:19:14.802Z",
"displayName": "My Display Name",
"id": 115,
"type": "user",
"updatedAt": "2018-04-18T23:42:11.406Z",
"deletedAt": "2018-04-18T23:42:11.406Z",
"email": "my.email.address@getodk.org"
}
]
array
|
HTTP Status: 403
Content Type: application/json
{
"code": 403.1,
"message": "The authenticated actor does not have rights to perform that action."
}
object
|
Creating a new User¶
POST /v1/users
All that is required to create a new user is an email address. That email address will receive a message instructing the new user on how to claim their new account and set a password.
Optionally, a password may also be supplied as a part of this request. If it is, the account is immediately usable with the given credentials. However, an email will still be dispatched with claim instructions as above.
Users are not able to do anything upon creation besides log in and change their own profile information. To allow Users to perform useful actions, you will need to assign them one or more Roles.
Request
Request body
{
"email": "my.email.address@getodk.org"
}
object
|
Response
HTTP Status: 200
Content Type: application/json
{
"createdAt": "2018-04-18T23:19:14.802Z",
"displayName": "My Display Name",
"id": 115,
"type": "user",
"updatedAt": "2018-04-18T23:42:11.406Z",
"deletedAt": "2018-04-18T23:42:11.406Z",
"email": "my.email.address@getodk.org"
}
object
|
HTTP Status: 400
Content Type: application/json
{
"code": "400",
"message": "Could not parse the given data (2 chars) as json."
}
object
|
HTTP Status: 403
Content Type: application/json
{
"code": 403.1,
"message": "The authenticated actor does not have rights to perform that action."
}
object
|
Getting User details¶
GET /v1/users/{actorId}
Typically, you supply the integer ID to get information about the user associated with that id.
It is also possible to supply the text current
instead of an integer ID; please see the following endpoint for documentation about this.
Request
Parameters
actorId |
Typically the integer ID of the Example: |
Response
HTTP Status: 200
Content Type: application/json
{
"createdAt": "2018-04-18T23:19:14.802Z",
"displayName": "My Display Name",
"id": 115,
"type": "user",
"updatedAt": "2018-04-18T23:42:11.406Z",
"deletedAt": "2018-04-18T23:42:11.406Z",
"email": "my.email.address@getodk.org"
}
object
|
HTTP Status: 403
Content Type: application/json
{
"code": 403.1,
"message": "The authenticated actor does not have rights to perform that action."
}
object
|
Deleting a User¶
DELETE /v1/users/{actorId}
Upon User deletion:
The account will be removed,
the user will be logged out of all existing sessions,
and should the user attempt to reset their password, they will receive an email informing them that their account has been removed.
The User record will remain on file within the database, so that when for example information about the creator of a Form or Submission is requested, basic details are still available on file. A new User account may be created with the same email address as any deleted accounts.
Request
Parameters
actorId |
Typically the integer ID of the Example: |
Response
HTTP Status: 200
Content Type: application/json
{
"success": true
}
object
|
HTTP Status: 403
Content Type: application/json
{
"code": 403.1,
"message": "The authenticated actor does not have rights to perform that action."
}
object
|
Modifying a User¶
PATCH /v1/users/{actorId}
You can PATCH
JSON data to update User details. Not all user information is modifiable; right now, the following fields may be updated:
displayName
sets the friendly display name the web interface uses to refer to the user.email
sets the email address associated with the account.
When user details are updated, the updatedAt
field will be automatically updated.
Request
Parameters
actorId |
The integer ID of the Example: |
Request body
{
"displayName": "New Name",
"email": "new.email.address@getodk.org"
}
object
|
Response
HTTP Status: 200
Content Type: application/json
{
"createdAt": "2018-04-18T23:19:14.802Z",
"displayName": "My Display Name",
"id": 115,
"type": "user",
"updatedAt": "2018-04-18T23:42:11.406Z",
"deletedAt": "2018-04-18T23:42:11.406Z",
"email": "my.email.address@getodk.org"
}
object
|
HTTP Status: 400
Content Type: application/json
{
"code": "400",
"message": "Could not parse the given data (2 chars) as json."
}
object
|
HTTP Status: 403
Content Type: application/json
{
"code": 403.1,
"message": "The authenticated actor does not have rights to perform that action."
}
object
|
Getting authenticated User details¶
GET /v1/users/current
Typically, you would get User details by the User's numeric Actor ID.
However, if you only have a Bearer token, for example, you don't have any information about the user attached to that session, including even the ID with which to get more information. So you can instead supply the text current
to get the user information associated with the authenticated session.
If you do use current
, you may request extended metadata. Supply an X-Extended-Metadata
header value of true
to additionally receive:
- an array of strings of the
verbs
the authenticated User/Actor is allowed to perform server-wide - an object containing the preferences of the authenticated User/Actor
Request
This endpoint doesn't take any request parameter or data
Response
HTTP Status: 200
Content Type: application/json
{
"createdAt": "2018-04-18T23:19:14.802Z",
"displayName": "My Display Name",
"id": 115,
"type": "user",
"updatedAt": "2018-04-18T23:42:11.406Z",
"deletedAt": "2018-04-18T23:42:11.406Z",
"email": "my.email.address@getodk.org"
}
object
|
HTTP Status: 200
Content Type: application/json; extended
{
"createdAt": "2018-04-18T23:19:14.802Z",
"displayName": "My Display Name",
"id": 115,
"type": "user",
"updatedAt": "2018-04-18T23:42:11.406Z",
"deletedAt": "2018-04-18T23:42:11.406Z",
"email": "my.email.address@getodk.org",
"verbs": [
"project.create",
"project.update"
],
"preferences": {
"site": {
"projectSortMode": "latest"
},
"projects": {
"1": {
"formTrashCollapsed": false
}
}
}
}
object
|
HTTP Status: 403
Content Type: application/json
{
"code": 403.1,
"message": "The authenticated actor does not have rights to perform that action."
}
object
|
Directly updating a user password¶
PUT /v1/users/{actorId}/password
To directly update a user password, you will need to reprove the user's intention by supplying the old
password alongside the new
. If you simply want to initiate an email-based password reset process, see the following endpoint.
Request
Parameters
actorId |
The integer ID of the Example: |
Request body
{
"old": "old.password",
"new": "new.password"
}
object
|
Response
HTTP Status: 200
Content Type: application/json
{
"success": true
}
object
|
HTTP Status: 403
Content Type: application/json
{
"code": 403.1,
"message": "The authenticated actor does not have rights to perform that action."
}
object
|
Initating a password reset¶
POST /v1/users/reset/initiate
Anybody can initate a reset of any user's password. An email will be sent with instructions on how to complete the password reset; it contains a token that is required to complete the process.
The optional query parameter invalidate
may be set to true
to immediately invalidate the user's current password, regardless of whether they complete the reset process. This can be done if, for example, their password has been compromised. In order to do this, though, the request must be performed as an authenticated user with permission to do this. If invalidation is attempted without the proper permissions, the entire request will fail.
If the email address provided does not match any user in the system, that address will still be sent an email informing them of the attempt and that no account was found.
Request
Parameters
invalidate (query) |
Specify Example: |
Request body
{
"email": "my.email.address@getodk.org"
}
object
|
Response
HTTP Status: 200
Content Type: application/json
{
"success": true
}
object
|
HTTP Status: 403
Content Type: application/json
{
"code": 403.1,
"message": "The authenticated actor does not have rights to perform that action."
}
object
|
User Preferences¶
The Central frontend uses this API to save various user preferences, such as the sorting order of certain listings. When a user starts a new session, these preferences are loaded in as part of the User object.
Preferences can be set site-wide or per-project.
Setting a project preference¶
PUT /v1/user-preferences/project/{projectId}/{propertyName}
Request
Parameters
projectId |
The integer ID of the Example: |
propertyName |
The name of the preference. Example: |
Request body
{
"propertyValue": true
}
any |
Response
HTTP Status: 200
Content Type: application/json
{
"success": true
}
object
|
HTTP Status: 403
Content Type: application/json
{
"code": 403.1,
"message": "The authenticated actor does not have rights to perform that action."
}
object
|
HTTP Status: 404
Content Type: application/json
{
"code": 404.1,
"message": "Could not find the resource you were looking for."
}
object
|
Deleting a project preference¶
DELETE /v1/user-preferences/project/{projectId}/{propertyName}
Request
Parameters
projectId |
The integer ID of the Example: |
propertyName |
The name of the preference. Example: |
Response
HTTP Status: 200
Content Type: application/json
{
"success": true
}
object
|
HTTP Status: 403
Content Type: application/json
{
"code": 403.1,
"message": "The authenticated actor does not have rights to perform that action."
}
object
|
HTTP Status: 404
Content Type: application/json
{
"code": 404.1,
"message": "Could not find the resource you were looking for."
}
object
|
Setting a sitewide preference¶
PUT /v1/user-preferences/site/{propertyName}
Request
Parameters
propertyName |
The name of the preference. Example: |
Request body
{
"propertyValue": true
}
any |
Response
HTTP Status: 200
Content Type: application/json
{
"success": true
}
object
|
HTTP Status: 403
Content Type: application/json
{
"code": 403.1,
"message": "The authenticated actor does not have rights to perform that action."
}
object
|
Deleting a sitewide preference¶
DELETE /v1/user-preferences/site/{propertyName}
Request
Parameters
propertyName |
The name of the preference. Example: |
Response
HTTP Status: 200
Content Type: application/json
{
"success": true
}
object
|
HTTP Status: 403
Content Type: application/json
{
"code": 404.1,
"message": "Could not find the resource you were looking for."
}
object
|
App Users¶
App Users may only be created, fetched, and manipulated within the nested Projects subresource, as App Users themselves are limited to the Project in which they are created. Through the App User
s API, you can create, list, and delete the App Users of any given Project. Because they have extremely limited permissions, App Users cannot manage themselves; only User
s may access this API.
For more information about the /projects
containing resource, please see the following section.
Listing all App Users¶
GET /v1/projects/{projectId}/app-users
Currently, there are no paging or filtering options, so listing App User
s will get you every App User in the system, every time.
This endpoint supports retrieving extended metadata; provide a header X-Extended-Metadata: true
to additionally retrieve the lastUsed
timestamp of each App User, as well as to retrieve the details of the Actor
the App User was createdBy
.
Request
Parameters
projectId |
The numeric ID of the Project Example: |
Response
HTTP Status: 200
Content Type: application/json
[
{
"createdAt": "2018-04-18T23:19:14.802Z",
"displayName": "My Display Name",
"id": 115,
"type": "user",
"updatedAt": "2018-04-18T23:42:11.406Z",
"deletedAt": "2018-04-18T23:42:11.406Z",
"token": "d1!E2GVHgpr4h9bpxxtqUJ7EVJ1Q$Dusm2RBXg8XyVJMCBCbvyE8cGacxUx3bcUT",
"projectId": 1,
"lastUsed": "2018-04-14T08:34:21.633Z"
}
]
array
|
HTTP Status: 200
Content Type: application/json; extended
[
{
"createdAt": "2018-04-18T23:19:14.802Z",
"displayName": "My Display Name",
"id": 115,
"type": "user",
"updatedAt": "2018-04-18T23:42:11.406Z",
"deletedAt": "2018-04-18T23:42:11.406Z",
"token": "d1!E2GVHgpr4h9bpxxtqUJ7EVJ1Q$Dusm2RBXg8XyVJMCBCbvyE8cGacxUx3bcUT",
"projectId": 1,
"createdBy": {
"createdAt": "2018-04-18T23:19:14.802Z",
"displayName": "My Display Name",
"id": 115,
"type": "user",
"updatedAt": "2018-04-18T23:42:11.406Z",
"deletedAt": "2018-04-18T23:42:11.406Z"
},
"lastUsed": "2018-04-14T08:34:21.633Z"
}
]
array
|
HTTP Status: 403
Content Type: application/json
{
"code": 403.1,
"message": "The authenticated actor does not have rights to perform that action."
}
object
|
Creating a new App User¶
POST /v1/projects/{projectId}/app-users
The only information required to create a new App User
is its displayName
(this is called "Nickname" in the administrative panel).
When an App User is created, they are assigned no rights. They will be able to authenticate and list forms on a mobile client, but the form list will be empty, as the list only includes Forms that the App User has read access to. Once an App User is created, you'll likely wish to use the Form Assignments resource to actually assign the app-user
role to them for the Forms you wish.
Request
Parameters
projectId |
The numeric ID of the Project Example: |
Request body
{
"displayName": "My Display Name"
}
object
|
Response
HTTP Status: 200
Content Type: application/json
{
"createdAt": "2018-04-18T23:19:14.802Z",
"displayName": "My Display Name",
"id": 115,
"type": "user",
"updatedAt": "2018-04-18T23:42:11.406Z",
"deletedAt": "2018-04-18T23:42:11.406Z",
"token": "d1!E2GVHgpr4h9bpxxtqUJ7EVJ1Q$Dusm2RBXg8XyVJMCBCbvyE8cGacxUx3bcUT",
"projectId": 1
}
object
|
HTTP Status: 400
Content Type: application/json
{
"code": "400",
"message": "Could not parse the given data (2 chars) as json."
}
object
|
HTTP Status: 403
Content Type: application/json
{
"code": 403.1,
"message": "The authenticated actor does not have rights to perform that action."
}
object
|
Deleting an App User¶
DELETE /v1/projects/{projectId}/app-users/{id}
You don't have to delete a App User
in order to cut off its access. Using a User
's credentials you can simply log the App User's session out using its token. This will end its session without actually deleting the App User, which allows you to still see it in the configuration panel and inspect its history. This is what the administrative panel does when you choose to "Revoke" the App User.
That said, if you do wish to delete the App User altogether, you can do so by issuing a DELETE
request to its resource path. App Users cannot delete themselves.
Request
Parameters
id |
The numeric ID of the App User Example: |
projectId |
The numeric ID of the Project Example: |
Response
HTTP Status: 200
Content Type: application/json
{
"success": true
}
object
|
HTTP Status: 403
Content Type: application/json
{
"code": 403.1,
"message": "The authenticated actor does not have rights to perform that action."
}
object
|
Roles¶
(introduced: version 0.5)
The Roles API lists and describes each known Role within the system. Right now, Roles may not be created or customized via the API, but this will likely change in the future.
Each Role contains information about the verbs it allows its assignees to perform. Some Roles have a system name associated with them; the Roles may always be referenced by this system name in request URLs, and system Roles are always read-only.
Listing all Roles¶
GET /v1/roles
Currently, there are no paging or filtering options, so listing Role
s will get you every Role in the system, every time. There are no authorization restrictions upon this endpoint: anybody is allowed to list all Role information at any time.
Request
This endpoint doesn't take any request parameter or data
Response
HTTP Status: 200
Content Type: application/json
[
{
"id": 4,
"name": "Project Manager",
"system": "manager",
"verbs": [
"project.update",
"project.delete"
],
"createdAt": "2018-01-19T23:58:03.395Z",
"updatedAt": "2018-03-21T12:45:02.312Z"
}
]
array
|
Getting Role Details¶
GET /v1/roles/{id}
Getting an individual Role does not reveal any additional information over listing all Roles. It is, however, useful for direct lookup of a specific role:
The id
parameter for Roles here and elsewhere will accept the numeric ID associated with that Role, or a system
name if there is one associated with the Role. Thus, you may request /v1/roles/admin
on any ODK Central server and receive information about the Administrator role.
As with Role listing, there are no authorization restrictions upon this endpoint: anybody is allowed to get information about any Role at any time.
Request
Parameters
id |
Typically the integer ID of the Example: |
Response
HTTP Status: 200
Content Type: application/json
{
"id": 4,
"name": "Project Manager",
"system": "manager",
"verbs": [
"project.update",
"project.delete"
],
"createdAt": "2018-01-19T23:58:03.395Z",
"updatedAt": "2018-03-21T12:45:02.312Z"
}
object
|
Assignments¶
(introduced: version 0.5)
There are multiple Assignments resources. This one, upon the API root (/v1/assignments
), manages Role assignment to the entire system (e.g. if you are assigned a Role that gives you form.create
, you may create a form anywhere on the entire server).
The Project Assignments resource, nested under Projects, manages Role assignment to that Project in particular, and all objects within it. And the Form Assignments resource allows even more granular assignments, to specific Forms within a Project. All of these resources have the same structure and take and return the same data types.
Assignments may be created (POST
) and deleted (DELETE
) like any other resource in the system. Here, creating an Assignment grants the referenced Actor the verbs associated with the referenced Role upon all system objects. The pathing for creation and deletion is not quite REST-standard: we represent the relationship between Role and Actor directly in the URL rather than as body data: assignments/{role}/{actor}
represents the assignment of the given Role to the given Actor.
Listing all Assignments¶
GET /v1/assignments
This will list every server-wide assignment, in the form of actorId
/roleId
pairs. It will not list Project-specific Assignments. To find those, you will need the Assignments subresource within Projects.
This endpoint supports retrieving extended metadata; provide a header X-Extended-Metadata: true
to expand the actorId
into a full actor
objects. The Role reference remains a numeric ID.
Request
This endpoint doesn't take any request parameter or data
Response
HTTP Status: 200
Content Type: application/json
[
{
"actorId": 115,
"roleId": 4
}
]
array
|
HTTP Status: 200
Content Type: application/json; extended
[
{
"actor": {
"createdAt": "2018-04-18T23:19:14.802Z",
"displayName": "My Display Name",
"id": 115,
"type": "user",
"updatedAt": "2018-04-18T23:42:11.406Z",
"deletedAt": "2018-04-18T23:42:11.406Z"
},
"roleId": 4
}
]
array
|
HTTP Status: 403
Content Type: application/json
{
"code": 403.1,
"message": "The authenticated actor does not have rights to perform that action."
}
object
|
Listing all Actors assigned some Role¶
GET /v1/assignments/{roleId}
Given a roleId
, which may be a numeric ID or a string role system
name, this endpoint lists all Actors
that have been assigned that Role on a server-wide basis.
Request
Parameters
roleId |
Typically the integer ID of the Example: |
Response
HTTP Status: 200
Content Type: application/json
[
{
"createdAt": "2018-04-18T23:19:14.802Z",
"displayName": "My Display Name",
"id": 115,
"type": "user",
"updatedAt": "2018-04-18T23:42:11.406Z",
"deletedAt": "2018-04-18T23:42:11.406Z"
}
]
array
|
HTTP Status: 403
Content Type: application/json
{
"code": 403.1,
"message": "The authenticated actor does not have rights to perform that action."
}
object
|
Assigning an Actor to a server-wide Role¶
POST /v1/assignments/{roleId}/{actorId}
Given a roleId
, which may be a numeric ID or a string role system
name, and a numeric actorId
, assigns that Role to that Actor across the entire server.
No POST
body data is required, and if provided it will be ignored.
Request
Parameters
roleId |
Typically the integer ID of the Example: |
actorId |
The integer ID of the Example: |
Response
HTTP Status: 200
Content Type: application/json
{
"success": true
}
object
|
HTTP Status: 403
Content Type: application/json
{
"code": 403.1,
"message": "The authenticated actor does not have rights to perform that action."
}
object
|
Stripping an Role Assignment from an Actor¶
DELETE /v1/assignments/{roleId}/{actorId}
Given a roleId
, which may be a numeric ID or a string role system
name, and a numeric actorId
, unassigns that Role from that Actor across the entire server.
Request
Parameters
roleId |
Typically the integer ID of the Example: |
actorId |
The integer ID of the Example: |
Response
HTTP Status: 200
Content Type: application/json
{
"success": true
}
object
|
HTTP Status: 403
Content Type: application/json
{
"code": 403.1,
"message": "The authenticated actor does not have rights to perform that action."
}
object
|