We offer a robust enterprise-grade multi-tenant digital asset custody solution.
Our solution employs Multi-Party Computation (MPC) to secure your keys. The standard installation consists of three cosigners, each deployed in separate locations, requiring at least two out of the three parties to execute the protocol to access the funds.
We currently support the following blockchain networks, with continuous additions of new chains and tokens:
Our system is structured around three primary concepts: vaults, accounts, and wallets. A single vault can house multiple accounts, while an account is limited to containing a wallet for a specific asset. An example setup is as follows:
Webhooks can be configured at both the wallet and vault levels to receive notifications regarding incoming and outgoing transactions. Notifications will be sent upon transactions reaching the mempool and upon a configurable number of confirmations.
Automatic fund sweeps can be established at the wallet level for enhanced financial management.
This API specification is in Beta and might change during the following weeks. We are working on releasing a stable specification early August 2023.
We provide 2 different environments: production and sandbox, you can setup your api key in the console under settings, the endpoints are as follows
Environment | Setup Link |
---|---|
Sandbox (testnet) | sandbox |
Prod (mainnet) | prod |
Once you setup your API key you will need to include the following headers in your API calls:
X-Cf-Key
: The key ID (uuid)
X-Cf-Secret
: The key secret (make sure to store it in a safe place on creation since it won't be retrievable after that)
Most List
operations will generate a paginated response returning at most 50 items, if there are more items you will get in the body a property named last
that you will need to include in the next GET
request as a path parameter e.g.:
{
"vaults": [...],
"last": "eyJjcmVhdGVkQXQiOiIyMDIzLTAzLTA1VDAyOjUwOjE0LjQ4OFoifQ=="
}
In order to get the next page you will have to do the following GET
:
https://{endpoint}/vaults?last=eyJjcmVhdGVkQXQiOiIyMDIzLTAzLTA1VDAyOjUwOjE0LjQ4OFoifQ==
Vauls are the main entity in the application. In order to create a wallet you will need to create first a vault and an account.
List of vaults in your organization, returns a paginated response
X-Cf-Key | string Example: {{apikey}} |
X-Cf-Secret | string Example: {{apisecret}} |
{- "vaults": [
- {
- "createdAt": "2023-06-18T15:10:23.766Z",
- "name": "first",
- "status": "COMPLETED",
- "vaultId": "df82cf8a-0d86-4a92-80ef-f5f39314e0e9"
}
]
}
Create a vault is an asynchronous operation, you will get a response with the status KEYGEN_IN_PROGRESS
and you will need to query the get vault endpoint to check the current status.
Normally you should not need to have more than 10 vaults in your organization.
X-Cf-Key | string Example: {{apikey}} |
X-Cf-Secret | string Example: {{apisecret}} |
externalId | string |
name | string |
{- "externalId": "your-unique-external-id",
- "name": "operations"
}
{- "createdAt": "2023-07-01T06:26:19.217Z",
- "name": "operations",
- "status": "KEYGEN_IN_PROGRESS",
- "vaultId": "ecd0d15b-43be-5182-8b57-d974fc49c36c"
}
List of vaults in your organization, returns a paginated response
vaultId required | string |
X-Cf-Key | string Example: {{apikey}} |
X-Cf-Secret | string Example: {{apisecret}} |
{- "createdAt": "2023-07-01T06:26:19.217Z",
- "name": "operations",
- "status": "COMPLETED",
- "vaultId": "ecd0d15b-43be-5182-8b57-d974fc49c36c"
}
Lists the accounts in a vault, the response is paginated
vaultId required | string |
X-Cf-Key | string Example: {{apikey}} |
X-Cf-Secret | string Example: {{apisecret}} |
{- "accounts": [
- {
- "accountId": "8f479e39-e32f-52ba-9109-505c3dce9a6b",
- "createdAt": "2023-07-01T06:41:08.893Z",
- "externalId": "your-unique-external-id",
- "name": "customer 007",
- "updatedAt": "2023-07-01T06:41:08.893Z",
- "vaultId": "ecd0d15b-43be-5182-8b57-d974fc49c36c"
}
]
}
Create a vault is an asynchronous operation, you will get a response with the status KEYGEN_IN_PROGRESS
and you will need to query the get vault endpoint to check the current status.
Normally you should not need to have more than 10 vaults in your organization.
vaultId required | string |
X-Cf-Key | string Example: {{apikey}} |
X-Cf-Secret | string Example: {{apisecret}} |
externalId | string |
name | string |
{- "externalId": "your-unique-external-id",
- "name": "customer 007"
}
{- "accountId": "8f479e39-e32f-52ba-9109-505c3dce9a6b",
- "createdAt": "2023-07-01T06:41:08.893Z",
- "externalId": "your-unique-external-id",
- "name": "customer 007",
- "updatedAt": "2023-07-01T06:41:08.893Z",
- "vaultId": "ecd0d15b-43be-5182-8b57-d974fc49c36c"
}
Wallets have an asset id, address(es) and a balance.
Currently if you want to create an ERC-20 token wallet you will need to create first the native wallet for that, so if I want to create a USDT
wallet on the Ethereum network, I will need to create a ETH
wallet and then a USDT_ETH
wallet.
Miner fees sending tokens will be paid from the native wallet for the chain, sending USDT
in the Ethereum network will incur in a miner fee payable by the ETH
wallet.
Balances and amounts are expressed natively, meaning that they are whole numbers, e.g.:
Ethereum has 18 decimal places therefore the if a wallet has 1 ETH, the balance will be returned in wei: "1000000000000000000" (1 + 18 zeroes) On the other hand USDT has 6 decimal places so the balance of a wallet containing 1 USDT will show as: "1000000" (1 + 6 zeroes)
Lists all the wallets in the vault, the response is paginated
vaultId required | string |
X-Cf-Key | string Example: {{apikey}} |
X-Cf-Secret | string Example: {{apisecret}} |
{- "wallets": [
- {
- "accountId": "8f479e39-e32f-52ba-9109-505c3dce9a6b",
- "address": "0xc6a08B2Dbd23166FaDa6D6e01a61e94F8AE63825",
- "assetId": "ETH_GOERLI",
- "balance": "5000000000000000",
- "createdAt": "2023-07-01T07:13:03.441Z",
- "name": "ETH Goerli",
- "vaultId": "ecd0d15b-43be-5182-8b57-d974fc49c36c",
- "walletId": "0a2d3e91-0e2e-5c17-8675-d140ec0b21bb"
}, - {
- "accountId": "4cb2e67e-657a-53b1-a451-a8d9def5a6fc",
- "address": "0xe5eE8842A21c712f0033ADA9BA674F714D34C511",
- "assetId": "ETH_GOERLI",
- "balance": "0",
- "createdAt": "2023-07-01T07:32:48.385Z",
- "name": "ETH Goerli",
- "vaultId": "ecd0d15b-43be-5182-8b57-d974fc49c36c",
- "walletId": "f5a238e2-6e57-5716-b476-94bcc6da9188"
}
]
}
Creates a wallet, given the asset id and the account id
vaultId required | string |
X-Cf-Key | string Example: {{apikey}} |
X-Cf-Secret | string Example: {{apisecret}} |
accountId | string |
assetId | string |
name | string |
{- "accountId": "{{accountId}}",
- "assetId": "ETH_GOERLI",
- "name": "ETH Goerli"
}
{- "accountId": "8f479e39-e32f-52ba-9109-505c3dce9a6b",
- "address": "0xc6a08B2Dbd23166FaDa6D6e01a61e94F8AE63825",
- "assetId": "ETH_GOERLI",
- "balance": "0",
- "createdAt": "2023-07-01T07:13:03.441Z",
- "name": "ETH Goerli",
- "vaultId": "ecd0d15b-43be-5182-8b57-d974fc49c36c",
- "walletId": "0a2d3e91-0e2e-5c17-8675-d140ec0b21bb"
}
Get a wallet by ID
vaultId required | string |
walletId required | string |
X-Cf-Key | string Example: {{apikey}} |
X-Cf-Secret | string Example: {{apisecret}} |
{- "accountId": "8f479e39-e32f-52ba-9109-505c3dce9a6b",
- "address": "0xc6a08B2Dbd23166FaDa6D6e01a61e94F8AE63825",
- "assetId": "ETH_GOERLI",
- "balance": "5000000000000000",
- "createdAt": "2023-07-01T07:13:03.441Z",
- "name": "ETH Goerli",
- "vaultId": "ecd0d15b-43be-5182-8b57-d974fc49c36c",
- "walletId": "0a2d3e91-0e2e-5c17-8675-d140ec0b21bb"
}
Send a specific amount from this wallet. The amount must be expressed natively, so if you want to send 1 USDT, since USDT has 6 decimal places you must specify:
"amount": "1000000"
vaultId required | string |
walletId required | string |
X-Cf-Key | string Example: {{apikey}} |
X-Cf-Secret | string Example: {{apisecret}} |
amount | string |
assetId | string |
toAddress | string |
{- "amount": "2000000000000000",
- "assetId": "ETH_GOERLI",
- "toAddress": "0xe5eE8842A21c712f0033ADA9BA674F714D34C511"
}
{- "amount": "2000000000000000",
- "assetId": "ETH_GOERLI",
- "createdAt": "2023-07-01T07:36:40.757Z",
- "direction": "OUT",
- "from": "0xc6a08b2dbd23166fada6d6e01a61e94f8ae63825",
- "orgId": "e1d5928f-2c5d-4961-821d-7fcef8217e51",
- "status": "NEW",
- "to": "0xe5eE8842A21c712f0033ADA9BA674F714D34C511",
- "txId": "07bae4e2-45b2-478b-b365-c6e9775a0544",
- "type": "USER_SEND",
- "updatedAt": "2023-07-01T07:36:40.757Z",
- "vaultId": "ecd0d15b-43be-5182-8b57-d974fc49c36c",
- "walletId": "0a2d3e91-0e2e-5c17-8675-d140ec0b21bb"
}
Lists the transactions of a wallet, the result is paginated
vaultId required | string |
walletId required | string |
X-Cf-Key | string Example: {{apikey}} |
X-Cf-Secret | string Example: {{apisecret}} |
{- "transactions": [
- {
- "amount": "2000000000000000",
- "assetId": "ETH_GOERLI",
- "confirmations": 1,
- "createdAt": "2023-07-01T07:36:40.757Z",
- "direction": "OUT",
- "from": "0xc6a08b2dbd23166fada6d6e01a61e94f8ae63825",
- "minerFee": "31500000588000",
- "orgId": "e1d5928f-2c5d-4961-821d-7fcef8217e51",
- "status": "CONFIRMED",
- "to": "0xe5eE8842A21c712f0033ADA9BA674F714D34C511",
- "txHash": "0x719042e022f062e00ebf08247a9fc69d4134c6e8742333983b75aae6c7f7b0aa",
- "txId": "07bae4e2-45b2-478b-b365-c6e9775a0544",
- "type": "USER_SEND",
- "updatedAt": "2023-07-01T07:37:09.453Z",
- "vaultId": "ecd0d15b-43be-5182-8b57-d974fc49c36c",
- "walletId": "0a2d3e91-0e2e-5c17-8675-d140ec0b21bb"
}, - {
- "amount": "5000000000000000",
- "assetId": "ETH_GOERLI",
- "confirmations": 24,
- "createdAt": "2023-07-01T07:27:48.000Z",
- "direction": "IN",
- "from": "0xeb4f368a2e9f9082e4a89d507fe7af29d41cb083",
- "minerFee": "31500000546000",
- "orgId": "e1d5928f-2c5d-4961-821d-7fcef8217e51",
- "status": "COMPLETE",
- "to": "0xc6a08b2dbd23166fada6d6e01a61e94f8ae63825",
- "txHash": "0xd24ff24fce5207966292775472a0a2b64a5d6b6294f597244ae839fe2a5c12d5",
- "txId": "970a018f-5b9b-5f85-a808-ef0683311912",
- "type": "RECEIVE",
- "updatedAt": "2023-07-01T07:34:19.017Z",
- "vaultId": "ecd0d15b-43be-5182-8b57-d974fc49c36c",
- "walletId": "0a2d3e91-0e2e-5c17-8675-d140ec0b21bb"
}
]
}
Gets a transaction by ID
vaultId required | string |
walletId required | string |
txId required | string |
X-Cf-Key | string Example: {{apikey}} |
X-Cf-Secret | string Example: {{apisecret}} |
{- "amount": "2000000000000000",
- "assetId": "ETH_GOERLI",
- "confirmations": 21,
- "createdAt": "2023-07-01T07:36:40.757Z",
- "direction": "OUT",
- "from": "0xc6a08b2dbd23166fada6d6e01a61e94f8ae63825",
- "minerFee": "31500000588000",
- "orgId": "e1d5928f-2c5d-4961-821d-7fcef8217e51",
- "status": "COMPLETE",
- "to": "0xe5eE8842A21c712f0033ADA9BA674F714D34C511",
- "txHash": "0x719042e022f062e00ebf08247a9fc69d4134c6e8742333983b75aae6c7f7b0aa",
- "txId": "07bae4e2-45b2-478b-b365-c6e9775a0544",
- "type": "USER_SEND",
- "updatedAt": "2023-07-01T07:41:57.110Z",
- "vaultId": "ecd0d15b-43be-5182-8b57-d974fc49c36c",
- "walletId": "0a2d3e91-0e2e-5c17-8675-d140ec0b21bb"
}