Discussions - API Design
Request & Response Envelopes
All API should follow the standard request and response structures:
Request Structure:
{
"params": {
"msgid": "4f04da60-1e24-4d31-aa7b-1daf91c46341" -- optional, random uuid
},
"request": {
<RequestData of the API>
}
}
Response Structure:
{
"id": "API Id, e.g. api.forum.create",
"ver": "API version, e.g. 1.0",
"ets": "response timestamp in epoch format",
"params": {
"resmsgid": "4f04da60-1e24-4d31-aa7b-1daf91c46341" -- random uuid,
"msgid": "4f04da60-1e24-4d31-aa7b-1daf91c46341" -- the msgid sent in request
"status": "successful or failed",
"err": "error code if any",
"errmsg": "error message if there is any error",
},
"responseCode": "OK, CLIENT_ERROR, RESOURCE_NOT_FOUND, SERVER_ERROR",
"result": {
<ResponseData of the API>
}
}
NodeBB APIs
NodeBB provides the following APIs:
Read APIs: https://docs.nodebb.org/api/
Write APIs via plugin: nodebb-plugin-write-api/routes/v2/readme.md at master ยท NodeBB/nodebb-plugin-write-api
In addition to the above APIs, we need a few additional APIs to use NodeBB discussions in different Sunbird use cases.
Sunbird APIs
Sunbird Discussion APIs will use the following terminology:
Forum
A forum contains one or more discussion threads with one or more users participating in the discussions. Internally, forum is also a NodeBB category but there are some differences between a Sunbird forum and a NodeBB category:
Every forum is linked to a Sunbird context (e.g.: a Course batch).
A forum can have one or more NodeBB categories. E.g.: A course may have two categories - Announcements and General Discussions.
A forum can have a list of groups (with a set of permissions) associated with it & its sub-categories.
Hierarchy
Forum are organised into a hierarchy for easy discovery, navigation and access. For example, all forums belonging to an organisation can grouped under a parent forum. Example hierarchy:
CBSE
Course
Classroom Management
Announcements
General Discussions
Textbook
Science Textbook for Class X
Maths Textbook for Class IX
Contribution Projects
KVS 6-8 Hindi & Science Textbooks
Permissions
Permissions are the list of privileges that user or group has in a forum. Instead of using NodeBB privileges, Sunbird APIs will support the following permissions:
read - nodebb privileges: 'topics:read', 'read', 'find'
vote - nodebb privileges: 'posts:upvote', 'posts:downvote'
post - nodebb privileges: 'topics:reply', 'posts:edit', 'posts:delete'
topic - nodebb privileges: 'topics:create', 'topics:tag'
moderate - nodebb privileges: 'topics:delete', 'posts:view_deleted', 'purge', 'moderate', 'posts:history'
In addition to the above nodebb privileges, a user with โmoderateโ permission should be able to edit the current forum (using Update Forum API).
The permissions read, vote, post, topic and moderate are hierarchical. I.e., if a higher permission (e.g. topic) is given, all the below permissions (e.g. read, vote, and post) are automatically granted.
Create User API
Request Path: โ/api/user/v1/createโ
Method: โPOSTโ
Request Structure
{
"params": {
"msgid": "4f04da60-1e24-4d31-aa7b-1daf91c46341" -- random uuid
},
"request": {
โidentifierโ: โsunbird user idโ,
โusernameโ: โsunbird user nameโ
}
}
Response Structure
{
"id": "api.discussions.user.create",
"ver": "1.0",
"ets": "1600883199000",
"params": {
"resmsgid": "4f04da60-1e24-4d31-aa7b-1daf91c46341"
"msgid": "4f04da60-1e24-4d31-aa7b-1daf91c46341",
"status": "successful",
},
"responseCode": "OK",
"result": {
โidentifierโ: โsunbird oidc user idโ
}
}
Create User API will create a nodebb user for the given sunbird user id and username.
This API will be exposed by nodebb-plugin-sunbird-oidc plugin.
This API will throw an error if a user already exists with the given sunbird user id.
This API will create an email using the username and the configured email domain.
Create Forum API
Request Path: โ/api/forum/v1/createโ
Method:โPOSTโ
Request Structure
{
"params": {
"msgid": "4f04da60-1e24-4d31-aa7b-1daf91c46341" -- random uuid
},
"request": {
"name": "name of the forum",
โcontextโ: {
โtypeโ: โbatch / organisation / sb-group / content-collaboration / etcโ,
โidentifierโ: โbatch id or org id or sb-group id or content id, e.g: 012232323223232โ
},
โparentโ: "parent forum id under which the forum has to be created. optional, if not provided, this forum is created under the root",
โgroupsโ : [ // list of groups to be created and associated with the forum. group id will be auto-generated using a combination of the forum id and a random string/number
{ โpermissionsโ: [โreadโ, โvoteโ], "applyToChildren": "true/false" }, // permissions to be given for each associated group
{ โpermissionsโ: [โreadโ, โvoteโ, โpostโ, โtopicโ, โmoderateโ], "applyToChildren": "true/false" }
],
โprivilegesโ: [ // set of permissions to be given for existing users or groups can also be provided
{
โpermissionsโ: [โreadโ, โvoteโ, โpostโ, โtopicโ, โmoderateโ],
โusersโ: [โlist of sunbird user oidc idsโ],
โgroupsโ: [โlist of nodebb group idsโ],
"applyToChildren": "true/false, if these users/groups should be given same set of privileges in all the sub-forums"
},
{
โpermissionsโ: [โreadโ, โvoteโ, โpostโ, โtopicโ, โmoderateโ],
โusersโ: [โlist of sunbird user oidc idsโ],
โgroupsโ: [โlist of nodebb group idsโ],
"applyToChildren": "true/false"
}
],
"children": [ // list of sub-forums to be created
{
"name": "Name of the sub-forum",
"groups": [], // permissions for the sub-forum can be configured by associating groups or
"privileges": [] // giving privileges to existing users or groups
}
]
}
}
Response Structure
{
"id": "api.discussions.forum.create",
"ver": "1.0",
"ets": "1600883199000",
"params": {
"resmsgid": "4f04da60-1e24-4d31-aa7b-1daf91c46341"
"msgid": "4f04da60-1e24-4d31-aa7b-1daf91c46341",
"status": "successful",
},
"responseCode": "OK",
"result": {
โidentifierโ: โnewly created forum idโ
}
}
Update Forum API
Request Path: โ/api/forum/v1/update/<forum_id>โ
Method:โPATCHโ
Request Structure
{
"params": {
"msgid": "4f04da60-1e24-4d31-aa7b-1daf91c46341" -- random uuid
},
"request": {
"name": "updated name of the forum",
โgroupsโ : [ // list of groups to be additionally created and associated with the forum. group id will be auto-generated using a combination of the forum id and a random string/number
{ โpermissionsโ: [โreadโ, โvoteโ], "applyToChildren": "true/false" }, // permissions to be given for each associated group
{ โpermissionsโ: [โreadโ, โvoteโ, โpostโ, โtopicโ, โmoderateโ], "applyToChildren": "true/false" }
],
โprivilegesโ: [ // set of permissions to be given/updated for existing users or groups can also be provided
{
โpermissionsโ: [โreadโ, โvoteโ, โpostโ, โtopicโ, โmoderateโ],
โusersโ: [โlist of sunbird user oidc idsโ],
โgroupsโ: [โlist of nodebb group idsโ],
"applyToChildren": "true/false, if these users/groups should be given same set of privileges in all the sub-forums"
},
{
โpermissionsโ: null, // if permissions is set as null, access to the given list of users/groups should be removed
โusersโ: [โlist of sunbird user oidc idsโ],
โgroupsโ: [โlist of nodebb group idsโ],
"applyToChildren": "true/false"
}
],
"children": [ // list of new sub-forums to be created
{
"name": "Name of the sub-forum",
"groups": [], // permissions for the sub-forum can be configured by associating groups or
"privileges": [] // giving privileges to existing users or groups
}
]
}
}
Response Structure
{
"id": "api.discussions.forum.update",
"ver": "1.0",
"ets": "1600883199000",
"params": {
"resmsgid": "4f04da60-1e24-4d31-aa7b-1daf91c46341"
"msgid": "4f04da60-1e24-4d31-aa7b-1daf91c46341",
"status": "successful",
},
"responseCode": "OK",
"result": {
โidentifierโ: โforum idโ
}
}
Fetch Forum API
API to get details of a forum for a given context.
Request Path: โ/api/forum/v1/fetchโ
Method:โPOSTโ
Request Structure
{
"params": {
"msgid": "4f04da60-1e24-4d31-aa7b-1daf91c46341" -- random uuid
},
"request": {
โcontextโ: {
โtypeโ: โbatchโ,
โidentifierโ: โbatch id, e.g: 012232323223232โ
}
}
}
Response Structure
{
"id": "api.discussions.forum.fetch",
"ver": "1.0",
"ets": "1600883199000",
"params": {
"resmsgid": "4f04da60-1e24-4d31-aa7b-1daf91c46341"
"msgid": "4f04da60-1e24-4d31-aa7b-1daf91c46341",
"status": "successful",
},
"responseCode": "OK",
"result": {
โforumsโ: [
{
โidentifierโ: โforum idโ,
โnameโ: โname of the forumโ,
โgroupsโ: [ // list of groups which have access to the forum
{
โidentifierโ: โidentifier of the nodebb groupโ,
โpermissionsโ: [โreadโ, โvoteโ, โpostโ, โtopicโ, โmoderateโ]
}
],
โusersโ: [ // list of users who have access to the forum
{
โidentifierโ: โsunbird user oidc idโ,
โpermissionsโ: [โreadโ, โvoteโ, โpostโ, โtopicโ, โmoderateโ]
}
],
"children": [ // list of sub-forums
{
"identifier": "sub-forum id",
"name": "name of the sub-forum",
"groups": [], // list of groups which have access to the sub-forum
"users": [] // list of users who have access to the sub-forum
}
]
}
]
}
Read Forum API
API to get details of a forum by forum id.
Request Path: โ/api/forum/v1/read/<forum_id>โ
Method:โGETโ
Response Structure
{
"id": "api.discussions.forum.read",
"ver": "1.0",
"ets": "1600883199000",
"params": {
"resmsgid": "4f04da60-1e24-4d31-aa7b-1daf91c46341"
"msgid": "4f04da60-1e24-4d31-aa7b-1daf91c46341",
"status": "successful",
},
"responseCode": "OK",
"result": {
โforumsโ: [
{
โidentifierโ: โforum idโ,
โnameโ: โname of the forumโ,
โgroupsโ: [ // list of groups which have access to the forum
{
โidentifierโ: โidentifier of the nodebb groupโ,
โpermissionsโ: [โreadโ, โvoteโ, โpostโ, โtopicโ, โmoderateโ]
}
],
โusersโ: [ // list of users who have access to the forum
{
โidentifierโ: โsunbird user oidc idโ,
โpermissionsโ: [โreadโ, โvoteโ, โpostโ, โtopicโ, โmoderateโ]
}
],
"children": [ // list of sub-forums
{
"identifier": "sub-forum id",
"name": "name of the sub-forum",
"groups": [], // list of groups which have access to the sub-forum
"users": [] // list of users who have access to the sub-forum
}
]
}
]
}
API Security
API Token
All APIs (including NodeBB APIs) should be onboarded on to Sunbird API manager and appropriate ACLs should be configured for all the APIs. No API should be directly accessible.
All API requests should have a request header โauthorizationโ in which the API token should be passed:
-H 'authorization: Bearer {{api_token}}'
TODO: List the ACL - API mapping for all the APIs
User Auth Token
All Sunbird discussion APIs should have an additional security layer that validates the userโs access using the user token generated by keycloak. Keycloak token should be passed to the API using โx-authenticated-user-tokenโ header.
-H 'x-authenticated-user-token: {{keycloak_access_token}}'
API should check if the passed user token is valid
API should check if the user whose token is passed in the request has access to perform the required operation (e.g.: in Update Forum API, check if user has access to update the specific forum)