The repository provides a sample code for Callback Handler and Connector that can be used by the CyberApps.
NOTE: The code is provided for educational purposes only and is not intended for production usage.
- Python 3.6+
aiohttp
,names
andargon2-cffi
installed (provided in the requirements.txt)
The sample code is written in Python and provides:
- An example of the database built with SQLite3.
- A web server that implements a callback handler with:
- An implementation of predefined callbacks (application enablement, customer mapping).
- An example of custom callbacks (CRUD operations for users management).
- An example of the basic connector that posts new alerts and workloads to Acronis.
With the callback handler, your service can receive callback requests from Acronis to allow the users to manage and interact with your CyberApp from Acronis UI.
The following diagram shows how Acronis communicates with the callback handler:
The main scenarios that the callback handler must implement are application enablement and customer mapping. They are required for the service providers to be able to configure your CyberApp and establish the correspondence between the end customers in your system and Acronis.
Additionally, the callback handler may support custom scenarios that would allow service providers and end customers to interact with your service.
For example, a user may open a main menu item in the Acronis UI that shows the information about the users in your service. The UI will send a corresponding callback that will ask for the list of users and show it in the UI as the result.
For more details on implementation, see the Callback Handler section below.
With the connector, your service can send the requests to Acronis API to manage specific resources (such as your custom workloads and alerts) in the Acronis system.
The following diagram shows how the connector communicates with Acronis and its relationship to the callback handler:
For example, the connector may push the alerts from a particular organization in your service to the corresponding tenant in Acronis. Those alerts will be displayed in the Acronis UI in the Alerts section.
For more details on implementation, see the Connector section below.
The database includes 3 tables: users
, organizations
and organizations_mapping
.
The following diagram demonstrates the tables and relationships:
For demo purposes, the sample code matches the Acronis tenants hierarchy as closely as possible:
- There are two kinds of organizations:
- Partner - represents a service provider that provisions service to its end customers.
- Customer - represents an end customer organization that purchases and consumes the services.
- Partner organizations may have any number of nested partners.
- Customers are not allowed to have nested organizations.
The callback handler uses the aiohttp
HTTP server framework to accept the HTTP requests from Acronis and uses the database to manage:
- Registered users and perform authentication.
- Organizations available in the service.
- Organizations mapped to Acronis tenants.
The code that implements the callback handler logic is located in the ./server/
folder.
The routes for the HTTP server are defined in the ./server/routes.py
file. There are two routes defined:
GET /
- the web server index route. Can be used that the web server works.POST /callback
- the callback route that accepts the callbacks.
The processing functions for the routes are defined in the ./server/handlers.py
.
The code implements a simple access control system where the user is authenticated using a username and password. The password is hashed using the Argon2 algorithm and stored in the database.
To authenticate callback requests from Acronis, the callback handler expects the credentials in the X-CyberApp-Auth
header
in the following format: <identity>:{"password": "<password>"}
. Where:
<identity>
- the value of the input field with typeIdentity
.<password>
- the value of the password field with typeSecret
. The password field ID must bepassword
.
The code for callbacks is located in the ./server/callbacks/
folder. The callbacks are grouped into Python modules
and merged in the ./server/callbacks/__init__.py
file that provides the callbacks mapping as a result.
The callback handler implements enablement callbacks that are required to enable and manage your CyberApp successfully.
The sample code is located in ./server/callbacks/enablement.py
.
cti.a.p.acgw.callback.v1.0~a.p.enablement.read.v1.0
- Takes the user'sorganization_id
and performs the lookup in theorganizations
mapping table. The payload includes theorganization_id
column value in thevendor_tenant_id
field. If there's an entry found in theorganizations
mapping table, theacronis_tenant_id
column value is included in theacronis_tenant_id
field.cti.a.p.acgw.callback.v1.0~a.p.enablement.write.v1.0
- Takes the data from the callback's payload and writes the information into theorganizations
mapping table.cti.a.p.acgw.callback.v1.0~a.p.enablement.reset.v1.0
- Deletes all the entries related to the user'sorganization_id
from theorganizations
mapping table.
cti.a.p.acgw.callback.v1.0~a.p.topology.read.v1.0
- Takes the user'sorganization_id
and, if the user's organization is mapped, lists all its customer organizations.cti.a.p.acgw.callback.v1.0~a.p.tenant_mapping.read.v1.0
- Takes the user'sorganization_id
and joins theacronis_tenant_id
from the organizations mapping table for each mapped customer organization and lists the result.cti.a.p.acgw.callback.v1.0~a.p.tenant_mapping.write.v1.0
- Takes the user'sorganization_id
, verifies that provided organization IDs belong to that user, and writes the information into theorganizations
mapping table.
Users management callbacks demonstrate how users can be managed with basic CRUD operations.
The callback handler implements the following custom callbacks:
cti.a.p.acgw.callback.v1.0~{APP_CODE}.users_read.v1.0
- Reads the list of users that are registered in the user's organization.cti.a.p.acgw.callback.v1.0~{APP_CODE}.user_update.v1.0
- Updates a user by user ID. Can update username and email.cti.a.p.acgw.callback.v1.0~{APP_CODE}.user_write.v1.0
- Creates a new user in the user's organization without a password. Note that login is unique and an attempt to add a user with already existing login will fail.cti.a.p.acgw.callback.v1.0~{APP_CODE}.user_delete.v1.0
- Deletes a user by user ID.
Where {APP_CODE}
is the template variable that is substituted with the value specified in ./constants.py
.
Request
Callback ID: cti.a.p.acgw.callback.v1.0~{APP_CODE}.users_read.v1.0
Callback request type: cti.a.p.acgw.request.v1.0~{APP_CODE}.users_read.v1.0
Callback request schema:
{}
Responses
-
Status code:
200
Response type:
cti.a.p.acgw.response.v1.0~{APP_CODE}.users_read_success.v1.0
Schema:
{ "$schema": "http://json-schema.org/draft-04/schema", "type": "object", "properties": { "items": { "type": "array", "items": { "type": "object", "properties": { "id": { "type": "string" }, "name": { "type": "string" }, "email": { "type": "string" } }, "required": ["id", "name", "email"] } } } }
Request
Callback ID: cti.a.p.acgw.callback.v1.0~{APP_CODE}.user_write.v1.0
Callback request type: cti.a.p.acgw.request.v1.0~{APP_CODE}.user_write.v1.0
Callback request schema:
{
"$schema": "http://json-schema.org/draft-04/schema",
"type": "object",
"properties": {
"login": {
"type": "string"
},
"name": {
"type": "string"
},
"email": {
"type": "string"
}
},
"required": ["login", "name", "email"]
}
Responses
-
Status code:
200
Response type:
cti.a.p.acgw.response.v1.0~{APP_CODE}.user_write_success.v1.0
Schema:
{}
Request
Callback ID: cti.a.p.acgw.callback.v1.0~{APP_CODE}.user_update.v1.0
Callback request type: cti.a.p.acgw.request.v1.0~{APP_CODE}.user_update.v1.0
Callback request schema:
{
"$schema": "http://json-schema.org/draft-04/schema",
"type": "object",
"properties": {
"id": {
"type": "string"
},
"name": {
"type": "string"
},
"email": {
"type": "string"
}
},
"required": ["id", "name", "email"]
}
Responses
-
Status code:
200
Response type:
cti.a.p.acgw.response.v1.0~{APP_CODE}.user_update_success.v1.0
Schema:
{}
Request
Callback ID: cti.a.p.acgw.callback.v1.0~{APP_CODE}.user_delete.v1.0
Callback request type: cti.a.p.acgw.request.v1.0~{APP_CODE}.user_delete.v1.0
Callback request schema:
{
"$schema": "http://json-schema.org/draft-04/schema",
"type": "object",
"properties": {
"id": {
"type": "string"
}
},
"required": ["id"]
}
Responses
-
Status code:
200
Response type:
cti.a.p.acgw.response.v1.0~{APP_CODE}.user_delete_success.v1.0
Schema:
{}
The connector uses the aiohttp
HTTP client framework to communicate with Acronis API and uses the database
to access the information about organizations that have mapped Acronis tenants.
The connector implements:
- API client authentication in Acronis cloud using OAuth 2.0 with the
client_credentials
flow. - Sending information about the alerts using the Alert Manager API.
- Sending information about the workloads using the Workload Management API.
The sample code is located in the ./connector/
folder.
The code authenticates the API client and runs a basic loop where the API client pushes the alerts and workloads to all mapped tenants using Acronis API once in 30 minutes.
Note that the database does not store any actual alerts and workloads. The code implements a generator of random alerts and workloads to demonstrate the API requests. See ./connector/generator.py
and follow the instructions provided in this file to send your alerts and workloads.
The code is bundled together with the Postman collection that can be used to send requests to the callback handler directly.
An online version of the collection is also available here.
-
Install the requirement with the following command:
pip install -r requirements.txt
-
[During the first time run] Run the script to create a database:
python ./create_db.py
You will be prompted to enter the root username and password.
The script will create an organization and sample sub-organizations that can be used for customer mapping.
Additionally, sample users will be created in sample organizations and displayed in custom callbacks.
By default, the script will write the resulting data to the
vendor.db
database file.
To run the callback handler:
Run the following command:
python ./run_server.py
This will run HTTP server on port 8080 and connect to vendor.db
database file by default.
You can also specify SSL certificates to enable HTTPS:
python ./run_server.py --certfile <absolute_path_to_cert> --certkey <absolute_path_to_key>
This will run HTTPS server on port 443 and connect to vendor.db
database file by default.
To run the connector:
-
Rename
connector.example.json
toconnector.json
. -
Open
connector.json
and specify the client credentials in the corresponding fields. -
Run the following command:
python ./run_connector.py
This will connect to vendor.db
database file by default.