Most deployments of ACA-Py use a single wallet for all operations. This means all connections, credentials, keys, and everything else is stored in the same wallet and shared between all controllers of the agent. Multi-tenancy in ACA-Py allows multiple tenants to use the same ACA-Py instance with a different context. All tenants get their own encrypted wallet that only holds their own data.
This allows ACA-Py to be used for a wider range of use cases. One use case could be a company that creates a wallet for each department. Each department has full control over the actions they perform while having a shared instance for easy maintenance. Another use case could be for a Issuer-Hosted Custodial Agent. Sometimes it is required to host the agent on behalf of someone else.
- General Concept
- Multi-tenant Admin API
- Managed vs Unmanaged Mode
- Message Routing
- Webhooks
- Authentication
- Tenant Management
When multi-tenancy is enabled in ACA-Py there is still a single agent running, however, some of the resources are now shared between the tenants of the agent. Each tenant has their own wallet, with their own DIDs, connections, and credentials. Transports and most of the settings are still shared between agents. Each wallet uses the same endpoint, so to the outside world, it is not obvious multiple tenants are using the same agent.
Multi-tenancy in ACA-Py makes a distinction between a base wallet and sub wallets.
The wallets used by the different tenants are called sub wallets. A sub wallet is almost identical to a wallet when multi-tenancy is disabled. This means that you can do everything with it that a single-tenant ACA-Py instance can also do.
The base wallet however, takes on a different role and has limited functionality. Its main function is to manage the sub wallets, which can be done using the Multi-tenant Admin API. It stores all settings and information about the different sub wallets and will route incoming messages to the corresponding sub wallets. See Message Routing for more details. All other features are disabled for the base wallet. This means it cannot issue credentials, present proof, or do any of the other actions sub wallets can do. This is to keep a clear hierarchical difference between base and sub wallets. For this reason, the base wallet should generally not be provisioned using the --wallet-seed
argument as not only it is not necessary for sub wallet management operations, but it will also require this DID to be correctly registered on the ledger for the service to start-up correctly.
Multi-tenancy is disabled by default. You can enable support for multiple wallets using the --multitenant
startup parameter. To also be able to manage wallets for the tenants, the multi-tenant admin API can be enabled using the --multitenant-admin
startup parameter. See Multi-tenant Admin API below for more info on the admin API.
The --jwt-secret
startup parameter is required when multi-tenancy is enabled. This is used for JWT creation and verification. See Authentication below for more info.
Example:
# This enables multi-tenancy in ACA-Py
multitenant: true
# This enables the admin API for multi-tenancy. More information below
multitenant-admin: true
# This sets the secret used for JWT creation/verification for sub wallets
jwt-secret: Something very secret
With askar wallets it's possible to have all tenant wallets in a single wallet or each have an individual wallet. The default is to have each tenant in a separate wallet. This is done to keep the wallets separate and to allow for more flexibility in the future. If you want to have all tenants in a single wallet you can set the multitenancy-config
with the value {"wallet_type": "single-wallet-askar"}
. If you want to explicitly set the wallet type for each tenant you can do so by setting the multitenancy-config
with the value {"wallet_type": "basic"}
. See .vscode-sample/multitenant-admin.yml for an example.
## Multi-tenant Admin API
The multi-tenant admin API allows you to manage wallets in ACA-Py. Only the base wallet can manage wallets, so you can't for example create a wallet in the context of sub wallet (using the `Authorization` header as specified in [Authentication](#authentication)).
Multi-tenancy related actions are grouped under the `/multitenancy` path or the `multitenancy` topic in the SwaggerUI. As mentioned above, the multi-tenant admin API is disabled by default, event when multi-tenancy is enabled. This is to allow for more flexible agent configuration (e.g. horizontal scaling where only a single instance exposes the admin API). To enable the multi-tenant admin API, the `--multitenant-admin` startup parameter can be used.
See the SwaggerUI for the exact API definition for multi-tenancy.
## Managed vs Unmanaged Mode
Multi-tenancy in ACA-Py is designed with two key management modes in mind.
### Managed Mode
In **`managed`** mode, ACA-Py will manage the key for the wallet. This is the easiest configuration as it allows ACA-Py to fully control the wallet. When a message is received from another agent it can immediately unlock the wallet and process the message. The wallet key is stored encrypted in the base wallet.
### Unmanaged Mode
In **`unmanaged`** mode, ACA-Py won't manage the key for the wallet. The key is not stored in the base wallet, which means the key to unlock the wallet needs to be provided whenever the wallet is used. When a message from another agent is received, ACA-Py cannot immediately unlock the wallet and process the message. See [Authentication](#authentication) for more info.
It is important to note unmanaged mode doesn't provide a lot of security over managed mode. The key is still processed by the agent, and therefore trust is required. It could however provide some benefit in the case a multi-tenant agent is compromised, as the agent doesn't store the key to unlock the wallet.
> :warning: Although support for unmanaged mode is mostly in place, the receiving of messages from other agents in unmanaged mode is not supported yet. This means unmanaged mode can not be used yet.
### Mode Usage
The mode used can be specified when creating a wallet using the `key_management_mode` parameter.
```jsonc
// POST /multitenancy/wallet
{
// ... other params ...
"key_management_mode": "managed" // or "unmanaged"
}
In multi-tenant mode, when ACA-Py receives a message from another agent, it will need to determine which tenant to route the message to. Hyperledger Aries defines two types of routing methods, mediation and relaying.
See the Mediators and Relays RFC for an in-depth description of the difference between the two concepts.
In multi-tenant mode, ACA-Py still exposes a single endpoint for each transport. This means it can't route messages to sub wallets based on the endpoint. To resolve this the base wallet acts as a relay for all sub wallets. As can be seen in the architecture diagram above, all messages go through the base wallet. whenever a sub wallet creates a new key or connection, it will be registered at the base wallet. This allows the base wallet to look at the recipient keys for a message and determine which wallet it needs to route to.
ACA-Py allows messages to be routed through a mediator, and multi-tenancy can be used in combination with external mediators. The following scenarios are possible:
- The base wallet has a default mediator set that will be used by sub wallets.
- Use
--mediator-invitation
to connect to the mediator, request mediation, and set it as the default mediator - Use
default-mediator-id
if you're already connected to the mediator and mediation is granted (e.g. after restart). - When a sub wallet creates a connection or key it will be registered at the mediator via the base wallet connection. The base wallet will still act as a relay and route the messages to the correct sub wallets.
- Pro: Not every wallet needs to create a connection with the mediator
- Con: Sub wallets have no control over the mediator.
- Use
- Sub wallet creates a connection with mediator and requests mediation
- Use mediation as you would in a non-multi-tenant agent, however, the base wallet will still act as a relay.
- You can set the default mediator to use for connections (using the mediation API).
- Pro: Sub wallets have control over the mediator.
- Con: Every wallet
The main tradeoff between option 1. and 2. is redundancy and control. Option 1. doesn't require every sub wallet to create a new connection with the mediator and request mediation. When all sub wallets are going to use the same mediator, this can be a huge benefit. Option 2. gives more control over the mediator being used. This could be useful if e.g. all wallets use a different mediator.
A combination of option 1. and 2. is also possible. In this case, two mediators will be used and the sub wallet mediator will forward to the base wallet mediator, which will, in turn, forward to the ACA-Py instance.
+---------------------+ +----------------------+ +--------------------+
| Sub wallet mediator | ---> | Base wallet mediator | ---> | Multi-tenant agent |
+---------------------+ +----------------------+ +--------------------+
ACA-Py makes use of webhook events to call back to the controller. Multiple webhook targets can be specified, however, in multi-tenant mode, it may be desirable to specify different webhook targets per wallet.
When creating a wallet wallet_dispatch_type
be used to specify how webhooks for the wallet should be dispatched. The options are:
default
: Dispatch only to webhooks associated with this wallet.base
: Dispatch only to webhooks associated with the base wallet.both
: Dispatch to both webhook targets.
If either default
or both
is specified you can set the webhook URLs specific to this wallet using the wallet.webhook_urls
option.
Example:
// POST /multitenancy/wallet
{
// ... other params ...
"wallet_dispatch_type": "default",
"wallet_webhook_urls": [
"https://webhook-url.com/path",
"https://another-url.com/site"
]
}
When the webhook URLs of the base wallet are used or when multiple wallets specify the same webhook URL it can be hard to identify the wallet an event belongs to. To resolve this each webhook event will include the wallet id the event corresponds to.
For HTTP events the wallet id is included as the x-wallet-id
header. For WebSockets, the wallet id is included in the enclosing JSON object.
HTTP example:
POST <webhook-url>/{topic} [headers=x-wallet-id]
{
// event payload
}
WebSocket example:
{
"topic": "{topic}",
"wallet_id": "{wallet_id}",
"payload": {
// event payload
}
}
When multi-tenancy is not enabled you can authenticate with the agent using the x-api-key
header. As there is only a single wallet, this provides sufficient authentication and authorization.
For sub wallets, an additional authentication method is introduced using JSON Web Tokens (JWTs). A token
parameter is returned after creating a wallet or calling the get token endpoint. This token must be provided for every admin API call you want to perform for the wallet using the Bearer authorization scheme.
Example
GET /connections [headers="Authorization: Bearer {token}]
The Authorization
header is in addition to the Admin API key. So if the admin-api-key
is enabled (which should be enabled in production) both the Authorization
and the x-api-key
headers should be provided when making calls to a sub wallet. For calls to a base wallet, only the x-api-key
should be provided.
A token can be obtained in two ways. The first method is the token
parameter from the response of the create wallet (POST /multitenancy/wallet
) endpoint. The second option is using the get wallet token endpoint (POST /multitenancy/wallet/{wallet_id}/token
) endpoint.
This is the method you use to obtain a token when you haven't already registered a tenant. In this process you will first register a tenant then an object containing your tenant token
as well as other useful information like your wallet id
will be returned to you.
Example
new_tenant='{
"image_url": "https://aries.ca/images/sample.png",
"key_management_mode": "managed",
"label": "example-label-02",
"wallet_dispatch_type": "default",
"wallet_key": "example-encryption-key-02",
"wallet_name": "example-name-02",
"wallet_type": "askar",
"wallet_webhook_urls": [
"https://example.com/webhook"
]
}'
echo $new_tenant | curl -X POST "${ACAPY_ADMIN_URL}/multitenancy/wallet" \
-H "Content-Type: application/json" \
-H "X-Api-Key: $ACAPY_ADMIN_URL_API_KEY" \
-d @-
Response
{
"settings": {
"wallet.type": "askar",
"wallet.name": "example-name-02",
"wallet.webhook_urls": [
"https://example.com/webhook"
],
"wallet.dispatch_type": "default",
"default_label": "example-label-02",
"image_url": "https://aries.ca/images/sample.png",
"wallet.id": "3b64ad0d-f556-4c04-92bc-cd95bfde58cd"
},
"key_management_mode": "managed",
"updated_at": "2022-04-01T15:12:35.474975Z",
"wallet_id": "3b64ad0d-f556-4c04-92bc-cd95bfde58cd",
"created_at": "2022-04-01T15:12:35.474975Z",
"token": "eyJ0eXAiOiJKV1QiLCJhbGciOiJIUzI1NiJ9.eyJ3YWxsZXRfaWQiOiIzYjY0YWQwZC1mNTU2LTRjMDQtOTJiYy1jZDk1YmZkZTU4Y2QifQ.A4eWbSR2M1Z6mbjcSLOlciBuUejehLyytCVyeUlxI0E"
}
This method allows you to retrieve a tenant token
for an already registered tenant. To retrieve a token you will need an Admin API key (if your admin is protected with one), wallet_key
and the wallet_id
of the tenant. Note that calling the get tenant token endpoint will invalidate the old token. This is useful if the old token needs to be revoked, but does mean that you can't have multiple authentication tokens for the same wallet. Only the last generated token will always be valid.
Example
curl -X POST "${ACAPY_ADMIN_URL}/multitenancy/wallet/{wallet_id}/token" \
-H "Content-Type: application/json" \
-H "X-Api-Key: $ACAPY_ADMIN_URL_API_KEY" \
-d { "wallet_key": "example-encryption-key-02" }
Response
{
"token": "eyJ0eXAiOiJKV1QiLCJhbGciOiJIUzI1NiJ9.eyJ3YWxsZXRfaWQiOiIzYjY0YWQwZC1mNTU2LTRjMDQtOTJiYy1jZDk1YmZkZTU4Y2QifQ.A4eWbSR2M1Z6mbjcSLOlciBuUejehLyytCVyeUlxI0E"
}
In unmanaged mode, the get token endpoint also requires the wallet_key
parameter to be included in the request body. The wallet key will be included in the JWT so the wallet can be unlocked when making requests to the admin API.
{
"wallet_id": "wallet_id",
// "wallet_key" in only present in unmanaged mode
"wallet_key": "wallet_key"
}
In unmanaged mode, sending the
wallet_key
to unlock the wallet in every request is not “secure” but keeps it simple at the moment. Eventually, the authentication method should be pluggable, and unmanaged mode would just mean that the key to unlock the wallet is not managed by ACA-Py.
For deterministic JWT creation and verification between restarts and multiple instances, the same JWT secret would need to be used. Therefore a --jwt-secret
param is added to the ACA-Py agent that will be used for JWT creation and verification.
When using the SwaggerUI you can click the 🔒 icon next to each of the endpoints or the Authorize
button at the top to set the correct authentication headers. Make sure to also include the Bearer
part in the input field. This won't be automatically added.
After registering a tenant which effectively creates a subwallet, you may need to update the tenant information or delete it. The following describes how to accomplish both goals.
The following properties can be updated: image_url
, label
, wallet_dispatch_type
, and wallet_webhook_urls
for tenants of a multitenancy wallet. To update these properties you will PUT
a request json containing the properties you wish to update along with the updated values to the /multitenancy/wallet/${TENANT_WALLET_ID}
admin endpoint. If the Admin API endpoint is protected, you will also include the Admin API Key in the request header.
Example
update_tenant='{
"image_url": "https://aries.ca/images/sample-updated.png",
"label": "example-label-02-updated",
"wallet_webhook_urls": [
"https://example.com/webhook/updated"
]
}'
echo $update_tenant | curl -X PUT "${ACAPY_ADMIN_URL}/multitenancy/wallet/${TENANT_WALLET_ID}" \
-H "Content-Type: application/json" \
-H "x-api-key: $ACAPY_ADMIN_URL_API_KEY" \
-d @-
Response
{
"settings": {
"wallet.type": "askar",
"wallet.name": "example-name-02",
"wallet.webhook_urls": [
"https://example.com/webhook/updated"
],
"wallet.dispatch_type": "default",
"default_label": "example-label-02-updated",
"image_url": "https://aries.ca/images/sample-updated.png",
"wallet.id": "3b64ad0d-f556-4c04-92bc-cd95bfde58cd"
},
"key_management_mode": "managed",
"updated_at": "2022-04-01T16:23:58.642004Z",
"wallet_id": "3b64ad0d-f556-4c04-92bc-cd95bfde58cd",
"created_at": "2022-04-01T15:12:35.474975Z"
}
An Admin API Key is all that is ALLOWED to be included in a request header during an update. Including the Bearer token header will result in a 404: Unauthorized error
The following information is required to delete a tenant:
- wallet_id
- wallet_key
- {Admin_Api_Key} if admin is protected
Example
curl -X POST "${ACAPY_ADMIN_URL}/multitenancy/wallet/{wallet_id}/remove" \
-H "Content-Type: application/json" \
-H "x-api-key: $ACAPY_ADMIN_URL_API_KEY" \
-d '{ "wallet_key": "example-encryption-key-02" }'
Response
{}
To allow the configuring of ACA-Py startup parameters/environment variables at a tenant/subwallet level. PR#2233 will provide the ability to update the following subset of settings when creating or updating the subwallet:
Labels | Setting | |
---|---|---|
ACAPY_LOG_LEVEL | log-level | log.level |
ACAPY_INVITE_PUBLIC | invite-public | debug.invite_public |
ACAPY_PUBLIC_INVITES | public-invites | public_invites |
ACAPY_AUTO_ACCEPT_INVITES | auto-accept-invites | debug.auto_accept_invites |
ACAPY_AUTO_ACCEPT_REQUESTS | auto-accept-requests | debug.auto_accept_requests |
ACAPY_AUTO_PING_CONNECTION | auto-ping-connection | auto_ping_connection |
ACAPY_MONITOR_PING | monitor-ping | debug.monitor_ping |
ACAPY_AUTO_RESPOND_MESSAGES | auto-respond-messages | debug.auto_respond_messages |
ACAPY_AUTO_RESPOND_CREDENTIAL_OFFER | auto-respond-credential-offer | debug.auto_respond_credential_offer |
ACAPY_AUTO_RESPOND_CREDENTIAL_REQUEST | auto-respond-credential-request | debug.auto_respond_credential_request |
ACAPY_AUTO_VERIFY_PRESENTATION | auto-verify-presentation | debug.auto_verify_presentation |
ACAPY_NOTIFY_REVOCATION | notify-revocation | revocation.notify |
ACAPY_AUTO_REQUEST_ENDORSEMENT | auto-request-endorsement | endorser.auto_request |
ACAPY_AUTO_WRITE_TRANSACTIONS | auto-write-transactions | endorser.auto_write |
ACAPY_CREATE_REVOCATION_TRANSACTIONS | auto-create-revocation-transactions | endorser.auto_create_rev_reg |
ACAPY_ENDORSER_ROLE | endorser-protocol-role | endorser.protocol_role |
-
POST /multitenancy/wallet
Added
extra_settings
dict field to request schema.extra_settings
can be configured in the request body as below:Example Request
{ "wallet_name": " ... ", "default_label": " ... ", "wallet_type": " ... ", "wallet_key": " ... ", "key_management_mode": "managed", "wallet_webhook_urls": [], "wallet_dispatch_type": "base", "extra_settings": { "ACAPY_LOG_LEVEL": "INFO", "ACAPY_INVITE_PUBLIC": true, "public-invites": true }, }
echo $new_tenant | curl -X POST "${ACAPY_ADMIN_URL}/multitenancy/wallet" \ -H "Content-Type: application/json" \ -H "X-Api-Key: $ACAPY_ADMIN_URL_API_KEY" \ -d @-
-
PUT /multitenancy/wallet/{wallet_id}
Added
extra_settings
dict field to request schema.Example Request
{
"wallet_webhook_urls": [ ... ],
"wallet_dispatch_type": "default",
"label": " ... ",
"image_url": " ... ",
"extra_settings": {
"ACAPY_LOG_LEVEL": "INFO",
"ACAPY_INVITE_PUBLIC": true,
"ACAPY_PUBLIC_INVITES": false
},
}
echo $update_tenant | curl -X PUT "${ACAPY_ADMIN_URL}/multitenancy/wallet/${WALLET_ID}" \
-H "Content-Type: application/json" \
-H "x-api-key: $ACAPY_ADMIN_URL_API_KEY" \
-d @-