An Adapt extension for adapt-trackingHub that implements xAPI tracking. It can send xAPI statements to an LRS, and save and load the state to/from the LRS.
It implements several launch methods, so you can use your Adapt course in different xAPI environments:
- Rustici launch (tincan launch)
- ADL xapi-launch
- Spoor (custom method to allow xAPI tracking from within SCORM packages, specific to Adapt with adapt-contrib-spoor)
- Hardcoded (just for testing purposes, not to be used in a real xAPI enviroment).
- acu-splitAP-launch
Important: This extension works with adapt-trackingHub, so it is required that it be installed and enabled in your course.
This extension uses the xAPI Wrapper library provided by ADL.
Install adapt-trackingHub as explained in its documentation. It is recommended to get familiar with that extension and to read its Wiki, since it explains the main ideas behind trackingHub.
With the Adapt CLI installed, run the following from the command line:
adapt install adapt-tkhub-xAPI
To use it with the Authoring Tool, here in the github repository page click on 'Clone or download' and then click on the 'Download ZIP'. Upload that zip file to the Authoring Tool using the Plugin Manager. Within your course, click on 'Manage Extensions' and click on the green 'Add' button that appears to the right of adapt-trackingHub. Then open the 'Configuration Settings' for your course, and find the 'Extensions' section, click on the tkhub-xAPI button and configure it.
If you are using the framework, the settings for this extension should be added to course/config.json, in _tkhub-xAPI
. If you are using the Authoring Tool, the settings are in your coures, in 'Configuration Settings', in the extension titled tkhub xAPI Channel Handler.
Here is the description of the settings:
_isEnabled
: True or false (defaults to true), to enable or disable this extension globally. Useful to turn on or off all the xAPI channels._channels
: An Array that contains objects with the settings for each channel you want to define.
The settings for each channel are:
_name
: An arbitrary name for the channel. It is used for logging to the browser's console, so you can clearly see from which channels the messages are coming._isEnabled
,_reportsEvents
,_tracksState
,_isStateSource
,_isStateStore
,_isLaunchManager
and_ignoreEvents
: These are explained in the adatp-trackingHub documentation. These settings (plus_name
) are common to any type of channel. That is, any channel handler should implement them.
The following settings are specific for the adapt-tkhub-xAPI extension:
_isFakeLRS
: True or false (defaults to false). Setting it to true will cause trackingHub to not attempt to send the statements to the LRS, it will just log the statement to the console, so you can see it. This way, you can develop/test without having a real LRS available._xapiLaunchMethod
: The launch method to use. It depends on how you plan to deploy your Adapt course. It has to be one ofhardcoded
,rustici
,adlxapi
, orspoor
. Depending on which launch method is used, you will need to supply info to some of the following settings or others. Please see the Launch Methods page in the Wiki for more information._homePage
: A URL to a system, used as an identifier. You only need to specify this setting if your channel uses the 'spoor' launch method._mbox
: A string. It can be anything you want, as long as its format is like "mailto:johndoe@example.com". You only need to specify this setting if your channel uses the 'hardcoded' launch method._fullName
: A string. It can be anything you want, just a person's name. You only need to specify this setting if your channel uses the 'hardcoded' launch method._endPoint
: It is the endPoint for the LRS that you want to use. You only need to specify this setting if your channel uses the 'hardcoded' or 'spoor' launch methods._userName
: A username for basic authentication in the LRS. You only need to specfy this setting if your channel uses the 'harcoded' or 'spoor' launch methods._password
: The password for basic authentication in the LRS. You only need to specfy this setting if your channel uses the 'harcoded' or 'spoor' launch methods._isCachedLocally
: True or false (defaults to false). If set to 'true', this channel will store state/statements locally (in the browser) before being sent to the LRS. In the event of connection problem the state will be available through the local cache and statements will be sent when the LRS is reachable._isMobile
: True or false (defaults to false). If set to 'true', this channel will attempt to work despite internet connectivity issues. If unable to retrieve remote state locally cached state or an empty default state will be used._generateIds
: True or false (defaults to true). Controls if statement ids are generated locally (true) or by the LRS (false)
As you can see, some settings are specific to some launch methods. However, there is no harm if you set them all in the configuration, as shown in example.json. The extension will only use the ones it needs, depending on the launch method selected.
Important: The settings related to caching are irrelevant if the launch method is 'adlxapi', and no caching will be done in this environment. The adl-xapi-launch algorithm involves a launch server that provides a proxy to the LRS. The algorithm requires that this server/proxy maintains a pretty tight control on the sessions/launches, apparently accepting only data that was generated in the current session. The proxy also modifies the statements before forwarding them to the LRS (adding some context data, if not already there, and signing the statement!). All this seems to interfere with the normal caching/recovering technique that works with the other launch enviroments (where statements cached from a past session are sent in a newer session). Maybe it's possible to do get it to work, but this is not a priority right now.
About Statement Id generation: According to the xAPI specification:
- Learning Record Providers SHOULD generate the statement Id (an UUID) and put it in the statement before sending it to the LRS, so when the LRS receives it, it will store it (if it is a valid, correct statement) with that id.
- If the LRS receives a statement without an Id, (but the rest of the statement is correct), the LRS should generate and assign an Id to the statement before storing it.
- If the LRS receives a statement with an Id and it already has a statement with that Id, but the statements are not equal (equality as defined in xAPI), then the LRS will NOT store the incoming statement.
There exists the possibility (and it has been experienced by a contributor/user of this extension) that javascript generates Id that it has generated before (something which, in theory, should never happen) thus assigning the 'same' Id to different statements. This would cause the LRS to reject the statement with the 'duplicate id'. A solution to this is to not follow recommendation 1 and just let the LRS generate all Ids.
For this reason, a setting (generateIds) has been provided so that the course author can choose whether to generate Ids locally or not.
For more in-deph information, please visit the Wiki.