"Copyright 2013 Advanced Distributed Learning (ADL) Initiative, U.S. Department of Defense
Licensed under the Apache License, Version 2.0 (the "License"). You may not use this file except in compliance with the License. You may obtain a copy of the License at http://www.apache.org/licenses/LICENSE-2.0
Unless required by applicable law or agreed to in writing, software distributed under the License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the License for the specific language governing permissions and limitations under the License."
This document was authored by members of the Experience API Working Group (see list in CONTRIBUTING.md) in support of the Office of the Deputy Assistant Secretary of Defense (Readiness) Advanced Distributed Learning (ADL) Initiative. Please send all feedback and inquiries to helpdesk@adlnet.gov
- Part One: About the Experience API
- 1.0. Introduction
- 2.0. How To Use This Document
- 3.0. Serialization and JavaScript Object Notation
- 4.0. Definitions
- 5.0. xAPI Components
- 6.0. Extending xAPI
- 7.0. Profiles and Communities of Practice
- Appendices
- Part Two: Experience API Data
- 1.0. Documents
- 2.0. Statements
- 2.1. Purpose
- 2.2. Formatting Requirements
- 2.3. Statement Lifecycle
- 2.3.1. Statement Immutability
- 2.3.2. Voiding
- 2.4. Statement Properties
- 2.5. Retrieval of Statements
- 2.6. Signed Statements
- 3.0. Metadata
- 3.1. IRI Requirements
- 3.2. Hosted Metadata
- 4.0. Special Data Types and Rules
- 4.1. Extensions
- 4.2. Language Maps
- 4.3. IRIs
- 4.4. UUIDs
- 4.5. ISO 8601 Timestamps
- 4.6. ISO 8601 Durations
- Appendices
- Part Three: Data Processing, Validation, and Security
- 1.0. Requests
- 1.1. HEAD Request Implementation
- 1.2. Headers
- 1.3. Alternate Request Syntax
- 1.4. Encoding
- 1.5. Content Types
- 1.5.1. Application/JSON
- 1.5.2. Multipart/Mixed
- 2.0. Resources
- 2.1. Statement Resource
- 2.2. Documents Resources
- 2.3. State Resource
- 2.4. Agents Resource
- 2.5. Activities Resource
- 2.6. Agent Profile Resource
- 2.7. Activity Profile Resource
- 2.8. About Resource
- 3.0. Data Validation
- 3.1. Concurrency
- 3.2. Error Codes
- 3.3 Versioning
- 4.0. Authentication
- 5.0 Security
- Appendices
- 1.0. Requests
This third part details the more technical side of the Experience API, dealing with how Statements are transferred between Learning Record Provider and LRS. A number of libraries are available for a range of technologies (including JavaScript) which handle this part of the specification. It therefore might not be necessary for Learning Record Providers to fully understand every detail of this part of the specification.
xAPI tracking is done via HTTP Requests from the Learning Record Provider (client) to the LRS (server). This specification offers guidance in some aspects of this communication. Where no guidance is offered, it is recommended that those implementing xAPI use current industry best practices.
The behavior of the LRS in response to PUT, POST, GET and DELETE requests is outlined in Resources below. All resources that support GET requests also support HEAD. The LRS will respond to HEAD requests by returning the meta information only, using the HTTP headers, and not the actual document.
Clients accessing the LRS might need to check if a particular Statement exists, or determine the modification date of documents such as those in State, Activity Profile, or Agent Profile Resources. Particularly for large documents, it is more efficient not to retrieve the entire document just to check its modification date.
- The LRS MUST respond to any HTTP HEAD request as it would have responded to an otherwise
identical HTTP GET request except:
- The message-body MUST be omitted.
- The Content-Length header MAY be omitted, in order to avoid wasting LRS resources.
Some header parameters used within xAPI data transfer are standard HTTP headers. Others are specific to this specification. The following request headers are expected to be used by the Learning Record Providers in some or all of the types of request and situations described in this specification:
- Accept
- Accept-Encoding
- Accept-Language
- Authorization
- Content-Type
- Content-Length
- Content-Transfer-Encoding
- If-Match
- If-None-Match
- X-Experience-API-Version
The following response headers are expected to be used by the LRS. Again, not all of these apply to every type of request and/or situations:
- Content-Type
- Content-Length
- Last-Modified
- ETag
- Status
- X-Experience-API-Version
- X-Experience-API-Consistent-Through
The lists above are not intended to be exhaustive. See requirements throughout this document for more details.
One of the goals of the xAPI is to allow cross-domain tracking, and even though xAPI seeks to enable tracking from applications other than browsers, browsers still need to be supported. For example, Internet Explorer 8 and 9 do not implement Cross Origin Resource Sharing, but rather use their own Cross Domain Request API, which cannot use all of the xAPI as described above due to only supporting "GET" and "POST", and not allowing HTTP headers to be set.
The following describes alternate syntax to use only when unable to use the usual syntax for specific calls due to the restrictions mentioned above. This alternate syntax can also be used to GET Statements due to limits on query string length.
See Appendix C: Cross Domain Request Example for an example.
Method:
- All xAPI requests issued MUST be POST.
- The intended xAPI method MUST be included as the value of the "method" query string parameter.
- The Learning Record Provider MUST NOT include any other query string parameters on the request.
Example: http://example.com/xAPI/statements?method=PUT
Content:
- If the xAPI call involved sending content, the Learning Record Provider MUST URL encode that content and include it as a form parameter called "content".
- The LRS MUST interpret this content as a UTF-8 string. Storing binary data is not supported with this syntax.
Headers:
- The Learning Record Provider MAY include any header parameters required by this specification which are expected to appear in the HTTP header as form parameters with the same names. This applies to the following parameters: Authorization, X-Experience-API-Version, Content-Type, Content-Length, If-Match and If-None-Match. It does not apply to Content-Transfer-Encoding.
- The LRS MUST treat the form parameters listed above as header parameters.
- The Learning Record Provider MUST include other header parameters not listed above in the HTTP header as normal.
- The Learning Record Provider SHOULD* still include a Content-Type header (in the HTTP header) for this type of request with a value of 'application/x-www-form-urlencoded'.
- The Content-Type form parameter SHOULD* specify the content type of the content within the content form parameter.
- The Learning Record Provider SHOULD* still include a Content-Length header (in the HTTP header) for this type of request indicating the overall length of the request's content.
- The Content-Length form parameter SHOULD* specify the length of the content within the content form parameter and will therefore be a lower figure than the length listed in the Content-Length header.
Query string parameters:
- Any query string parameters other than "method" MUST instead be included as a form parameter with the same name.
- The LRS MUST treat any form parameters other than "content" or the header parameters listed above as query string parameters.
Attachments: Note that due to issues relating to encoding, it is not possible to send binary data attachments using this syntax. See Attachments
- The LRS MUST support the syntax above.
There might be cases where there is a requirement for the Learning Record Provider to support applications or browsers where the Client code is hosted on a different scheme (HTTP or HTTPS) from the LRS. A proxy is only needed IF you want to support HTTP to HTTPS requests from IE 9 or lower. You can do HTTP to HTTPS (or HTTPS to HTTP!) without a proxy if you use a modern browser. Two simple solutions might be to 1) set up a proxy pass through on the same scheme as the Client code to the LRS or 2) to host an intermediary server-side LRS on the same scheme as the Client code to route Statements to the target LRS.
Strongly consider security risks before making the decision to use implementations that use HTTP.
- All strings MUST be encoded and interpreted as UTF-8.
Requests and responses within this specification normally use an application/json
content type. Exceptions to this are:
- Documents can have any content type.
- Statement requests that can sometimes include Attachments use the
multipart/mixed
content type.
Requests within this specification normally use an application/json
content type.
- When receiving a PUT or POST with a document type of
application/json
, an LRS MUST accept batches of Statements which contain no Attachment Objects. - When receiving a PUT or POST with a document type of
application/json
, an LRS MUST accept batches of Statements which contain only Attachment Objects with a populated fileUrl.
The multipart/mixed
content type is used for requests that could include Attachments. This does not mean that all
"multipart/mixed" requests necessarily do include Attachments.
-
A Statement request including zero or more Attachments is construed as described below.
-
The Statement is sent using a Content-Type of
multipart/mixed
. Any Attachments are placed at the end of such transmissions. -
The LRS decides whether to accept or reject the Statement based on the information in the first part.
-
If it accepts the request, it can match the raw data of an Attachment(s) with the Attachment header by comparing the SHA-2 of the raw data to the SHA-2 declared in the header. It MUST not do so any other way.
A request transmitting a Statement batch, Statement results, or single Statement that includes Attachments MUST satisfy one of the following criteria:
- It MUST be of type
application/json
and include a fileUrl for every Attachment EXCEPT for Statement results when the "attachments" filter isfalse
. - It MUST conform to the definition of "multipart/mixed" in RFC 2046 and:
- The first part of the multipart document MUST contain the Statements themselves,
with type
application/json
. - Each additional part contains the raw data for an Attachment and forms a logical part of the Statement. This capability will be available when issuing PUT or POST requests against the Statement Resource.
- MUST include an X-Experience-API-Hash parameter in each part's header after the first (Statements) part.
- MUST include a Content-Transfer-Encoding parameter with a value of
binary
in each part's header after the first (Statements) part. - SHOULD only include one copy of an Attachment's data when the same Attachment is used in multiple Statements that are sent together.
- SHOULD include a Content-Type parameter in each part's header. For the first part
(containing the Statement) this MUST be
application/json
. - Where parameters have a corresponding property within the attachment Object (outlined in the table above), and both the parameter and property are specified for a given Attachment, the value of these parameters and properties MUST match.
- The first part of the multipart document MUST contain the Statements themselves,
with type
- An LRS MUST include Attachments in the Transmission Format described above when requested by the Client (see Statement Resource).
- An LRS MUST NOT pull Statements from another LRS without requesting Attachments.
- An LRS MUST NOT push Statements into another LRS without including Attachment data received, if any, for those Attachments.
- When receiving a PUT or POST with a document type of
multipart/mixed
, an LRS MUST accept batches of Statements that contain Attachments in the Transmission Format described above. - When receiving a PUT or POST with a document type of
multipart/mixed
, an LRS MUST reject batches of Statements having Attachments that neither contain a fileUrl nor match a received Attachment part based on their hash. - When receiving a PUT or POST with a document type of
multipart/mixed
, an LRS SHOULD assume a Content-Transfer-Encoding of binary for Attachment parts. - An LRS MAY reject (batches of) Statements that are larger than the LRS is configured to allow.
- When receiving a PUT or POST with a document type of
multipart/mixed
, an LRS SHOULD* accept batches of Statements which contain no Attachment Objects. - When receiving a PUT or POST with a document type of
multipart/mixed
, an LRS SHOULD* accept batches of Statements which contain only Attachment Objects with a populated fileUrl.
Note: There is no requirement that Statement batches using the "mime/multipart" format contain Attachments.
- A Learning Record Provider MAY send Statements with Attachments as described above.
- A Learning Record Provider MAY send multiple Statements where some or all have Attachments if using POST.
- A Learning Record Provider MAY send batches of type
application/json
where every attachment Object has a fileUrl, ignoring all requirements based on the "multipart/mixed" format. - A Learning Record Provider SHOULD use SHA-256, SHA-384, or SHA-512 to populate the "sha2" property.
The File URL is intended to provide a location from which the Attachment can be received. There are, however, no requirements for the owner of the Attachment to make the Attachment data available at the location indefinitely or to make the Attachment publically available without security restrictions. When determining Attachment hosting arrangements, those creating Statements using the "fileUrl" property are encouraged to consider the needs of end recipient(s) of the Statement, especially if the Attachment content is not included with the Statement.
- The Attachment data SHOULD be retrievable at the URL indicated by the fileUrl.
- The owner of the attachment MAY stop providing the attachment data at this IRL at any time.
- Security restrictions MAY be applied to those attempting to access the Attachment data at this IRL.
The period of time an Attachment is made available for, and the security restrictions applied to hosted attachments, are out of scope of this specification.
Below is an example of a very simple Statement with an Attachment. Please note the following:
- The boundary in the sample was chosen to demonstrate valid character classes;
- The selected boundary does not appear in any of the parts;
- For readability the sample attachment is of type "text/plain". Even if it had been a binary type like "image/jpeg", no encoding would be done, the raw octets would be included;
- Per RFC 2046, the boundary is
<CRLF>
followed by -- followed by the boundary string declared in the header.
Note: Don't forget the <CRLF>
when building or parsing these messages.
Headers:
Content-Type: multipart/mixed; boundary="abcABC0123'()+_,-./:=?"
X-Experience-API-Version:1.0.0
Content:
--abcABC0123'()+_,-./:=?
Content-Type:application/json
{
"actor": {
"mbox": "mailto:sample.agent@example.com",
"name": "Sample Agent",
"objectType": "Agent"
},
"verb": {
"id": "http://adlnet.gov/expapi/verbs/answered",
"display": {
"en-US": "answered"
}
},
"object": {
"id": "http://www.example.com/tincan/activities/multipart",
"objectType": "Activity",
"definition": {
"name": {
"en-US": "Multi Part Activity"
},
"description": {
"en-US": "Multi Part Activity Description"
}
}
},
"attachments": [
{
"usageType": "http://example.com/attachment-usage/test",
"display": { "en-US": "A test attachment" },
"description": { "en-US": "A test attachment (description)" },
"contentType": "text/plain; charset=ascii",
"length": 27,
"sha2": "495395e777cd98da653df9615d09c0fd6bb2f8d4788394cd53c56a3bfdcd848a"
}
]
}
--abcABC0123'()+_,-./:=?
Content-Type:text/plain
Content-Transfer-Encoding:binary
X-Experience-API-Hash:495395e777cd98da653df9615d09c0fd6bb2f8d4788394cd53c56a3bfdcd848a
here is a simple attachment
--abcABC0123'()+_,-./:=?--
The LRS is interacted with via RESTful HTTP methods to the resources outlined in this section. The Statement Resource can be used by itself to track learning records. Other resources provide additional functionality.
An LRS will support all of the resources described in this section. It's also possible for a tool which is not an LRS to choose to follow the LRS requirements of one or more of the resources and methods described in this section. For example a tool might implement POST Statements for the purposes of receiving incoming Statements forwarded by an LRS. Such a system is not considered to be an LRS or 'partial LRS'; it is simply not an LRS.
Note: In all of the example endpoints where xAPI resources are located given in the specification, http://example.com/xAPI/
is the example base endpoint of the LRS. All other IRI syntax after this represents the particular resource used.
A full list of the endpoints is included in Appendix B: Table of All Resources.
- The LRS MUST support all of the resources described in this section.
- If the LRS implements OAuth 1.0, the LRS MUST also support all of the OAuth resources described in OAuth Authorization Scope.
- The LRS MAY support additional resources not described in this specification.
- Past, current and future versions of this specification do not and will not define endpoints
with path segments starting with
extensions/
. LRSs supporting additional resources not defined in this specification SHOULD define their endpoints with path segments starting withextensions/
.
The basic communication mechanism of the Experience API.
Example endpoint: http://example.com/xAPI/statements
Stores a single Statement with the given id. POST can also be used to store single Statements.
Content: The Statement object to be stored.
Returns: 204 No Content
Parameter | Type | Default | Description | Required |
---|---|---|---|---|
statementId | String | Id of Statement to record | Required |
-
The LRS MAY respond before Statements that have been stored are available for retrieval.
-
An LRS MUST NOT make any modifications to its state based on receiving a Statement with a statementId that it already has a Statement for. Whether it responds with
409 Conflict
or204 No Content
, it MUST NOT modify the Statement or any other Object. -
If the LRS receives a Statement with an id it already has a Statement for, it SHOULD verify the received Statement matches the existing one and SHOULD return
409 Conflict
if they do not match. See Statement comparison requirements. -
If the LRS receives a batch of Statements containing two or more Statements with the same id, it SHOULD* reject the batch and return
400 Bad Request
.
- Learning Record Providers SHOULD POST Statements including the Statement "id" property instead of using PUT.
- When PUTing Statements, the "id" property of the Statement SHOULD be used.
- Where provided, the "id" property of the Statement MUST match the "statementId" parameter of the request.
Example endpoint: http://example.com/xAPI/statements
Stores a Statement, or a set of Statements.
Content: An array of Statements or a single Statement to be stored.
Returns: 200 OK
, Array of Statement id(s) (UUID) in the same order as the corresponding stored Statements.
- The LRS MAY respond before Statements that have been stored are available for retrieval.
- GET Statements MAY be called using POST and form parameters if necessary as query strings have limits. See Alternate Request Syntax for more details.
- The LRS MUST differentiate a POST to add a Statement or to list Statements based on the parameters passed. See Alternate Request Syntax for more details.
- An LRS MUST NOT make any modifications to its state based on receiving a Statement
with an id that it already has a Statement for. Whether it responds with
409 Conflict
or204 No Content
, it MUST NOT modify the Statement or any other Object. - If the LRS receives a Statement with an id it already has a Statement for, it SHOULD
verify the received Statement matches the existing one and SHOULD return
409 Conflict
if they do not match. See Statement comparison requirements. - If the LRS receives a batch of Statements containing two or more Statements with the same id,
it SHOULD* reject the batch and return
400 Bad Request
.
Example endpoint: http://example.com/xAPI/statements
This method is called to fetch a single Statement or multiple Statements. If the statementId or voidedStatementId parameter is specified a single Statement is returned.
Otherwise returns: A StatementResult Object, a list of Statements in reverse chronological order based on "stored" time, subject to permissions and maximum list length. If additional results are available, an IRL to retrieve them will be included in the StatementResult Object.
Content: None.
Returns: 200 OK
, Statement or Statement Result
Parameter | Type | Default | Description | Required |
---|---|---|---|---|
statementId | String | Id of Statement to fetch | Optional | |
voidedStatementId | String | Id of voided Statement to fetch. see Voided Statements | Optional | |
agent | Agent or Identified Group Object (JSON) | Filter, only return Statements for which the specified Agent or Group is
the Actor or Object of the Statement.
See agent/group Object definition for details. |
Optional | |
verb | Verb id (IRI) | Filter, only return Statements matching the specified Verb id. | Optional | |
activity | Activity id (IRI) | Filter, only return Statements for which the Object of the Statement is an Activity with the specified id. | Optional | |
registration | UUID | Filter, only return Statements matching the specified registration id. Note that although frequently a unique registration will be used for one Actor assigned to one Activity, this cannot be assumed. If only Statements for a certain Actor or Activity are required, those parameters also need to be specified. | Optional | |
related_activities | Boolean | false | Apply the Activity filter broadly. Include Statements for which the Object, any of the context Activities, or any of those properties in a contained SubStatement match the Activity parameter, instead of that parameter's normal behavior. Matching is defined in the same way it is for the "activity" parameter. | Optional |
related_agents | Boolean | false | Apply the Agent filter broadly. Include Statements for which the Actor, Object, Authority, Instructor, Team, or any of these properties in a contained SubStatement match the Agent parameter, instead of that parameter's normal behavior. Matching is defined in the same way it is for the "agent" parameter. | Optional |
since | Timestamp | Only Statements stored since the specified Timestamp (exclusive) are returned. | Optional | |
until | Timestamp | Only Statements stored at or before the specified Timestamp are returned. | Optional | |
limit | Nonnegative Integer | 0 | Maximum number of Statements to return. 0 indicates return the maximum the server will allow. | Optional |
format | String: (ids , exact , or canonical ) |
exact | If ids , only include minimum information necessary in Agent, Activity, Verb
and Group Objects to identify them. For Anonymous Groups this means including
the minimum information needed to identify each member.
If exact , return Agent, Activity, Verb and Group Objects populated exactly as they
were when the Statement was received. An LRS requesting Statements for the purpose
of importing them would use a format of "exact" in order to maintain
Statement Immutability.
If canonical , return Activity Objects and Verbs populated with the canonical
definition of the Activity Objects and Display of the Verbs as determined by the LRS, after
applying the language filtering process defined below,
and return the original Agent and Group Objects as in "exact" mode.
|
Optional |
attachments | Boolean | false | If true , the LRS uses the multipart response format and includes all attachments as
described previously. If false , the LRS sends the prescribed response with Content-Type
application/json and does not send attachment data. |
Optional |
ascending | Boolean | false | If true , return results in ascending order of stored time |
Optional |
Note: The values of Boolean parameters are represented as true
or false
as in JSON.
-
The LRS MUST reject with a
400 Bad Request
error any requests to this resource which contain both statementId and voidedStatementId parameters -
The LRS MUST reject with an
400 Bad Request
error any requests to this resource which contain statementId or voidedStatementId parameters, and also contain any other parameter besides "attachments" or "format". -
The LRS MAY apply additional query filter criteria based on permissions associated with the credentials used.
-
In the event that no Statements are found matching the query filter criteria, the LRS MUST still return
200 OK
and a StatementResult Object. In this case, the "statements" property will contain an empty array. -
The LRS MUST include the header "X-Experience-API-Consistent-Through", in ISO 8601 combined date and time format, on all responses to Statements Resource requests, with a value of the timestamp for which all Statements that have or will have a "stored" property before that time are known with reasonable certainty to be available for retrieval. This time SHOULD take into account any temporary condition, such as excessive load, which might cause a delay in Statements becoming available for retrieval. It is expected that this will be a recent timestamp, even if there are no recently received Statements.
-
If the "attachment" property of a GET Statement is used and is set to
true
, the LRS MUST use the multipart response format and include all Attachments as described in Part Two. -
If the "attachment" property of a GET statement is used and is set to
false
, the LRS MUST NOT include Attachment raw data and MUST reportapplication/json
. -
The LRS SHOULD* include a "Last-Modified" header which matches the "stored" Timestamp of the Statement.
This section outlines rules by which Statements targeting other Statements can sometimes be considered to meet the filter conditions of a query even if they do not match the original query's filter parameters. These rules do not apply when retrieving a single Statement using "statementId" or "voidedStatementId" query parameters.
'Targeting Statements' means that one Statement (the targeting Statement) includes the Statement id of another Statement (the targeted Statement) as a Statement Reference as the Object of the Statement.
For filter parameters which are not time or sequence based (that is, other than "since", "until", or "limit"), Statements which target another Statement (by using a StatementRef as the Object of the Statement) will meet the filter condition if the targeted Statement meets the filter condition.
The time and sequence based parameters MUST still be applied to the Statement making the StatementRef in this manner. This rule applies recursively, so that "Statement a" is a match when a targets b which targets c and the filter conditions described above match for "Statement c".
For example, consider the Statement 'Ben passed explosives training', and a follow up Statement: "Andrew confirmed <StatementRef to original Statement>". The follow up Statement will not mention 'Ben' or 'explosives training', but when fetching Statements with an Actor filter of 'Ben' or an Activity filter of 'explosives training', both Statements match and will be returned so long as they fall into the time or sequence being fetched.
Note: StatementRefs used as a value of the "Statement" property within Context do not affect how Statements are filtered.
-
Activity Objects contain Language Map Objects within their "name", "description" and various interaction components. The LRS MUST return only one language in each of these maps.
-
The LRS MAY maintain canonical versions of language maps against any IRI identifying an object containing language maps. This includes the language map stored in the Verb's "display" property and potentially some language maps used within extensions.
-
If the LRS maintains a canonical version of a language map, it SHOULD* return this canonical language map when canonical format is used to retrieve Statements.
-
The LRS SHOULD* return only one language within each language map for which it returns a canonical map.
-
In order to choose the most relevant language, the LRS MUST apply the "Accept-Language" header as described in RFC 2616 (HTTP 1.1), except that this logic MUST be applied to each language map individually to select which language entry to include, rather than to the resource (list of Statements) as a whole.
Part Two describes the process by which Statements can be voided. This section describes how voided Statements are handled by the LRS when queried.
Clients can identify the presence and Statement id of any voided Statements by the target of the voiding Statement. Aside from debugging tools, many Learning Record Consumers will not want to display voiding Statements to their users and will not display these as part of activity streams and other reports.
-
The LRS MUST not return any Statement which has been voided, unless that Statement has been requested by voidedStatementId. The process described in the section on filter conditions for StatementRefs is no exception to this requirement. The process of retrieving voiding Statements is to request each individually by voidedStatementId.
-
The LRS MUST still return any Statements targeting the voided Statement, following the process and conditions described in the section on filter conditions for StatementRefs. This includes the voiding Statement, which cannot be voided.
The Experience API provides a facility for Learning Record Providers to save arbitrary data in the form of documents, perhaps related to an Activity, Agent, or combination of both.
Note that the following table shows generic properties, not a JSON Object as many other tables in this specification do. The id is stored in the IRL, "updated" is HTTP header information, and "contents" is the HTTP document itself (as opposed to an Object).
Property | Type | Description |
---|---|---|
id | String | Set by Learning Record Provider, unique within the scope of the Agent or Activity. |
updated | Timestamp | When the document was most recently modified. |
contents | Arbitrary binary data | The contents of the document |
The three Document Resources provide document storage. The details of each resource are found in the following sections, and the information in this section applies to all three resources.
Resource | Method | Endpoint | Example |
---|---|---|---|
State Resource | POST | activities/state | http://example.com/xAPI/activities/state |
Activity Profile Resource | POST | activities/profile | http://example.com/xAPI/activities/profile |
Agent Profile Resource | POST | agents/profile | http://example.com/xAPI/agents/profile |
-
A Learning Record Provider MAY send documents to any of the Document Resources for Activities and Agents that the LRS does not have prior knowledge of.
-
The LRS MUST NOT reject documents on the basis of not having prior knowledge of the Activity and/or Agent.
The "Last Modified" header is set by the LRS when returning single or multiple documents in response to a GET request.
- When returning a single document, the LRS SHOULD* include a "Last-Modified" header indicating when the document was last modified.
- When returning multiple documents, the LRS SHOULD* include a "Last-Modified" header indicating when the most recently modified document was last modified.
If a Learning Record Provider stores variables as JSON Objects in a document with content type application/json
,
they can manipulate them as sets of variables using POST.
The following process walks through that process and the process requirements.
For example, a document contains:
{
"x" : "foo",
"y" : "bar"
}
When an LRS receives a POST request with content type application/json
for an existing document also of
content type application/json
, it MUST merge the posted document with the existing document.
In this context, merge is defined as:
- de-serialize the Objects represented by each document.
- for each property directly defined on the Object being posted, set the corresponding property on the existing Object equal to the value from the posted Object.
- store any valid json serialization of the existing Object as the document referenced in the request.
Note that only top-level properties are merged, even if a top-level property is an Object. The entire contents of each original property are replaced with the entire contents of each new property.
For example, this document is POSTed with the same id as the existing document above:
{
"x" : "bash",
"z" : "faz"
}
the resulting document stored in the LRS is:
{
"x" : "bash",
"y" : "bar",
"z" : "faz"
}
-
If the document being posted or any existing document does not have a Content-Type of
application/json
, or if either document cannot be parsed as a JSON Object, the LRS MUST respond with HTTP status code400 Bad Request
, and MUST NOT update the target document as a result of the request. -
If the merge is successful, the LRS MUST respond with HTTP status code
204 No Content
. -
If a Learning Record Provider needs to delete a property, it SHOULD use a PUT request to replace the whole document as described below.
Generally, this is a scratch area for Learning Record Providers that do not have their own internal storage, or need to persist state across devices.
The semantics of the call are driven by the "stateId" parameter. If it is included, the GET and DELETE methods will act upon a single defined state document identified by "stateId". Otherwise, GET will return the available ids, and DELETE will delete all state in the context given through the other parameters. This resource has Concurrency controls associated with it.
Example endpoint: http://example.com/xAPI/activities/state
Stores, changes, fetches, or deletes the document specified by the given "stateId" that exists in the context of the specified Activity, Agent, and registration (if specified).
Content (PUT | POST): The document to be stored or updated.
Content (GET | DELETE): None.
Returns (PUT | POST | DELETE): 204 No Content
Returns (GET): 200 OK
, the State document
Parameter | Type | Description | Required |
---|---|---|---|
activityId | Activity id (IRI) | The Activity id associated with this state. | Required |
agent | Agent Object (JSON) | The Agent associated with this state. | Required |
registration | UUID | The registration associated with this state. | Optional |
stateId | String | The id for this state, within the given context. | Required |
Example endpoint: http://example.com/xAPI/activities/state
Fetches State ids of all state data for this context (Activity + Agent [ + registration if specified]). If "since" parameter is specified, this is limited to entries that have been stored or updated since the specified timestamp (exclusive).
Content: None.
Returns: 200 OK
, Array of State id(s)
Parameter | Type | Description | Required |
---|---|---|---|
activityId | Activity id (IRI) | The Activity id associated with these states. | Required |
agent | Agent object (JSON) | The Agent associated with these states. | Required |
registration | UUID | The Registration associated with these states. | Optional |
since | Timestamp | Only ids of states stored since the specified Timestamp (exclusive) are returned. | Optional |
Example endpoint: http://example.com/xAPI/activities/state
Deletes all state data for this context (Activity + Agent [+ registration if specified]).
Content: None.
Returns: 204 No Content
Parameter | Type | Description | Required |
---|---|---|---|
activityId | Activity id (IRI) | The Activity id associated with this state. | Required |
agent | Agent object (JSON) | The Agent associated with this state. | Required |
registration | UUID | The Registration associated with this state. | Optional |
The Agents Resource provides a method to retrieve a special Object with combined information about an Agent derived from an outside service, such as a directory service. This resource has Concurrency controls associated with it.
Example endpoint: http://example.com/xAPI/agents
Return a special, Person Object for a specified Agent. The Person Object is very similar to an Agent Object, but instead of each attribute having a single value, each attribute has an array value, and it is legal to include multiple identifying properties. This is different from the FOAF concept of person, person is being used here to indicate a person-centric view of the LRS Agent data, but Agents just refer to one persona (a person in one context).
The "agent" parameter is a normal Agent Object with a single identifier and no arrays. It is not a Person Object, nor is it a Group.
Content: None.
Returns: 200 OK
, Person Object
Parameter | Type | Description | Required |
---|---|---|---|
agent | Agent object (JSON) | The Agent representation to use in fetching expanded Agent information. | Required |
- An LRS capable of returning multiple identifying properties for a Person Object SHOULD require the connecting credentials have increased, explicitly given permissions.
- An LRS SHOULD reject insufficiently privileged requests with
403 Forbidden
. - If an LRS does not have any additional information about an Agent to return, the LRS MUST still return a Person Object when queried, but that Person Object will only include the information associated with the requested Agent.
Note: This means that if a request is made for an Agent which the LRS has no prior knowledge of, it will still return a Person object containing the information about the Agent it received in the request.
Property | Type | Description | Required |
---|---|---|---|
objectType | String | Person |
Required |
name | Array of strings. | List of names of Agents retrieved. | Optional |
mbox | Array of IRIs in the form "mailto:email address". | List of e-mail addresses of Agents retrieved. | Optional |
mbox_sha1sum | Array of strings. | List of the SHA1 hashes of mailto IRIs (such as go in an mbox property). | Optional |
openid* | Array of strings. | List of openids that uniquely identify the Agents retrieved. | Optional |
account* | Array of account objects. | List of accounts to match. Complete account Objects (homePage and name) MUST be provided. | Optional |
See also: Agent.
-
All array properties MUST be populated with members with the same definition as the similarly named property from Agent Objects.
-
Additional properties not listed here SHOULD* NOT be added to this object and each property MUST occur only once.
The Activities Resource provides a method to retrieve a full description of an Activity from the LRS. This resource has Concurrency controls associated with it.
Example endpoint: http://example.com/xAPI/activities
Loads the complete Activity Object specified.
Content: None.
Returns: 200 OK
, Content
Parameter | Type | Description | Required |
---|---|---|---|
activityId | Activity id (IRI) | The id associated with the Activities to load. | Required |
- If an LRS does not have a canonical definition of the Activity to return, the LRS SHOULD* still return an Activity Object when queried.
The Agent Profile Resource is much like the State Resource, allowing for arbitrary key / document pairs to be saved which are related to an Agent.
The semantics of the request are driven by the "profileId" parameter. If it is included, the GET method will act upon a single defined document identified by "profileId". Otherwise, GET will return the available ids.
Example endpoint: http://example.com/xAPI/agents/profile
Stores, changes, fetches, or deletes the specified Profile document in the context of the specified Agent.
Content (PUT | POST): The document to be stored or updated.
Content (GET | DELETE): None.
Returns (PUT | POST | DELETE): 204 No Content
Returns (GET): 200 OK
, the Profile document
Parameter | Type | Description | Required |
---|---|---|---|
agent | Agent object (JSON) | The Agent associated with this Profile document. | Required |
profileId | String | The profile id associated with this Profile document. | Required |
Note: The "agent" parameter is an Agent Object and not a Group. Learning Record Providers wishing to store data against an Identified Group can use the Identified Group's identifier within an Agent Object.
Example endpoint: http://example.com/xAPI/agents/profile
Fetches Profile ids of all Profile documents for an Agent. If "since" parameter is specified, this is limited to entries that have been stored or updated since the specified Timestamp (exclusive).
Content: None.
Returns: 200 OK
, Array of Profile id(s)
Parameter | Type | Description | Required |
---|---|---|---|
agent | Agent object (JSON) | The Agent associated with this Profile document. | Required |
since | Timestamp | Only ids of Profiles stored since the specified Timestamp (exclusive) are returned. | Optional |
The Activity Profile Resource is much like the State Resource, allowing for arbitrary key / document pairs to be saved which are related to an Activity.
The semantics of the request are driven by the "profileId" parameter. If it is included, the GET method will act upon a single defined document identified by "profileId". Otherwise, GET will return the available ids.
Example endpoint: http://example.com/xAPI/activities/profile
Stores, changes, fetches, or deletes the specified Profile document in the context of the specified Activity.
Content (PUT | POST): The document to be stored or updated.
Content (GET | DELETE): None.
Returns (PUT | POST | DELETE) 204 No Content
Returns (GET): 200 OK
, the Profile document
Parameter | Type | Description | Required |
---|---|---|---|
activityId | Activity id (IRI) | The Activity id associated with this Profile document. | Required |
profileId | String | The profile id associated with this Profile document. | Required |
Example endpoint: http://example.com/xAPI/activities/profile
Fetches Profile ids of all Profile documents for an Activity. If "since" parameter is specified, this is limited to entries that have been stored or updated since the specified Timestamp (exclusive).
Content: None.
Returns: 200 OK
, Array of Profile id(s)
Parameter | Type | Description | Required |
---|---|---|---|
activityId | Activity id (IRI) | The Activity id associated with these Profile documents. | Required |
since | Timestamp | Only ids of Profile documents stored since the specified Timestamp (exclusive) are returned. | Optional |
Returns JSON Object containing information about this LRS, including the xAPI version supported.
Primarily this resource exists to allow Clients that support multiple xAPI versions to decide which version to use when communicating with the LRS. Extensions are included to allow other uses to emerge.
Example endpoint: http://example.com/xAPI/about
Content: None.
Returns: 200 OK
, JSON object containing basic metadata about this LRS
Property | Type | Description | Required |
---|---|---|---|
version | Array of version strings | xAPI versions this LRS supports | Required |
extensions | Object | A map of other properties as needed | Optional |
- An LRS MUST return the JSON document described above, with a "version" property that includes
the latest minor and patch version the LRS conforms to, for each major version.
- For version 1.0.0 of this specification, this means that
1.0.0
MUST be included;0.9
and0.95
MAY be included. (For the purposes of this requirement,0.9
and0.95
are considered major versions.)
- For version 1.0.0 of this specification, this means that
- Additional properties MUST NOT be added to this object outside of extensions and each property MUST occur only once.
- An LRS SHOULD allow unauthenticated access to this resource.
- An LRS MUST NOT reject requests based on their version header as would otherwise be required by Versioning.
The function of the LRS within the xAPI is to store and retrieve Statements. As long as it has sufficient information to perform these tasks, it is expected that it does them. Validation of Statements in the Experience API is focused solely on syntax, not semantics. Enforcing the rules that ensure valid meaning among Verb definitions, Activity types, and extensions is the responsibility of the Learning Record Provider sending the Statement.
- The LRS SHOULD enforce rules regarding structure.
- The LRS SHOULD NOT enforce rules regarding meaning.
Concurrency control makes certain that a client does not PUT, POST or DELETE documents based on old data into an LRS.
xAPI will use HTTP 1.1 entity tags (ETags) to implement optimistic concurrency control in the portions of the API where PUT, POST or DELETE might overwrite or remove existing data, being:
- State Resource
- Agent Profile Resource
- Activity Profile Resource
The State Resource will permit PUT, POST and DELETE requests without concurrency headers, since state conflicts are unlikely. The requirements below only apply to Agent Profile Resource and Activity Profile Resource.
-
A Client making a PUT request to either the Agent Profile Resource or Activity Profile Resource MUST include the "If-Match" header or the If-None-Match header.
-
A Client making a POST request to either the Agent Profile Resource or Activity Profile Resource SHOULD* include the "If-Match" header or the If-None-Match header.
-
A Client making a DELETE request to either the Agent Profile Resource or Activity Profile Resource SHOULD* include the "If-Match" header.
-
Clients SHOULD* use the ETag value provided by the LRS rather than calculating it themselves.
- An LRS responding to a GET request MUST add an ETag HTTP header to the response.
- An LRS responding to a GET request without using a transfer encoding or using the identity transfer encoding MUST calculate the value of the ETag header to be a hexadecimal string of the SHA-1 digest of the contents. This hexadecimal string SHOULD be rendered using numbers and lowercase characters only; uppercase characters SHOULD NOT be used. The requirement to calculate the ETag this way will be removed in a future version of the specification.
- An LRS responding to a GET request using any non-identity transfer encoding MUST NOT calculate the included ETag as above, due to the interpretation of ETags by existing web infrastructure.
- As defined in RFC 2616, an LRS responding to a GET request MUST enclose the header in quotes.
- An LRS responding to a PUT request MUST handle the "If-Match" header as described in RFC2616, HTTP 1.1 if it contains an ETag, in order to detect modifications made after the Client last fetched the document.
- An LRS responding to a PUT request MUST handle the "If-None-Match" header as described in RFC2616, HTTP 1.1 if it contains "*", in order to to detect when there is a resource present that the Client is not aware of.
- An LRS responding to a POST or DELETE request SHOULD* handle the "If-Match" header as described in RFC2616, HTTP 1.1 if it contains an ETag, in order to detect modifications made after the Client last fetched the document.
- An LRS responding to a POST request SHOULD* handle the "If-None-Match" header as described in RFC2616, HTTP 1.1 if it contains "*", in order to to detect when there is a resource present that the Client is not aware of.
If the header precondition in either of the PUT request cases above fails, the LRS:
- MUST return HTTP status
412 Precondition Failed
. - MUST NOT make a modification to the resource.
If the header precondition in any of the POST or DELETE request cases above fails, the LRS:
- SHOULD* return HTTP status
412 Precondition Failed
. - SHOULD* NOT make a modification to the resource.
If a PUT request is received without either header for a resource that already exists, the LRS:
- MUST return HTTP status
409 Conflict
. - MUST return a response explaining that the Learning Record Provider SHOULD
- check the current state of the resource.
- set the "If-Match" header with the current ETag to resolve the conflict.
- MUST NOT make a modification to the resource.
This specification defines requirements, some of which are imposed on the LRS to accept or reject requests, return responses and perform other behaviors in certain conditions. In cases where an LRS is required to reject a request, the appropriate error code is listed as part of the requirement.
None of these requirements contradict the idea that the LRS is also allowed to be configurable to reject requests and respond or behave differently on the basis of conditions that are out of scope this specification.
One of these conditions is permission. For example, the LRS might assign permissions to a particular set of credentials such that those credentials can only issue Statements relating to a particular agent. It could then reject any statements using those credentials not relating to that agent. The permissions that can be assigned by an LRS are out of scope of this specification, aside from the list of recommended OAuth Authorization scope values in section 4.2.
Permissions can also affect the response returned by an LRS to GET requests. For example, a set of credentials might have permission only to view Statements about a particular Actor, in which case the LRS will filter any returned Statements to exclude any Statements not relating to that Actor. See GET Statements for details.
In cases explicitly allowed by this specification, the credentials used can also affect the LRS behavior in handling a request, for example the LRS will normally overwrite the "authority" property of a Statement, but can sometimes accept a submitted authority if it has a strong trust relationship associated with the credentials used to submit the Statement. See Authority for details.
Permissions set by an LRS could cause a technically conformant LRS to fail conformance testing. This could occur where requests made by the testing software are rejected on the basis of permissions. For this reason the LRS needs to be configurable, or credentials used for testing need to have sufficient permissions granted, such that permission restrictions do not affect the result of conformance testing.
Another condition is where the request sent is beyond the size limits set by the LRS. It would be unreasonable to expect the LRS to always accept requests of any size. The LRS can choose any size limit it sees fit, but needs to be configurable so as not to apply size limits during conformance testing. Of course, some size limits will still exist during conformance testing due to limitations of hardware, etc. but it is expected that these limits are sufficiently high so as not to affect the running of tests.
The LRS can also reject requests or revoke credentials in case of suspected malicious intent, for example an unexpected large number of requests made in a short period of time. It is expected that that limits will be sufficiently high such that the rate of requests made during conformance testing will not trigger any rate limits.
The list below offers some general guidance on HTTP error codes that could be returned from various methods in the API.
-
400 Bad Request
- Indicates an error condition caused by an invalid or missing argument. The term "invalid arguments" includes malformed JSON or invalid Object structures. -
401 Unauthorized
- Indicates that authentication is required, or in the case authentication has been posted in the request, that the given credentials have been refused. -
403 Forbidden
- Indicates that the request is unauthorized for the given credentials. Note this is different than refusing the credentials given. In this case, the credentials have been validated, but the authenticated Client is not allowed to perform the given action. -
404 Not Found
- Indicates the requested resource was not found. May be returned by any method that returns a uniquely identified resource, for instance, any State, Agent Profile, or Activity Profile Resource request targeting a specific document, or the method to retrieve a single Statement. -
409 Conflict
- Indicates an error condition due to a conflict with the current state of a resource, in the case of State Resource, Agent Profile Resource or Activity Profile Resource requests, or in the Statement Resource PUT or POST calls. See Section 3.1 Concurrency for more details. -
412 Precondition Failed
- Indicates an error condition due to a failure of a precondition posted with the request, in the case of State or Agent Profile or Activity Profile API requests. See Section 6.3 Concurrency for more details. -
413 Request Entity Too Large
- Indicates that the LRS has rejected the Statement or document because its size (or the size of an Attachment included in the request) is larger than the maximum allowed by the LRS. -
429 Too Many Requests
- Indicates that the LRS has rejected the request because it has received too many requests from the Client or set of credentials in a given amount of time. -
500 Internal Server Error
- Indicates a general error condition, typically an unexpected exception in processing on the server.
-
An LRS MUST return the error code most appropriate to the error condition from the list above.
-
An LRS SHOULD return a message in the response explaining the cause of the error.
-
An LRS SHOULD use content negotiation as described in RFC 7231 to decide the format of the error.
-
An LRS SHOULD allow for plain text, HTML, and JSON responses for errors (using content negotiation).
-
A Learning Record Provider SHOULD send an "Accept" header with requests to enable content negotiation.
-
The LRS SHOULD* reject any request with
400 Bad Request
status where the content type header does not match the content included in the request or where the structure of the request does not match the structure outlined in this specification for a particular content type. For example, if the content of the request is formatted as JSON, the content type is expected to beapplication/json
. If the content type is application/x-www-form-urlencoded it is expected that the request will include a method parameter as outlined in Alternate Request Syntax. -
The LRS MUST reject with
400 Bad Request
status any requests that use any parameters which the LRS does not recognize in their intended context in this specification. ( Note: LRSs MAY recognize and act on parameters not in this specification). -
The LRS MUST reject with
400 Bad Request
status any requests that use any parameters matching parameters described in this specification in all but case. -
The LRS MUST reject a batch of statements if any Statement within that batch is rejected.
-
The LRS MUST reject with
403 Forbidden
status any request rejected by the LRS where the credentials associated with the request do not have permission to make that request. -
The LRS MUST reject with
413 Request Entity Too Large
status any request rejected by the LRS where the size of the Attachment, Statement or document is larger than the maximum allowed by the LRS. -
The LRS MAY choose any Attachment, Statement and document size limits and MAY vary this limit on any basis, e.g., per authority.
-
The LRS MUST reject with
429 Too Many Requests
status any request rejected by the LRS where the request is rejected due to too many requests being received by a particular Client or set of credentials in a given amount of time. -
The LRS MAY choose any rate limit and MAY vary this limit on any basis, e.g., per authority.
The following requirements exist for the purposes of conformance testing, to ensure that any limitations or permissions implemented by the LRS do not affect the running of conformance testing software.
-
The LRS SHOULD* be configurable not to reject any requests from a particular set of credentials on the basis of permissions. This set of credentials SHOULD* be used for conformance testing but MAY be deleted/deactivated on live systems.
-
The LRS MUST be configurable to accept Attachments, Statements or documents of any reasonable size (see above).
-
The LRS MUST be configurable to accept requests at any reasonable rate.
Future revisions of the specification might introduce changes such as properties added to Statements. Using Semantic Versioning will allow Clients and LRSs to remain interoperable as the specification changes.
Starting with version 1.0.0, xAPI will be versioned according to Semantic Versioning 1.0.0.
Every request from a Client and every response from the LRS includes an HTTP header with the name X-Experience-API-Version
and the version as the value. For example, X-Experience-API-Version : 1.0.3
for version 1.0.3;
see the Revision History for the current version of this specification.
Note: For patch versions of the specification later than 1.0.0, the "X-Experience-API-Version" header will not match the
statement version property which is always 1.0.0
for all 1.0.x versions of the spec. The
"X-Experience-API-Version" header enables the LRS and Client to determine the exact patch version of the specification being
followed. While no communication incompatibility should arise among 1.0.x versions, there are sometimes clarifications
of previously intended behavior.
- The LRS MUST include the "X-Experience-API-Version" header in every response.
- The LRS MUST set this header to the latest patch version.
- The LRS MUST accept requests with a version header of
1.0
as if the version header was1.0.0
. - The LRS MUST reject requests with version header prior to version 1.0.0 unless such requests are routed to a fully conformant implementation of the prior version specified in the header.
- The LRS MUST reject requests without a version header unless such requests are routed to a fully conformant 0.9 implementation.
- The LRS MUST accept requests with a version header starting with
1.0.
if the request is otherwise valid. - The LRS MUST reject requests with a version header of
1.1.0
or greater. - The LRS MUST make these rejects by responding with a
400 Bad Request
error including a short description of the problem.
- The Client MUST include the "X-Experience-API-Version" header in every request.
- The Client MUST set this header to the latest patch version.
- The Client SHOULD tolerate receiving responses with a version of
1.0.0
or greater. - The Client SHOULD tolerate receiving data structures with additional properties.
- The Client SHOULD ignore any properties not defined in version 1.0.0 of the spec.
- Statements of newer versions MUST NOT be converted into a prior version format, e.g., in order to handle version differences.
- Statements of prior versions MAY be converted into a newer version only by following the methods described in Appendix A: Converting Statements to 1.0.0.
In order to balance interoperability and the varying security requirements of different environments, several authentication options are defined.
The following authentication methods are defined within the specification. Any given LRS will implement at least one of these methods and might implement additional methods not defined within this specification.
- OAuth 1.0 (RFC 5849), with signature methods of "HMAC-SHA1", "RSA-SHA1", and "PLAINTEXT"
- HTTP Basic Authentication
- Common Access Cards
While Common Access Cards are defined as an authentication method within this specification, the implementation details of this authentication method are not defined. The xAPI Working Group encourages LRS developers implementing Common Access Cards as an authentication method to collaborate in defining the details of this authentication method in a future version of this specification.
No further details are provided in this specification to describe HTTP Basic Authentication as this authentication method is clearly and completely defined in RFC 7235.
-
The LRS MUST support authentication using at least one of the authentication methods defined in this specification.
-
The LRS MUST handle making, or delegating, decisions on the validity of Statements, and determining what operations might be performed based on the credentials used.
The matrix and requirements below describe the possible authentication scenarios used within OAuth and recommends the authentication workflow to be used in these scenarios. The process described for each scenario is not intended to be comprehensive, but rather outline variations to the standard OAuth workflow.
The requirements in this section only apply if the LRS supports OAuth 1.0.
A registered application is an application that will authenticate to the LRS as an OAuth consumer that has been registered with the LRS.
A known user is a user account on the LRS, or on a system which the LRS trusts to define users.
Known user | User unknown | |
---|---|---|
Application is registered | Standard workflow for OAuth. | LRS trusts application to access xAPI without additional user credentials. OAuth token steps are not invoked |
Application is not registered | The application Agent is not identified as a registered Agent and the LRS cannot make assumptions on its identity. | |
No application | HTTP Basic Authentication is used instead of OAuth, since no application is involved. | |
No authentication | MAY be supported by the LRS, possibly for testing purposes. |
- The LRS MUST record the application's name and a unique consumer key (identifier).
- The LRS MUST provide a mechanism to complete this registration, or delegate to another system that provides such a mechanism.
- The LRS MUST be able to be configured for complete support of the xAPI:
- With any of the methods below.
- In any of the workflow scenarios below.
- The LRS MAY (for security reasons):
- Support a subset of the methods below.
- Limit the known users or registered applications.
- The LRS SHOULD at a minimum supply OAuth with "HMAC-SHA1" and "RSA-SHA1" signatures.
Process: The standard workflow for OAUth 1.0 is used.
Requirements:
- The LRS MUST support the resources in OAuth Authorization Scope to complete the standard OAuth workflow (details not in this specification).
- If this form of authentication is used to record Statements and no authority is specified, the LRS SHOULD record the authority as a Group consisting of an Agent representing the registered application, and an Agent representing the known user.
Process: The LRS honors requests that are signed using OAuth with the registered application's credentials and with an empty token and token secret.
Requirements:
- If this form of authentication is used to record Statements, the LRS SHOULD record the authority as the Agent representing the registered application.
Process: The Learning Record Provider uses a consumer secret consisting of an empty string to call the Temporary Credential Request endpoint specifying the "consumer_name" and other usual parameters. The "consumer_name" contains a string representing the application requesting access.
The Learning Record Provider then sends the user's browser to the Resource Owner Authorization using the temporary credentials obtained from the LRS. The Resource Owner Authorization presents a page displaying the "consumer_name" plus a warning that the identity of the application requesting authorization cannot be verified.
Otherwise the process follows the standard OAuth workflow.
Requirements:
- If this form of authentication is used to record Statements, the LRS MUST record an authority that includes both that application and the authenticating user, as a Group, since OAuth specifies an application.
Process: Use a username/password combination provided by the LRS for use by the known user.
Requirements:
- If this form of authentication is used to record Statements, the LRS SHOULD record the authority as the Agent representing the known user.
- Requests MUST include headers for HTTP Basic Authentication based on a username and password containing zero or more space characters.
- Requests SHOULD* include headers for HTTP Basic Authentication based on a username and password
each consisting of an empty string. In this case the HTTP Basic Authentication header will be
Basic
followed by a base64 encoded version of the string:
. This results in the stringBasic Og==
.
This is in order to distinguish an explicitly unauthenticated request from a request that needs to be given a HTTP Basic Authentication challenge.
These are recommendations for scopes designed to enable an LRS and an application communicating using the xAPI to negotiate a level of access which accomplishes what the application needs while minimizing the potential for misuse. The limitations of each scope are in addition to any security limitations placed on the user account associated with the request.
Elements of this section draw on OAuth 2.0 despite this section describing requirements for LRS supporting OAuth 1.0.
The requirements in this section only apply if the LRS supports OAuth 1.0.
The following table lists xAPI scope values:
Scope | Permission |
---|---|
statements/write | write any Statement |
statements/read/mine | read Statements written by "me", that is with an authority matching what the LRS would assign if writing a Statement with the current token. |
statements/read | read any Statement |
state | read/write state data, limited to Activities and Actors associated with the current token to the extent it is possible to determine this relationship. |
define | (re)Define Activities and Actors. If storing a Statement when this is not granted, ids will be saved and the LRS MAY save the original Statement for audit purposes, but SHOULD NOT update its canonical representation of any Actors or Activities. |
profile | read/write Profile document data, limited to Activities and Actors associated with the current token to the extent it is possible to determine this relationship. |
all/read | unrestricted read access |
all | unrestricted access |
Name | Endpoint | Example |
---|---|---|
Temporary Credential Request | OAuth/initiate | http://example.com/xAPI/OAuth/initiate |
Resource Owner Authorization | OAuth/authorize | http://example.com/xAPI/OAuth/authorize |
Token Request | OAuth/token | http://example.com/xAPI/OAuth/token |
The list of scopes determines the set of permissions that is being requested. For example, an instructor might grant "statements/read" to an application (Client), but the LRS would still limit that tool to Statements that the instructor could read if querying the LRS with their credentials directly (such as Statements relating to their students).
- The LRS MUST accept a scope parameter as defined in OAuth 2.0.
- The LRS MUST assume a requested scope of "statements/write" and "statements/read/mine" if no scope is specified.
- The LRS MUST support the scope of "all" as a minimum.
- The LRS MAY support other scopes.
- The Client SHOULD request only the minimal needed scopes, to increase the chances that the request will be granted.
- The parameters "consumer_name" and "scope" are not part of OAuth 1.0, and therefore if used MUST be passed as query string or form parameters, not in the OAuth header.
Security beyond authentication (including the interpretation of OAuth authorization scopes) is beyond the current scope of this document and left to the individual LRS provider as an implementation detail. Implementors are encouraged to follow industry best practices, e.g., The HTTPS-Only Standard from the office of the White House CIO.
It is possible that security concerns may arise in the implementation of this specification, and implementers might choose to break a conformance requirement for the sake of security. In these cases, implementers are encouraged to consider both the security and interoperability implications of their implementation decisions. In any case, the LRS will still need to be configurable such that it is able to pass conformance tests.
While other security concerns are beyond the scope of this specification, the xAPI Community remains dedicated to determining security best practices. This effort has begun at xAPIsec. Participation is highly encouraged.
This is a 1.0.0 specification, and as such implementers do not have to consider prior versions of the specification. However, prior versions did see notable adoption. This data conversion is specified in order to preserve the data tracked using earlier versions, and make it available to new implementers in a consistent manner.
A 1.0.0 Client or other system converting a Statement created in 0.9 MUST follow the steps below:
- If the Statement has been voided or uses Verbs, Activity types, or properties not included in the 0.9 specification, do not convert it.
- Prefix "verb" with
http://adlnet.gov/expapi/verbs/
. - Prefix any Activity ids which are not full absolute IRIs with
tag:adlnet.gov,2013:expapi:0.9:activities:
- Prefix any extension keys which are not full absolute IRIs with
tag:adlnet.gov,2013:expapi:0.9:extensions:
- Prefix Activity types with
http://adlnet.gov/expapi/activities/
- for each Agent (Actor):
- Search for Inverse Functional Identifiers in this order: "mbox, mbox_sha1sum, openid, account". Keep the first populated Inverse Functional Identifier found and discard the rest.
- For the above Inverse Functional Identifier, take the first element in the array and use that as the value of that Inverse Functional Identifier property, discarding any remaining elements.
- If the "name" property is present, set it equal to the first element in the "name" array, discard the remaining elements.
- Remove all remaining properties.
- Remove the "voided" property from the Statement, if present. Remember, if the value of the
voided property is
true
, then the Statement MUST NOT be converted. - Add
version": "1.0.0
- If an authority was not previously set, set the authority to an Agent identified by an account
with a homePage set to the home page corresponding to the system performing the conversion and an accountName of
unknown
. - If the "statement" property in Context was set, remove it from the Statement.
- Preserve all other properties without modification, including the "stored" property. The "stored" property will still be updated if the Statement is sent to an LRS.
A 1.0.0 Client or other system converting a Statement created in 0.95 MUST follow the steps below:
- If the Statement is voided, do not convert it.
- Remove the "voided" property from the Statement, if present. Remember, if the value of the "voided"
property is
true
, then the Statement MUST NOT be converted. - Add
version": "1.0.0
- If an authority was not previously set, set the authority to an Agent identified by an account
with a homePage set to the home page corresponding to the system performing the conversion and an accountName of
unknown
. - If the Statement property in Context was set to anything other than a StatementRef, remove it from the Statement.
- Preserve all other properties without modification, including the "stored" property. The "stored" property will still be updated if the Statement is sent to an LRS.
A 0.9 Statement:
{
"id": "d1eec41f-1e93-4ed6-acbf-5c4bd0c24269",
"actor": {
"objectType": "Person",
"name": [
"Joe Schmoe",
"Joseph Schmoseph"
],
"mbox": [
"mailto:joe@example.com"
],
"openid": [
"http://openid.com/joe-schmoe"
]
},
"verb": "completed",
"inProgress": false,
"object": {
"objectType": "Activity",
"id": "http://www.example.com/activities/001",
"definition": {
"name": {
"en-US": "Example Activity"
},
"type": "course"
}
},
"result": {
"completion": true
},
"context": {
"instructor": {
"objectType": "Person",
"lastName": [
"Dad"
],
"firstName": [
"Joe's"
],
"mbox": [
"mailto:joesdad@example.com"
]
},
"contextActivities": {
"parent": {
"objectType": "Activity",
"id": "non-absolute-activity-id",
"definition": {
"name": {
"en-US": "Another Activity"
}
}
}
}
},
"timestamp": "2012-06-01T19:09:13.245Z",
"stored": "2012-06-29T15:41:39.165Z"
}
Converted to 1.0.0:
{
"version": "1.0.0",
"id": "d1eec41f-1e93-4ed6-acbf-5c4bd0c24269",
"actor": {
"objectType": "Agent",
"name": "Joe Schmoe",
"mbox": "mailto:joe@example.com"
},
"verb": {
"id": "http://adlnet.gov/expapi/verbs/completed",
"display": {
"en-US": "completed"
}
},
"object": {
"objectType": "Activity",
"id": "http://www.example.com/activities/001",
"definition": {
"name": {
"en-US": "Example Activity"
},
"type": "http://adlnet.gov/expapi/activities/course"
}
},
"result": {
"completion": true
},
"context": {
"instructor": {
"objectType": "Agent",
"mbox": "mailto:joesdad@example.com"
},
"contextActivities": {
"parent": [
{
"objectType": "Activity",
"id": "tag:adlnet.gov,2013:expapi:0.9:activities:non-absolute-activity-id",
"definition": {
"name": {
"en-US": "Another Activity"
}
}
}
]
}
},
"timestamp": "2012-06-01T19:09:13.245Z",
"stored": "2012-06-29T15:41:39.165Z",
"authority": {
"objectType": "Agent",
"account": {
"homePage": "http://www.example.com",
"name": "unknown"
}
}
}
Base Resource Endpoint of the LRS Precedes Each Endpoint | Function |
---|---|
statements | Statement Storage/Retrieval |
agents | Agent Object Storage/Retrieval |
agents/profile | Agent Profile Resource |
activities | Activity Object Storage/Retrieval |
activities/profile | Activity Profile Resource |
activities/state | State Resource |
about | LRS Information |
Base Resource Endpoint of the LRS Precedes Each Endpoint | Function |
---|---|
OAuth/initiate | Temporary Credential Request |
OAuth/authorize | Resource Owner Authorization |
OAuth/token | Token Request |
Alternate Request Syntax outlines alternative syntax for use when the normal syntax cannot be used due to browser or querystring length restrictions. This appendix provides an example of a PUT request to the Statements Resource following this format.
Request using normal syntax:
URL: http://example.com/xAPI/statements
Method: PUT
Query String Parameters:
statementId=c70c2b85-c294-464f-baca-cebd4fb9b348
Request Headers:
Accept:*/*
Accept-Encoding:gzip, deflate, sdch
Accept-Language:en-US,en;q=0.8
Authorization: Basic VGVzdFVzZXI6cGFzc3dvcmQ=
Content-Type: application/json
X-Experience-API-Version: 1.0.3
Content-Length: 351
Content:
{"id":"c70c2b85-c294-464f-baca-cebd4fb9b348","timestamp":"2014-12-29T12:09:37.468Z","actor":{"objectType":"Agent","mbox":"mailto:example@example.com","name":"Test User"},"verb":{"id":"http://adlnet.gov/expapi/verbs/experienced","display":{"en-US":"experienced"}},"object":{"id":"http://example.com/xAPI/activities/myactivity","objectType":"Activity"}}
Request using alternative syntax:
URL: http://example.com/xAPI/statements
Method: POST
Query String Parameters:
method=PUT
Request Headers:
Accept:*/*
Accept-Encoding:gzip, deflate, sdch
Accept-Language:en-US,en;q=0.8
Content-Type: application/x-www-form-urlencoded
Content-Length: 745
Content (with added line breaks and not URL encoded for readability):
statementId=c70c2b85-c294-464f-baca-cebd4fb9b348
&Authorization=Basic VGVzdFVzZXI6cGFzc3dvcmQ=
&X-Experience-API-Version=1.0.3
&Content-Type=application/json
&Content-Length=351
&content={"id":"c70c2b85-c294-464f-baca-cebd4fb9b348","timestamp":"2014-12-29T12:09:37.468Z","actor":{"objectType":"Agent","mbox":"mailto:example@example.com","name":"Test User"},"verb":{"id":"http://adlnet.gov/expapi/verbs/experienced","display":{"en-US":"experienced"}},"object":{"id":"http://example.com/xAPI/activities/myactivity","objectType":"Activity"}}