diff --git a/master/CHANGELOG/index.html b/master/CHANGELOG/index.html index 0aed403a1..3038868d9 100644 --- a/master/CHANGELOG/index.html +++ b/master/CHANGELOG/index.html @@ -1956,6 +1956,8 @@
cachetools
to 5.3.2
database
and storage
backends under the unified data
backend
interface [BC]{
"actor": null,
- "id": "80fd65b0-85f8-4b64-a678-bd39624402fe",
+ "id": "292e8ffe-4ed4-460b-b307-16a7a5c14213",
"object": {
"id": "string"
},
@@ -2333,7 +2333,7 @@ PUT /xAPI/statements
{
"actor": null,
- "id": "282bcef8-665d-4e92-ac4a-049c5cfb93e4",
+ "id": "6b700e1f-3e01-4218-85c1-7d3be1b1f319",
"object": {
"id": "string"
},
diff --git a/master/index.html b/master/index.html
index 20f3cf670..012b89588 100644
--- a/master/index.html
+++ b/master/index.html
@@ -944,7 +944,7 @@ Install from PyPIpython3.11 -m venv <path-to-virtual-environment>
+
-
diff --git a/master/search/search_index.json b/master/search/search_index.json
index 6b3656d47..b390ec1d7 100644
--- a/master/search/search_index.json
+++ b/master/search/search_index.json
@@ -1 +1 @@
-{"config":{"lang":["en"],"separator":"[\\s\\-]+","pipeline":["stopWordFilter"]},"docs":[{"location":"","title":"Ralph","text":"
\u2699\ufe0f The ultimate toolbox for your learning analytics (expect some xAPI \u2764\ufe0f)
Ralph is a toolbox for your learning analytics, it can be used as a:
- LRS, a HTTP API server to collect xAPI statements (learning events), following the ADL LRS standard
- command-line interface (CLI), to build data pipelines the UNIX-way\u2122\ufe0f,
- library, to fetch learning events from various backends, (de)serialize or convert them from and to various standard formats such as xAPI, or openedx
"},{"location":"#installation","title":"Installation","text":""},{"location":"#install_from_pypi","title":"Install from PyPI","text":"Ralph is distributed as a standard python package; it can be installed via pip
or any other python package manager (e.g. Poetry, Pipenv, etc.):
Use a virtual environment for installation To maintain a clean and controlled environment when installing ralph-malph
, consider using a virtual environment.
-
Create a virtual environment:
python3.11 -m venv <path-to-virtual-environment>\n
-
Activate the virtual environment:
source venv/bin/activate\n
If you only need to integrate learning statement models feature in your project, you don\u2019t need to install the backends
, cli
or lrs
extra dependencies, the core library is what you need:
pip install ralph-malph\n
If you want to use the Ralph LRS server, add the lrs
flavour in your installation. You also have to choose the type of backend you will use for LRS data storage (backend-clickhouse
,backend-es
,backend-mongo
).
- Install the core package with the LRS and the Elasticsearch backend. For example:
pip install ralph-malph[backend-es,lrs]\n
- Add the
cli
flavour if you want to use the LRS on the command line:
pip install ralph-malph[backend-es,lrs,cli]\n
- If you have various uses for Ralph\u2019s features or would like to discover all the existing functionnalities, it is recommended to install the full package:
pip install ralph-malph[backend-clickhouse,backend-es,backend-ldp,backend-lrs,backend-mongo,backend-s3,backend-swift,backend-ws,cli,lrs]\n
"},{"location":"#install_from_dockerhub","title":"Install from DockerHub","text":"Ralph is distributed as a Docker image. If Docker is installed on your machine, it can be pulled from DockerHub:
docker run --rm -i fundocker/ralph:latest ralph --help\n
Use a ralph
alias in your local environment Simplify your workflow by creating an alias for easy access to Ralph commands:
alias ralph=\"docker run --rm -i fundocker/ralph:latest ralph\"\n
"},{"location":"#wip_start_using_ralph","title":"[WIP] Start using Ralph","text":""},{"location":"#contributing_to_ralph","title":"Contributing to Ralph","text":"If you\u2019re interested in contributing to Ralph, whether it\u2019s by reporting issues, suggesting improvements, or submitting code changes, please head over to our dedicated Contributing to Ralph page. There, you\u2019ll find detailed guidelines and instructions on how to take part in the project.
We look forward to your contributions and appreciate your commitment to making Ralph a more valuable tool for everyone.
"},{"location":"#contributors","title":"Contributors","text":""},{"location":"#license","title":"License","text":"This work is released under the MIT License (see LICENSE).
"},{"location":"CHANGELOG/","title":"Changelog","text":"All notable changes to this project will be documented in this file.
The format is based on Keep a Changelog, and this project adheres to Semantic Versioning.
"},{"location":"CHANGELOG/#unreleased","title":"Unreleased","text":""},{"location":"CHANGELOG/#added","title":"Added","text":" - Implement Pydantic model for LRS Statements resource query parameters
- Implement xAPI LMS Profile statements validation
EdX
to xAPI
converters for enrollment events - Backends: Add
Writable
and Listable
interfaces to distinguish supported functionalities among data
backends - Backends: Add
get_backends
function to automatically discover backends for CLI and LRS usage
"},{"location":"CHANGELOG/#changed","title":"Changed","text":" - Upgrade
cachetools
to 5.3.2
- Refactor
database
and storage
backends under the unified data
backend interface [BC] - Refactor LRS
query_statements
and query_statements_by_ids
backends methods under the unified lrs
backend interface [BC] - Refactor LRS Statements resource query parameters defined for
ralph
API - User credentials must now include an \u201cagent\u201d field which can be created using the cli
GET /statements
now has \u201cmine\u201d option which matches statements that have an authority field matching that of the user - CLI: change
push
to write
and fetch
to read
[BC] - Upgrade
fastapi
to 0.104.1
- Upgrade
sentry_sdk
to 1.34.0
- Upgrade
uvicorn
to 0.24.0.post1
- API: Invalid parameters now return 400 status code
- API: Forwarding PUT now uses PUT (instead of POST)
- Models: The xAPI
context.contextActivities.category
field is now mandatory in the video and virtual classroom profiles. [BC] - Backends:
LRSHTTP
methods must not be used in asyncio
events loop (BC) - Add variable to override PVC name in arnold deployment
- Backends: add
max_statements
option to AsyncLRSHTTP
- API: Incoming statements are enriched with
id
, timestamp
, stored
and authority
- Backends: update
statementId
and voidedStatementId
to snake_case, with camelCase alias, in LRSStatementsQuery
- API: Add
RALPH_LRS_RESTRICT_BY_AUTHORITY
option making ?mine=True
implicit - CLI: list cli usage strings in alphabetical order
- API: Add
RALPH_LRS_RESTRICT_BY_SCOPE
option enabling endpoint access control by user scopes - Backends: Replace reference to a JSON column in ClickHouse with function calls on the String column [BC]
- API: enhance \u2018limit\u2019 query parameter\u2019s validation
- API: Variable
RUNSERVER_AUTH_BACKEND
becomes RUNSERVER_AUTH_BACKENDS
, and multiple authentication methods are supported simultaneously
"},{"location":"CHANGELOG/#fixed","title":"Fixed","text":" - API: Fix a typo (\u2018attachements\u2019 -> \u2018attachments\u2019) to ensure compliance with the LRS specification and prevent potential silent bugs.
"},{"location":"CHANGELOG/#removed","title":"Removed","text":" school
, course
, module
context extensions in Edx to xAPI base converter name
field in VideoActivity
xAPI model mistakenly used in video
profile - drop support for python 3.7
"},{"location":"CHANGELOG/#390_-_2023-07-21","title":"3.9.0 - 2023-07-21","text":""},{"location":"CHANGELOG/#changed_1","title":"Changed","text":" - Upgrade
fastapi
to 0.100.0
- Upgrade
sentry_sdk
to 1.28.1
- Upgrade
uvicorn
to 0.23.0
- Enforce valid IRI for
activity
parameter in GET /statements
- Change how duplicate xAPI statements are handled for
clickhouse
backend
"},{"location":"CHANGELOG/#380_-_2023-06-21","title":"3.8.0 - 2023-06-21","text":""},{"location":"CHANGELOG/#added_1","title":"Added","text":" - Implement edX open response assessment events pydantic models
- Implement edx peer instruction events pydantic models
- Implement xAPI VideoDownloaded pydantic model (using xAPI TinCan
downloaded
verb)
"},{"location":"CHANGELOG/#changed_2","title":"Changed","text":" - Allow to use a query for HTTP backends in the CLI
"},{"location":"CHANGELOG/#370_-_2023-06-13","title":"3.7.0 - 2023-06-13","text":""},{"location":"CHANGELOG/#added_2","title":"Added","text":" - Implement asynchronous
async_lrs
backend - Implement synchronous
lrs
backend - Implement xAPI virtual classroom pydantic models
- Allow to insert custom endpoint url for S3 service
- Cache the HTTP Basic auth credentials to improve API response time
- Support OpenID Connect authentication method
"},{"location":"CHANGELOG/#changed_3","title":"Changed","text":" - Clean xAPI pydantic models naming convention
- Upgrade
fastapi
to 0.97.0
- Upgrade
sentry_sdk
to 1.25.1
- Set Clickhouse
client_options
to a dedicated pydantic model - Upgrade
httpx
to 0.24.1
- Force a valid (JSON-formatted) IFI to be passed for the
/statements
GET query agent
filtering - Upgrade
cachetools
to 5.3.1
"},{"location":"CHANGELOG/#removed_1","title":"Removed","text":" verb
.display
field no longer mandatory in xAPI models and for converter
"},{"location":"CHANGELOG/#360_-_2023-05-17","title":"3.6.0 - 2023-05-17","text":""},{"location":"CHANGELOG/#added_3","title":"Added","text":" - Allow to ignore health check routes for Sentry transactions
"},{"location":"CHANGELOG/#changed_4","title":"Changed","text":" - Upgrade
sentry_sdk
to 1.22.2
- Upgrade
uvicorn
to 0.22.0
- LRS
/statements
GET
method returns a code 400 with certain parameters as per the xAPI specification - Use batch/v1 api in cronjob_pipeline manifest
- Use autoscaling/v2 in HorizontalPodAutoscaler manifest
"},{"location":"CHANGELOG/#fixed_1","title":"Fixed","text":" - Fix the
more
IRL building in LRS /statements
GET requests
"},{"location":"CHANGELOG/#351_-_2023-04-18","title":"3.5.1 - 2023-04-18","text":""},{"location":"CHANGELOG/#changed_5","title":"Changed","text":" - Upgrade
httpx
to 0.24.0
- Upgrade
fastapi
to 0.95.1
- Upgrade
sentry_sdk
to 1.19.1
- Upgrade
uvicorn
to 0.21.1
"},{"location":"CHANGELOG/#fixed_2","title":"Fixed","text":" - An issue with starting Ralph in pre-built Docker containers
- Fix double quoting in ClickHouse backend server parameters
- An issue Ralph starting when ClickHouse is down
"},{"location":"CHANGELOG/#350_-_2023-03-08","title":"3.5.0 - 2023-03-08","text":""},{"location":"CHANGELOG/#added_4","title":"Added","text":" - Implement PUT verb on statements endpoint
- Add ClickHouse database backend support
"},{"location":"CHANGELOG/#changed_6","title":"Changed","text":" - Make trailing slashes optional on statements endpoint
- Upgrade
sentry_sdk
to 1.16.0
"},{"location":"CHANGELOG/#340_-_2023-03-01","title":"3.4.0 - 2023-03-01","text":""},{"location":"CHANGELOG/#changed_7","title":"Changed","text":" - Upgrade
fastapi
to 0.92.0
- Upgrade
sentry_sdk
to 1.15.0
"},{"location":"CHANGELOG/#fixed_3","title":"Fixed","text":" - Restore sentry integration in the LRS server
"},{"location":"CHANGELOG/#330_-_2023-02-03","title":"3.3.0 - 2023-02-03","text":""},{"location":"CHANGELOG/#added_5","title":"Added","text":" - Restore python 3.7+ support for library usage (models)
"},{"location":"CHANGELOG/#changed_8","title":"Changed","text":" - Allow xAPI extra fields in
extensions
fields
"},{"location":"CHANGELOG/#321_-_2023-02-01","title":"3.2.1 - 2023-02-01","text":""},{"location":"CHANGELOG/#changed_9","title":"Changed","text":" - Relax required Python version to
3.7
+
"},{"location":"CHANGELOG/#320_-_2023-01-25","title":"3.2.0 - 2023-01-25","text":""},{"location":"CHANGELOG/#added_6","title":"Added","text":" - Add a new
auth
subcommand to generate required credentials file for the LRS - Implement support for AWS S3 storage backend
- Add CLI
--version
option
"},{"location":"CHANGELOG/#changed_10","title":"Changed","text":" - Upgrade
fastapi
to 0.89.1
- Upgrade
httpx
to 0.23.3
- Upgrade
sentry_sdk
to 1.14.0
- Upgrade
uvicorn
to 0.20.0
- Tray: add the
ca_certs
path for the ES backend client option (LRS) - Improve Sentry integration for the LRS
- Update handbook link to
https://handbook.openfun.fr
- Upgrade base python version to 3.11 for the development stack and Docker image
"},{"location":"CHANGELOG/#fixed_4","title":"Fixed","text":" - Restore ES and Mongo backends ability to use client options
"},{"location":"CHANGELOG/#310_-_2022-11-17","title":"3.1.0 - 2022-11-17","text":""},{"location":"CHANGELOG/#added_7","title":"Added","text":" - EdX to xAPI converters for video events
"},{"location":"CHANGELOG/#changed_11","title":"Changed","text":" - Improve Ralph\u2019s library integration by unpinning dependencies (and prefer ranges)
- Upgrade
fastapi
to 0.87.0
"},{"location":"CHANGELOG/#removed_2","title":"Removed","text":" - ModelRules constraint
"},{"location":"CHANGELOG/#300_-_2022-10-19","title":"3.0.0 - 2022-10-19","text":""},{"location":"CHANGELOG/#added_8","title":"Added","text":" - Implement edX video browser events pydantic models
- Create a
post
endpoint for statements implementing the LRS spec - Implement support for the MongoDB database backend
- Implement support for custom queries when using database backends
get
method (used in the fetch
command) - Add dotenv configuration file support and
python-dotenv
dependency - Add
host
and port
options for the runserver
cli command - Add support for database selection when running the Ralph LRS server
- Implement support for xAPI statement forwarding
- Add database backends
status
checking - Add
health
LRS router - Tray: add LRS server support
"},{"location":"CHANGELOG/#changed_12","title":"Changed","text":" - Migrate to
python-legacy
handler for mkdocstrings
package - Upgrade
click
to 8.1.3
- Upgrade
elasticsearch
to 8.3.3
- Upgrade
fastapi
to 0.79.1
- Upgrade
ovh
to 1.0.0
- Upgrade
pydantic
to 1.9.2
- Upgrade
pymongo
to 4.2.0
- Upgrade
python-keystoneclient
to 5.0.0
- Upgrade
python-swiftclient
to 4.0.1
- Upgrade
requests
to 2.28.1
- Upgrade
sentry_sdk
to 1.9.5
- Upgrade
uvicorn
to 0.18.2
- Upgrade
websockets
to 10.3
- Make backends yield results instead of writing to standard streams (BC)
- Use pydantic settings management instead of global variables in defaults.py
- Rename backend and parser parameter environment variables (BC)
- Make project dependencies management more modular for library usage
"},{"location":"CHANGELOG/#removed_3","title":"Removed","text":" - Remove YAML configuration file support and
pyyaml
dependency (BC)
"},{"location":"CHANGELOG/#fixed_5","title":"Fixed","text":" - Tray: do not create a cronjobs list when no cronjob has been defined
- Restore history mixin logger
"},{"location":"CHANGELOG/#210_-_2022-04-13","title":"2.1.0 - 2022-04-13","text":""},{"location":"CHANGELOG/#added_9","title":"Added","text":" - Implement edX problem interaction events pydantic models
- Implement edX textbook interaction events pydantic models
ws
websocket stream backend (compatible with the fetch
command) - bundle
jq
, curl
and wget
in the fundocker/ralph
Docker image - Tray: enable ralph app deployment command configuration
- Add a
runserver
command with basic auth and a Whoami
route - Create a
get
endpoint for statements implementing the LRS spec - Add optional fields to BaseXapiModel
"},{"location":"CHANGELOG/#changed_13","title":"Changed","text":" - Upgrade
uvicorn
to 0.17.4
- Upgrade
elasticsearch
to 7.17.0
- Upgrade
sentry_sdk
to 1.5.5
- Upgrade
fastapi
to 0.73.0
- Upgrade
pyparsing
to 3.0.7
- Upgrade
pydantic
to 1.9.0
- Upgrade
python-keystoneclient
to 4.4.0
- Upgrade
python-swiftclient
to 3.13.0
- Upgrade
pyyaml
to 6.0
- Upgrade
requests
to 2.27.1
- Upgrade
websockets
to 10.1
"},{"location":"CHANGELOG/#201_-_2021-07-15","title":"2.0.1 - 2021-07-15","text":""},{"location":"CHANGELOG/#changed_14","title":"Changed","text":" - Upgrade
elasticsearch
to 7.13.3
"},{"location":"CHANGELOG/#fixed_6","title":"Fixed","text":" - Restore elasticsearch backend datastream compatibility for bulk operations
"},{"location":"CHANGELOG/#200_-_2021-07-09","title":"2.0.0 - 2021-07-09","text":""},{"location":"CHANGELOG/#added_10","title":"Added","text":" - xAPI video
interacted
pydantic models - xAPI video
terminated
pydantic models - xAPI video
completed
pydantic models - xAPI video
seeked
pydantic models - xAPI video
initialized
pydantic models - xAPI video
paused
pydantic models convert
command to transform edX events to xAPI format - EdX to xAPI converters for page
viewed
andpage_close
events - Implement core event format converter
- xAPI video
played
pydantic models - xAPI page
viewed
and page terminated
pydantic models - Implement edX navigational events pydantic models
- Implement edX enrollment events pydantic models
- Install security updates in project Docker images
- Model selector to retrieve associated pydantic model of a given event
validate
command to lint edX events using pydantic models - Support all available bulk operation types for the elasticsearch backend (create, index, update, delete) using the
--es-op-type
option
"},{"location":"CHANGELOG/#changed_15","title":"Changed","text":" - Upgrade
elasticsearch
to 7.13.2
- Upgrade
python-swiftclient
to 3.12.0
- Upgrade
click
to 8.0.1
- Upgrade
click-option-group
to 0.5.3
- Upgrade
pydantic
to 1.8.2
- Upgrade
sentry_sdk
to 1.1.0
- Rename edX models
- Migrate model tests from factories to hypothesis strategies
- Tray: switch from openshift to k8s (BC)
- Tray: remove useless deployment probes
"},{"location":"CHANGELOG/#fixed_7","title":"Fixed","text":" - Tray: remove
version
immutable field in DC selector
"},{"location":"CHANGELOG/#120_-_2021-02-26","title":"1.2.0 - 2021-02-26","text":""},{"location":"CHANGELOG/#added_11","title":"Added","text":" - edX server event pydantic model and factory
- edX page_close browser event pydantic model and factory
- Tray: allow to specify a self-generated elasticsearch cluster CA certificate
"},{"location":"CHANGELOG/#fixed_8","title":"Fixed","text":" - Tray: add missing Swift variables in the secret
- Tray: fix pods anti-affinity selector
"},{"location":"CHANGELOG/#removed_4","title":"Removed","text":" pandas
is no longer required
"},{"location":"CHANGELOG/#110_-_2021-02-04","title":"1.1.0 - 2021-02-04","text":""},{"location":"CHANGELOG/#added_12","title":"Added","text":" - Support for Swift storage backend
- Use the
push
command --ignore-errors
option to ignore ES bulk import errors - The elasticsearch backend now accepts passing all supported client options
"},{"location":"CHANGELOG/#changed_16","title":"Changed","text":" - Upgrade
pyyaml
to 5.4.1
- Upgrade
pandas
to 1.2.1
"},{"location":"CHANGELOG/#removed_5","title":"Removed","text":" click_log
is no longer required as we are able to configure logging
"},{"location":"CHANGELOG/#100_-_2021-01-13","title":"1.0.0 - 2021-01-13","text":""},{"location":"CHANGELOG/#added_13","title":"Added","text":" - Implement base CLI commands (list, extract, fetch & push) for supported backends
- Support for ElasticSearch database backend
- Support for LDP storage backend
- Support for FS storage backend
- Parse (gzipped) tracking logs in GELF format
- Support for application\u2019s configuration file
- Add optional sentry integration
- Distribute Arnold\u2019s tray to deploy Ralph in a k8s cluster as cronjobs
"},{"location":"LICENSE/","title":"License","text":"MIT License
Copyright (c) 2020-present France Universit\u00e9 Num\u00e9rique
Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the \u201cSoftware\u201d), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions:
The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software.
THE SOFTWARE IS PROVIDED \u201cAS IS\u201d, WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
"},{"location":"UPGRADE/","title":"Upgrade","text":"All instructions to upgrade this project from one release to the next will be documented in this file. Upgrades must be run sequentially, meaning you should not skip minor/major releases while upgrading (fix releases can be skipped).
This project adheres to Semantic Versioning.
"},{"location":"UPGRADE/#3xx_to_400","title":"3.x.x to 4.0.0","text":""},{"location":"UPGRADE/#upgrade_user_credentials","title":"Upgrade user credentials","text":"To conform to xAPI specifications, we need to represent users as xAPI Agents. You must therefore delete and re-create the credentials file using the updated cli, OR you can modify it directly to add the agents
field. The credentials file is located in { RALPH_APP_DIR }/{ RALPH_AUTH_FILE }
(defaults to .ralph/auth.json
). Each user profile must follow the following pattern (see this post for examples of valid agent objects) :
{\n \"username\": \"USERNAME_UNCHANGED\",\n \"hash\": \"PASSWORD_HASH_UNCHANGED\",\n \"scopes\": [ LIST_OF_SCOPES_UNCHANGED ],\n \"agent\": { A_VALID_AGENT_OBJECT }\n}\n
Agent can take one of the following forms, as specified by the xAPI specification: - mbox: \"agent\": {\n \"mbox\": \"mailto:john.doe@example.com\"\n}\n
- mbox_sha1sum: \"agent\": {\n \"mbox_sha1sum\": \"ebd31e95054c018b10727ccffd2ef2ec3a016ee9\",\n}\n
- openid: \"agent\": {\n \"openid\": \"http://foo.openid.example.org/\"\n}\n
- account: \"agent\": {\n \"account\": {\n \"name\": \"simonsAccountName\",\n \"homePage\": \"http://www.exampleHomePage.com\"\n}\n
For example here is a valid auth.json
file:
[\n {\n \"username\": \"john.doe@example.com\",\n \"hash\": \"$2b$12$yBXrzIuRIk6yaft5KUgVFOIPv0PskCCh9PXmF2t7pno.qUZ5LK0D2\",\n \"scopes\": [\"example_scope\"],\n \"agent\": {\n \"mbox\": \"mailto:john.doe@example.com\"\n }\n },\n {\n \"username\": \"simon.says@example.com\",\n \"hash\": \"$2b$12$yBXrzIuRIk6yaft5KUgVFOIPv0PskCCh9PXmF2t7pno.qUZ5LK0D2\",\n \"scopes\": [\"second_scope\", \"third_scope\"],\n \"agent\": {\n \"account\": {\n \"name\": \"simonsAccountName\",\n \"homePage\": \"http://www.exampleHomePage.com\"\n }\n }\n }\n]\n
"},{"location":"UPGRADE/#upgrade_history_syntax","title":"Upgrade history syntax","text":"CLI syntax has been changed from fetch
& push
to read
& write
affecting the command history. You must replace the command history after updating: - locate your history file path, which is in { RALPH_APP_DIR }/history.json
(defaults to .ralph/history.json
) - run the commands below to update history
sed -i 's/\"fetch\"/\"read\"/g' { my_history_file_path }\nsed -i 's/\"push\"/\"write\"/g' { my_history_file_path }\n
"},{"location":"UPGRADE/#upgrade_clickhouse_schema","title":"Upgrade ClickHouse schema","text":"If you are using the ClickHouse backend, schema changes have been made to drop the existing JSON column in favor of the String version of the same data. See this issue for details.
Ralph does not manage the ClickHouse schema, so if you have existing data you will need to manually alter it as an admin user. Note: this will rewrite the statements table, which may take a long time if you have many rows. The command to run is:
-- If RALPH_BACKENDS__DATA__CLICKHOUSE__DATABASE is 'xapi'\n-- and RALPH_BACKENDS__DATA__CLICKHOUSE__EVENT_TABLE_NAME is 'test'\n\nALTER TABLE xapi.test DROP COLUMN event, RENAME COLUMN event_str to event;\n
"},{"location":"commands/","title":"Commands","text":""},{"location":"commands/#ralph","title":"ralph","text":"The cli is a stream-based tool to play with your logs.
It offers functionalities to: - Validate or convert learning data in different standards - Read and write learning data to various databases or servers - Manage an instance of a Ralph LRS server
Usage:
ralph [OPTIONS] COMMAND [ARGS]...\n
Options:
-v, --verbosity LVL Either CRITICAL, ERROR, WARNING, INFO (default) or\n DEBUG\n --version Show the version and exit.\n --help Show this message and exit.\n
"},{"location":"commands/#ralph-auth","title":"ralph auth","text":"Generate credentials for LRS HTTP basic authentication.
Usage:
ralph auth [OPTIONS]\n
Options:
-u, --username TEXT The user for which we generate credentials.\n [required]\n -p, --password TEXT The password to encrypt for this user. Will\n be prompted if missing. [required]\n -s, --scope TEXT The user scope(s). This option can be\n provided multiple times. [required]\n -M, --agent-ifi-mbox TEXT The mbox Inverse Functional Identifier of\n the associated agent.\n -S, --agent-ifi-mbox-sha1sum TEXT\n The mbox-sha1sum Inverse Functional\n Identifier of the associated agent.\n -O, --agent-ifi-openid TEXT The openid Inverse Functional Identifier of\n the associated agent.\n -A, --agent-ifi-account TEXT...\n Input \"{name} {homePage}\". The account\n Inverse Functional Identifier of the\n associated agent.\n -N, --agent-name TEXT The name of the associated agent.\n -w, --write-to-disk Write new credentials to the LRS\n authentication file.\n --help Show this message and exit.\n
"},{"location":"commands/#ralph-convert","title":"ralph convert","text":"Convert input events to a given format.
Usage:
ralph convert [OPTIONS]\n
Options:
From edX to xAPI converter options: \n -u, --uuid-namespace TEXT The UUID namespace to use for the `ID` field\n generation\n -p, --platform-url TEXT The `actor.account.homePage` to use in the\n xAPI statements [required]\n -f, --from [edx] Input events format to convert [required]\n -t, --to [xapi] Output events format [required]\n -I, --ignore-errors Continue writing regardless of raised errors\n -F, --fail-on-unknown Stop converting at first unknown event\n --help Show this message and exit.\n
"},{"location":"commands/#ralph-extract","title":"ralph extract","text":"Extract input events from a container format using a dedicated parser.
Usage:
ralph extract [OPTIONS]\n
Options:
-p, --parser [gelf|es] Container format parser used to extract events\n [required]\n --help Show this message and exit.\n
"},{"location":"commands/#ralph-validate","title":"ralph validate","text":"Validate input events of given format.
Usage:
ralph validate [OPTIONS]\n
Options:
-f, --format [edx|xapi] Input events format to validate [required]\n -I, --ignore-errors Continue validating regardless of raised errors\n -F, --fail-on-unknown Stop validating at first unknown event\n --help Show this message and exit.\n
"},{"location":"contribute/","title":"Contributing to Ralph","text":"Thank you for considering contributing to Ralph! We appreciate your interest and support. This documentation provides guidelines on how to contribute effectively to our project.
"},{"location":"contribute/#issues","title":"Issues","text":"Issues are a valuable way to contribute to Ralph. They can include bug reports, feature requests, and general questions or discussions. When creating or interacting with issues, please keep the following in mind:
"},{"location":"contribute/#1_search_for_existing_issues","title":"1. Search for existing issues","text":"Before creating a new issue, search the existing issues to see if your concern has already been raised. If you find a related issue, you can add your input or follow the discussion. Feel free to engage in discussions, offer help, or provide feedback on existing issues. Your input is valuable in shaping the project\u2019s future.
"},{"location":"contribute/#2_creating_a_new_issue","title":"2. Creating a new issue","text":"Use the provided issue template that fits the best to your concern. Provide as much information as possible when writing your issue. Your issue will be reviewed by a project maintainer and you may be offered to open a PR if you want to contribute to the code. If not, and if your issue is relevant, a contributor will apply the changes to the project. The issue will then be automatically closed when the PR is merged.
Issues will be closed by project maintainers if they are deemed invalid. You can always reopen an issue if you believe it hasn\u2019t been adequately addressed.
"},{"location":"contribute/#3_code_of_conduct_in_discussion","title":"3. Code of conduct in discussion","text":" - Be respectful and considerate when participating in discussions.
- Avoid using offensive language, and maintain a positive and collaborative tone.
- Stay on topic and avoid derailing discussions.
"},{"location":"contribute/#discussions","title":"Discussions","text":"Discussions in the Ralph repository are a place for open-ended conversations, questions, and general community interactions. Here\u2019s how to effectively use discussions:
"},{"location":"contribute/#1_creating_a_discussion","title":"1. Creating a discussion","text":" - Use a clear and concise title that summarizes the topic.
- In the description, provide context and details regarding the discussion.
- Use labels to categorize the discussion (e.g., \u201cquestion,\u201d \u201cgeneral discussion,\u201d \u201cannouncements,\u201d etc.).
"},{"location":"contribute/#2_participating_in_discussions","title":"2. Participating in discussions","text":" - Engage in conversations respectfully, respecting others\u2019 opinions.
- Avoid spamming or making off-topic comments.
- Help answer questions when you can.
"},{"location":"contribute/#pull_requests_pr","title":"Pull Requests (PR)","text":"Contributing to Ralph through pull requests is a powerful way to advance the project. If you want to make changes or add new features, please follow these steps to submit a PR:
"},{"location":"contribute/#1_fork_the_repository","title":"1. Fork the repository","text":"Begin by forking Ralph project\u2019s repository.
"},{"location":"contribute/#2_clone_the_fork","title":"2. Clone the fork","text":"Clone the forked repository to your local machine and change the directory to the project folder using the following commands (replace <your_fork>
with your GitHub username):
git clone https://github.com/<your_fork>/ralph.git\ncd ralph\n
"},{"location":"contribute/#3_create_a_new_branch","title":"3. Create a new branch","text":"Create a new branch for your changes, ideally with a descriptive name:
git checkout -b your-new-feature\n
"},{"location":"contribute/#4_make_changes","title":"4. Make changes","text":"Implement the changes or additions to the code, ensuring it follows OpenFUN coding and documentation standards.
For comprehensive guidance on starting your development journey with Ralph and preparing your pull request, please refer to our dedicated Start developing with Ralph tutorial.
When committing your changes, please adhere to OpenFUN commit practices. Follow the low granularity commit splitting approach and use commit messages based on the Angular commit message guidelines.
"},{"location":"contribute/#5_push_changes","title":"5. Push changes","text":"Push your branch to your GitHub repository:
git push origin feature/your-new-feature\n
"},{"location":"contribute/#6_create_a_pull_request","title":"6. Create a pull request","text":"To initiate a Pull Request (PR), head to Ralph project\u2019s GitHub repository and click on New Pull Request.
Set your branch as the source and Ralph project\u2019s master
branch as the target.
Provide a clear title for your PR and make use of the provided PR body template to document the changes made by your PR. This helps streamline the review process and maintain a well-documented project history.
"},{"location":"contribute/#7_review_and_discussion","title":"7. Review and discussion","text":"Ralph project maintainers will review your PR. Be prepared to make necessary changes or address any feedback. Patience during this process is appreciated.
"},{"location":"contribute/#8_merge","title":"8. Merge","text":"Once your PR is approved, Ralph maintainers will merge your changes into the main project. Congratulations, you\u2019ve successfully contributed to Ralph! \ud83c\udf89
"},{"location":"api/","title":"LRS HTTP server","text":"Ralph comes with an API server that aims to implement the Learning Record Store (LRS) specification (still a work in progress).
"},{"location":"api/#getting_started","title":"Getting started","text":"The API server supports the following authentication methods:
- HTTP Basic Authentication (default method)
- OpenID Connect authentication on top of OAuth2.0
"},{"location":"api/#http_basic_authentication","title":"HTTP Basic Authentication","text":"The default method for securing Ralph API server is with HTTP basic authentication.
The API server can be started up with the following command:
ralph runserver --backend es\n
The --backend
(or -b
) option specifies which database backend to use for LRS data storage and retrieval. See Ralph\u2019s backends documentation for more details.
However, before you can start your API server and make requests against it, you need to set up your credentials.
"},{"location":"api/#creating_a_credentials_file","title":"Creating a credentials file","text":"The credentials file is expected to be a valid JSON file. Its location is specified by the RALPH_AUTH_FILE
configuration value. By default, ralph
will look for the auth.json
file in the application directory (see click documentation for details).
The expected format is a list of entries (JSON objects) each containing the username, the user\u2019s bcrypt
hashed+salted password, scopes they can access, and an agent
object used to represent the user in the LRS. The agent
is constrained by LRS specifications, and must use one of four valid Inverse Functional Identifiers.
[\n {\n \"username\": \"john.doe@example.com\",\n \"hash\": \"$2b$12$yBXrzIuRIk6yaft5KUgVFOIPv0PskCCh9PXmF2t7pno.qUZ5LK0D2\",\n \"scopes\": [\"example_scope\"],\n \"agent\": {\n \"mbox\": \"mailto:john.doe@example.com\"\n }\n },\n {\n \"username\": \"simon.says@example.com\",\n \"hash\": \"$2b$12$yBXrzIuRIk6yaft5KUgVFOIPv0PskCCh9PXmF2t7pno.qUZ5LK0D2\",\n \"scopes\": [\"second_scope\", \"third_scope\"],\n \"agent\": {\n \"account\": {\n \"name\": \"simonsAccountName\",\n \"homePage\": \"http://www.exampleHomePage.com\"\n }\n }\n }\n]\n
To create a new user credentials, Ralph\u2019s CLI provides a dedicated command:
ralph auth \\\n --username janedoe \\\n --password supersecret \\\n --scope janedoe_scope \\\n --agent-ifi-mbox mailto:janedoe@example.com \\\n # or --agent-ifi-mbox-sha1sum ebd31e95054c018b10727ccffd2ef2ec3a016ee9 \\\n # or --agent-ifi-openid \"http://jane.openid.example.org/\" \\\n # or --agent-ifi-account exampleAccountname http://www.exampleHomePage.com \\\n -w\n
This command updates your credentials file with the new janedoe
user.
Note that running this command requires that you installed Ralph with the CLI optional dependencies, e.g. pip install ralph-malph[cli]
(which we highly recommend).
"},{"location":"api/#making_a_get_request","title":"Making a GET request","text":"The first request that can be answered by the ralph API server is a whoami
request, which checks if the user is authenticated and returns their username and permission scopes.
Use curl to get http://localhost:8100/whoami
:
curl http://localhost:8100/whoami\n< HTTP/1.1 401 Unauthorized\n< {\"error\":\"Not authenticated\"}\n
Send your username and password to the API server through HTTP Basic Auth:
curl --user john.doe@example.com:PASSWORD http://localhost:8100/whoami\n< HTTP/1.1 200 OK\n< {\"scopes\":[\"example_scope\"], \"agent\": {\"mbox\": \"mailto:john.doe@example.com\"}}\n
"},{"location":"api/#openid_connect_authentication","title":"OpenID Connect authentication","text":"Ralph LRS API server supports OpenID Connect (OIDC) on top of OAuth 2.0 for authentication and authorization.
To enable OIDC auth, you should modify the RALPH_RUNSERVER_AUTH_BACKENDS
environment variable by adding (or replacing by) oidc
:
RALPH_RUNSERVER_AUTH_BACKENDS=basic,oidc\n
and you should define the RALPH_RUNSERVER_AUTH_OIDC_ISSUER_URI
environment variable with your identity provider\u2019s Issuer Identifier URI as follows:
RALPH_RUNSERVER_AUTH_OIDC_ISSUER_URI=http://{provider_host}:{provider_port}/auth/realms/{realm_name}\n
This address must be accessible to the LRS on startup as it will perform OIDC Discovery to retrieve public keys and other information about the OIDC environment.
It is also strongly recommended that you set the optional RALPH_RUNSERVER_AUTH_OIDC_AUDIENCE
environment variable to the origin address of the LRS itself (ex. \u201chttp://localhost:8100\u201d) to enable verification that a given token was issued specifically for the LRS.
"},{"location":"api/#identity_providers","title":"Identity Providers","text":"OIDC support is currently developed and tested against Keycloak but may work with other identity providers that implement the specification.
The Learning analytics playground repository contains a Docker Compose file and configuration for a demo instance of Keycloak with a ralph
client.
"},{"location":"api/#making_a_get_request_1","title":"Making a GET request","text":"The first request that can be answered by the ralph API server is a whoami
request, which checks if the user is authenticated and returns their username and permission scopes.
Use curl to get http://localhost:8100/whoami
:
curl http://localhost:8100/whoami\n< HTTP/1.1 401 Unauthorized\n< {\"detail\":\"Could not validate credentials\"}\n
With the Keycloak instance running, use curl to get access token from Keycloak:
curl --request POST 'http://localhost:8080/auth/realms/fun-mooc/protocol/openid-connect/token' \\\n--header 'Content-Type: application/x-www-form-urlencoded' \\\n--data-urlencode 'client_id=ralph' \\\n--data-urlencode 'client_secret=super-secret' \\\n--data-urlencode 'username=ralph_admin' \\\n--data-urlencode 'password=funfunfun' \\\n--data-urlencode 'grant_type=password'\n
which outputs (tokens truncated for example purpose):
{ \n \"access_token\":\"eyJhbGciOiJSUzI1NiIsInR5cCIgOiAiSldUIiwia2lkIiA6ICJSTWlLM\",\n \"expires_in\":300,\n \"refresh_expires_in\":1800,\n \"refresh_token\":\"eyJhbGciOiJIUzI1NiIsInR5cCIgOiAiSldUIiwia2lkIiA6ICI4MDc5NjExM\",\n \"token_type\":\"Bearer\",\n \"not-before-policy\":0,\n \"session_state\":\"22a36735-e35f-496b-a243-152d32ebff45\",\n \"scope\":\"profile email\"\n}\n
Send the access token to the API server as a Bearer header:
curl http://localhost:8100/whoami --header \"Authorization: Bearer eyJhbGciOiJSUzI1NiIsInR5cCIgOiAiSldUIiwia2lkIiA6ICJSTWlLM\"\n< HTTP/1.1 200 OK\n< {\"username\":\"ralph_admin\",\"scopes\":[\"all\"]}\n
"},{"location":"api/#security","title":"Security","text":"By default, all authenticated users have full read and write access to the server. You may use the options below to restrict behavior.
"},{"location":"api/#filtering_results_by_authority_multitenancy","title":"Filtering results by authority (multitenancy)","text":"In Ralph LRS, all incoming statements are assigned an authority
(or ownership) derived from the user that makes the call. You may restrict read access to users \u201cown\u201d statements (thus enabling multitenancy) by setting the following environment variable:
RALPH_LRS_RESTRICT_BY_AUTHORITY = True # Default: False\n
WARNING: Two accounts with different credentials may share the same authority
meaning they can access the same statements. It is the administrators responsability to ensure that authority
is properly assigned.
NB: If not using \u201cscopes\u201d, or for users with limited \u201cscopes\u201d, using this option will make the use of option ?mine=True
implicit when fetching statement.
"},{"location":"api/#scopes","title":"Scopes","text":"In Ralph, users are assigned scopes which may be used to restrict endpoint access or functionalities. You may enable this option by setting the following environment variable:
RALPH_LRS_RESTRICT_BY_SCOPES = True # Default: False\n
Valid scopes are a slight variation on those proposed by the xAPI specification:
- statements/write
- statements/read/mine
- statements/read
- state/write
- state/read
- define
- profile/write
- profile/read
- all/read
- all
"},{"location":"api/#forwarding_statements","title":"Forwarding statements","text":"Ralph\u2019s API server can be configured to forward xAPI statements it receives to other LRSes.
To configure statement forwarding, you need to create a .env
file in the current directory and define the RALPH_XAPI_FORWARDINGS
variable or define the RALPH_XAPI_FORWARDINGS
environment variable.
The value of the RALPH_XAPI_FORWARDINGS
variable should be a JSON encoded list of dictionaries where each dictionary defines a forwarding configuration and consists of the following key/value pairs:
key value type description is_active
boolean
Specifies whether or not this forwarding configuration should take effect. url
URL
Specifies the endpoint URL where forwarded statements should be send. basic_username
string
Specifies the basic auth username. basic_password
string
Specifies the basic auth password. max_retries
number
Specifies the number of times a failed forwarding request should be retried. timeout
number
Specifies the duration in seconds of network inactivity leading to a timeout. Note that for a forwarding configuration to be valid it is required that all key/value pairs are defined.
Example of a valid forwarding configuration:
RALPH_XAPI_FORWARDINGS='\n[\n {\n \"is_active\": true,\n \"url\": \"http://lrs1.example.com/xAPI/statements/\",\n \"basic_username\": \"admin1@example.com\",\n \"basic_password\": \"PASSWORD1\",\n \"max_retries\": 1,\n \"timeout\": 5\n },\n {\n \"is_active\": true,\n \"url\": \"http://lrs2.example.com/xAPI/statements/\",\n \"basic_username\": \"admin2@example.com\",\n \"basic_password\": \"PASSWORD2\",\n \"max_retries\": 5,\n \"timeout\": 0.2\n }\n]\n'\n
"},{"location":"api/#sentry_configuration","title":"Sentry configuration","text":"Ralph provides Sentry integration to monitor its LRS server and its CLI. To activate Sentry integration, one should define the following environment variables:
RALPH_SENTRY_DSN={PROTOCOL}://{PUBLIC_KEY}:{SECRET_KEY}@{HOST}{PATH}/{PROJECT_ID}\nRALPH_EXECUTION_ENVIRONMENT=development\n
The Sentry DSN (Data Source Name) can be found in your project settings from Sentry application. The execution environment should reflect the environment Ralph has been deployed in (e.g. production
).
You may also want to monitor the performance of Ralph by configuring the CLI and LRS traces sample rates:
RALPH_SENTRY_CLI_TRACES_SAMPLE_RATE=0.1\nRALPH_SENTRY_LRS_TRACES_SAMPLE_RATE=0.3\n
Note that a sampling rate of 1.0
means 100% of transactions are sent to sentry and 0.1
only 10%.
If you want to lower noisy transactions (e.g. in a Kubernetes cluster), you can disable health checks related ones:
RALPH_SENTRY_IGNORE_HEALTH_CHECKS=True\n
"},{"location":"api/#additional_configuration","title":"Additional configuration","text":""},{"location":"api/#http_basic_auth_caching","title":"HTTP Basic auth caching","text":"HTTP Basic auth implementation uses the secure and standard bcrypt algorithm to hash/salt passwords before storing them. This implementation comes with a performance cost. To speed up requests, credentials are stored in an LRU cache with a Time To Live
. To configure this cache, you can define the following environment variables:
- the maximum number of entries in the cache. Select a value greater than the maximum number of individual user credentials, for better performance. Defaults to 100.
RALPH_AUTH_CACHE_MAX_SIZE=100\n
- the Time To Live
of the cache entries in seconds. Defaults to 3600s. RALPH_AUTH_CACHE_TTL=3600\n
"},{"location":"api/#api_documentation","title":"API documentation","text":""},{"location":"api/#fastapi_010","title":"FastAPI 0.1.0","text":""},{"location":"api/#endpoints","title":"Endpoints","text":""},{"location":"api/#get_xapistatements","title":"GET /xAPI/statements/","text":"Get
Description Read a single xAPI Statement or multiple xAPI Statements.
LRS Specification: https://github.com/adlnet/xAPI-Spec/blob/1.0.3/xAPI- Communication.md#213-get-statements
Input parameters
Parameter In Type Default Nullable Description HTTPBasic
header string N/A No Basic authentication HTTPBasic
header string N/A No Basic authentication activity
query string No Filter, only return Statements for which the Object of the Statement is an Activity with the specified id agent
query string No Filter, only return Statements for which the specified Agent or Group is the Actor or Object of the Statement ascending
query boolean False No If \"true\", return results in ascending order of stored time attachments
query boolean False No **Not implemented** 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. format
query string exact No **Not implemented** 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. limit
query integer 100 No Maximum number of Statements to return. 0 indicates return the maximum the server will allow mine
query boolean False No If \"true\", return only the results for which the authority matches the \"agent\" associated to the user that is making the query. pit_id
query string No Point-in-time ID to ensure consistency of search requests through multiple pages.NB: for internal use, not part of the LRS specification. registration
query string No **Not implemented** Filter, only return Statements matching the specified registration id related_activities
query boolean False No **Not implemented** 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 behaviour related_agents
query boolean False No **Not implemented** 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 behaviour. search_after
query string No Sorting data to allow pagination through large number of search results. NB: for internal use, not part of the LRS specification. since
query string No Only Statements stored since the specified Timestamp (exclusive) are returned statementId
query string No Id of Statement to fetch until
query string No Only Statements stored at or before the specified Timestamp are returned verb
query string No Filter, only return Statements matching the specified Verb id voidedStatementId
query string No **Not implemented** Id of voided Statement to fetch Response 200 OK
application/json Schema of the response body {\n \"type\": \"object\",\n \"title\": \"Response Get Xapi Statements Get\"\n}\n
Response 422 Unprocessable Entity
application/json {\n \"detail\": [\n {\n \"loc\": [\n null\n ],\n \"msg\": \"string\",\n \"type\": \"string\"\n }\n ]\n}\n
\u26a0\ufe0f This example has been generated automatically from the schema and it is not accurate. Refer to the schema for more information. Schema of the response body {\n \"properties\": {\n \"detail\": {\n \"items\": {\n \"$ref\": \"#/components/schemas/ValidationError\"\n },\n \"type\": \"array\",\n \"title\": \"Detail\"\n }\n },\n \"type\": \"object\",\n \"title\": \"HTTPValidationError\"\n}\n
"},{"location":"api/#put_xapistatements","title":"PUT /xAPI/statements/","text":"Put
Description Store a single statement as a single member of a set.
LRS Specification: https://github.com/adlnet/xAPI-Spec/blob/1.0.3/xAPI- Communication.md#211-put-statements
Input parameters
Parameter In Type Default Nullable Description HTTPBasic
header string N/A No Basic authentication HTTPBasic
header string N/A No Basic authentication statementId
query string No Request body
application/json {\n \"actor\": null,\n \"id\": \"80fd65b0-85f8-4b64-a678-bd39624402fe\",\n \"object\": {\n \"id\": \"string\"\n },\n \"verb\": {\n \"id\": \"string\"\n }\n}\n
\u26a0\ufe0f This example has been generated automatically from the schema and it is not accurate. Refer to the schema for more information. Schema of the request body {\n \"properties\": {\n \"actor\": {\n \"anyOf\": [\n {\n \"$ref\": \"#/components/schemas/BaseXapiAgentWithMbox\"\n },\n {\n \"$ref\": \"#/components/schemas/BaseXapiAgentWithMboxSha1Sum\"\n },\n {\n \"$ref\": \"#/components/schemas/BaseXapiAgentWithOpenId\"\n },\n {\n \"$ref\": \"#/components/schemas/BaseXapiAgentWithAccount\"\n },\n {\n \"$ref\": \"#/components/schemas/BaseXapiAnonymousGroup\"\n },\n {\n \"$ref\": \"#/components/schemas/BaseXapiIdentifiedGroupWithMbox\"\n },\n {\n \"$ref\": \"#/components/schemas/BaseXapiIdentifiedGroupWithMboxSha1Sum\"\n },\n {\n \"$ref\": \"#/components/schemas/BaseXapiIdentifiedGroupWithOpenId\"\n },\n {\n \"$ref\": \"#/components/schemas/BaseXapiIdentifiedGroupWithAccount\"\n }\n ],\n \"title\": \"Actor\"\n },\n \"id\": {\n \"type\": \"string\",\n \"format\": \"uuid\",\n \"title\": \"Id\"\n },\n \"object\": {\n \"$ref\": \"#/components/schemas/LaxObjectField\"\n },\n \"verb\": {\n \"$ref\": \"#/components/schemas/LaxVerbField\"\n }\n },\n \"type\": \"object\",\n \"required\": [\n \"actor\",\n \"object\",\n \"verb\"\n ],\n \"title\": \"LaxStatement\",\n \"description\": \"Pydantic model for lax statement.\\n\\nIt accepts without validating all fields beyond the bare minimum required to\\nqualify an object as an XAPI statement.\"\n}\n
Response 204 No Content
Response 400 Bad Request
application/json {\n \"detail\": \"string\"\n}\n
\u26a0\ufe0f This example has been generated automatically from the schema and it is not accurate. Refer to the schema for more information. Schema of the response body {\n \"properties\": {\n \"detail\": {\n \"type\": \"string\",\n \"title\": \"Detail\"\n }\n },\n \"type\": \"object\",\n \"required\": [\n \"detail\"\n ],\n \"title\": \"ErrorDetail\",\n \"description\": \"Pydantic model for errors raised detail.\\n\\nType for return value for errors raised in API endpoints.\\nUseful for OpenAPI documentation generation.\"\n}\n
Response 409 Conflict
application/json {\n \"detail\": \"string\"\n}\n
\u26a0\ufe0f This example has been generated automatically from the schema and it is not accurate. Refer to the schema for more information. Schema of the response body {\n \"properties\": {\n \"detail\": {\n \"type\": \"string\",\n \"title\": \"Detail\"\n }\n },\n \"type\": \"object\",\n \"required\": [\n \"detail\"\n ],\n \"title\": \"ErrorDetail\",\n \"description\": \"Pydantic model for errors raised detail.\\n\\nType for return value for errors raised in API endpoints.\\nUseful for OpenAPI documentation generation.\"\n}\n
Response 422 Unprocessable Entity
application/json {\n \"detail\": [\n {\n \"loc\": [\n null\n ],\n \"msg\": \"string\",\n \"type\": \"string\"\n }\n ]\n}\n
\u26a0\ufe0f This example has been generated automatically from the schema and it is not accurate. Refer to the schema for more information. Schema of the response body {\n \"properties\": {\n \"detail\": {\n \"items\": {\n \"$ref\": \"#/components/schemas/ValidationError\"\n },\n \"type\": \"array\",\n \"title\": \"Detail\"\n }\n },\n \"type\": \"object\",\n \"title\": \"HTTPValidationError\"\n}\n
"},{"location":"api/#post_xapistatements","title":"POST /xAPI/statements/","text":"Post
Description Store a set of statements (or a single statement as a single member of a set).
NB: at this time, using POST to make a GET request, is not supported. LRS Specification: https://github.com/adlnet/xAPI-Spec/blob/1.0.3/xAPI- Communication.md#212-post-statements
Input parameters
Parameter In Type Default Nullable Description HTTPBasic
header string N/A No Basic authentication HTTPBasic
header string N/A No Basic authentication Request body
application/json Schema of the request body {\n \"anyOf\": [\n {\n \"$ref\": \"#/components/schemas/LaxStatement\"\n },\n {\n \"items\": {\n \"$ref\": \"#/components/schemas/LaxStatement\"\n },\n \"type\": \"array\"\n }\n ],\n \"title\": \"Statements\"\n}\n
Response 200 OK
application/json [\n null\n]\n
\u26a0\ufe0f This example has been generated automatically from the schema and it is not accurate. Refer to the schema for more information. Schema of the response body {\n \"items\": {},\n \"type\": \"array\",\n \"title\": \"Response Post Xapi Statements Post\"\n}\n
Response 400 Bad Request
application/json {\n \"detail\": \"string\"\n}\n
\u26a0\ufe0f This example has been generated automatically from the schema and it is not accurate. Refer to the schema for more information. Schema of the response body {\n \"properties\": {\n \"detail\": {\n \"type\": \"string\",\n \"title\": \"Detail\"\n }\n },\n \"type\": \"object\",\n \"required\": [\n \"detail\"\n ],\n \"title\": \"ErrorDetail\",\n \"description\": \"Pydantic model for errors raised detail.\\n\\nType for return value for errors raised in API endpoints.\\nUseful for OpenAPI documentation generation.\"\n}\n
Response 409 Conflict
application/json {\n \"detail\": \"string\"\n}\n
\u26a0\ufe0f This example has been generated automatically from the schema and it is not accurate. Refer to the schema for more information. Schema of the response body {\n \"properties\": {\n \"detail\": {\n \"type\": \"string\",\n \"title\": \"Detail\"\n }\n },\n \"type\": \"object\",\n \"required\": [\n \"detail\"\n ],\n \"title\": \"ErrorDetail\",\n \"description\": \"Pydantic model for errors raised detail.\\n\\nType for return value for errors raised in API endpoints.\\nUseful for OpenAPI documentation generation.\"\n}\n
Response 422 Unprocessable Entity
application/json {\n \"detail\": [\n {\n \"loc\": [\n null\n ],\n \"msg\": \"string\",\n \"type\": \"string\"\n }\n ]\n}\n
\u26a0\ufe0f This example has been generated automatically from the schema and it is not accurate. Refer to the schema for more information. Schema of the response body {\n \"properties\": {\n \"detail\": {\n \"items\": {\n \"$ref\": \"#/components/schemas/ValidationError\"\n },\n \"type\": \"array\",\n \"title\": \"Detail\"\n }\n },\n \"type\": \"object\",\n \"title\": \"HTTPValidationError\"\n}\n
"},{"location":"api/#get_xapistatements_1","title":"GET /xAPI/statements","text":"Get
Description Read a single xAPI Statement or multiple xAPI Statements.
LRS Specification: https://github.com/adlnet/xAPI-Spec/blob/1.0.3/xAPI- Communication.md#213-get-statements
Input parameters
Parameter In Type Default Nullable Description HTTPBasic
header string N/A No Basic authentication HTTPBasic
header string N/A No Basic authentication activity
query string No Filter, only return Statements for which the Object of the Statement is an Activity with the specified id agent
query string No Filter, only return Statements for which the specified Agent or Group is the Actor or Object of the Statement ascending
query boolean False No If \"true\", return results in ascending order of stored time attachments
query boolean False No **Not implemented** 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. format
query string exact No **Not implemented** 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. limit
query integer 100 No Maximum number of Statements to return. 0 indicates return the maximum the server will allow mine
query boolean False No If \"true\", return only the results for which the authority matches the \"agent\" associated to the user that is making the query. pit_id
query string No Point-in-time ID to ensure consistency of search requests through multiple pages.NB: for internal use, not part of the LRS specification. registration
query string No **Not implemented** Filter, only return Statements matching the specified registration id related_activities
query boolean False No **Not implemented** 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 behaviour related_agents
query boolean False No **Not implemented** 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 behaviour. search_after
query string No Sorting data to allow pagination through large number of search results. NB: for internal use, not part of the LRS specification. since
query string No Only Statements stored since the specified Timestamp (exclusive) are returned statementId
query string No Id of Statement to fetch until
query string No Only Statements stored at or before the specified Timestamp are returned verb
query string No Filter, only return Statements matching the specified Verb id voidedStatementId
query string No **Not implemented** Id of voided Statement to fetch Response 200 OK
application/json Schema of the response body {\n \"type\": \"object\",\n \"title\": \"Response Get Xapi Statements Get\"\n}\n
Response 422 Unprocessable Entity
application/json {\n \"detail\": [\n {\n \"loc\": [\n null\n ],\n \"msg\": \"string\",\n \"type\": \"string\"\n }\n ]\n}\n
\u26a0\ufe0f This example has been generated automatically from the schema and it is not accurate. Refer to the schema for more information. Schema of the response body {\n \"properties\": {\n \"detail\": {\n \"items\": {\n \"$ref\": \"#/components/schemas/ValidationError\"\n },\n \"type\": \"array\",\n \"title\": \"Detail\"\n }\n },\n \"type\": \"object\",\n \"title\": \"HTTPValidationError\"\n}\n
"},{"location":"api/#put_xapistatements_1","title":"PUT /xAPI/statements","text":"Put
Description Store a single statement as a single member of a set.
LRS Specification: https://github.com/adlnet/xAPI-Spec/blob/1.0.3/xAPI- Communication.md#211-put-statements
Input parameters
Parameter In Type Default Nullable Description HTTPBasic
header string N/A No Basic authentication HTTPBasic
header string N/A No Basic authentication statementId
query string No Request body
application/json {\n \"actor\": null,\n \"id\": \"282bcef8-665d-4e92-ac4a-049c5cfb93e4\",\n \"object\": {\n \"id\": \"string\"\n },\n \"verb\": {\n \"id\": \"string\"\n }\n}\n
\u26a0\ufe0f This example has been generated automatically from the schema and it is not accurate. Refer to the schema for more information. Schema of the request body {\n \"properties\": {\n \"actor\": {\n \"anyOf\": [\n {\n \"$ref\": \"#/components/schemas/BaseXapiAgentWithMbox\"\n },\n {\n \"$ref\": \"#/components/schemas/BaseXapiAgentWithMboxSha1Sum\"\n },\n {\n \"$ref\": \"#/components/schemas/BaseXapiAgentWithOpenId\"\n },\n {\n \"$ref\": \"#/components/schemas/BaseXapiAgentWithAccount\"\n },\n {\n \"$ref\": \"#/components/schemas/BaseXapiAnonymousGroup\"\n },\n {\n \"$ref\": \"#/components/schemas/BaseXapiIdentifiedGroupWithMbox\"\n },\n {\n \"$ref\": \"#/components/schemas/BaseXapiIdentifiedGroupWithMboxSha1Sum\"\n },\n {\n \"$ref\": \"#/components/schemas/BaseXapiIdentifiedGroupWithOpenId\"\n },\n {\n \"$ref\": \"#/components/schemas/BaseXapiIdentifiedGroupWithAccount\"\n }\n ],\n \"title\": \"Actor\"\n },\n \"id\": {\n \"type\": \"string\",\n \"format\": \"uuid\",\n \"title\": \"Id\"\n },\n \"object\": {\n \"$ref\": \"#/components/schemas/LaxObjectField\"\n },\n \"verb\": {\n \"$ref\": \"#/components/schemas/LaxVerbField\"\n }\n },\n \"type\": \"object\",\n \"required\": [\n \"actor\",\n \"object\",\n \"verb\"\n ],\n \"title\": \"LaxStatement\",\n \"description\": \"Pydantic model for lax statement.\\n\\nIt accepts without validating all fields beyond the bare minimum required to\\nqualify an object as an XAPI statement.\"\n}\n
Response 204 No Content
Response 400 Bad Request
application/json {\n \"detail\": \"string\"\n}\n
\u26a0\ufe0f This example has been generated automatically from the schema and it is not accurate. Refer to the schema for more information. Schema of the response body {\n \"properties\": {\n \"detail\": {\n \"type\": \"string\",\n \"title\": \"Detail\"\n }\n },\n \"type\": \"object\",\n \"required\": [\n \"detail\"\n ],\n \"title\": \"ErrorDetail\",\n \"description\": \"Pydantic model for errors raised detail.\\n\\nType for return value for errors raised in API endpoints.\\nUseful for OpenAPI documentation generation.\"\n}\n
Response 409 Conflict
application/json {\n \"detail\": \"string\"\n}\n
\u26a0\ufe0f This example has been generated automatically from the schema and it is not accurate. Refer to the schema for more information. Schema of the response body {\n \"properties\": {\n \"detail\": {\n \"type\": \"string\",\n \"title\": \"Detail\"\n }\n },\n \"type\": \"object\",\n \"required\": [\n \"detail\"\n ],\n \"title\": \"ErrorDetail\",\n \"description\": \"Pydantic model for errors raised detail.\\n\\nType for return value for errors raised in API endpoints.\\nUseful for OpenAPI documentation generation.\"\n}\n
Response 422 Unprocessable Entity
application/json {\n \"detail\": [\n {\n \"loc\": [\n null\n ],\n \"msg\": \"string\",\n \"type\": \"string\"\n }\n ]\n}\n
\u26a0\ufe0f This example has been generated automatically from the schema and it is not accurate. Refer to the schema for more information. Schema of the response body {\n \"properties\": {\n \"detail\": {\n \"items\": {\n \"$ref\": \"#/components/schemas/ValidationError\"\n },\n \"type\": \"array\",\n \"title\": \"Detail\"\n }\n },\n \"type\": \"object\",\n \"title\": \"HTTPValidationError\"\n}\n
"},{"location":"api/#post_xapistatements_1","title":"POST /xAPI/statements","text":"Post
Description Store a set of statements (or a single statement as a single member of a set).
NB: at this time, using POST to make a GET request, is not supported. LRS Specification: https://github.com/adlnet/xAPI-Spec/blob/1.0.3/xAPI- Communication.md#212-post-statements
Input parameters
Parameter In Type Default Nullable Description HTTPBasic
header string N/A No Basic authentication HTTPBasic
header string N/A No Basic authentication Request body
application/json Schema of the request body {\n \"anyOf\": [\n {\n \"$ref\": \"#/components/schemas/LaxStatement\"\n },\n {\n \"items\": {\n \"$ref\": \"#/components/schemas/LaxStatement\"\n },\n \"type\": \"array\"\n }\n ],\n \"title\": \"Statements\"\n}\n
Response 200 OK
application/json [\n null\n]\n
\u26a0\ufe0f This example has been generated automatically from the schema and it is not accurate. Refer to the schema for more information. Schema of the response body {\n \"items\": {},\n \"type\": \"array\",\n \"title\": \"Response Post Xapi Statements Post\"\n}\n
Response 400 Bad Request
application/json {\n \"detail\": \"string\"\n}\n
\u26a0\ufe0f This example has been generated automatically from the schema and it is not accurate. Refer to the schema for more information. Schema of the response body {\n \"properties\": {\n \"detail\": {\n \"type\": \"string\",\n \"title\": \"Detail\"\n }\n },\n \"type\": \"object\",\n \"required\": [\n \"detail\"\n ],\n \"title\": \"ErrorDetail\",\n \"description\": \"Pydantic model for errors raised detail.\\n\\nType for return value for errors raised in API endpoints.\\nUseful for OpenAPI documentation generation.\"\n}\n
Response 409 Conflict
application/json {\n \"detail\": \"string\"\n}\n
\u26a0\ufe0f This example has been generated automatically from the schema and it is not accurate. Refer to the schema for more information. Schema of the response body {\n \"properties\": {\n \"detail\": {\n \"type\": \"string\",\n \"title\": \"Detail\"\n }\n },\n \"type\": \"object\",\n \"required\": [\n \"detail\"\n ],\n \"title\": \"ErrorDetail\",\n \"description\": \"Pydantic model for errors raised detail.\\n\\nType for return value for errors raised in API endpoints.\\nUseful for OpenAPI documentation generation.\"\n}\n
Response 422 Unprocessable Entity
application/json {\n \"detail\": [\n {\n \"loc\": [\n null\n ],\n \"msg\": \"string\",\n \"type\": \"string\"\n }\n ]\n}\n
\u26a0\ufe0f This example has been generated automatically from the schema and it is not accurate. Refer to the schema for more information. Schema of the response body {\n \"properties\": {\n \"detail\": {\n \"items\": {\n \"$ref\": \"#/components/schemas/ValidationError\"\n },\n \"type\": \"array\",\n \"title\": \"Detail\"\n }\n },\n \"type\": \"object\",\n \"title\": \"HTTPValidationError\"\n}\n
"},{"location":"api/#get_lbheartbeat","title":"GET /lbheartbeat","text":"Lbheartbeat
Description Load balancer heartbeat.
Return a 200 when the server is running.
Response 200 OK
application/json Schema of the response body"},{"location":"api/#get_heartbeat","title":"GET /heartbeat","text":"Heartbeat
Description Application heartbeat.
Return a 200 if all checks are successful.
Response 200 OK
application/json Schema of the response body"},{"location":"api/#get_whoami","title":"GET /whoami","text":"Whoami
Description Return the current user\u2019s username along with their scopes.
Input parameters
Parameter In Type Default Nullable Description HTTPBasic
header string N/A No Basic authentication Response 200 OK
application/json Schema of the response body {\n \"type\": \"object\",\n \"title\": \"Response Whoami Whoami Get\"\n}\n
"},{"location":"api/#schemas","title":"Schemas","text":""},{"location":"api/#basexapiaccount","title":"BaseXapiAccount","text":"Name Type homePage
string name
string"},{"location":"api/#basexapiagentwithaccount","title":"BaseXapiAgentWithAccount","text":"Name Type account
BaseXapiAccount name
string objectType
string"},{"location":"api/#basexapiagentwithmbox","title":"BaseXapiAgentWithMbox","text":"Name Type mbox
string name
string objectType
string"},{"location":"api/#basexapiagentwithmboxsha1sum","title":"BaseXapiAgentWithMboxSha1Sum","text":"Name Type mbox_sha1sum
string name
string objectType
string"},{"location":"api/#basexapiagentwithopenid","title":"BaseXapiAgentWithOpenId","text":"Name Type name
string objectType
string openid
string(uri)"},{"location":"api/#basexapianonymousgroup","title":"BaseXapiAnonymousGroup","text":"Name Type member
Array<> name
string objectType
string"},{"location":"api/#basexapiidentifiedgroupwithaccount","title":"BaseXapiIdentifiedGroupWithAccount","text":"Name Type account
BaseXapiAccount member
Array<> name
string objectType
string"},{"location":"api/#basexapiidentifiedgroupwithmbox","title":"BaseXapiIdentifiedGroupWithMbox","text":"Name Type mbox
string member
Array<> name
string objectType
string"},{"location":"api/#basexapiidentifiedgroupwithmboxsha1sum","title":"BaseXapiIdentifiedGroupWithMboxSha1Sum","text":"Name Type mbox_sha1sum
string member
Array<> name
string objectType
string"},{"location":"api/#basexapiidentifiedgroupwithopenid","title":"BaseXapiIdentifiedGroupWithOpenId","text":"Name Type member
Array<> name
string objectType
string openid
string(uri)"},{"location":"api/#errordetail","title":"ErrorDetail","text":"Name Type detail
string"},{"location":"api/#httpvalidationerror","title":"HTTPValidationError","text":"Name Type detail
Array<ValidationError>"},{"location":"api/#laxobjectfield","title":"LaxObjectField","text":"Name Type id
string(uri)"},{"location":"api/#laxstatement","title":"LaxStatement","text":"Name Type actor
id
string(uuid) object
LaxObjectField verb
LaxVerbField"},{"location":"api/#laxverbfield","title":"LaxVerbField","text":"Name Type id
string(uri)"},{"location":"api/#validationerror","title":"ValidationError","text":"Name Type loc
Array<> msg
string type
string"},{"location":"api/#security_schemes","title":"Security schemes","text":"Name Type Scheme Description HTTPBasic http basic"},{"location":"backends/","title":"Backends for data storage","text":"Ralph supports various backends that can be accessed to read from or write to (learning events or random data). Implemented backends are listed below along with their configuration parameters. If your favourite data storage method is missing, feel free to submit your implementation or get in touch!
"},{"location":"backends/#key_concepts","title":"Key concepts","text":"Each backend has its own parameter requirements. These parameters can be set as command line options or environment variables; the later is the recommended solution for sensitive data such as service credentials. For example, the os_username
(OpenStack user name) parameter of the OpenStack Swift backend, can be set as a command line option using swift
as the option prefix (and replacing underscores in its name by dashes):
ralph list --backend swift --swift-os-username johndoe # [...] more options\n
Alternatively, this parameter can be set as an environment variable (in upper case, prefixed by the program name, e.g. RALPH_
):
export RALPH_BACKENDS__DATA__SWIFT__OS_USERNAME=\"johndoe\"\nralph list --backend swift # [...] more options\n
The general patterns for backend parameters are:
--{{ backend_name }}-{{ parameter | underscore_to_dash }}
for command options, and, RALPH_BACKENDS__{{ backend_type | uppercase }}__{{ backend_name | uppercase }}__{{ parameter | uppercase }}
for environment variables, where the backend_type
is one of DATA
, LRS
and STREAM
.
"},{"location":"backends/#elasticsearch","title":"Elasticsearch","text":"Elasticsearch backend is mostly used for indexation purpose (as a datalake) but it can also be used to fetch indexed data from it.
Elasticsearch data backend default configuration.
Attributes:
Name Type Description ALLOW_YELLOW_STATUS
bool
Whether to consider Elasticsearch yellow health status to be ok.
CLIENT_OPTIONS
dict
A dictionary of valid options for the Elasticsearch class initialization.
DEFAULT_CHUNK_SIZE
int
The default chunk size for reading batches of documents.
DEFAULT_INDEX
str
The default index to use for querying Elasticsearch.
HOSTS
str or tuple
The comma-separated list of Elasticsearch nodes to connect to.
LOCALE_ENCODING
str
The encoding used for reading/writing documents.
POINT_IN_TIME_KEEP_ALIVE
str
The duration for which Elasticsearch should keep a point in time alive.
REFRESH_AFTER_WRITE
str or bool
Whether the Elasticsearch index should be refreshed after the write operation.
"},{"location":"backends/#mongodb","title":"MongoDB","text":"MongoDB backend is mostly used for indexation purpose (as a datalake) but it can also be used to fetch collections of documents from it.
MongoDB data backend default configuration.
Attributes:
Name Type Description CONNECTION_URI
str
The MongoDB connection URI.
DEFAULT_DATABASE
str
The MongoDB database to connect to.
DEFAULT_COLLECTION
str
The MongoDB database collection to get objects from.
CLIENT_OPTIONS
MongoClientOptions
A dictionary of MongoDB client options.
DEFAULT_CHUNK_SIZE
int
The default chunk size to use when none is provided.
LOCALE_ENCODING
str
The locale encoding to use when none is provided.
"},{"location":"backends/#clickhouse","title":"ClickHouse","text":"The ClickHouse backend can be used as a data lake and to fetch collections of documents from it.
Represent the ClickHouse data backend default configuration.
Attributes:
Name Type Description HOST
str
ClickHouse server host to connect to.
PORT
int
ClickHouse server port to connect to.
DATABASE
str
ClickHouse database to connect to.
EVENT_TABLE_NAME
str
Table where events live.
USERNAME
str
ClickHouse username to connect as (optional).
PASSWORD
str
Password for the given ClickHouse username (optional).
CLIENT_OPTIONS
ClickHouseClientOptions
A dictionary of valid options for the ClickHouse client connection.
DEFAULT_CHUNK_SIZE
int
The default chunk size for reading/writing.
LOCALE_ENCODING
str
The locale encoding to use when none is provided.
"},{"location":"backends/#ovh_-_log_data_platform_ldp","title":"OVH - Log Data Platform (LDP)","text":"LDP is a nice service built by OVH on top of Graylog to follow, analyse and store your logs. Learning events (aka tracking logs) can be stored in GELF format using this backend.
Read-only backend
For now the LDP backend is read-only as we consider that it is mostly used to collect primary logs and not as a Ralph target. Feel free to get in touch to prove us wrong, or better: submit your proposal for the write
method implementation.
To access OVH\u2019s LDP API, you need to register Ralph as an authorized application and generate an application key, an application secret and a consumer key.
While filling the registration form available at: eu.api.ovh.com/createToken/, be sure to give an appropriate validity time span to your token and allow only GET requests on the /dbaas/logs/*
path.
OVH LDP (Log Data Platform) data backend default configuration.
Attributes:
Name Type Description APPLICATION_KEY
str
The OVH API application key (AK).
APPLICATION_SECRET
str
The OVH API application secret (AS).
CONSUMER_KEY
str
The OVH API consumer key (CK).
DEFAULT_STREAM_ID
str
The default stream identifier to query.
ENDPOINT
str
The OVH API endpoint.
REQUEST_TIMEOUT
int
HTTP request timeout in seconds.
SERVICE_NAME
str
The default LDP account name.
For more information about OVH\u2019s API client parameters, please refer to the project\u2019s documentation: github.com/ovh/python-ovh.
"},{"location":"backends/#openstack_swift","title":"OpenStack Swift","text":"Swift is the OpenStack object storage service. This storage backend is fully supported (read and write operations) to stream and store log archives.
Parameters correspond to a standard authentication using OpenStack Keystone service and configuration to work with the target container.
Represent the SWIFT data backend default configuration.
Attributes:
Name Type Description AUTH_URL
str
The authentication URL.
USERNAME
str
The name of the openstack swift user.
PASSWORD
str
The password of the openstack swift user.
IDENTITY_API_VERSION
str
The keystone API version to authenticate to.
TENANT_ID
str
The identifier of the tenant of the container.
TENANT_NAME
str
The name of the tenant of the container.
PROJECT_DOMAIN_NAME
str
The project domain name.
REGION_NAME
str
The region where the container is.
OBJECT_STORAGE_URL
str
The default storage URL.
USER_DOMAIN_NAME
str
The user domain name.
DEFAULT_CONTAINER
str
The default target container.
LOCALE_ENCODING
str
The encoding used for reading/writing documents.
"},{"location":"backends/#amazon_s3","title":"Amazon S3","text":"S3 is the Amazon Simple Storage Service. This storage backend is fully supported (read and write operations) to stream and store log archives.
Parameters correspond to a standard authentication with AWS CLI and configuration to work with the target bucket.
S3 data backend default configuration.
Attributes:
Name Type Description ACCESS_KEY_ID
str
The access key id for the S3 account.
SECRET_ACCESS_KEY
str
The secret key for the S3 account.
SESSION_TOKEN
str
The session token for the S3 account.
ENDPOINT_URL
str
The endpoint URL of the S3.
DEFAULT_REGION
str
The default region used in instantiating the client.
DEFAULT_BUCKET_NAME
str
The default bucket name targeted.
DEFAULT_CHUNK_SIZE
str
The default chunk size for reading and writing objects.
LOCALE_ENCODING
str
The encoding used for writing dictionaries to objects.
"},{"location":"backends/#file_system","title":"File system","text":"The file system backend is a dummy template that can be used to develop your own backend. It is a \u201cdummy\u201d backend as it is not intended for practical use (UNIX ls
and cat
would be more practical).
The only required parameter is the path
we want to list or stream content from.
FileSystem data backend default configuration.
Attributes:
Name Type Description DEFAULT_CHUNK_SIZE
int
The default chunk size for reading files.
DEFAULT_DIRECTORY_PATH
str or Path
The default target directory path where to perform list, read and write operations.
DEFAULT_QUERY_STRING
str
The default query string to match files for the read operation.
LOCALE_ENCODING
str
The encoding used for writing dictionaries to files.
The ClickHouse client options supported in Ralph can be found in these locations:
- Python driver specific
- General ClickHouse client settings
"},{"location":"backends/#learning_record_store_lrs_-_http_backend_interface","title":"Learning Record Store (LRS) - HTTP backend interface","text":"Warning
HTTP
backend type will soon be merged into the Data
backend type.
PR #521 prepares Data
backend alignments with HTTP
backend read
method signature
The LRS backend is used to store and retrieve xAPI statements from various systems that follow the xAPI specification (such as our own Ralph LRS, which can be run from this package). LRS systems are mostly used in e-learning infrastructures.
LRS HTTP backend default configuration.
Attributes:
Name Type Description BASE_URL
AnyHttpUrl
LRS server URL.
USERNAME
str
Basic auth username for LRS authentication.
PASSWORD
str
Basic auth password for LRS authentication.
HEADERS
dict
Headers defined for the LRS server connection.
STATUS_ENDPOINT
str
Endpoint used to check server status.
STATEMENTS_ENDPOINT
str
Default endpoint for LRS statements resource.
"},{"location":"backends/#websocket_-_stream_backend_interface","title":"WebSocket - Stream backend interface","text":"The webSocket backend is read-only and can be used to get real-time events.
If you use OVH\u2019s Logs Data Platform (LDP), you can retrieve a WebSocket URI to test your data stream by following instructions from the official documentation.
Websocket stream backend default configuration.
Attributes:
Name Type Description URI
str
The URI to connect to.
"},{"location":"models/","title":"Learning statement models","text":"The learning statement models validation and conversion tools in Ralph empower you to work with an LRS and ensure the quality of xAPI statements. These features not only enhance the integrity of your learning data but also facilitate integration and compliance with industry standards.
This section provides insights into the supported models, their conversion, and validation.
"},{"location":"models/#supported_statements","title":"Supported statements","text":"Learning statement models encompass a wide array of xAPI and OpenEdx statement types, ensuring comprehensive support for your e-learning data.
-
xAPI statements models:
- LMS
- Video
- Virtual classroom
-
OpenEdx statements models:
- Enrollment
- Navigational
- Open Reponse Assessment
- Peer instruction
- Problem interaction
- Textbook interaction
- Video interaction
"},{"location":"models/#statements_validation","title":"Statements validation","text":"In learning analytics, the validation of statements takes on significant importance. These statements, originating from diverse sources, systems or applications, must align with specific standards such as xAPI for the best known. The validation process becomes essential in ensuring that these statements meet the required standards, facilitating data quality and reliability.
Ralph allows you to automate the validation process in your production stack. OpenEdx related events and xAPI statements are supported.
Warning
For now, validation is effective only with supported learning statement models on Ralph. About xAPI statements, an issue is open to extend validation to any xAPI statement.
Check out tutorials to test the validation feature:
validate
with Ralph as a CLI validate
with Ralph as a library
"},{"location":"models/#statements_conversion","title":"Statements conversion","text":"Ralph currently supports conversion from OpenEdx learning events to xAPI statements. Here is the up-to-date conversion sets availables:
FROM TO edx.course.enrollment.activated registered to a course edx.course.enrollment.deactivated unregistered to a course load_video/edx.video.loaded initialized a video play_video/edx.video.played played a video pause_video/edx.video.paused paused a video stop_video/edx.video.stopped terminated a video seek_video/edx.video.position.changed seeked in a video Check out tutorials to test the conversion feature:
convert
with Ralph as a CLI convert
with Ralph as a library
"},{"location":"tutorials/cli/","title":"How to use Ralph as a CLI ?","text":"WIP.
"},{"location":"tutorials/cli/#prerequisites","title":"Prerequisites","text":" - Ralph should be properly installed to be used as a
CLI
. Follow Installation section for more information - [Recommended] To easily manipulate JSON streams, please install
jq
on your machine
"},{"location":"tutorials/cli/#validate_command","title":"validate
command","text":"In this tutorial, we\u2019ll walk you through the process of using validate
command to check the validity of xAPI statements.
"},{"location":"tutorials/cli/#with_an_invalid_xapi_statement","title":"With an invalid xAPI statement","text":"First, let\u2019s test the validate
command with a dummy JSON
string.
- Create in the terminal a dummy statement as follows:
invalid_statement='{\"foo\": \"invalid xapi\"}'\n
- Run validation on this statement with this command:
echo \"$invalid_statement\" | ralph validate -f xapi \n
- You should observe the following output from the terminal:
INFO ralph.cli Validating xapi events (ignore_errors=False | fail-on-unknown=False)\nERROR ralph.models.validator No matching pydantic model found for input event\nINFO ralph.models.validator Total events: 1, Invalid events: 1\n
"},{"location":"tutorials/cli/#with_a_valid_xapi_statement","title":"With a valid xAPI statement","text":"Now, let\u2019s test the validate
command with a valid xAPI statement.
The tutorial is made on a completed video
xAPI statement.
Info According to the specification, an xAPI statement to be valid should contain, at least the three following fields:
- an
actor
(with a correct IFI), - a
verb
(with an id
property), - an
object
(with an id
property).
- Create in the terminal a valid xAPI statement as follows:
valid_statement='{\"actor\": {\"mbox\": \"mailto:johndoe@example.com\", \"name\": \"John Doe\"}, \"verb\": {\"id\": \"http://adlnet.gov/expapi/verbs/completed\"}, \"object\": {\"id\": \"http://example.com/video/001-introduction\"}, \"timestamp\": \"2023-10-31T15:30:00Z\"}'\n
- Run validation on this statement with this command:
echo \"$valid_statement\" | bin/ralph validate -f xapi \n
- You should observe the following output from the terminal:
INFO ralph.cli Validating xapi events (ignore_errors=False | fail-on-unknown=False)\nINFO ralph.models.validator Total events: 1, Invalid events: 1\n
"},{"location":"tutorials/cli/#convert_command","title":"convert
command","text":"In this tutorial, you\u2019ll learn how to convert OpenEdx events into xAPI statements with Ralph.
Note
Please note that this feature is currently only supported for a set of OpenEdx events. When converting Edx events to xAPI statements, always refer to the list of supported event types to ensure accurate and successful conversion.
For this example, let\u2019s choose the page_close
OpenEdx event that is converted into a terminated a page
xAPI statement.
- Create in the terminal a
page_close
OpenEdx event as follows:
edx_statements={\"username\": \"\", \"ip\": \"0.0.0.0\", \"agent\": \"0\", \"host\": \"0\", \"referer\": \"\", \"accept_language\": \"0\", \"context\": {\"course_id\": \"\", \"course_user_tags\": null, \"module\": null, \"org_id\": \"0\", \"path\": \".\", \"user_id\": null}, \"time\": \"2000-01-01T00:00:00\", \"page\": \"http://A.ac/\", \"event_source\": \"browser\", \"session\": \"\", \"event\": \"{}\", \"event_type\": \"page_close\", \"name\": \"page_close\"}\n
- Convert this statement into a
terminated a page
statement with this command:
echo \"$edx_statements\" | \\ \nralph convert \\\n --platform-url \"http://lms-example.com\" \\\n --uuid-namespace \"ee241f8b-174f-5bdb-bae9-c09de5fe017f\" \\\n --from edx \\\n --to xapi | \\\n jq\n
- You should observe the following output from the terminal:
INFO ralph.cli Converting edx events to xapi format (ignore_errors=False | fail-on-unknown=False)\nINFO ralph.models.converter Total events: 1, Invalid events: 0\n{\n \"id\": \"8670c7d4-5485-52bd-b10a-a8ae27a51501\",\n \"actor\": {\n \"account\": {\n \"homePage\": \"http://lms-example.com\",\n \"name\": \"anonymous\"\n }\n },\n \"verb\": {\n \"id\": \"http://adlnet.gov/expapi/verbs/terminated\"\n },\n \"object\": {\n \"id\": \"http://A.ac/\",\n \"definition\": {\n \"type\": \"http://activitystrea.ms/schema/1.0/page\"\n }\n },\n \"timestamp\": \"2000-01-01T00:00:00\",\n \"version\": \"1.0.0\"\n}\n
\ud83c\udf89 Congratulations! You just have converted an event generated from OpenEdx LMS to a standardised xAPI statement!
Store locally converted statements To stored the converted statements locally on your machine, send the output of the convert
command to a JSON
file as follows:
echo \"$edx_statements\" | \\ \nralph convert \\\n --platform-url \"http://lms-example.com\" \\\n --uuid-namespace \"ee241f8b-174f-5bdb-bae9-c09de5fe017f\" \\\n --from edx \\\n --to xapi \\\n > converted_event.json\n
"},{"location":"tutorials/development_contributing/","title":"How to start developing with Ralph?","text":""},{"location":"tutorials/development_contributing/#prepare_your_environment","title":"Prepare your environment","text":"Welcome to our developer contribution guidelines! If you\u2019re interested in contributing to our project, there are a few prerequisites to get you started. Ralph development environment is containerized with Docker for consistency. Before diving in, ensure you have the following installed:
- Docker Engine
- Docker Compose
- make
"},{"location":"tutorials/development_contributing/#bootstrap_ralph_for_development","title":"Bootstrap Ralph for development","text":"To start playing with ralph
, you should first bootstrap
using:
make bootstrap\n
Once the project has been bootstrapped, you may want to edit generated .env
file to set up available backend parameters that will be injected into the running container as environment variables to configure Ralph (see backends documentation):
# Elasticsearch backend\nRALPH_BACKENDS__LRS__ES__HOSTS=http://elasticsearch:9200\nRALPH_BACKENDS__LRS__ES__INDEX=statements\nRALPH_BACKENDS__LRS__ES__TEST_HOSTS=http://elasticsearch:9200\nRALPH_BACKENDS__LRS__ES__TEST_INDEX=test-index\n\n# [...]\n
Uncomment lines to define environment variables Lines starting with a #
are considered as commented and thus will have no effect while running Ralph.
"},{"location":"tutorials/development_contributing/#working_with_backends","title":"Working with backends","text":"To configure the backends, we provide default parameters in the .env.dist
template which is copied to the .env
file when bootstrapping the project for the first time. You can uncomment them so that they are properly injected in running containers.
In order to run the Elasticsearch backend locally on GNU/Linux operating systems, ensure that your virtual memory limits are not too low and increase them if needed by typing this command from your terminal (as root
or using sudo
):
sysctl -w vm.max_map_count=262144
Reference: https://www.elastic.co/guide/en/elasticsearch/reference/master/vm-max-map-count.html
Once configured, start the backend container using:
make run-[BACKEND]\n
Substitute [BACKEND]
by the backend name, e.g. es
for Elasticsearch or swift
for OpenStack Swift:
# Start Elasticsearch backend\nmake run-es\n# Start Swift backend\nmake run-swift\n# Start Mongo backend\nmake run-mongo\n# Start ClickHouse backend\nmake run-clickhouse\n# Start all backends\nmake run-all\n
Disk space for Elasticsearch
Ensure that you have at least 10% of available disk space on your machine to run Elasticsearch.
Now that you have started at least the elasticsearch
and swift
backends, it\u2019s time to play with them:
# Store a JSON file in the Swift backend\necho '{\"id\": 1, \"foo\": \"bar\"}' | \\\n ./bin/ralph write -b swift -t foo.json\n\n# Check that we have created a new JSON file in the Swift backend\nbin/ralph list -b swift\n>>> foo.json\n\n# Read the content of the JSON file and index it in Elasticsearch\nbin/ralph read -b swift -t foo.json | \\\n bin/ralph write -b es\n\n# Check that we have properly indexed the JSON file in Elasticsearch\nbin/ralph read -b es\n>>> {\"id\": 1, \"foo\": \"bar\"}\n
"},{"location":"tutorials/development_contributing/#wip_working_with_the_lrs","title":"[WIP] Working with the LRS","text":""},{"location":"tutorials/development_contributing/#working_with_ralphs_tray","title":"Working with Ralph\u2019s tray","text":"Ralph is distributed along with its tray (a deployable package for Kubernetes clusters using Arnold). If you intend to work on this tray, please refer to Arnold\u2019s documentation first.
"},{"location":"tutorials/development_contributing/#prerequisites","title":"Prerequisites","text":" - Kubectl (>
v.1.23.5
): This CLI is used to communicate with the running Kubernetes instance you will use. - k3d (>
v.5.0.0
): This tool is used to set up and run a lightweight Kubernetes cluster, in order to have a local environment (it is required to complete quickstart instructions below to avoid depending on an existing Kubernetes cluster). - curl is required by Arnold\u2019s CLI.
- gnupg to encrypt Ansible vaults passwords and collaborate with your team.
"},{"location":"tutorials/development_contributing/#create_a_local_k3d_cluster","title":"Create a local k3d
cluster","text":"To create (or run) a local kubernetes cluster, we use k3d
. The cluster\u2019s bootstrapping should be run via:
make k3d-cluster\n
Running a k3d-cluster locally supposes that the 80 and 443 ports of your machine are available, so that the ingresses created for your project responds properly. If one or both ports are already used by another service running on your machine, the make k3d-cluster
command may fail.
You can check that your cluster is running using the k3d cluster
command:
k3d cluster list\n
You should expect the following output:
NAME SERVERS AGENTS LOADBALANCER\nralph 1/1 0/0 true\n
As you can see, we are running a single node cluster called ralph
.
"},{"location":"tutorials/development_contributing/#bootstrap_an_arnold_project","title":"Bootstrap an Arnold project","text":"Once your Kubernetes cluster is running, you need to create a standard Arnold project describing applications and environments you need to deploy:
make arnold-bootstrap\n
Once bootstrapped, Arnold should have created a group_vars
directory.
Run the following command to discover the directory tree.
tree group_vars\n
The output should be as follows:
group_vars\n\u251c\u2500\u2500 common\n\u2514\u2500\u2500 customer\n \u2514\u2500\u2500 ralph\n \u251c\u2500\u2500 development\n \u2502\u00a0\u00a0 \u251c\u2500\u2500 main.yml\n \u2502\u00a0\u00a0 \u2514\u2500\u2500 secrets\n \u2502\u00a0\u00a0 \u251c\u2500\u2500 databases.vault.yml\n \u2502\u00a0\u00a0 \u251c\u2500\u2500 elasticsearch.vault.yml\n \u2502\u00a0\u00a0 \u2514\u2500\u2500 ralph.vault.yml\n \u2514\u2500\u2500 main.yml\n\n5 directories, 5 files\n
To create the LRS credentials file, you need to provide a list of accounts allowed to request the LRS in Ralph\u2019s vault:
# Setup your kubernetes environment\nsource .k3d-cluster.env.sh\n\n# Decrypt the vault\nbin/arnold -d -c ralph -e development -- vault -a ralph decrypt\n
Edit the vault file to add a new account for the foo
user with the bar
password and a relevant scope:
# group_vars/customer/ralph/development/secrets/ralph.vault.yml\n#\n# [...]\n#\n# LRS\nLRS_AUTH:\n - username: \"foo\"\n hash: \"$2b$12$lCggI749U6TrzK7Qyr7xGe1KVSAXdPjtkMew.BD6lzIk//T5YSb72\"\n scopes:\n - \"all\"\n
The password hash has been generated using bcrypt
as explained in the API user guide.
And finally (re-)encrypt Ralph\u2019s vault:
bin/arnold -d -c ralph -e development -- vault -a ralph encrypt\n
You are now ready to create the related Kubernetes Secret while initializing Arnold project in the next step.
"},{"location":"tutorials/development_contributing/#prepare_working_namespace","title":"Prepare working namespace","text":"You are now ready to create required Kubernetes objects to start working on Ralph\u2019s deployment:
make arnold-init\n
At this point an Elasticsearch cluster should be running on your Kubernetes cluster:
kubectl -n development-ralph get -l app=elasticsearch pod\nNAME READY STATUS RESTARTS AGE\nelasticsearch-node-0 1/1 Running 0 69s\nelasticsearch-node-1 1/1 Running 0 69s\nelasticsearch-node-2 1/1 Running 0 69s\nes-index-template-j-221010-09h25m24s-nx5qz 0/1 Completed 0 49s\n
We are now ready to deploy Ralph to Kubernetes!
"},{"location":"tutorials/development_contributing/#deploy_code_repeat","title":"Deploy, code, repeat","text":"To test your local docker image, you need to build it and publish it to the local kubernetes cluster docker registry using the k3d-push
Makefile rule:
make k3d-push\n
Note
Each time you modify Ralph\u2019s application or its Docker image, you will need to make this update.
Now that your Docker image is published, it\u2019s time to deploy it!
make arnold-deploy\n
To test this deployment, let\u2019s try to make an authenticated request to the LRS:
curl -sLk \\\n --user foo:bar \\\n \"https://$(\\\n kubectl -n development-ralph \\\n get \\\n ingress/ralph-app-current \\\n -o jsonpath='{.spec.rules[0].host}')/whoami\"\n
And why not send test statements from Potsie\u2019s repository:
curl -sL \\\n https://github.com/openfun/potsie/raw/main/fixtures/elasticsearch/lrs.json.gz | \\\ngunzip | \\\nhead -n 100 | \\\njq -s . | \\\nsed \"s/@timestamp/timestamp/g\" | \\\ncurl -sLk \\\n --user foo:bar \\\n -X POST \\\n -H \"Content-Type: application/json\" \\\n \"https://$(\\\n kubectl -n development-ralph \\\n get \\\n ingress/ralph-app-current \\\n -o jsonpath='{.spec.rules[0].host}')/xAPI/statements/\" \\\n -d @-\n
Install jq
This example requires jq
command to serialize the request payload (xAPI statements). When dealing with JSON data, we strongly recommend installing it to manipulate them from the command line.
"},{"location":"tutorials/development_contributing/#perform_arnolds_operations","title":"Perform Arnold\u2019s operations","text":"If you want to run the bin/arnold
script to run specific Arnold commands, you must ensure that your environment is properly set and that Arnold runs in development mode (i.e. using the -d
flag):
source .k3d-cluster.env.sh\nbin/arnold -d -c ralph -e development -- vault -a ralph view\n
"},{"location":"tutorials/development_contributing/#stop_k3d_cluster","title":"Stop k3d
cluster","text":"When finished to work on the Tray, you can stop the k3d
cluster using the k3d-stop
helper:
make k3d-stop\n
"},{"location":"tutorials/development_contributing/#test_your_development","title":"Test your development","text":"To run tests on your code, either use the test
Make target or the bin/pytest
script to pass specific arguments to the test runner:
# Run all tests\nmake test\n\n# Run pytest with options\nbin/pytest -x -k mixins\n\n# Run pytest with options and more debugging logs\nbin/pytest tests/api -x -vvv -s --log-level=DEBUG -k mixins\n
"},{"location":"tutorials/development_contributing/#lint_your_code","title":"Lint your code","text":"To lint your code, either use the lint
meta target or one of the linting tools we use:
# Run all linters\nmake lint\n\n# Run ruff linter\nmake lint-ruff\n\n# Run ruff linter and resolve fixable errors\nmake lint-ruff-fix\n\n# List available linters\nmake help | grep lint-\n
"},{"location":"tutorials/development_contributing/#document_your_modification","title":"Document your modification","text":"In case you need to document your code, use the following targets:
# Build documentation site\nmake docs-build\n\n# Run mkdocs live server for dev docs\nmake docs-serve\n
"},{"location":"tutorials/library/","title":"How to use Ralph as a library ?","text":"WIP.
"},{"location":"tutorials/library/#validate_method","title":"validate
method","text":"WIP.
"},{"location":"tutorials/library/#convert_method","title":"convert
method","text":"WIP.
"}]}
\ No newline at end of file
+{"config":{"lang":["en"],"separator":"[\\s\\-]+","pipeline":["stopWordFilter"]},"docs":[{"location":"","title":"Ralph","text":" \u2699\ufe0f The ultimate toolbox for your learning analytics (expect some xAPI \u2764\ufe0f)
Ralph is a toolbox for your learning analytics, it can be used as a:
- LRS, a HTTP API server to collect xAPI statements (learning events), following the ADL LRS standard
- command-line interface (CLI), to build data pipelines the UNIX-way\u2122\ufe0f,
- library, to fetch learning events from various backends, (de)serialize or convert them from and to various standard formats such as xAPI, or openedx
"},{"location":"#installation","title":"Installation","text":""},{"location":"#install_from_pypi","title":"Install from PyPI","text":"Ralph is distributed as a standard python package; it can be installed via pip
or any other python package manager (e.g. Poetry, Pipenv, etc.):
Use a virtual environment for installation To maintain a clean and controlled environment when installing ralph-malph
, consider using a virtual environment.
-
Create a virtual environment:
python3.12 -m venv <path-to-virtual-environment>\n
-
Activate the virtual environment:
source venv/bin/activate\n
If you only need to integrate learning statement models feature in your project, you don\u2019t need to install the backends
, cli
or lrs
extra dependencies, the core library is what you need:
pip install ralph-malph\n
If you want to use the Ralph LRS server, add the lrs
flavour in your installation. You also have to choose the type of backend you will use for LRS data storage (backend-clickhouse
,backend-es
,backend-mongo
).
- Install the core package with the LRS and the Elasticsearch backend. For example:
pip install ralph-malph[backend-es,lrs]\n
- Add the
cli
flavour if you want to use the LRS on the command line:
pip install ralph-malph[backend-es,lrs,cli]\n
- If you have various uses for Ralph\u2019s features or would like to discover all the existing functionnalities, it is recommended to install the full package:
pip install ralph-malph[backend-clickhouse,backend-es,backend-ldp,backend-lrs,backend-mongo,backend-s3,backend-swift,backend-ws,cli,lrs]\n
"},{"location":"#install_from_dockerhub","title":"Install from DockerHub","text":"Ralph is distributed as a Docker image. If Docker is installed on your machine, it can be pulled from DockerHub:
docker run --rm -i fundocker/ralph:latest ralph --help\n
Use a ralph
alias in your local environment Simplify your workflow by creating an alias for easy access to Ralph commands:
alias ralph=\"docker run --rm -i fundocker/ralph:latest ralph\"\n
"},{"location":"#wip_start_using_ralph","title":"[WIP] Start using Ralph","text":""},{"location":"#contributing_to_ralph","title":"Contributing to Ralph","text":"If you\u2019re interested in contributing to Ralph, whether it\u2019s by reporting issues, suggesting improvements, or submitting code changes, please head over to our dedicated Contributing to Ralph page. There, you\u2019ll find detailed guidelines and instructions on how to take part in the project.
We look forward to your contributions and appreciate your commitment to making Ralph a more valuable tool for everyone.
"},{"location":"#contributors","title":"Contributors","text":""},{"location":"#license","title":"License","text":"This work is released under the MIT License (see LICENSE).
"},{"location":"CHANGELOG/","title":"Changelog","text":"All notable changes to this project will be documented in this file.
The format is based on Keep a Changelog, and this project adheres to Semantic Versioning.
"},{"location":"CHANGELOG/#unreleased","title":"Unreleased","text":""},{"location":"CHANGELOG/#added","title":"Added","text":" - Implement Pydantic model for LRS Statements resource query parameters
- Implement xAPI LMS Profile statements validation
EdX
to xAPI
converters for enrollment events - Backends: Add
Writable
and Listable
interfaces to distinguish supported functionalities among data
backends - Backends: Add
get_backends
function to automatically discover backends for CLI and LRS usage
"},{"location":"CHANGELOG/#changed","title":"Changed","text":" - Upgrade base python version to 3.12 for the development stack and Docker image
- Upgrade
cachetools
to 5.3.2
- Refactor
database
and storage
backends under the unified data
backend interface [BC] - Refactor LRS
query_statements
and query_statements_by_ids
backends methods under the unified lrs
backend interface [BC] - Refactor LRS Statements resource query parameters defined for
ralph
API - User credentials must now include an \u201cagent\u201d field which can be created using the cli
GET /statements
now has \u201cmine\u201d option which matches statements that have an authority field matching that of the user - CLI: change
push
to write
and fetch
to read
[BC] - Upgrade
fastapi
to 0.104.1
- Upgrade
sentry_sdk
to 1.34.0
- Upgrade
uvicorn
to 0.24.0.post1
- API: Invalid parameters now return 400 status code
- API: Forwarding PUT now uses PUT (instead of POST)
- Models: The xAPI
context.contextActivities.category
field is now mandatory in the video and virtual classroom profiles. [BC] - Backends:
LRSHTTP
methods must not be used in asyncio
events loop (BC) - Add variable to override PVC name in arnold deployment
- Backends: add
max_statements
option to AsyncLRSHTTP
- API: Incoming statements are enriched with
id
, timestamp
, stored
and authority
- Backends: update
statementId
and voidedStatementId
to snake_case, with camelCase alias, in LRSStatementsQuery
- API: Add
RALPH_LRS_RESTRICT_BY_AUTHORITY
option making ?mine=True
implicit - CLI: list cli usage strings in alphabetical order
- API: Add
RALPH_LRS_RESTRICT_BY_SCOPE
option enabling endpoint access control by user scopes - Backends: Replace reference to a JSON column in ClickHouse with function calls on the String column [BC]
- API: enhance \u2018limit\u2019 query parameter\u2019s validation
- API: Variable
RUNSERVER_AUTH_BACKEND
becomes RUNSERVER_AUTH_BACKENDS
, and multiple authentication methods are supported simultaneously
"},{"location":"CHANGELOG/#fixed","title":"Fixed","text":" - API: Fix a typo (\u2018attachements\u2019 -> \u2018attachments\u2019) to ensure compliance with the LRS specification and prevent potential silent bugs.
"},{"location":"CHANGELOG/#removed","title":"Removed","text":" school
, course
, module
context extensions in Edx to xAPI base converter name
field in VideoActivity
xAPI model mistakenly used in video
profile - drop support for python 3.7
"},{"location":"CHANGELOG/#390_-_2023-07-21","title":"3.9.0 - 2023-07-21","text":""},{"location":"CHANGELOG/#changed_1","title":"Changed","text":" - Upgrade
fastapi
to 0.100.0
- Upgrade
sentry_sdk
to 1.28.1
- Upgrade
uvicorn
to 0.23.0
- Enforce valid IRI for
activity
parameter in GET /statements
- Change how duplicate xAPI statements are handled for
clickhouse
backend
"},{"location":"CHANGELOG/#380_-_2023-06-21","title":"3.8.0 - 2023-06-21","text":""},{"location":"CHANGELOG/#added_1","title":"Added","text":" - Implement edX open response assessment events pydantic models
- Implement edx peer instruction events pydantic models
- Implement xAPI VideoDownloaded pydantic model (using xAPI TinCan
downloaded
verb)
"},{"location":"CHANGELOG/#changed_2","title":"Changed","text":" - Allow to use a query for HTTP backends in the CLI
"},{"location":"CHANGELOG/#370_-_2023-06-13","title":"3.7.0 - 2023-06-13","text":""},{"location":"CHANGELOG/#added_2","title":"Added","text":" - Implement asynchronous
async_lrs
backend - Implement synchronous
lrs
backend - Implement xAPI virtual classroom pydantic models
- Allow to insert custom endpoint url for S3 service
- Cache the HTTP Basic auth credentials to improve API response time
- Support OpenID Connect authentication method
"},{"location":"CHANGELOG/#changed_3","title":"Changed","text":" - Clean xAPI pydantic models naming convention
- Upgrade
fastapi
to 0.97.0
- Upgrade
sentry_sdk
to 1.25.1
- Set Clickhouse
client_options
to a dedicated pydantic model - Upgrade
httpx
to 0.24.1
- Force a valid (JSON-formatted) IFI to be passed for the
/statements
GET query agent
filtering - Upgrade
cachetools
to 5.3.1
"},{"location":"CHANGELOG/#removed_1","title":"Removed","text":" verb
.display
field no longer mandatory in xAPI models and for converter
"},{"location":"CHANGELOG/#360_-_2023-05-17","title":"3.6.0 - 2023-05-17","text":""},{"location":"CHANGELOG/#added_3","title":"Added","text":" - Allow to ignore health check routes for Sentry transactions
"},{"location":"CHANGELOG/#changed_4","title":"Changed","text":" - Upgrade
sentry_sdk
to 1.22.2
- Upgrade
uvicorn
to 0.22.0
- LRS
/statements
GET
method returns a code 400 with certain parameters as per the xAPI specification - Use batch/v1 api in cronjob_pipeline manifest
- Use autoscaling/v2 in HorizontalPodAutoscaler manifest
"},{"location":"CHANGELOG/#fixed_1","title":"Fixed","text":" - Fix the
more
IRL building in LRS /statements
GET requests
"},{"location":"CHANGELOG/#351_-_2023-04-18","title":"3.5.1 - 2023-04-18","text":""},{"location":"CHANGELOG/#changed_5","title":"Changed","text":" - Upgrade
httpx
to 0.24.0
- Upgrade
fastapi
to 0.95.1
- Upgrade
sentry_sdk
to 1.19.1
- Upgrade
uvicorn
to 0.21.1
"},{"location":"CHANGELOG/#fixed_2","title":"Fixed","text":" - An issue with starting Ralph in pre-built Docker containers
- Fix double quoting in ClickHouse backend server parameters
- An issue Ralph starting when ClickHouse is down
"},{"location":"CHANGELOG/#350_-_2023-03-08","title":"3.5.0 - 2023-03-08","text":""},{"location":"CHANGELOG/#added_4","title":"Added","text":" - Implement PUT verb on statements endpoint
- Add ClickHouse database backend support
"},{"location":"CHANGELOG/#changed_6","title":"Changed","text":" - Make trailing slashes optional on statements endpoint
- Upgrade
sentry_sdk
to 1.16.0
"},{"location":"CHANGELOG/#340_-_2023-03-01","title":"3.4.0 - 2023-03-01","text":""},{"location":"CHANGELOG/#changed_7","title":"Changed","text":" - Upgrade
fastapi
to 0.92.0
- Upgrade
sentry_sdk
to 1.15.0
"},{"location":"CHANGELOG/#fixed_3","title":"Fixed","text":" - Restore sentry integration in the LRS server
"},{"location":"CHANGELOG/#330_-_2023-02-03","title":"3.3.0 - 2023-02-03","text":""},{"location":"CHANGELOG/#added_5","title":"Added","text":" - Restore python 3.7+ support for library usage (models)
"},{"location":"CHANGELOG/#changed_8","title":"Changed","text":" - Allow xAPI extra fields in
extensions
fields
"},{"location":"CHANGELOG/#321_-_2023-02-01","title":"3.2.1 - 2023-02-01","text":""},{"location":"CHANGELOG/#changed_9","title":"Changed","text":" - Relax required Python version to
3.7
+
"},{"location":"CHANGELOG/#320_-_2023-01-25","title":"3.2.0 - 2023-01-25","text":""},{"location":"CHANGELOG/#added_6","title":"Added","text":" - Add a new
auth
subcommand to generate required credentials file for the LRS - Implement support for AWS S3 storage backend
- Add CLI
--version
option
"},{"location":"CHANGELOG/#changed_10","title":"Changed","text":" - Upgrade
fastapi
to 0.89.1
- Upgrade
httpx
to 0.23.3
- Upgrade
sentry_sdk
to 1.14.0
- Upgrade
uvicorn
to 0.20.0
- Tray: add the
ca_certs
path for the ES backend client option (LRS) - Improve Sentry integration for the LRS
- Update handbook link to
https://handbook.openfun.fr
- Upgrade base python version to 3.11 for the development stack and Docker image
"},{"location":"CHANGELOG/#fixed_4","title":"Fixed","text":" - Restore ES and Mongo backends ability to use client options
"},{"location":"CHANGELOG/#310_-_2022-11-17","title":"3.1.0 - 2022-11-17","text":""},{"location":"CHANGELOG/#added_7","title":"Added","text":" - EdX to xAPI converters for video events
"},{"location":"CHANGELOG/#changed_11","title":"Changed","text":" - Improve Ralph\u2019s library integration by unpinning dependencies (and prefer ranges)
- Upgrade
fastapi
to 0.87.0
"},{"location":"CHANGELOG/#removed_2","title":"Removed","text":" - ModelRules constraint
"},{"location":"CHANGELOG/#300_-_2022-10-19","title":"3.0.0 - 2022-10-19","text":""},{"location":"CHANGELOG/#added_8","title":"Added","text":" - Implement edX video browser events pydantic models
- Create a
post
endpoint for statements implementing the LRS spec - Implement support for the MongoDB database backend
- Implement support for custom queries when using database backends
get
method (used in the fetch
command) - Add dotenv configuration file support and
python-dotenv
dependency - Add
host
and port
options for the runserver
cli command - Add support for database selection when running the Ralph LRS server
- Implement support for xAPI statement forwarding
- Add database backends
status
checking - Add
health
LRS router - Tray: add LRS server support
"},{"location":"CHANGELOG/#changed_12","title":"Changed","text":" - Migrate to
python-legacy
handler for mkdocstrings
package - Upgrade
click
to 8.1.3
- Upgrade
elasticsearch
to 8.3.3
- Upgrade
fastapi
to 0.79.1
- Upgrade
ovh
to 1.0.0
- Upgrade
pydantic
to 1.9.2
- Upgrade
pymongo
to 4.2.0
- Upgrade
python-keystoneclient
to 5.0.0
- Upgrade
python-swiftclient
to 4.0.1
- Upgrade
requests
to 2.28.1
- Upgrade
sentry_sdk
to 1.9.5
- Upgrade
uvicorn
to 0.18.2
- Upgrade
websockets
to 10.3
- Make backends yield results instead of writing to standard streams (BC)
- Use pydantic settings management instead of global variables in defaults.py
- Rename backend and parser parameter environment variables (BC)
- Make project dependencies management more modular for library usage
"},{"location":"CHANGELOG/#removed_3","title":"Removed","text":" - Remove YAML configuration file support and
pyyaml
dependency (BC)
"},{"location":"CHANGELOG/#fixed_5","title":"Fixed","text":" - Tray: do not create a cronjobs list when no cronjob has been defined
- Restore history mixin logger
"},{"location":"CHANGELOG/#210_-_2022-04-13","title":"2.1.0 - 2022-04-13","text":""},{"location":"CHANGELOG/#added_9","title":"Added","text":" - Implement edX problem interaction events pydantic models
- Implement edX textbook interaction events pydantic models
ws
websocket stream backend (compatible with the fetch
command) - bundle
jq
, curl
and wget
in the fundocker/ralph
Docker image - Tray: enable ralph app deployment command configuration
- Add a
runserver
command with basic auth and a Whoami
route - Create a
get
endpoint for statements implementing the LRS spec - Add optional fields to BaseXapiModel
"},{"location":"CHANGELOG/#changed_13","title":"Changed","text":" - Upgrade
uvicorn
to 0.17.4
- Upgrade
elasticsearch
to 7.17.0
- Upgrade
sentry_sdk
to 1.5.5
- Upgrade
fastapi
to 0.73.0
- Upgrade
pyparsing
to 3.0.7
- Upgrade
pydantic
to 1.9.0
- Upgrade
python-keystoneclient
to 4.4.0
- Upgrade
python-swiftclient
to 3.13.0
- Upgrade
pyyaml
to 6.0
- Upgrade
requests
to 2.27.1
- Upgrade
websockets
to 10.1
"},{"location":"CHANGELOG/#201_-_2021-07-15","title":"2.0.1 - 2021-07-15","text":""},{"location":"CHANGELOG/#changed_14","title":"Changed","text":" - Upgrade
elasticsearch
to 7.13.3
"},{"location":"CHANGELOG/#fixed_6","title":"Fixed","text":" - Restore elasticsearch backend datastream compatibility for bulk operations
"},{"location":"CHANGELOG/#200_-_2021-07-09","title":"2.0.0 - 2021-07-09","text":""},{"location":"CHANGELOG/#added_10","title":"Added","text":" - xAPI video
interacted
pydantic models - xAPI video
terminated
pydantic models - xAPI video
completed
pydantic models - xAPI video
seeked
pydantic models - xAPI video
initialized
pydantic models - xAPI video
paused
pydantic models convert
command to transform edX events to xAPI format - EdX to xAPI converters for page
viewed
andpage_close
events - Implement core event format converter
- xAPI video
played
pydantic models - xAPI page
viewed
and page terminated
pydantic models - Implement edX navigational events pydantic models
- Implement edX enrollment events pydantic models
- Install security updates in project Docker images
- Model selector to retrieve associated pydantic model of a given event
validate
command to lint edX events using pydantic models - Support all available bulk operation types for the elasticsearch backend (create, index, update, delete) using the
--es-op-type
option
"},{"location":"CHANGELOG/#changed_15","title":"Changed","text":" - Upgrade
elasticsearch
to 7.13.2
- Upgrade
python-swiftclient
to 3.12.0
- Upgrade
click
to 8.0.1
- Upgrade
click-option-group
to 0.5.3
- Upgrade
pydantic
to 1.8.2
- Upgrade
sentry_sdk
to 1.1.0
- Rename edX models
- Migrate model tests from factories to hypothesis strategies
- Tray: switch from openshift to k8s (BC)
- Tray: remove useless deployment probes
"},{"location":"CHANGELOG/#fixed_7","title":"Fixed","text":" - Tray: remove
version
immutable field in DC selector
"},{"location":"CHANGELOG/#120_-_2021-02-26","title":"1.2.0 - 2021-02-26","text":""},{"location":"CHANGELOG/#added_11","title":"Added","text":" - edX server event pydantic model and factory
- edX page_close browser event pydantic model and factory
- Tray: allow to specify a self-generated elasticsearch cluster CA certificate
"},{"location":"CHANGELOG/#fixed_8","title":"Fixed","text":" - Tray: add missing Swift variables in the secret
- Tray: fix pods anti-affinity selector
"},{"location":"CHANGELOG/#removed_4","title":"Removed","text":" pandas
is no longer required
"},{"location":"CHANGELOG/#110_-_2021-02-04","title":"1.1.0 - 2021-02-04","text":""},{"location":"CHANGELOG/#added_12","title":"Added","text":" - Support for Swift storage backend
- Use the
push
command --ignore-errors
option to ignore ES bulk import errors - The elasticsearch backend now accepts passing all supported client options
"},{"location":"CHANGELOG/#changed_16","title":"Changed","text":" - Upgrade
pyyaml
to 5.4.1
- Upgrade
pandas
to 1.2.1
"},{"location":"CHANGELOG/#removed_5","title":"Removed","text":" click_log
is no longer required as we are able to configure logging
"},{"location":"CHANGELOG/#100_-_2021-01-13","title":"1.0.0 - 2021-01-13","text":""},{"location":"CHANGELOG/#added_13","title":"Added","text":" - Implement base CLI commands (list, extract, fetch & push) for supported backends
- Support for ElasticSearch database backend
- Support for LDP storage backend
- Support for FS storage backend
- Parse (gzipped) tracking logs in GELF format
- Support for application\u2019s configuration file
- Add optional sentry integration
- Distribute Arnold\u2019s tray to deploy Ralph in a k8s cluster as cronjobs
"},{"location":"LICENSE/","title":"License","text":"MIT License
Copyright (c) 2020-present France Universit\u00e9 Num\u00e9rique
Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the \u201cSoftware\u201d), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions:
The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software.
THE SOFTWARE IS PROVIDED \u201cAS IS\u201d, WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
"},{"location":"UPGRADE/","title":"Upgrade","text":"All instructions to upgrade this project from one release to the next will be documented in this file. Upgrades must be run sequentially, meaning you should not skip minor/major releases while upgrading (fix releases can be skipped).
This project adheres to Semantic Versioning.
"},{"location":"UPGRADE/#3xx_to_400","title":"3.x.x to 4.0.0","text":""},{"location":"UPGRADE/#upgrade_user_credentials","title":"Upgrade user credentials","text":"To conform to xAPI specifications, we need to represent users as xAPI Agents. You must therefore delete and re-create the credentials file using the updated cli, OR you can modify it directly to add the agents
field. The credentials file is located in { RALPH_APP_DIR }/{ RALPH_AUTH_FILE }
(defaults to .ralph/auth.json
). Each user profile must follow the following pattern (see this post for examples of valid agent objects) :
{\n \"username\": \"USERNAME_UNCHANGED\",\n \"hash\": \"PASSWORD_HASH_UNCHANGED\",\n \"scopes\": [ LIST_OF_SCOPES_UNCHANGED ],\n \"agent\": { A_VALID_AGENT_OBJECT }\n}\n
Agent can take one of the following forms, as specified by the xAPI specification: - mbox: \"agent\": {\n \"mbox\": \"mailto:john.doe@example.com\"\n}\n
- mbox_sha1sum: \"agent\": {\n \"mbox_sha1sum\": \"ebd31e95054c018b10727ccffd2ef2ec3a016ee9\",\n}\n
- openid: \"agent\": {\n \"openid\": \"http://foo.openid.example.org/\"\n}\n
- account: \"agent\": {\n \"account\": {\n \"name\": \"simonsAccountName\",\n \"homePage\": \"http://www.exampleHomePage.com\"\n}\n
For example here is a valid auth.json
file:
[\n {\n \"username\": \"john.doe@example.com\",\n \"hash\": \"$2b$12$yBXrzIuRIk6yaft5KUgVFOIPv0PskCCh9PXmF2t7pno.qUZ5LK0D2\",\n \"scopes\": [\"example_scope\"],\n \"agent\": {\n \"mbox\": \"mailto:john.doe@example.com\"\n }\n },\n {\n \"username\": \"simon.says@example.com\",\n \"hash\": \"$2b$12$yBXrzIuRIk6yaft5KUgVFOIPv0PskCCh9PXmF2t7pno.qUZ5LK0D2\",\n \"scopes\": [\"second_scope\", \"third_scope\"],\n \"agent\": {\n \"account\": {\n \"name\": \"simonsAccountName\",\n \"homePage\": \"http://www.exampleHomePage.com\"\n }\n }\n }\n]\n
"},{"location":"UPGRADE/#upgrade_history_syntax","title":"Upgrade history syntax","text":"CLI syntax has been changed from fetch
& push
to read
& write
affecting the command history. You must replace the command history after updating: - locate your history file path, which is in { RALPH_APP_DIR }/history.json
(defaults to .ralph/history.json
) - run the commands below to update history
sed -i 's/\"fetch\"/\"read\"/g' { my_history_file_path }\nsed -i 's/\"push\"/\"write\"/g' { my_history_file_path }\n
"},{"location":"UPGRADE/#upgrade_clickhouse_schema","title":"Upgrade ClickHouse schema","text":"If you are using the ClickHouse backend, schema changes have been made to drop the existing JSON column in favor of the String version of the same data. See this issue for details.
Ralph does not manage the ClickHouse schema, so if you have existing data you will need to manually alter it as an admin user. Note: this will rewrite the statements table, which may take a long time if you have many rows. The command to run is:
-- If RALPH_BACKENDS__DATA__CLICKHOUSE__DATABASE is 'xapi'\n-- and RALPH_BACKENDS__DATA__CLICKHOUSE__EVENT_TABLE_NAME is 'test'\n\nALTER TABLE xapi.test DROP COLUMN event, RENAME COLUMN event_str to event;\n
"},{"location":"commands/","title":"Commands","text":""},{"location":"commands/#ralph","title":"ralph","text":"The cli is a stream-based tool to play with your logs.
It offers functionalities to: - Validate or convert learning data in different standards - Read and write learning data to various databases or servers - Manage an instance of a Ralph LRS server
Usage:
ralph [OPTIONS] COMMAND [ARGS]...\n
Options:
-v, --verbosity LVL Either CRITICAL, ERROR, WARNING, INFO (default) or\n DEBUG\n --version Show the version and exit.\n --help Show this message and exit.\n
"},{"location":"commands/#ralph-auth","title":"ralph auth","text":"Generate credentials for LRS HTTP basic authentication.
Usage:
ralph auth [OPTIONS]\n
Options:
-u, --username TEXT The user for which we generate credentials.\n [required]\n -p, --password TEXT The password to encrypt for this user. Will\n be prompted if missing. [required]\n -s, --scope TEXT The user scope(s). This option can be\n provided multiple times. [required]\n -M, --agent-ifi-mbox TEXT The mbox Inverse Functional Identifier of\n the associated agent.\n -S, --agent-ifi-mbox-sha1sum TEXT\n The mbox-sha1sum Inverse Functional\n Identifier of the associated agent.\n -O, --agent-ifi-openid TEXT The openid Inverse Functional Identifier of\n the associated agent.\n -A, --agent-ifi-account TEXT...\n Input \"{name} {homePage}\". The account\n Inverse Functional Identifier of the\n associated agent.\n -N, --agent-name TEXT The name of the associated agent.\n -w, --write-to-disk Write new credentials to the LRS\n authentication file.\n --help Show this message and exit.\n
"},{"location":"commands/#ralph-convert","title":"ralph convert","text":"Convert input events to a given format.
Usage:
ralph convert [OPTIONS]\n
Options:
From edX to xAPI converter options: \n -u, --uuid-namespace TEXT The UUID namespace to use for the `ID` field\n generation\n -p, --platform-url TEXT The `actor.account.homePage` to use in the\n xAPI statements [required]\n -f, --from [edx] Input events format to convert [required]\n -t, --to [xapi] Output events format [required]\n -I, --ignore-errors Continue writing regardless of raised errors\n -F, --fail-on-unknown Stop converting at first unknown event\n --help Show this message and exit.\n
"},{"location":"commands/#ralph-extract","title":"ralph extract","text":"Extract input events from a container format using a dedicated parser.
Usage:
ralph extract [OPTIONS]\n
Options:
-p, --parser [gelf|es] Container format parser used to extract events\n [required]\n --help Show this message and exit.\n
"},{"location":"commands/#ralph-validate","title":"ralph validate","text":"Validate input events of given format.
Usage:
ralph validate [OPTIONS]\n
Options:
-f, --format [edx|xapi] Input events format to validate [required]\n -I, --ignore-errors Continue validating regardless of raised errors\n -F, --fail-on-unknown Stop validating at first unknown event\n --help Show this message and exit.\n
"},{"location":"contribute/","title":"Contributing to Ralph","text":"Thank you for considering contributing to Ralph! We appreciate your interest and support. This documentation provides guidelines on how to contribute effectively to our project.
"},{"location":"contribute/#issues","title":"Issues","text":"Issues are a valuable way to contribute to Ralph. They can include bug reports, feature requests, and general questions or discussions. When creating or interacting with issues, please keep the following in mind:
"},{"location":"contribute/#1_search_for_existing_issues","title":"1. Search for existing issues","text":"Before creating a new issue, search the existing issues to see if your concern has already been raised. If you find a related issue, you can add your input or follow the discussion. Feel free to engage in discussions, offer help, or provide feedback on existing issues. Your input is valuable in shaping the project\u2019s future.
"},{"location":"contribute/#2_creating_a_new_issue","title":"2. Creating a new issue","text":"Use the provided issue template that fits the best to your concern. Provide as much information as possible when writing your issue. Your issue will be reviewed by a project maintainer and you may be offered to open a PR if you want to contribute to the code. If not, and if your issue is relevant, a contributor will apply the changes to the project. The issue will then be automatically closed when the PR is merged.
Issues will be closed by project maintainers if they are deemed invalid. You can always reopen an issue if you believe it hasn\u2019t been adequately addressed.
"},{"location":"contribute/#3_code_of_conduct_in_discussion","title":"3. Code of conduct in discussion","text":" - Be respectful and considerate when participating in discussions.
- Avoid using offensive language, and maintain a positive and collaborative tone.
- Stay on topic and avoid derailing discussions.
"},{"location":"contribute/#discussions","title":"Discussions","text":"Discussions in the Ralph repository are a place for open-ended conversations, questions, and general community interactions. Here\u2019s how to effectively use discussions:
"},{"location":"contribute/#1_creating_a_discussion","title":"1. Creating a discussion","text":" - Use a clear and concise title that summarizes the topic.
- In the description, provide context and details regarding the discussion.
- Use labels to categorize the discussion (e.g., \u201cquestion,\u201d \u201cgeneral discussion,\u201d \u201cannouncements,\u201d etc.).
"},{"location":"contribute/#2_participating_in_discussions","title":"2. Participating in discussions","text":" - Engage in conversations respectfully, respecting others\u2019 opinions.
- Avoid spamming or making off-topic comments.
- Help answer questions when you can.
"},{"location":"contribute/#pull_requests_pr","title":"Pull Requests (PR)","text":"Contributing to Ralph through pull requests is a powerful way to advance the project. If you want to make changes or add new features, please follow these steps to submit a PR:
"},{"location":"contribute/#1_fork_the_repository","title":"1. Fork the repository","text":"Begin by forking Ralph project\u2019s repository.
"},{"location":"contribute/#2_clone_the_fork","title":"2. Clone the fork","text":"Clone the forked repository to your local machine and change the directory to the project folder using the following commands (replace <your_fork>
with your GitHub username):
git clone https://github.com/<your_fork>/ralph.git\ncd ralph\n
"},{"location":"contribute/#3_create_a_new_branch","title":"3. Create a new branch","text":"Create a new branch for your changes, ideally with a descriptive name:
git checkout -b your-new-feature\n
"},{"location":"contribute/#4_make_changes","title":"4. Make changes","text":"Implement the changes or additions to the code, ensuring it follows OpenFUN coding and documentation standards.
For comprehensive guidance on starting your development journey with Ralph and preparing your pull request, please refer to our dedicated Start developing with Ralph tutorial.
When committing your changes, please adhere to OpenFUN commit practices. Follow the low granularity commit splitting approach and use commit messages based on the Angular commit message guidelines.
"},{"location":"contribute/#5_push_changes","title":"5. Push changes","text":"Push your branch to your GitHub repository:
git push origin feature/your-new-feature\n
"},{"location":"contribute/#6_create_a_pull_request","title":"6. Create a pull request","text":"To initiate a Pull Request (PR), head to Ralph project\u2019s GitHub repository and click on New Pull Request.
Set your branch as the source and Ralph project\u2019s master
branch as the target.
Provide a clear title for your PR and make use of the provided PR body template to document the changes made by your PR. This helps streamline the review process and maintain a well-documented project history.
"},{"location":"contribute/#7_review_and_discussion","title":"7. Review and discussion","text":"Ralph project maintainers will review your PR. Be prepared to make necessary changes or address any feedback. Patience during this process is appreciated.
"},{"location":"contribute/#8_merge","title":"8. Merge","text":"Once your PR is approved, Ralph maintainers will merge your changes into the main project. Congratulations, you\u2019ve successfully contributed to Ralph! \ud83c\udf89
"},{"location":"api/","title":"LRS HTTP server","text":"Ralph comes with an API server that aims to implement the Learning Record Store (LRS) specification (still a work in progress).
"},{"location":"api/#getting_started","title":"Getting started","text":"The API server supports the following authentication methods:
- HTTP Basic Authentication (default method)
- OpenID Connect authentication on top of OAuth2.0
"},{"location":"api/#http_basic_authentication","title":"HTTP Basic Authentication","text":"The default method for securing Ralph API server is with HTTP basic authentication.
The API server can be started up with the following command:
ralph runserver --backend es\n
The --backend
(or -b
) option specifies which database backend to use for LRS data storage and retrieval. See Ralph\u2019s backends documentation for more details.
However, before you can start your API server and make requests against it, you need to set up your credentials.
"},{"location":"api/#creating_a_credentials_file","title":"Creating a credentials file","text":"The credentials file is expected to be a valid JSON file. Its location is specified by the RALPH_AUTH_FILE
configuration value. By default, ralph
will look for the auth.json
file in the application directory (see click documentation for details).
The expected format is a list of entries (JSON objects) each containing the username, the user\u2019s bcrypt
hashed+salted password, scopes they can access, and an agent
object used to represent the user in the LRS. The agent
is constrained by LRS specifications, and must use one of four valid Inverse Functional Identifiers.
[\n {\n \"username\": \"john.doe@example.com\",\n \"hash\": \"$2b$12$yBXrzIuRIk6yaft5KUgVFOIPv0PskCCh9PXmF2t7pno.qUZ5LK0D2\",\n \"scopes\": [\"example_scope\"],\n \"agent\": {\n \"mbox\": \"mailto:john.doe@example.com\"\n }\n },\n {\n \"username\": \"simon.says@example.com\",\n \"hash\": \"$2b$12$yBXrzIuRIk6yaft5KUgVFOIPv0PskCCh9PXmF2t7pno.qUZ5LK0D2\",\n \"scopes\": [\"second_scope\", \"third_scope\"],\n \"agent\": {\n \"account\": {\n \"name\": \"simonsAccountName\",\n \"homePage\": \"http://www.exampleHomePage.com\"\n }\n }\n }\n]\n
To create a new user credentials, Ralph\u2019s CLI provides a dedicated command:
ralph auth \\\n --username janedoe \\\n --password supersecret \\\n --scope janedoe_scope \\\n --agent-ifi-mbox mailto:janedoe@example.com \\\n # or --agent-ifi-mbox-sha1sum ebd31e95054c018b10727ccffd2ef2ec3a016ee9 \\\n # or --agent-ifi-openid \"http://jane.openid.example.org/\" \\\n # or --agent-ifi-account exampleAccountname http://www.exampleHomePage.com \\\n -w\n
This command updates your credentials file with the new janedoe
user.
Note that running this command requires that you installed Ralph with the CLI optional dependencies, e.g. pip install ralph-malph[cli]
(which we highly recommend).
"},{"location":"api/#making_a_get_request","title":"Making a GET request","text":"The first request that can be answered by the ralph API server is a whoami
request, which checks if the user is authenticated and returns their username and permission scopes.
Use curl to get http://localhost:8100/whoami
:
curl http://localhost:8100/whoami\n< HTTP/1.1 401 Unauthorized\n< {\"error\":\"Not authenticated\"}\n
Send your username and password to the API server through HTTP Basic Auth:
curl --user john.doe@example.com:PASSWORD http://localhost:8100/whoami\n< HTTP/1.1 200 OK\n< {\"scopes\":[\"example_scope\"], \"agent\": {\"mbox\": \"mailto:john.doe@example.com\"}}\n
"},{"location":"api/#openid_connect_authentication","title":"OpenID Connect authentication","text":"Ralph LRS API server supports OpenID Connect (OIDC) on top of OAuth 2.0 for authentication and authorization.
To enable OIDC auth, you should modify the RALPH_RUNSERVER_AUTH_BACKENDS
environment variable by adding (or replacing by) oidc
:
RALPH_RUNSERVER_AUTH_BACKENDS=basic,oidc\n
and you should define the RALPH_RUNSERVER_AUTH_OIDC_ISSUER_URI
environment variable with your identity provider\u2019s Issuer Identifier URI as follows:
RALPH_RUNSERVER_AUTH_OIDC_ISSUER_URI=http://{provider_host}:{provider_port}/auth/realms/{realm_name}\n
This address must be accessible to the LRS on startup as it will perform OIDC Discovery to retrieve public keys and other information about the OIDC environment.
It is also strongly recommended that you set the optional RALPH_RUNSERVER_AUTH_OIDC_AUDIENCE
environment variable to the origin address of the LRS itself (ex. \u201chttp://localhost:8100\u201d) to enable verification that a given token was issued specifically for the LRS.
"},{"location":"api/#identity_providers","title":"Identity Providers","text":"OIDC support is currently developed and tested against Keycloak but may work with other identity providers that implement the specification.
The Learning analytics playground repository contains a Docker Compose file and configuration for a demo instance of Keycloak with a ralph
client.
"},{"location":"api/#making_a_get_request_1","title":"Making a GET request","text":"The first request that can be answered by the ralph API server is a whoami
request, which checks if the user is authenticated and returns their username and permission scopes.
Use curl to get http://localhost:8100/whoami
:
curl http://localhost:8100/whoami\n< HTTP/1.1 401 Unauthorized\n< {\"detail\":\"Could not validate credentials\"}\n
With the Keycloak instance running, use curl to get access token from Keycloak:
curl --request POST 'http://localhost:8080/auth/realms/fun-mooc/protocol/openid-connect/token' \\\n--header 'Content-Type: application/x-www-form-urlencoded' \\\n--data-urlencode 'client_id=ralph' \\\n--data-urlencode 'client_secret=super-secret' \\\n--data-urlencode 'username=ralph_admin' \\\n--data-urlencode 'password=funfunfun' \\\n--data-urlencode 'grant_type=password'\n
which outputs (tokens truncated for example purpose):
{ \n \"access_token\":\"eyJhbGciOiJSUzI1NiIsInR5cCIgOiAiSldUIiwia2lkIiA6ICJSTWlLM\",\n \"expires_in\":300,\n \"refresh_expires_in\":1800,\n \"refresh_token\":\"eyJhbGciOiJIUzI1NiIsInR5cCIgOiAiSldUIiwia2lkIiA6ICI4MDc5NjExM\",\n \"token_type\":\"Bearer\",\n \"not-before-policy\":0,\n \"session_state\":\"22a36735-e35f-496b-a243-152d32ebff45\",\n \"scope\":\"profile email\"\n}\n
Send the access token to the API server as a Bearer header:
curl http://localhost:8100/whoami --header \"Authorization: Bearer eyJhbGciOiJSUzI1NiIsInR5cCIgOiAiSldUIiwia2lkIiA6ICJSTWlLM\"\n< HTTP/1.1 200 OK\n< {\"username\":\"ralph_admin\",\"scopes\":[\"all\"]}\n
"},{"location":"api/#security","title":"Security","text":"By default, all authenticated users have full read and write access to the server. You may use the options below to restrict behavior.
"},{"location":"api/#filtering_results_by_authority_multitenancy","title":"Filtering results by authority (multitenancy)","text":"In Ralph LRS, all incoming statements are assigned an authority
(or ownership) derived from the user that makes the call. You may restrict read access to users \u201cown\u201d statements (thus enabling multitenancy) by setting the following environment variable:
RALPH_LRS_RESTRICT_BY_AUTHORITY = True # Default: False\n
WARNING: Two accounts with different credentials may share the same authority
meaning they can access the same statements. It is the administrators responsability to ensure that authority
is properly assigned.
NB: If not using \u201cscopes\u201d, or for users with limited \u201cscopes\u201d, using this option will make the use of option ?mine=True
implicit when fetching statement.
"},{"location":"api/#scopes","title":"Scopes","text":"In Ralph, users are assigned scopes which may be used to restrict endpoint access or functionalities. You may enable this option by setting the following environment variable:
RALPH_LRS_RESTRICT_BY_SCOPES = True # Default: False\n
Valid scopes are a slight variation on those proposed by the xAPI specification:
- statements/write
- statements/read/mine
- statements/read
- state/write
- state/read
- define
- profile/write
- profile/read
- all/read
- all
"},{"location":"api/#forwarding_statements","title":"Forwarding statements","text":"Ralph\u2019s API server can be configured to forward xAPI statements it receives to other LRSes.
To configure statement forwarding, you need to create a .env
file in the current directory and define the RALPH_XAPI_FORWARDINGS
variable or define the RALPH_XAPI_FORWARDINGS
environment variable.
The value of the RALPH_XAPI_FORWARDINGS
variable should be a JSON encoded list of dictionaries where each dictionary defines a forwarding configuration and consists of the following key/value pairs:
key value type description is_active
boolean
Specifies whether or not this forwarding configuration should take effect. url
URL
Specifies the endpoint URL where forwarded statements should be send. basic_username
string
Specifies the basic auth username. basic_password
string
Specifies the basic auth password. max_retries
number
Specifies the number of times a failed forwarding request should be retried. timeout
number
Specifies the duration in seconds of network inactivity leading to a timeout. Note that for a forwarding configuration to be valid it is required that all key/value pairs are defined.
Example of a valid forwarding configuration:
RALPH_XAPI_FORWARDINGS='\n[\n {\n \"is_active\": true,\n \"url\": \"http://lrs1.example.com/xAPI/statements/\",\n \"basic_username\": \"admin1@example.com\",\n \"basic_password\": \"PASSWORD1\",\n \"max_retries\": 1,\n \"timeout\": 5\n },\n {\n \"is_active\": true,\n \"url\": \"http://lrs2.example.com/xAPI/statements/\",\n \"basic_username\": \"admin2@example.com\",\n \"basic_password\": \"PASSWORD2\",\n \"max_retries\": 5,\n \"timeout\": 0.2\n }\n]\n'\n
"},{"location":"api/#sentry_configuration","title":"Sentry configuration","text":"Ralph provides Sentry integration to monitor its LRS server and its CLI. To activate Sentry integration, one should define the following environment variables:
RALPH_SENTRY_DSN={PROTOCOL}://{PUBLIC_KEY}:{SECRET_KEY}@{HOST}{PATH}/{PROJECT_ID}\nRALPH_EXECUTION_ENVIRONMENT=development\n
The Sentry DSN (Data Source Name) can be found in your project settings from Sentry application. The execution environment should reflect the environment Ralph has been deployed in (e.g. production
).
You may also want to monitor the performance of Ralph by configuring the CLI and LRS traces sample rates:
RALPH_SENTRY_CLI_TRACES_SAMPLE_RATE=0.1\nRALPH_SENTRY_LRS_TRACES_SAMPLE_RATE=0.3\n
Note that a sampling rate of 1.0
means 100% of transactions are sent to sentry and 0.1
only 10%.
If you want to lower noisy transactions (e.g. in a Kubernetes cluster), you can disable health checks related ones:
RALPH_SENTRY_IGNORE_HEALTH_CHECKS=True\n
"},{"location":"api/#additional_configuration","title":"Additional configuration","text":""},{"location":"api/#http_basic_auth_caching","title":"HTTP Basic auth caching","text":"HTTP Basic auth implementation uses the secure and standard bcrypt algorithm to hash/salt passwords before storing them. This implementation comes with a performance cost. To speed up requests, credentials are stored in an LRU cache with a Time To Live
. To configure this cache, you can define the following environment variables:
- the maximum number of entries in the cache. Select a value greater than the maximum number of individual user credentials, for better performance. Defaults to 100.
RALPH_AUTH_CACHE_MAX_SIZE=100\n
- the Time To Live
of the cache entries in seconds. Defaults to 3600s. RALPH_AUTH_CACHE_TTL=3600\n
"},{"location":"api/#api_documentation","title":"API documentation","text":""},{"location":"api/#fastapi_010","title":"FastAPI 0.1.0","text":""},{"location":"api/#endpoints","title":"Endpoints","text":""},{"location":"api/#get_xapistatements","title":"GET /xAPI/statements/","text":"Get
Description Read a single xAPI Statement or multiple xAPI Statements.
LRS Specification: https://github.com/adlnet/xAPI-Spec/blob/1.0.3/xAPI- Communication.md#213-get-statements
Input parameters
Parameter In Type Default Nullable Description HTTPBasic
header string N/A No Basic authentication HTTPBasic
header string N/A No Basic authentication activity
query string No Filter, only return Statements for which the Object of the Statement is an Activity with the specified id agent
query string No Filter, only return Statements for which the specified Agent or Group is the Actor or Object of the Statement ascending
query boolean False No If \"true\", return results in ascending order of stored time attachments
query boolean False No **Not implemented** 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. format
query string exact No **Not implemented** 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. limit
query integer 100 No Maximum number of Statements to return. 0 indicates return the maximum the server will allow mine
query boolean False No If \"true\", return only the results for which the authority matches the \"agent\" associated to the user that is making the query. pit_id
query string No Point-in-time ID to ensure consistency of search requests through multiple pages.NB: for internal use, not part of the LRS specification. registration
query string No **Not implemented** Filter, only return Statements matching the specified registration id related_activities
query boolean False No **Not implemented** 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 behaviour related_agents
query boolean False No **Not implemented** 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 behaviour. search_after
query string No Sorting data to allow pagination through large number of search results. NB: for internal use, not part of the LRS specification. since
query string No Only Statements stored since the specified Timestamp (exclusive) are returned statementId
query string No Id of Statement to fetch until
query string No Only Statements stored at or before the specified Timestamp are returned verb
query string No Filter, only return Statements matching the specified Verb id voidedStatementId
query string No **Not implemented** Id of voided Statement to fetch Response 200 OK
application/json Schema of the response body {\n \"type\": \"object\",\n \"title\": \"Response Get Xapi Statements Get\"\n}\n
Response 422 Unprocessable Entity
application/json {\n \"detail\": [\n {\n \"loc\": [\n null\n ],\n \"msg\": \"string\",\n \"type\": \"string\"\n }\n ]\n}\n
\u26a0\ufe0f This example has been generated automatically from the schema and it is not accurate. Refer to the schema for more information. Schema of the response body {\n \"properties\": {\n \"detail\": {\n \"items\": {\n \"$ref\": \"#/components/schemas/ValidationError\"\n },\n \"type\": \"array\",\n \"title\": \"Detail\"\n }\n },\n \"type\": \"object\",\n \"title\": \"HTTPValidationError\"\n}\n
"},{"location":"api/#put_xapistatements","title":"PUT /xAPI/statements/","text":"Put
Description Store a single statement as a single member of a set.
LRS Specification: https://github.com/adlnet/xAPI-Spec/blob/1.0.3/xAPI- Communication.md#211-put-statements
Input parameters
Parameter In Type Default Nullable Description HTTPBasic
header string N/A No Basic authentication HTTPBasic
header string N/A No Basic authentication statementId
query string No Request body
application/json {\n \"actor\": null,\n \"id\": \"292e8ffe-4ed4-460b-b307-16a7a5c14213\",\n \"object\": {\n \"id\": \"string\"\n },\n \"verb\": {\n \"id\": \"string\"\n }\n}\n
\u26a0\ufe0f This example has been generated automatically from the schema and it is not accurate. Refer to the schema for more information. Schema of the request body {\n \"properties\": {\n \"actor\": {\n \"anyOf\": [\n {\n \"$ref\": \"#/components/schemas/BaseXapiAgentWithMbox\"\n },\n {\n \"$ref\": \"#/components/schemas/BaseXapiAgentWithMboxSha1Sum\"\n },\n {\n \"$ref\": \"#/components/schemas/BaseXapiAgentWithOpenId\"\n },\n {\n \"$ref\": \"#/components/schemas/BaseXapiAgentWithAccount\"\n },\n {\n \"$ref\": \"#/components/schemas/BaseXapiAnonymousGroup\"\n },\n {\n \"$ref\": \"#/components/schemas/BaseXapiIdentifiedGroupWithMbox\"\n },\n {\n \"$ref\": \"#/components/schemas/BaseXapiIdentifiedGroupWithMboxSha1Sum\"\n },\n {\n \"$ref\": \"#/components/schemas/BaseXapiIdentifiedGroupWithOpenId\"\n },\n {\n \"$ref\": \"#/components/schemas/BaseXapiIdentifiedGroupWithAccount\"\n }\n ],\n \"title\": \"Actor\"\n },\n \"id\": {\n \"type\": \"string\",\n \"format\": \"uuid\",\n \"title\": \"Id\"\n },\n \"object\": {\n \"$ref\": \"#/components/schemas/LaxObjectField\"\n },\n \"verb\": {\n \"$ref\": \"#/components/schemas/LaxVerbField\"\n }\n },\n \"type\": \"object\",\n \"required\": [\n \"actor\",\n \"object\",\n \"verb\"\n ],\n \"title\": \"LaxStatement\",\n \"description\": \"Pydantic model for lax statement.\\n\\nIt accepts without validating all fields beyond the bare minimum required to\\nqualify an object as an XAPI statement.\"\n}\n
Response 204 No Content
Response 400 Bad Request
application/json {\n \"detail\": \"string\"\n}\n
\u26a0\ufe0f This example has been generated automatically from the schema and it is not accurate. Refer to the schema for more information. Schema of the response body {\n \"properties\": {\n \"detail\": {\n \"type\": \"string\",\n \"title\": \"Detail\"\n }\n },\n \"type\": \"object\",\n \"required\": [\n \"detail\"\n ],\n \"title\": \"ErrorDetail\",\n \"description\": \"Pydantic model for errors raised detail.\\n\\nType for return value for errors raised in API endpoints.\\nUseful for OpenAPI documentation generation.\"\n}\n
Response 409 Conflict
application/json {\n \"detail\": \"string\"\n}\n
\u26a0\ufe0f This example has been generated automatically from the schema and it is not accurate. Refer to the schema for more information. Schema of the response body {\n \"properties\": {\n \"detail\": {\n \"type\": \"string\",\n \"title\": \"Detail\"\n }\n },\n \"type\": \"object\",\n \"required\": [\n \"detail\"\n ],\n \"title\": \"ErrorDetail\",\n \"description\": \"Pydantic model for errors raised detail.\\n\\nType for return value for errors raised in API endpoints.\\nUseful for OpenAPI documentation generation.\"\n}\n
Response 422 Unprocessable Entity
application/json {\n \"detail\": [\n {\n \"loc\": [\n null\n ],\n \"msg\": \"string\",\n \"type\": \"string\"\n }\n ]\n}\n
\u26a0\ufe0f This example has been generated automatically from the schema and it is not accurate. Refer to the schema for more information. Schema of the response body {\n \"properties\": {\n \"detail\": {\n \"items\": {\n \"$ref\": \"#/components/schemas/ValidationError\"\n },\n \"type\": \"array\",\n \"title\": \"Detail\"\n }\n },\n \"type\": \"object\",\n \"title\": \"HTTPValidationError\"\n}\n
"},{"location":"api/#post_xapistatements","title":"POST /xAPI/statements/","text":"Post
Description Store a set of statements (or a single statement as a single member of a set).
NB: at this time, using POST to make a GET request, is not supported. LRS Specification: https://github.com/adlnet/xAPI-Spec/blob/1.0.3/xAPI- Communication.md#212-post-statements
Input parameters
Parameter In Type Default Nullable Description HTTPBasic
header string N/A No Basic authentication HTTPBasic
header string N/A No Basic authentication Request body
application/json Schema of the request body {\n \"anyOf\": [\n {\n \"$ref\": \"#/components/schemas/LaxStatement\"\n },\n {\n \"items\": {\n \"$ref\": \"#/components/schemas/LaxStatement\"\n },\n \"type\": \"array\"\n }\n ],\n \"title\": \"Statements\"\n}\n
Response 200 OK
application/json [\n null\n]\n
\u26a0\ufe0f This example has been generated automatically from the schema and it is not accurate. Refer to the schema for more information. Schema of the response body {\n \"items\": {},\n \"type\": \"array\",\n \"title\": \"Response Post Xapi Statements Post\"\n}\n
Response 400 Bad Request
application/json {\n \"detail\": \"string\"\n}\n
\u26a0\ufe0f This example has been generated automatically from the schema and it is not accurate. Refer to the schema for more information. Schema of the response body {\n \"properties\": {\n \"detail\": {\n \"type\": \"string\",\n \"title\": \"Detail\"\n }\n },\n \"type\": \"object\",\n \"required\": [\n \"detail\"\n ],\n \"title\": \"ErrorDetail\",\n \"description\": \"Pydantic model for errors raised detail.\\n\\nType for return value for errors raised in API endpoints.\\nUseful for OpenAPI documentation generation.\"\n}\n
Response 409 Conflict
application/json {\n \"detail\": \"string\"\n}\n
\u26a0\ufe0f This example has been generated automatically from the schema and it is not accurate. Refer to the schema for more information. Schema of the response body {\n \"properties\": {\n \"detail\": {\n \"type\": \"string\",\n \"title\": \"Detail\"\n }\n },\n \"type\": \"object\",\n \"required\": [\n \"detail\"\n ],\n \"title\": \"ErrorDetail\",\n \"description\": \"Pydantic model for errors raised detail.\\n\\nType for return value for errors raised in API endpoints.\\nUseful for OpenAPI documentation generation.\"\n}\n
Response 422 Unprocessable Entity
application/json {\n \"detail\": [\n {\n \"loc\": [\n null\n ],\n \"msg\": \"string\",\n \"type\": \"string\"\n }\n ]\n}\n
\u26a0\ufe0f This example has been generated automatically from the schema and it is not accurate. Refer to the schema for more information. Schema of the response body {\n \"properties\": {\n \"detail\": {\n \"items\": {\n \"$ref\": \"#/components/schemas/ValidationError\"\n },\n \"type\": \"array\",\n \"title\": \"Detail\"\n }\n },\n \"type\": \"object\",\n \"title\": \"HTTPValidationError\"\n}\n
"},{"location":"api/#get_xapistatements_1","title":"GET /xAPI/statements","text":"Get
Description Read a single xAPI Statement or multiple xAPI Statements.
LRS Specification: https://github.com/adlnet/xAPI-Spec/blob/1.0.3/xAPI- Communication.md#213-get-statements
Input parameters
Parameter In Type Default Nullable Description HTTPBasic
header string N/A No Basic authentication HTTPBasic
header string N/A No Basic authentication activity
query string No Filter, only return Statements for which the Object of the Statement is an Activity with the specified id agent
query string No Filter, only return Statements for which the specified Agent or Group is the Actor or Object of the Statement ascending
query boolean False No If \"true\", return results in ascending order of stored time attachments
query boolean False No **Not implemented** 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. format
query string exact No **Not implemented** 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. limit
query integer 100 No Maximum number of Statements to return. 0 indicates return the maximum the server will allow mine
query boolean False No If \"true\", return only the results for which the authority matches the \"agent\" associated to the user that is making the query. pit_id
query string No Point-in-time ID to ensure consistency of search requests through multiple pages.NB: for internal use, not part of the LRS specification. registration
query string No **Not implemented** Filter, only return Statements matching the specified registration id related_activities
query boolean False No **Not implemented** 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 behaviour related_agents
query boolean False No **Not implemented** 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 behaviour. search_after
query string No Sorting data to allow pagination through large number of search results. NB: for internal use, not part of the LRS specification. since
query string No Only Statements stored since the specified Timestamp (exclusive) are returned statementId
query string No Id of Statement to fetch until
query string No Only Statements stored at or before the specified Timestamp are returned verb
query string No Filter, only return Statements matching the specified Verb id voidedStatementId
query string No **Not implemented** Id of voided Statement to fetch Response 200 OK
application/json Schema of the response body {\n \"type\": \"object\",\n \"title\": \"Response Get Xapi Statements Get\"\n}\n
Response 422 Unprocessable Entity
application/json {\n \"detail\": [\n {\n \"loc\": [\n null\n ],\n \"msg\": \"string\",\n \"type\": \"string\"\n }\n ]\n}\n
\u26a0\ufe0f This example has been generated automatically from the schema and it is not accurate. Refer to the schema for more information. Schema of the response body {\n \"properties\": {\n \"detail\": {\n \"items\": {\n \"$ref\": \"#/components/schemas/ValidationError\"\n },\n \"type\": \"array\",\n \"title\": \"Detail\"\n }\n },\n \"type\": \"object\",\n \"title\": \"HTTPValidationError\"\n}\n
"},{"location":"api/#put_xapistatements_1","title":"PUT /xAPI/statements","text":"Put
Description Store a single statement as a single member of a set.
LRS Specification: https://github.com/adlnet/xAPI-Spec/blob/1.0.3/xAPI- Communication.md#211-put-statements
Input parameters
Parameter In Type Default Nullable Description HTTPBasic
header string N/A No Basic authentication HTTPBasic
header string N/A No Basic authentication statementId
query string No Request body
application/json {\n \"actor\": null,\n \"id\": \"6b700e1f-3e01-4218-85c1-7d3be1b1f319\",\n \"object\": {\n \"id\": \"string\"\n },\n \"verb\": {\n \"id\": \"string\"\n }\n}\n
\u26a0\ufe0f This example has been generated automatically from the schema and it is not accurate. Refer to the schema for more information. Schema of the request body {\n \"properties\": {\n \"actor\": {\n \"anyOf\": [\n {\n \"$ref\": \"#/components/schemas/BaseXapiAgentWithMbox\"\n },\n {\n \"$ref\": \"#/components/schemas/BaseXapiAgentWithMboxSha1Sum\"\n },\n {\n \"$ref\": \"#/components/schemas/BaseXapiAgentWithOpenId\"\n },\n {\n \"$ref\": \"#/components/schemas/BaseXapiAgentWithAccount\"\n },\n {\n \"$ref\": \"#/components/schemas/BaseXapiAnonymousGroup\"\n },\n {\n \"$ref\": \"#/components/schemas/BaseXapiIdentifiedGroupWithMbox\"\n },\n {\n \"$ref\": \"#/components/schemas/BaseXapiIdentifiedGroupWithMboxSha1Sum\"\n },\n {\n \"$ref\": \"#/components/schemas/BaseXapiIdentifiedGroupWithOpenId\"\n },\n {\n \"$ref\": \"#/components/schemas/BaseXapiIdentifiedGroupWithAccount\"\n }\n ],\n \"title\": \"Actor\"\n },\n \"id\": {\n \"type\": \"string\",\n \"format\": \"uuid\",\n \"title\": \"Id\"\n },\n \"object\": {\n \"$ref\": \"#/components/schemas/LaxObjectField\"\n },\n \"verb\": {\n \"$ref\": \"#/components/schemas/LaxVerbField\"\n }\n },\n \"type\": \"object\",\n \"required\": [\n \"actor\",\n \"object\",\n \"verb\"\n ],\n \"title\": \"LaxStatement\",\n \"description\": \"Pydantic model for lax statement.\\n\\nIt accepts without validating all fields beyond the bare minimum required to\\nqualify an object as an XAPI statement.\"\n}\n
Response 204 No Content
Response 400 Bad Request
application/json {\n \"detail\": \"string\"\n}\n
\u26a0\ufe0f This example has been generated automatically from the schema and it is not accurate. Refer to the schema for more information. Schema of the response body {\n \"properties\": {\n \"detail\": {\n \"type\": \"string\",\n \"title\": \"Detail\"\n }\n },\n \"type\": \"object\",\n \"required\": [\n \"detail\"\n ],\n \"title\": \"ErrorDetail\",\n \"description\": \"Pydantic model for errors raised detail.\\n\\nType for return value for errors raised in API endpoints.\\nUseful for OpenAPI documentation generation.\"\n}\n
Response 409 Conflict
application/json {\n \"detail\": \"string\"\n}\n
\u26a0\ufe0f This example has been generated automatically from the schema and it is not accurate. Refer to the schema for more information. Schema of the response body {\n \"properties\": {\n \"detail\": {\n \"type\": \"string\",\n \"title\": \"Detail\"\n }\n },\n \"type\": \"object\",\n \"required\": [\n \"detail\"\n ],\n \"title\": \"ErrorDetail\",\n \"description\": \"Pydantic model for errors raised detail.\\n\\nType for return value for errors raised in API endpoints.\\nUseful for OpenAPI documentation generation.\"\n}\n
Response 422 Unprocessable Entity
application/json {\n \"detail\": [\n {\n \"loc\": [\n null\n ],\n \"msg\": \"string\",\n \"type\": \"string\"\n }\n ]\n}\n
\u26a0\ufe0f This example has been generated automatically from the schema and it is not accurate. Refer to the schema for more information. Schema of the response body {\n \"properties\": {\n \"detail\": {\n \"items\": {\n \"$ref\": \"#/components/schemas/ValidationError\"\n },\n \"type\": \"array\",\n \"title\": \"Detail\"\n }\n },\n \"type\": \"object\",\n \"title\": \"HTTPValidationError\"\n}\n
"},{"location":"api/#post_xapistatements_1","title":"POST /xAPI/statements","text":"Post
Description Store a set of statements (or a single statement as a single member of a set).
NB: at this time, using POST to make a GET request, is not supported. LRS Specification: https://github.com/adlnet/xAPI-Spec/blob/1.0.3/xAPI- Communication.md#212-post-statements
Input parameters
Parameter In Type Default Nullable Description HTTPBasic
header string N/A No Basic authentication HTTPBasic
header string N/A No Basic authentication Request body
application/json Schema of the request body {\n \"anyOf\": [\n {\n \"$ref\": \"#/components/schemas/LaxStatement\"\n },\n {\n \"items\": {\n \"$ref\": \"#/components/schemas/LaxStatement\"\n },\n \"type\": \"array\"\n }\n ],\n \"title\": \"Statements\"\n}\n
Response 200 OK
application/json [\n null\n]\n
\u26a0\ufe0f This example has been generated automatically from the schema and it is not accurate. Refer to the schema for more information. Schema of the response body {\n \"items\": {},\n \"type\": \"array\",\n \"title\": \"Response Post Xapi Statements Post\"\n}\n
Response 400 Bad Request
application/json {\n \"detail\": \"string\"\n}\n
\u26a0\ufe0f This example has been generated automatically from the schema and it is not accurate. Refer to the schema for more information. Schema of the response body {\n \"properties\": {\n \"detail\": {\n \"type\": \"string\",\n \"title\": \"Detail\"\n }\n },\n \"type\": \"object\",\n \"required\": [\n \"detail\"\n ],\n \"title\": \"ErrorDetail\",\n \"description\": \"Pydantic model for errors raised detail.\\n\\nType for return value for errors raised in API endpoints.\\nUseful for OpenAPI documentation generation.\"\n}\n
Response 409 Conflict
application/json {\n \"detail\": \"string\"\n}\n
\u26a0\ufe0f This example has been generated automatically from the schema and it is not accurate. Refer to the schema for more information. Schema of the response body {\n \"properties\": {\n \"detail\": {\n \"type\": \"string\",\n \"title\": \"Detail\"\n }\n },\n \"type\": \"object\",\n \"required\": [\n \"detail\"\n ],\n \"title\": \"ErrorDetail\",\n \"description\": \"Pydantic model for errors raised detail.\\n\\nType for return value for errors raised in API endpoints.\\nUseful for OpenAPI documentation generation.\"\n}\n
Response 422 Unprocessable Entity
application/json {\n \"detail\": [\n {\n \"loc\": [\n null\n ],\n \"msg\": \"string\",\n \"type\": \"string\"\n }\n ]\n}\n
\u26a0\ufe0f This example has been generated automatically from the schema and it is not accurate. Refer to the schema for more information. Schema of the response body {\n \"properties\": {\n \"detail\": {\n \"items\": {\n \"$ref\": \"#/components/schemas/ValidationError\"\n },\n \"type\": \"array\",\n \"title\": \"Detail\"\n }\n },\n \"type\": \"object\",\n \"title\": \"HTTPValidationError\"\n}\n
"},{"location":"api/#get_lbheartbeat","title":"GET /lbheartbeat","text":"Lbheartbeat
Description Load balancer heartbeat.
Return a 200 when the server is running.
Response 200 OK
application/json Schema of the response body"},{"location":"api/#get_heartbeat","title":"GET /heartbeat","text":"Heartbeat
Description Application heartbeat.
Return a 200 if all checks are successful.
Response 200 OK
application/json Schema of the response body"},{"location":"api/#get_whoami","title":"GET /whoami","text":"Whoami
Description Return the current user\u2019s username along with their scopes.
Input parameters
Parameter In Type Default Nullable Description HTTPBasic
header string N/A No Basic authentication Response 200 OK
application/json Schema of the response body {\n \"type\": \"object\",\n \"title\": \"Response Whoami Whoami Get\"\n}\n
"},{"location":"api/#schemas","title":"Schemas","text":""},{"location":"api/#basexapiaccount","title":"BaseXapiAccount","text":"Name Type homePage
string name
string"},{"location":"api/#basexapiagentwithaccount","title":"BaseXapiAgentWithAccount","text":"Name Type account
BaseXapiAccount name
string objectType
string"},{"location":"api/#basexapiagentwithmbox","title":"BaseXapiAgentWithMbox","text":"Name Type mbox
string name
string objectType
string"},{"location":"api/#basexapiagentwithmboxsha1sum","title":"BaseXapiAgentWithMboxSha1Sum","text":"Name Type mbox_sha1sum
string name
string objectType
string"},{"location":"api/#basexapiagentwithopenid","title":"BaseXapiAgentWithOpenId","text":"Name Type name
string objectType
string openid
string(uri)"},{"location":"api/#basexapianonymousgroup","title":"BaseXapiAnonymousGroup","text":"Name Type member
Array<> name
string objectType
string"},{"location":"api/#basexapiidentifiedgroupwithaccount","title":"BaseXapiIdentifiedGroupWithAccount","text":"Name Type account
BaseXapiAccount member
Array<> name
string objectType
string"},{"location":"api/#basexapiidentifiedgroupwithmbox","title":"BaseXapiIdentifiedGroupWithMbox","text":"Name Type mbox
string member
Array<> name
string objectType
string"},{"location":"api/#basexapiidentifiedgroupwithmboxsha1sum","title":"BaseXapiIdentifiedGroupWithMboxSha1Sum","text":"Name Type mbox_sha1sum
string member
Array<> name
string objectType
string"},{"location":"api/#basexapiidentifiedgroupwithopenid","title":"BaseXapiIdentifiedGroupWithOpenId","text":"Name Type member
Array<> name
string objectType
string openid
string(uri)"},{"location":"api/#errordetail","title":"ErrorDetail","text":"Name Type detail
string"},{"location":"api/#httpvalidationerror","title":"HTTPValidationError","text":"Name Type detail
Array<ValidationError>"},{"location":"api/#laxobjectfield","title":"LaxObjectField","text":"Name Type id
string(uri)"},{"location":"api/#laxstatement","title":"LaxStatement","text":"Name Type actor
id
string(uuid) object
LaxObjectField verb
LaxVerbField"},{"location":"api/#laxverbfield","title":"LaxVerbField","text":"Name Type id
string(uri)"},{"location":"api/#validationerror","title":"ValidationError","text":"Name Type loc
Array<> msg
string type
string"},{"location":"api/#security_schemes","title":"Security schemes","text":"Name Type Scheme Description HTTPBasic http basic"},{"location":"backends/","title":"Backends for data storage","text":"Ralph supports various backends that can be accessed to read from or write to (learning events or random data). Implemented backends are listed below along with their configuration parameters. If your favourite data storage method is missing, feel free to submit your implementation or get in touch!
"},{"location":"backends/#key_concepts","title":"Key concepts","text":"Each backend has its own parameter requirements. These parameters can be set as command line options or environment variables; the later is the recommended solution for sensitive data such as service credentials. For example, the os_username
(OpenStack user name) parameter of the OpenStack Swift backend, can be set as a command line option using swift
as the option prefix (and replacing underscores in its name by dashes):
ralph list --backend swift --swift-os-username johndoe # [...] more options\n
Alternatively, this parameter can be set as an environment variable (in upper case, prefixed by the program name, e.g. RALPH_
):
export RALPH_BACKENDS__DATA__SWIFT__OS_USERNAME=\"johndoe\"\nralph list --backend swift # [...] more options\n
The general patterns for backend parameters are:
--{{ backend_name }}-{{ parameter | underscore_to_dash }}
for command options, and, RALPH_BACKENDS__{{ backend_type | uppercase }}__{{ backend_name | uppercase }}__{{ parameter | uppercase }}
for environment variables, where the backend_type
is one of DATA
, LRS
and STREAM
.
"},{"location":"backends/#elasticsearch","title":"Elasticsearch","text":"Elasticsearch backend is mostly used for indexation purpose (as a datalake) but it can also be used to fetch indexed data from it.
Elasticsearch data backend default configuration.
Attributes:
Name Type Description ALLOW_YELLOW_STATUS
bool
Whether to consider Elasticsearch yellow health status to be ok.
CLIENT_OPTIONS
dict
A dictionary of valid options for the Elasticsearch class initialization.
DEFAULT_CHUNK_SIZE
int
The default chunk size for reading batches of documents.
DEFAULT_INDEX
str
The default index to use for querying Elasticsearch.
HOSTS
str or tuple
The comma-separated list of Elasticsearch nodes to connect to.
LOCALE_ENCODING
str
The encoding used for reading/writing documents.
POINT_IN_TIME_KEEP_ALIVE
str
The duration for which Elasticsearch should keep a point in time alive.
REFRESH_AFTER_WRITE
str or bool
Whether the Elasticsearch index should be refreshed after the write operation.
"},{"location":"backends/#mongodb","title":"MongoDB","text":"MongoDB backend is mostly used for indexation purpose (as a datalake) but it can also be used to fetch collections of documents from it.
MongoDB data backend default configuration.
Attributes:
Name Type Description CONNECTION_URI
str
The MongoDB connection URI.
DEFAULT_DATABASE
str
The MongoDB database to connect to.
DEFAULT_COLLECTION
str
The MongoDB database collection to get objects from.
CLIENT_OPTIONS
MongoClientOptions
A dictionary of MongoDB client options.
DEFAULT_CHUNK_SIZE
int
The default chunk size to use when none is provided.
LOCALE_ENCODING
str
The locale encoding to use when none is provided.
"},{"location":"backends/#clickhouse","title":"ClickHouse","text":"The ClickHouse backend can be used as a data lake and to fetch collections of documents from it.
Represent the ClickHouse data backend default configuration.
Attributes:
Name Type Description HOST
str
ClickHouse server host to connect to.
PORT
int
ClickHouse server port to connect to.
DATABASE
str
ClickHouse database to connect to.
EVENT_TABLE_NAME
str
Table where events live.
USERNAME
str
ClickHouse username to connect as (optional).
PASSWORD
str
Password for the given ClickHouse username (optional).
CLIENT_OPTIONS
ClickHouseClientOptions
A dictionary of valid options for the ClickHouse client connection.
DEFAULT_CHUNK_SIZE
int
The default chunk size for reading/writing.
LOCALE_ENCODING
str
The locale encoding to use when none is provided.
"},{"location":"backends/#ovh_-_log_data_platform_ldp","title":"OVH - Log Data Platform (LDP)","text":"LDP is a nice service built by OVH on top of Graylog to follow, analyse and store your logs. Learning events (aka tracking logs) can be stored in GELF format using this backend.
Read-only backend
For now the LDP backend is read-only as we consider that it is mostly used to collect primary logs and not as a Ralph target. Feel free to get in touch to prove us wrong, or better: submit your proposal for the write
method implementation.
To access OVH\u2019s LDP API, you need to register Ralph as an authorized application and generate an application key, an application secret and a consumer key.
While filling the registration form available at: eu.api.ovh.com/createToken/, be sure to give an appropriate validity time span to your token and allow only GET requests on the /dbaas/logs/*
path.
OVH LDP (Log Data Platform) data backend default configuration.
Attributes:
Name Type Description APPLICATION_KEY
str
The OVH API application key (AK).
APPLICATION_SECRET
str
The OVH API application secret (AS).
CONSUMER_KEY
str
The OVH API consumer key (CK).
DEFAULT_STREAM_ID
str
The default stream identifier to query.
ENDPOINT
str
The OVH API endpoint.
REQUEST_TIMEOUT
int
HTTP request timeout in seconds.
SERVICE_NAME
str
The default LDP account name.
For more information about OVH\u2019s API client parameters, please refer to the project\u2019s documentation: github.com/ovh/python-ovh.
"},{"location":"backends/#openstack_swift","title":"OpenStack Swift","text":"Swift is the OpenStack object storage service. This storage backend is fully supported (read and write operations) to stream and store log archives.
Parameters correspond to a standard authentication using OpenStack Keystone service and configuration to work with the target container.
Represent the SWIFT data backend default configuration.
Attributes:
Name Type Description AUTH_URL
str
The authentication URL.
USERNAME
str
The name of the openstack swift user.
PASSWORD
str
The password of the openstack swift user.
IDENTITY_API_VERSION
str
The keystone API version to authenticate to.
TENANT_ID
str
The identifier of the tenant of the container.
TENANT_NAME
str
The name of the tenant of the container.
PROJECT_DOMAIN_NAME
str
The project domain name.
REGION_NAME
str
The region where the container is.
OBJECT_STORAGE_URL
str
The default storage URL.
USER_DOMAIN_NAME
str
The user domain name.
DEFAULT_CONTAINER
str
The default target container.
LOCALE_ENCODING
str
The encoding used for reading/writing documents.
"},{"location":"backends/#amazon_s3","title":"Amazon S3","text":"S3 is the Amazon Simple Storage Service. This storage backend is fully supported (read and write operations) to stream and store log archives.
Parameters correspond to a standard authentication with AWS CLI and configuration to work with the target bucket.
S3 data backend default configuration.
Attributes:
Name Type Description ACCESS_KEY_ID
str
The access key id for the S3 account.
SECRET_ACCESS_KEY
str
The secret key for the S3 account.
SESSION_TOKEN
str
The session token for the S3 account.
ENDPOINT_URL
str
The endpoint URL of the S3.
DEFAULT_REGION
str
The default region used in instantiating the client.
DEFAULT_BUCKET_NAME
str
The default bucket name targeted.
DEFAULT_CHUNK_SIZE
str
The default chunk size for reading and writing objects.
LOCALE_ENCODING
str
The encoding used for writing dictionaries to objects.
"},{"location":"backends/#file_system","title":"File system","text":"The file system backend is a dummy template that can be used to develop your own backend. It is a \u201cdummy\u201d backend as it is not intended for practical use (UNIX ls
and cat
would be more practical).
The only required parameter is the path
we want to list or stream content from.
FileSystem data backend default configuration.
Attributes:
Name Type Description DEFAULT_CHUNK_SIZE
int
The default chunk size for reading files.
DEFAULT_DIRECTORY_PATH
str or Path
The default target directory path where to perform list, read and write operations.
DEFAULT_QUERY_STRING
str
The default query string to match files for the read operation.
LOCALE_ENCODING
str
The encoding used for writing dictionaries to files.
The ClickHouse client options supported in Ralph can be found in these locations:
- Python driver specific
- General ClickHouse client settings
"},{"location":"backends/#learning_record_store_lrs_-_http_backend_interface","title":"Learning Record Store (LRS) - HTTP backend interface","text":"Warning
HTTP
backend type will soon be merged into the Data
backend type.
PR #521 prepares Data
backend alignments with HTTP
backend read
method signature
The LRS backend is used to store and retrieve xAPI statements from various systems that follow the xAPI specification (such as our own Ralph LRS, which can be run from this package). LRS systems are mostly used in e-learning infrastructures.
LRS HTTP backend default configuration.
Attributes:
Name Type Description BASE_URL
AnyHttpUrl
LRS server URL.
USERNAME
str
Basic auth username for LRS authentication.
PASSWORD
str
Basic auth password for LRS authentication.
HEADERS
dict
Headers defined for the LRS server connection.
STATUS_ENDPOINT
str
Endpoint used to check server status.
STATEMENTS_ENDPOINT
str
Default endpoint for LRS statements resource.
"},{"location":"backends/#websocket_-_stream_backend_interface","title":"WebSocket - Stream backend interface","text":"The webSocket backend is read-only and can be used to get real-time events.
If you use OVH\u2019s Logs Data Platform (LDP), you can retrieve a WebSocket URI to test your data stream by following instructions from the official documentation.
Websocket stream backend default configuration.
Attributes:
Name Type Description URI
str
The URI to connect to.
"},{"location":"models/","title":"Learning statement models","text":"The learning statement models validation and conversion tools in Ralph empower you to work with an LRS and ensure the quality of xAPI statements. These features not only enhance the integrity of your learning data but also facilitate integration and compliance with industry standards.
This section provides insights into the supported models, their conversion, and validation.
"},{"location":"models/#supported_statements","title":"Supported statements","text":"Learning statement models encompass a wide array of xAPI and OpenEdx statement types, ensuring comprehensive support for your e-learning data.
-
xAPI statements models:
- LMS
- Video
- Virtual classroom
-
OpenEdx statements models:
- Enrollment
- Navigational
- Open Reponse Assessment
- Peer instruction
- Problem interaction
- Textbook interaction
- Video interaction
"},{"location":"models/#statements_validation","title":"Statements validation","text":"In learning analytics, the validation of statements takes on significant importance. These statements, originating from diverse sources, systems or applications, must align with specific standards such as xAPI for the best known. The validation process becomes essential in ensuring that these statements meet the required standards, facilitating data quality and reliability.
Ralph allows you to automate the validation process in your production stack. OpenEdx related events and xAPI statements are supported.
Warning
For now, validation is effective only with supported learning statement models on Ralph. About xAPI statements, an issue is open to extend validation to any xAPI statement.
Check out tutorials to test the validation feature:
validate
with Ralph as a CLI validate
with Ralph as a library
"},{"location":"models/#statements_conversion","title":"Statements conversion","text":"Ralph currently supports conversion from OpenEdx learning events to xAPI statements. Here is the up-to-date conversion sets availables:
FROM TO edx.course.enrollment.activated registered to a course edx.course.enrollment.deactivated unregistered to a course load_video/edx.video.loaded initialized a video play_video/edx.video.played played a video pause_video/edx.video.paused paused a video stop_video/edx.video.stopped terminated a video seek_video/edx.video.position.changed seeked in a video Check out tutorials to test the conversion feature:
convert
with Ralph as a CLI convert
with Ralph as a library
"},{"location":"tutorials/cli/","title":"How to use Ralph as a CLI ?","text":"WIP.
"},{"location":"tutorials/cli/#prerequisites","title":"Prerequisites","text":" - Ralph should be properly installed to be used as a
CLI
. Follow Installation section for more information - [Recommended] To easily manipulate JSON streams, please install
jq
on your machine
"},{"location":"tutorials/cli/#validate_command","title":"validate
command","text":"In this tutorial, we\u2019ll walk you through the process of using validate
command to check the validity of xAPI statements.
"},{"location":"tutorials/cli/#with_an_invalid_xapi_statement","title":"With an invalid xAPI statement","text":"First, let\u2019s test the validate
command with a dummy JSON
string.
- Create in the terminal a dummy statement as follows:
invalid_statement='{\"foo\": \"invalid xapi\"}'\n
- Run validation on this statement with this command:
echo \"$invalid_statement\" | ralph validate -f xapi \n
- You should observe the following output from the terminal:
INFO ralph.cli Validating xapi events (ignore_errors=False | fail-on-unknown=False)\nERROR ralph.models.validator No matching pydantic model found for input event\nINFO ralph.models.validator Total events: 1, Invalid events: 1\n
"},{"location":"tutorials/cli/#with_a_valid_xapi_statement","title":"With a valid xAPI statement","text":"Now, let\u2019s test the validate
command with a valid xAPI statement.
The tutorial is made on a completed video
xAPI statement.
Info According to the specification, an xAPI statement to be valid should contain, at least the three following fields:
- an
actor
(with a correct IFI), - a
verb
(with an id
property), - an
object
(with an id
property).
- Create in the terminal a valid xAPI statement as follows:
valid_statement='{\"actor\": {\"mbox\": \"mailto:johndoe@example.com\", \"name\": \"John Doe\"}, \"verb\": {\"id\": \"http://adlnet.gov/expapi/verbs/completed\"}, \"object\": {\"id\": \"http://example.com/video/001-introduction\"}, \"timestamp\": \"2023-10-31T15:30:00Z\"}'\n
- Run validation on this statement with this command:
echo \"$valid_statement\" | bin/ralph validate -f xapi \n
- You should observe the following output from the terminal:
INFO ralph.cli Validating xapi events (ignore_errors=False | fail-on-unknown=False)\nINFO ralph.models.validator Total events: 1, Invalid events: 1\n
"},{"location":"tutorials/cli/#convert_command","title":"convert
command","text":"In this tutorial, you\u2019ll learn how to convert OpenEdx events into xAPI statements with Ralph.
Note
Please note that this feature is currently only supported for a set of OpenEdx events. When converting Edx events to xAPI statements, always refer to the list of supported event types to ensure accurate and successful conversion.
For this example, let\u2019s choose the page_close
OpenEdx event that is converted into a terminated a page
xAPI statement.
- Create in the terminal a
page_close
OpenEdx event as follows:
edx_statements={\"username\": \"\", \"ip\": \"0.0.0.0\", \"agent\": \"0\", \"host\": \"0\", \"referer\": \"\", \"accept_language\": \"0\", \"context\": {\"course_id\": \"\", \"course_user_tags\": null, \"module\": null, \"org_id\": \"0\", \"path\": \".\", \"user_id\": null}, \"time\": \"2000-01-01T00:00:00\", \"page\": \"http://A.ac/\", \"event_source\": \"browser\", \"session\": \"\", \"event\": \"{}\", \"event_type\": \"page_close\", \"name\": \"page_close\"}\n
- Convert this statement into a
terminated a page
statement with this command:
echo \"$edx_statements\" | \\ \nralph convert \\\n --platform-url \"http://lms-example.com\" \\\n --uuid-namespace \"ee241f8b-174f-5bdb-bae9-c09de5fe017f\" \\\n --from edx \\\n --to xapi | \\\n jq\n
- You should observe the following output from the terminal:
INFO ralph.cli Converting edx events to xapi format (ignore_errors=False | fail-on-unknown=False)\nINFO ralph.models.converter Total events: 1, Invalid events: 0\n{\n \"id\": \"8670c7d4-5485-52bd-b10a-a8ae27a51501\",\n \"actor\": {\n \"account\": {\n \"homePage\": \"http://lms-example.com\",\n \"name\": \"anonymous\"\n }\n },\n \"verb\": {\n \"id\": \"http://adlnet.gov/expapi/verbs/terminated\"\n },\n \"object\": {\n \"id\": \"http://A.ac/\",\n \"definition\": {\n \"type\": \"http://activitystrea.ms/schema/1.0/page\"\n }\n },\n \"timestamp\": \"2000-01-01T00:00:00\",\n \"version\": \"1.0.0\"\n}\n
\ud83c\udf89 Congratulations! You just have converted an event generated from OpenEdx LMS to a standardised xAPI statement!
Store locally converted statements To stored the converted statements locally on your machine, send the output of the convert
command to a JSON
file as follows:
echo \"$edx_statements\" | \\ \nralph convert \\\n --platform-url \"http://lms-example.com\" \\\n --uuid-namespace \"ee241f8b-174f-5bdb-bae9-c09de5fe017f\" \\\n --from edx \\\n --to xapi \\\n > converted_event.json\n
"},{"location":"tutorials/development_contributing/","title":"How to start developing with Ralph?","text":""},{"location":"tutorials/development_contributing/#prepare_your_environment","title":"Prepare your environment","text":"Welcome to our developer contribution guidelines! If you\u2019re interested in contributing to our project, there are a few prerequisites to get you started. Ralph development environment is containerized with Docker for consistency. Before diving in, ensure you have the following installed:
- Docker Engine
- Docker Compose
- make
"},{"location":"tutorials/development_contributing/#bootstrap_ralph_for_development","title":"Bootstrap Ralph for development","text":"To start playing with ralph
, you should first bootstrap
using:
make bootstrap\n
Once the project has been bootstrapped, you may want to edit generated .env
file to set up available backend parameters that will be injected into the running container as environment variables to configure Ralph (see backends documentation):
# Elasticsearch backend\nRALPH_BACKENDS__LRS__ES__HOSTS=http://elasticsearch:9200\nRALPH_BACKENDS__LRS__ES__INDEX=statements\nRALPH_BACKENDS__LRS__ES__TEST_HOSTS=http://elasticsearch:9200\nRALPH_BACKENDS__LRS__ES__TEST_INDEX=test-index\n\n# [...]\n
Uncomment lines to define environment variables Lines starting with a #
are considered as commented and thus will have no effect while running Ralph.
"},{"location":"tutorials/development_contributing/#working_with_backends","title":"Working with backends","text":"To configure the backends, we provide default parameters in the .env.dist
template which is copied to the .env
file when bootstrapping the project for the first time. You can uncomment them so that they are properly injected in running containers.
In order to run the Elasticsearch backend locally on GNU/Linux operating systems, ensure that your virtual memory limits are not too low and increase them if needed by typing this command from your terminal (as root
or using sudo
):
sysctl -w vm.max_map_count=262144
Reference: https://www.elastic.co/guide/en/elasticsearch/reference/master/vm-max-map-count.html
Once configured, start the backend container using:
make run-[BACKEND]\n
Substitute [BACKEND]
by the backend name, e.g. es
for Elasticsearch or swift
for OpenStack Swift:
# Start Elasticsearch backend\nmake run-es\n# Start Swift backend\nmake run-swift\n# Start Mongo backend\nmake run-mongo\n# Start ClickHouse backend\nmake run-clickhouse\n# Start all backends\nmake run-all\n
Disk space for Elasticsearch
Ensure that you have at least 10% of available disk space on your machine to run Elasticsearch.
Now that you have started at least the elasticsearch
and swift
backends, it\u2019s time to play with them:
# Store a JSON file in the Swift backend\necho '{\"id\": 1, \"foo\": \"bar\"}' | \\\n ./bin/ralph write -b swift -t foo.json\n\n# Check that we have created a new JSON file in the Swift backend\nbin/ralph list -b swift\n>>> foo.json\n\n# Read the content of the JSON file and index it in Elasticsearch\nbin/ralph read -b swift -t foo.json | \\\n bin/ralph write -b es\n\n# Check that we have properly indexed the JSON file in Elasticsearch\nbin/ralph read -b es\n>>> {\"id\": 1, \"foo\": \"bar\"}\n
"},{"location":"tutorials/development_contributing/#wip_working_with_the_lrs","title":"[WIP] Working with the LRS","text":""},{"location":"tutorials/development_contributing/#working_with_ralphs_tray","title":"Working with Ralph\u2019s tray","text":"Ralph is distributed along with its tray (a deployable package for Kubernetes clusters using Arnold). If you intend to work on this tray, please refer to Arnold\u2019s documentation first.
"},{"location":"tutorials/development_contributing/#prerequisites","title":"Prerequisites","text":" - Kubectl (>
v.1.23.5
): This CLI is used to communicate with the running Kubernetes instance you will use. - k3d (>
v.5.0.0
): This tool is used to set up and run a lightweight Kubernetes cluster, in order to have a local environment (it is required to complete quickstart instructions below to avoid depending on an existing Kubernetes cluster). - curl is required by Arnold\u2019s CLI.
- gnupg to encrypt Ansible vaults passwords and collaborate with your team.
"},{"location":"tutorials/development_contributing/#create_a_local_k3d_cluster","title":"Create a local k3d
cluster","text":"To create (or run) a local kubernetes cluster, we use k3d
. The cluster\u2019s bootstrapping should be run via:
make k3d-cluster\n
Running a k3d-cluster locally supposes that the 80 and 443 ports of your machine are available, so that the ingresses created for your project responds properly. If one or both ports are already used by another service running on your machine, the make k3d-cluster
command may fail.
You can check that your cluster is running using the k3d cluster
command:
k3d cluster list\n
You should expect the following output:
NAME SERVERS AGENTS LOADBALANCER\nralph 1/1 0/0 true\n
As you can see, we are running a single node cluster called ralph
.
"},{"location":"tutorials/development_contributing/#bootstrap_an_arnold_project","title":"Bootstrap an Arnold project","text":"Once your Kubernetes cluster is running, you need to create a standard Arnold project describing applications and environments you need to deploy:
make arnold-bootstrap\n
Once bootstrapped, Arnold should have created a group_vars
directory.
Run the following command to discover the directory tree.
tree group_vars\n
The output should be as follows:
group_vars\n\u251c\u2500\u2500 common\n\u2514\u2500\u2500 customer\n \u2514\u2500\u2500 ralph\n \u251c\u2500\u2500 development\n \u2502\u00a0\u00a0 \u251c\u2500\u2500 main.yml\n \u2502\u00a0\u00a0 \u2514\u2500\u2500 secrets\n \u2502\u00a0\u00a0 \u251c\u2500\u2500 databases.vault.yml\n \u2502\u00a0\u00a0 \u251c\u2500\u2500 elasticsearch.vault.yml\n \u2502\u00a0\u00a0 \u2514\u2500\u2500 ralph.vault.yml\n \u2514\u2500\u2500 main.yml\n\n5 directories, 5 files\n
To create the LRS credentials file, you need to provide a list of accounts allowed to request the LRS in Ralph\u2019s vault:
# Setup your kubernetes environment\nsource .k3d-cluster.env.sh\n\n# Decrypt the vault\nbin/arnold -d -c ralph -e development -- vault -a ralph decrypt\n
Edit the vault file to add a new account for the foo
user with the bar
password and a relevant scope:
# group_vars/customer/ralph/development/secrets/ralph.vault.yml\n#\n# [...]\n#\n# LRS\nLRS_AUTH:\n - username: \"foo\"\n hash: \"$2b$12$lCggI749U6TrzK7Qyr7xGe1KVSAXdPjtkMew.BD6lzIk//T5YSb72\"\n scopes:\n - \"all\"\n
The password hash has been generated using bcrypt
as explained in the API user guide.
And finally (re-)encrypt Ralph\u2019s vault:
bin/arnold -d -c ralph -e development -- vault -a ralph encrypt\n
You are now ready to create the related Kubernetes Secret while initializing Arnold project in the next step.
"},{"location":"tutorials/development_contributing/#prepare_working_namespace","title":"Prepare working namespace","text":"You are now ready to create required Kubernetes objects to start working on Ralph\u2019s deployment:
make arnold-init\n
At this point an Elasticsearch cluster should be running on your Kubernetes cluster:
kubectl -n development-ralph get -l app=elasticsearch pod\nNAME READY STATUS RESTARTS AGE\nelasticsearch-node-0 1/1 Running 0 69s\nelasticsearch-node-1 1/1 Running 0 69s\nelasticsearch-node-2 1/1 Running 0 69s\nes-index-template-j-221010-09h25m24s-nx5qz 0/1 Completed 0 49s\n
We are now ready to deploy Ralph to Kubernetes!
"},{"location":"tutorials/development_contributing/#deploy_code_repeat","title":"Deploy, code, repeat","text":"To test your local docker image, you need to build it and publish it to the local kubernetes cluster docker registry using the k3d-push
Makefile rule:
make k3d-push\n
Note
Each time you modify Ralph\u2019s application or its Docker image, you will need to make this update.
Now that your Docker image is published, it\u2019s time to deploy it!
make arnold-deploy\n
To test this deployment, let\u2019s try to make an authenticated request to the LRS:
curl -sLk \\\n --user foo:bar \\\n \"https://$(\\\n kubectl -n development-ralph \\\n get \\\n ingress/ralph-app-current \\\n -o jsonpath='{.spec.rules[0].host}')/whoami\"\n
And why not send test statements from Potsie\u2019s repository:
curl -sL \\\n https://github.com/openfun/potsie/raw/main/fixtures/elasticsearch/lrs.json.gz | \\\ngunzip | \\\nhead -n 100 | \\\njq -s . | \\\nsed \"s/@timestamp/timestamp/g\" | \\\ncurl -sLk \\\n --user foo:bar \\\n -X POST \\\n -H \"Content-Type: application/json\" \\\n \"https://$(\\\n kubectl -n development-ralph \\\n get \\\n ingress/ralph-app-current \\\n -o jsonpath='{.spec.rules[0].host}')/xAPI/statements/\" \\\n -d @-\n
Install jq
This example requires jq
command to serialize the request payload (xAPI statements). When dealing with JSON data, we strongly recommend installing it to manipulate them from the command line.
"},{"location":"tutorials/development_contributing/#perform_arnolds_operations","title":"Perform Arnold\u2019s operations","text":"If you want to run the bin/arnold
script to run specific Arnold commands, you must ensure that your environment is properly set and that Arnold runs in development mode (i.e. using the -d
flag):
source .k3d-cluster.env.sh\nbin/arnold -d -c ralph -e development -- vault -a ralph view\n
"},{"location":"tutorials/development_contributing/#stop_k3d_cluster","title":"Stop k3d
cluster","text":"When finished to work on the Tray, you can stop the k3d
cluster using the k3d-stop
helper:
make k3d-stop\n
"},{"location":"tutorials/development_contributing/#test_your_development","title":"Test your development","text":"To run tests on your code, either use the test
Make target or the bin/pytest
script to pass specific arguments to the test runner:
# Run all tests\nmake test\n\n# Run pytest with options\nbin/pytest -x -k mixins\n\n# Run pytest with options and more debugging logs\nbin/pytest tests/api -x -vvv -s --log-level=DEBUG -k mixins\n
"},{"location":"tutorials/development_contributing/#lint_your_code","title":"Lint your code","text":"To lint your code, either use the lint
meta target or one of the linting tools we use:
# Run all linters\nmake lint\n\n# Run ruff linter\nmake lint-ruff\n\n# Run ruff linter and resolve fixable errors\nmake lint-ruff-fix\n\n# List available linters\nmake help | grep lint-\n
"},{"location":"tutorials/development_contributing/#document_your_modification","title":"Document your modification","text":"In case you need to document your code, use the following targets:
# Build documentation site\nmake docs-build\n\n# Run mkdocs live server for dev docs\nmake docs-serve\n
"},{"location":"tutorials/library/","title":"How to use Ralph as a library ?","text":"WIP.
"},{"location":"tutorials/library/#validate_method","title":"validate
method","text":"WIP.
"},{"location":"tutorials/library/#convert_method","title":"convert
method","text":"WIP.
"}]}
\ No newline at end of file
diff --git a/master/sitemap.xml b/master/sitemap.xml
index 1e8af8535..94301bbdc 100644
--- a/master/sitemap.xml
+++ b/master/sitemap.xml
@@ -2,62 +2,62 @@
https://openfun.github.io/ralph/latest/
- 2023-11-24
+ 2023-11-30
daily
https://openfun.github.io/ralph/latest/CHANGELOG/
- 2023-11-24
+ 2023-11-30
daily
https://openfun.github.io/ralph/latest/LICENSE/
- 2023-11-24
+ 2023-11-30
daily
https://openfun.github.io/ralph/latest/UPGRADE/
- 2023-11-24
+ 2023-11-30
daily
https://openfun.github.io/ralph/latest/commands/
- 2023-11-24
+ 2023-11-30
daily
https://openfun.github.io/ralph/latest/contribute/
- 2023-11-24
+ 2023-11-30
daily
https://openfun.github.io/ralph/latest/api/
- 2023-11-24
+ 2023-11-30
daily
https://openfun.github.io/ralph/latest/backends/
- 2023-11-24
+ 2023-11-30
daily
https://openfun.github.io/ralph/latest/models/
- 2023-11-24
+ 2023-11-30
daily
https://openfun.github.io/ralph/latest/tutorials/cli/
- 2023-11-24
+ 2023-11-30
daily
https://openfun.github.io/ralph/latest/tutorials/development_contributing/
- 2023-11-24
+ 2023-11-30
daily
https://openfun.github.io/ralph/latest/tutorials/library/
- 2023-11-24
+ 2023-11-30
daily
\ No newline at end of file
diff --git a/master/sitemap.xml.gz b/master/sitemap.xml.gz
index bb6c71173..f7219564e 100644
Binary files a/master/sitemap.xml.gz and b/master/sitemap.xml.gz differ