VOOT x.x is a simple protocol for cross-domain read-only access to information about users and their group membership within an organization or aggregated across organizations. It can be seen as making a subset of LDAP-like information available through a web service.
The API is loosely based on the OpenSocial specification and previous standardization attempts as part of the [GN3-JRA3-T2] project, but this is just for historical reasons and not all requirements of OpenSocial nor previous versions of VOOT are met. One such example is that only the JSON data format is supported. This specification does however aim to be backwards compatible.
The key words "MUST", "MUST NOT", "REQUIRED", "SHALL", "SHALL NOT", "SHOULD", "SHOULD NOT", "RECOMMENDED", "MAY", and "OPTIONAL" in this document are to be interpreted as described in [RFC 2119].
All the use cases that are valid for accessing data from an organization's LDAP are also valid for VOOT. For instance, requesting information about users, their group memberships and the members of a group. VOOT however is not meant to perform user authentication.
Web applications sometimes have the ability to interface with LDAP directly to access information about users in the organization's directory (and authenticate users), but this is notoriously hard or even impossible to get working cross-domain in a secure fashion. This is where VOOT steps in.
This document describes the protocol used between application (client) and VOOT provider. The client uses a REST protocol to interact with the provider. The protocol is protected using a mechanism described in the next section.
+--------+ +----------+ +-----------+
| VOOT | | VOOT | | User |
| Client +--------------+ Provider +------------+ Group |
| | VOOT/OAuth | | LDAP/SQL | Directory |
+--------+ +----------+ +-----------+
An application (client) has two ways to authenticate to the VOOT provider:
- Using Basic Authentication [RFC 2617] if the VOOT provider fully trusts the client as the client will have full access to all data at the provider;
- OAuth 2.0 [RFC 6749] if there is some trust between the VOOT provider and the client where it is left to the user to explicitly authorize the client that wants to access data from the provider. The user gives the client only access to their data, and not to that of other users.
Technically, OAuth 2.0 is an authorization framework and not primarily intedend for authentication, however, from the point of view of the VOOT provider the client authenticates using an access token.
The VOOT API supports three calls:
- Retrieve a list of groups the user is a member of;
- Retrieve the list of people belonging to a certain group;
- Retrieve information about the user.
For the API calls one has to specify the user identifier for which the
information is retrieved. For API calls using OAuth 2.0 the @me
placeholder
identifier MUST be supported which is replaced in the provider with the actual
user identifier that authorized the client to act on its behalf. With this
placeholder the client accessing information about the user does not need to
know the unique identifier of the user at the provider.
For Basic Authentication an actual user identifier and group identifier MUST be
specified, @me
is not supported here. It is out of scope how the client
obtains the identifiers of the users and groups.
If the user john
authorized a client to act on its behalf (with OAuth 2.0),
the following calls have identical results:
Retrieve group membership for the user john
:
/groups/@me
/groups/john
Retrieve group members for the group students
where user john
is a member
of:
/people/@me/students
/people/john/students
Retrieve user information for user john
:
/people/@me
/people/john
For Basic Authentication only the calls that mention the actual user identifier is supported.
This call retrieves a list of groups the user is a member of:
/groups/@me
This call MUST be supported. The result can include the following keys with
information about the groups, where only id
MUST be present:
id
title
description
voot_membership_role
The id
field contains a local (to the provider) unique identifier of the
group. It SHOULD be opaque to the client. The title
field contains the
short human readable name of the group. The description
field can contain a
longer description of the group with possibly its purpose. The field
voot_membership_role
indicates the role the user has in that particular
group. It can be any of these values: admin
, manager
or member
.
This call retrieves a list of all members of a group the user is a member of:
/people/@me/{groupId}
Where {groupId}
is replaced with a group identifier which was obtained
through the call used to retrieve Group Membership for a particular user.
This call MAY be supported. The result can include the following keys with
information about the users that are a member of the group, where only id
MUST be present:
id
displayName
voot_membership_role
The id
field contains a local (to this provider) unique identifier of the
user. It SHOULD be opaque to the client. The displayName
field contains the
name by which the user prefers to be addressed and can possibly be set by the
user themselves at the provider. The displayName
field is OPTIONAL. The field
voot_membership_role
indicates the role the user has in that particular
group. It can be any of these values: admin
, manager
or member
.
The user MUST be a member of the group being queried.
This call retrieves additional information about a user.
/people/@me
This call MAY be supported. The result can include the following keys with
information about the user, where only id
MUST be present:
id
displayName
commonName
emails
The id
field contains a local (to this provider) unique identifier of the
user. It SHOULD be opaque to the client. The displayName
field contains the
name by which the user prefers to be addressed and can possibly be set by the
user themselves at the provider. The displayName
field is OPTIONAL. The
commonName
field contains the official full name of the user. This field
cannot be modified by the user themselves. The emails
field contains a list
of email addresses belonging to the user.
The API calls have three OPTIONAL parameters that manipulate the result obtained from the provider:
sortBy
startIndex
count
The sortBy
parameter determines the key in the result that is used for sorting
the groups or group members. It can be any of the mentioned keys mentioned in
the previous section:
id
displayName
commonName
emails
title
description
voot_membership_role
If the key is not available in the set being sorted, the results are not sorted or sorted at the providers discretion. It is up to the provider whether or not to sort and by what key in what order if these parameters are not present. If the results are to be sorted, the value SHOULD be compared as strings and SHOULD be sorted case-insensitive in ascending order.
The startIndex
parameter determines the offset as an integer >= 0 at which
the start for giving back results. The count
parameter, an integer >= 0
indicates the number of results to be given back. The startIndex
and count
parameters can be used to implement paging by returning only a subset of the
results. These parameters are OPTIONAL, if they are not provided or invalid the
provider MUST consider startIndex
equal to 0
and count
equal to the total
number of items available in the entire set.
The sorting, if requested, MUST be performed on the provider before applying
limiting the results based on the startIndex
and count
parameters.
For the API call requesting user information the sortBy
parameter has no
effect. Using startIndex
and count
is possible, however they are of little
use as there always will be only one answer.
All responses mentioned above have the same JSON format. There are always four keys:
startIndex
itemsPerPage
totalResults
entry
Where startIndex
contains the offset from which the results are returned,
this is usually equals to the requested startIndex
. The itemsPerPage
contains the actual number of results in the set, as part of entry
, returned.
The totalResults
field contains the full number of elements available, not
depending on the startIndex
and count
parameters.
The entry
key contains a list of items, either groups, people or person
information. Below are some examples.
Below are some API examples for retrieve group membership, a list of group members and information about the user.
This is an example of the response to the query:
Host: provider.example.org
GET /groups/@me?sortBy=title HTTP/1.1
The response looks like this:
HTTP/1.1 200 OK
Content-Type: application/json
{
"entry": [
{
"description": "Group containing employees.",
"id": "employees",
"title": "Employees",
"voot_membership_role": "admin"
},
{
"description": "Group containing everyone at this institute.",
"id": "members",
"title": "Members",
"voot_membership_role": "member"
}
],
"itemsPerPage": 2,
"startIndex": "0",
"totalResults": 2
}
This is an example of the response to the query:
Host: provider.example.org
GET /people/@me/members?sortBy=displayName&startIndex=3&count=2 HTTP/1.1
The response looks like this:
HTTP/1.1 200 OK
Content-Type: application/json
{
"entry": [
{
"id": "mwisdom",
"voot_membership_role": "member"
},
{
"id": "bmcatee",
"voot_membership_role": "member"
}
],
"itemsPerPage": 2,
"startIndex": "3",
"totalResults": "7"
}
This is an example of the response to the query:
Host: provider.example.org
GET /people/@me HTTP/1.1
The response looks like this:
HTTP/1.1 200 OK
Content-Type: application/json
{
"entry": {
"commonName": "Mr. Admin I. Strator",
"displayName": "admin",
"emails": [
"admin@example.org",
"postmaster@example.org"
],
"id": "admin"
},
"itemsPerPage": 1,
"startIndex": 0,
"totalResults": 1
}
Handling failures of Authentication, either Basic or Bearer are handled in the
ways described in [RFC 2617] and [RFC 6750]. This will involve sending the
WWW-Authenticate
header if something is wrong, for example an invalid
OAuth 2.0 access token will result in the following response:
HTTP/1.1 401 Unauthorized
WWW-Authenticate: Bearer realm="Resource Server",error="invalid_token",error_description="the access token is not valid"
Content-Type: application/json
{"error":"invalid_token","error_description":"the access token is not valid"}
There are also some request errors defined, i.e.: invalid requests to the
provider that should be dealt with in a certain manner. Only the call that
retrieves group membership MUST be supported, the other calls do not need to be
supported. When this call is disabled a response code of 400 Bad Request
is
returned with error
set to invalid_request
.
The error response is returned as JSON, for example:
HTTP/1.1 404 Not Found
Content-Type: application/json
{
"error": "invalid_user",
}
The error
field MUST be present.
The call looks like this:
/groups/@me
- If Basic Authentication is used and
@me
is used an error response with code404 Not Found
is returned. Theerror
field containsinvalid_user
. If a user identifier is specified instead of@me
for providers not supporting the use of user identifiers the same error is returned; - If the specified user does not exist at the provider an error response with
code
404 Not Found
is returned. Theerror
field containsinvalid_user
; - If any other error occurs an error response with code
500 Internal Server Error
is returned. Theerror
field containsinternal_server_error
.
The call looks like this:
/people/@me/members
- If Basic Authentication is used and
@me
is used an error response with code404 Not Found
is returned. Theerror
field containsinvalid_user
. If a user identifier is specified instead of@me
for providers not supporting the use of user identifiers the same error is returned; - If the specified user does not exist at the provider an error response with
code
404 Not Found
is returned. Theerror
field containsinvalid_user
; - If the specified user is not a member of the group an error response with
code
403 Forbidden
is returned. Theerror
field containsnot_a_member
. This response MUST be returned when the user is not a member, no matter whether the group exists or not; - If any other error occurs an error response with code
500 Internal Server Error
is returned. Theerror
field containsinternal_server_error
.
The call looks like this:
/people/@me
-
If Basic Authentication is used and
@me
is used an error response with code404 Not Found
is returned. Theerror
field containsinvalid_user
. If a user identifier is specified instead of@me
for providers not supporting the use of user identifiers the same error is returned; -
If the specified user does not exist at the provider an error response with code
404 Not Found
is returned. Theerror
field containsinvalid_user
; -
If any other error occurs an error response with code
500 Internal Server Error
is returned. Theerror
field containsinternal_server_error
.
One of the use cases is to make it possible to combine data from various group providers using one API service. This way group membership information can be aggregated from various sources. The proxy provides an OAuth 2.0 protected API to clients and in the backend uses Basic Authentication to talk to the group providers from which it needs to aggregate data.
+-------+ +----------+
| | | VOOT |
| +--------------+ Provider |
| | VOOT/Basic | A |
| VOOT | +----------+
--------------+ Proxy |
VOOT/OAuth | | +----------+
| | | VOOT |
| +--------------+ Provider |
| | VOOT/Basic | B |
+-------+ +----------+
From the client's point of view there should be no difference in the API compared to talking directly to a group provider. There are however some special error cases that should be considered. For instance if (one of) the remote group providers is not available. Also the group identifiers that were scoped locally per group provider need to be modified to include a "scope", i.e. to indicate to what group provider they belong.
For example the user john
, which is a local identifier at a group provider
can occur in multiple group providers, so it needs to be matched to this
particular group provider, for example prefixed with the identifier of the
group provider. The prefixed value SHOULD be opaque to the client as well.
The VOOT protocol is not meant to authenticate users at a service. For this, other mechanisms such as federated identity protocols like SAML, OpenID or BrowserID exists. What needs to be considered here is making sure the identifiers used in the authentication layer are the same as used by the group provider when exposing this information to clients.
In SAML for instance there is the persistent opaque identifier provided to RPs through the NameID value. This is an identifier that remains constant for a particular user per RP. This way, RPs cannot match the same user between themselves based on this identifier alone.
This only needs to be considered when implementing the call that retrieves group members. For just retrieving the group identifiers there can be any mechanism to make the values opaque as this information is typically not provided through the SAML assertion.
ADD MORE STUFF
In order to maintain user privacy only the group membership API call should be allowed by third party clients. The other calls are not needed to determine group membership, e.g. to base application authorization on. If a user is a member of a particular group certain privileges may be granted based on just this fact.
Only the @me
user identifier should be allowed as to avoid needing to obtain
unique user identifiers.
If one makes use of the proxy scenario where the proxy provider is trusted,
Basic Authentication can be used with for instance the local uid
of the user.
The proxy then SHOULD take care of making this information opaque towards the
client and generate new identifiers for the same user for different clients.
The calls to retrieve group members are unacceptable from the point of view of
user privacy. Even is OAuth 2.0 is used this would leak user identifiers of
other people to clients without the users that are a member of this group ever
gave permissions for that. The impact of this can be minimized by making sure
only the unique, opaque user identifier of the user is mentioned in the result
and not other identifying information such as displayName
.
- RFC 2119 Key words for use in RFCs to Indicate Requirement Levels
- RFC 2617 HTTP Authentication: Basic and Digest Access Authentication
- RFC 6749 The OAuth 2.0 Authorization Framework
- RFC 6750 The OAuth 2.0 Authorization Framework: Bearer Token Usage
- Maarten Kremers
<maarten.kremers AT surfnet.nl>
- Andreas Åkre Solberg
<x@example.org>
- Leif Johansson
<y@example.org>
- François Kooman
<francois.kooman@surfnet.nl>