Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
This guide is intended for Platforms to learn about how they can integrate the Liquid Collective protocol with the support of Alluvial's APIs.
To help illustrate how a Platform utilizes the Alluvial APIs to support its Liquid Collective protocol integration, this guide introduces a few fictional characters:
Acme Custodian Corp: Financial institution that is a Platform for the Liquid Collective protocol and will offer its customers the ability to participate in the Liquid Collective protocol by staking ETH.
Alice: user of Acme Custodian Corp
Bob: user of Acme Custodian Corp
A custodial wallet's private key(s) are held by either a 3rd party or by the financial institution that is servicing the wallet account on behalf of the customer. In this example, Alice and Bob have a custodial account with Acme Corp and Acme Corp owns / has access to the private key associated with Alice's and Bob's wallet addresses.
Below is an example of the flow of funds, showing how the staking action by Alice and Bob will flow through Acme's architecture and the Alluvial APIs.
Non-custodial (or self-managed) wallets are those in which the customer holds the private keys. In this example, Alice is the only one with access to the private key for her wallet.
The largest differences between the two flow of funds diagrams are the wallet ownership and actions (i.e. signing transactions).
A segregated account structure is one where each customer at the financial institution (ex. Acme Corp) has their own unique, separate account. In this example, both Alice and Bob have their own unique Ethereum address associated with their individual accounts at Acme Corp.
Based on the example segregated account structure, you can see how Acme Corp will map its internal accounts to the technical resources (Organization, Depositor, Wallet) created when interacting with the Alluvial API.
An omnibus account is an account structure in which all users' accounts are grouped into a single wallet address.
Below is an image of how to map Acme's architecture to resources (Organization, Depositor, and Wallet) created via Alluvial's API.
Alluvial isn't aware of, and at no time has control over, the underlying account structure Acme implements. Acme will create a Depositor object that represents all of the internal customers.
Review the Depositor states (deny, pause, reactivate, and remove) as applying these states will impact all underlying customers.
Review the Authentication Guide for the Alluvial API.
First create an account object for each of your users.
Request:
Response:
Attach a wallet object to each account. You can add the wallet to the Allowlist (on-chain) or the On-Platform list (off-chain).
The first example below shows how to add a wallet address to the Allowlist (which be default is added to the On-Platform list).
Request:
Response:
Platforms that enable mint/redeem may also need to add a wallet address to the On-Platform list separately. The following request is for a Platform that enables mint/redeem to add a wallet to the On-Platform list.
Request:
Request:
Now that you have an Account object created and Wallet objects associated, the user can stake ETH and mint LsETH. Before depositing, ensure that the Wallet object(s) have a status = ALLOWLISTED
.
To complete the staking process, review this guide.
Alluvial is a software development company supporting the development of the Liquid Collective protocol. We're focused on the overall growth and maturity of the ecosystem by fostering participation in proof of stake blockchains.
The Alluvial docs have three key sections:
To learn more about the Liquid Collective protocol and how it works, check out Liquid Collective's .
: This section contains several step by step guides for how technical teams can implement key Liquid Staking flows.
: This section shows all the raw API requests that Liquid Collective Platforms can call.
: This section includes guides for how to interact with the Liquid Collective protocol via other software platforms.
If you are an engineer or technical person, looking to understand the level of effort or work involved in implementing the key flows (ETH staking and LsETH redemptions) then we recommend reviewing the and guides. The is a helpful compliment to these key guides.
Using Alluvial APIs makes it easy to onboard and interact with the Liquid Collective protocol.
There are two types of Liquid Collective Platforms. Platforms either:
Enable mint/redeem: These Platforms enable direct Liquid Collective protocol interactions, including depositing ETH to Ethereum's deposit contract and redeeming LsETH for ETH (minting and redeeming). These Platforms offer KYC/AML and Sanctions Screening procedures for their users, then submit the user's wallet address to Liquid Collective's Allowlist smart contract upon successful completion.
Enable secondary interactions: These Platforms do not enable minting, redeeming, or direct Liquid Collective protocol interactions, but do enable secondary protocol interactions such as trading, lending, or other services not required to Allowlist KYC/AML and Sanctions Screened wallet addresses. Users of these Platforms can seamlessly interact with LsETH, and may accrue Ethereum's consensus and execution layer network rewards simply by holding LsETH.
There are two lists that Platforms can interact with:
Allowlist: Any address added to this list can mint and/or burn directly with the Liquid Collective protocol. Only wallets that have on-ramped via a mint/redeemed enabled Platform can be added to this list.
On-Platform: Platforms that only enable secondary interactions can only add wallets to the On-Platform list, and cannot interact with the Allowlist smart contract. When a wallet address is added to the Allowlist it is also automatically added to the On-Platform list. This list is used for calculating Platform rebates.
Below are two guides for Platforms to add their users to the appropriate lists.
Review the Authentication Guide for the Alluvial API.
Platforms that enable secondary protocol interactions, such as trading, lending, or other services, can create an Account object. The Account object is generally associated with a Platform's customer.
Request:
Response:
After an Account object is created, associate a Wallet object to an account. You can associate one or many Wallet objects to a given Account object.
Request:
Response:
Your Platform is all set! Wallets added to the On-Platform list will be used to calculate the relevant Protocol Service Fee Rebates.
This guide is a technical guide with examples of how to implement Liquid Collective’s redemption workflow into a Platform's workflow. For more information on Liquid Collective’s ETH staking withdrawal architecture, check out Liquid Collective’s LsETH redemption documentation
Review the Authentication Guide for the Alluvial API.
For this example, the implementation will use both smart contract calls and offchain calls via the Redemption API.
The Redemption API is an offchain API that exposes:
The list of redeem requests for an owner (wallet) including their redemption IDs
The status information about redeem requests (including id, redeemed amount, satisfaction status, claim status, etc.)
The heights of the withdrawal stack (ETH supplied) and the redeem queue (LsETH queued to redeem). Learn more here on different queues & stacks.
Time estimates for redemptions (projected and fulfilled)
For Contract Addresses: view this page. Make sure you use the Contract Address when sending txs.
For ABIs: This guide uses Goerli testnet and interacts with two contracts: LsETH and RedeemManager. RedeemManger contract is only needed if listening to RedeemManager contract events.
LsETH
0x1d8b30cC38Dba8aBce1ac29Ea27d9cFd05379A09
0x6edbde63319df1c54ee94075191c3d2ac5a1bf81
RedeemManager
0x0693875efbF04dDAd955c04332bA3324472DF980
0x3b377e3ac2cc844d8d27b8930f6a3035d3a3fc5b
The LsETH contract exposes a function called requestRedeem.
As an Platform, you allow users to submit (or to request that you submit on their behalf) the amount of LsETH they wish to redeem.
After successfully calling requestRedeem function an ID will be generated. Keep this redeemRequestId as you will use it later on.
Below is a code snippet that will call the requestRedeem function.
When creating a requestRedeem, the LsETH contract will emit a RequestedRedeem event. The event contains the redeem request IDs. Additionally, you can use the Redemption API to retrieve redeem request IDs associated with an owner, which is demonstrated later on in this guide.
Below is a code snippet for listening to the RequestedRedeem event via websocket connection.
Request:
run in terminal
Response:
As you can see in the response below, the redeem request ID is 18. This is the request ID associated with the request earlier in the “Create Redeem Request” section of this guide.
You can use the Redemption API to list the redeemRequests for a given recipient wallet.
You can get a list of all redemptions associated with a wallet using the curl request below.
Response:
You can see information related to the status of the redemption. Later on, you will see the status update as the redeemRequest becomes satisfied and can be claimed.
Now that you have a redeem ID, use the projections endpoint to get an estimate of how long it might take for the redeemRequest to be satisfied and become claimable.
Request:
Response:
The response below displays when the redeemRequest is likely to be redeemable. Once redeemable, you can use resolveRedeemRequests to get the matching withdrawal event ID.
A withdrawal event is triggered at the time of the protocol’s Oracle report. Oracles report every 24 hours.
Below is an example event.
Request:
This event will emit the newly created withdrawal ID, as you can see below:
Response:
In order to see if a redemption request can be satisfied, use the resolveRedeemRequests function call.
Request:
Response:
You receive a redeem ID of 10. 10 is the specific withdrawal event ID that is produced when the Liquid Collective Protocol has enough ETH to satisfy (either partial or full) the redemption. This means that the redeemRequest has been satisfied and the corresponding withdrawal event ID was returned.
A response of…
-1 means that the request is not satisfied yet
-2 means that the request is out of bounds
-3 means that the request has already been claimed
An alternative method to getting the Withdrawal event ID is using the /eth/v0/redeems/{id}
endpoint.
Request:
Response:
The response below indicates that this redemption can be fully satisfied, meaning there is enough supply in the withdrawal stack to fulfill this redemption. This is noted via the "status_satisfaction": "FULLY_SATISFIED"
and also the claimable_amount_lseth
is equal to the total_amount_lseth
.
For other potential states, check out the appendix at the bottom of the doc.
With both a request redeem ID and withdrawal event ID you can submit a claim for the redemption calling the claimRedeemRequests function.
Before claiming, ensure you have correct Withdrawal event ID. Withdrawal events IDs will change based on redemptions happening on the protocol.
Request:
After making a claim there will be two events triggered, which will be inspected next in this guide.
On successfully claiming a redeemRequest, a ClaimedRedeemRequest event is emitted. This event indicates the amount of LsETH claimed, the amount of ETH sent to the recipient, and the remaining LsETH amount that has been satisfied but not yet claimed.
A remaining LsETH amount of 0 means the request has been fully claimed. If the amount is greater than 0, the request has been partially claimed and another claim will be required to fully claim the request.
Request:
Response:
On successfully claiming a redemption request one or more SatisfiedRedeemRequest events are emitted.
Those events provide details about the withdrawal events that have satisfied a redemption request.
Request:
Response:
An alternative approach is to use the Alluvial API to call the /redeems
endpoint, as it will return the status of both satisfaction and claim status.
Request:
Non-finalized blocks are returned in the Alluvial API.
Response:
Below the endpoint returns that this request redeem has been fully satisfied and claimed.
You are now able to fully implement the redemption process for the Liquid Collective protocol!
The Alluvial API exposes information about both theWithdrawal stack and Redeem queue.
Platforms who want visibility into the current supply (amount of withdrawals) and demand (amount of redeems) can use the /redeems_info
endpoint.
Request:
Response:
Below you can see that the redeem queue is greater than the withdrawal stack. This indicates that more supply is needed to fulfill the redeem requests. Additional supply can enter via more deposits and/or exiting a validator.
In addition to exposing the Withdrawal stack and Redeem queue, Alluvial exposes an estimated time when redeems will be fulfilled.
Request:
Response:
The current projections timestamp logic is:
If the timestamp is in the past, it means that there is enough supply to fulfill demand and Alluvial returns the timestamp associated with the latest redeem request.
If the timestamp is in the future, it means there isn't enough supply to fulfill demand and time is needed to add supply into the Withdrawal stack.
Full code for making deposit transaction
Code is meant for testing purposes and should not be used directly for production workloads.
The redemption process is very dynamic. You can see a matrix of all the redeem statuses here. Below will show the difference between response payloads when requesting from either:
eth/v0/redeems/18
eth/v0/redeems?owner=
Why would this state happen?
This is the initial state for all redeem requests.
Why would this state happen?
This happens when the redemption request can be partially satisfied and the owner has not made a claimRedeemRequest against the redeem request.
Notice that the claimable_amount_lseth
is greater than 0 but less than the total_amount_lseth
.
Why would this state happen?
The amount of supply can fully satisfy the amount of LsETH being redeemed in this request.
Notice that total_amount_lseth
== claimable_amount_lseth
Why would this state happen?
The owner has claimed part of redeem request before the redeem request was fully satisfied. More supply will need to be added, hence the withdrawal_event_id
is == -1
, whereas before there would have been a positive integer associated with the withdrawal_event_id
(ex. 0, 1, 2, etc...).
Note that claimed_amount_lseth
is equal to previously claimed amount.
Why would this state happen?
The owner has claimed part of redeem request before the redeem request was fully satisfied. More supply was found, however not enough to fully satisfy this redeem request.
In the example below you can see that total_amount_lseth
> claimed_amount_lseth
+ claimable_amount_lseth
. There is 40000 LsETH still left to fully satisfy this request.
Why would this state happen?
The owner has claimed part of redeem request before the redeem request was fully satisfied.
In the example below you can see that total_amount_lseth
= claimed_amount_lseth
+ claimable_amount_lseth
.
Why would this state happen?
This is final state for all redeem requests. This indicates that the redeem request has been fully satisfied.
total_amount_lseth
== claimed_amount_lseth
and claimable_amount_lseth
== 0
withdrawal_event_id
reflects this state with the -3
ID.
This guide walks users through how to use authenticate requests using the Alluvial APIs.
Below is a ladder diagram showing the flow to create an access token.
This flow involves 3 parties:
Platform Server: the client looking to access the Alluvial API and which has previously been given a Client ID and Client Secret credentials.
Alluvial Authorization Server: responsible for validating credentials and generating JWT Access Token.
Alluvial API: the target resource to be accessed.
To obtain a Client ID and Client Secret reach out to your Alluvial representative.
To obtain an access token, use the a request below using your client id and secret.
Request:
Response:
Access Token should be reused for every request until it expires, in which case they should go through the Client Credential Flow again to obtain a fresh Access Token.
To use the access token, pass it via the HTTP header Authorization: Bearer
If you receive a 2xx response, you are now able to make fully authenticated requests.
If you receive a 4xx response, check if your access token is expired.
This page shows the process for creating an access token.
When onboarding with Alluvial, you will receive API credentials. These include a client_id and client_secret.
After receiving your credentials use the curl request below, updating the client_id and client_secret with your specific credentials.
Response
Request
Response
Make sure you are using the correct audience URL. Staging: Production:
You are now ready to make requests! Please check out our guides on and .
Redemption API is a collection of APIs that expose read data on redemptions.
To see APIs used in an example implementation check out the Redemption guide (below).
Alluvial exposes several non-authenticated APIs to support DApps
Ethereum Data API is a collection of APIs that expose read data from the Ethereum network.
Alluvial's APIs can also interact with other third party services, such as those offered by third party custodians that Platforms may use to custody funds staked using Liquid Collective.
Please note that these guides are to support Platforms who are connecting with third party products that are not offered by or in partnership or affiliation with Alluvial. Products and services offered by third parties are subject to separate terms and conditions. Visit the website of the third parties noted in these guides for more information on their offerings.
APIs that allow Platforms to create target discount schedules.
Alluvial's changelog records changes made to the Alluvial API specification, including new features, improvements, bug fixes, and more.
Discounting APIs are live! These APIs enable adding a new target rate in the event Platforms need to adjust the net Protocol Service Fee rate for specific accounts and/or wallets.
Summary APIs are now exposed within the Reporting API. Summary APIs allow for teams to get data for a range of dates rather than a specific day.
When creating a wallet, Platforms that allow for minting and redeeming, but only want to allow a wallet for secondary interactions can add these wallets to the on_platform list.
Expose redemption requests by address via /eth/v0/redeems
Expose redemption requests by id via /eth/v0/redeems/{idx}
Expose redemption manager (heights) via /eth/v0/redeems_info
Expose redeem projected redeemable at timestamp via /eth/v0/redeems/{idx}/projection
Expose manager projected fulfilled at timestamp via /eth/v0/redeems_info/projection
Remove address from Allowlister via PATCH /v0/depositors/{idOrKey}/wallets/{idOrAddress}/remove
Rather than removing the entire Depositor object, Platforms can now remove only the wallet address. This is the recommended approach for most cases where an a Platform needs to revoke a wallet address' Allowlist permissions.
The wallet will is still able to transfer LsETH to other wallets.
Alluvial's reporting API enables Platforms to request the ETH network rewards that their users have received. Below are two guides:
Platforms with omnibus account structure.
Platforms with segregated account structure.
Platforms that support an omnibus structure should use the /eth/v0/rewards
endpoint. This will return data in a lots structure. Lots represent LsETH balance changes for a given period. Rewards can then be calculated for each period. Total rewards for a user is the sum of rewards for each lot.
Below is an example request where a Platform wants to see the rewards for a wallet. Assume that the user has a total balance of 3 but acquired at different dates.
The user had 1 LsETH from 2024-04-01 to 2024-04-10. As such this will be the first lot added to the request body.
The user then acquired 2 more LsETH, for a total of 3, as of 2024-04-11.
In our request below assume that the current date for which we want data is til 2024-04-15.
Platforms that support a segregated structure can use two different endpoints to get reward data at the account or wallet level.
Use the endpoint /eth/v0/rewards/accounts
to display ETH network rewards at the account level.
Below is the response show the aggregated rewards for the account in addition for each address.
Use the endpoint /eth/v0/rewards/wallets
to display ETH network rewards at the wallet level.
Below is the response showing the aggregated rewards for the account in addition for each address.
The Alluvial API provides a staking rewards rate (SRR), which is calculated as a 7 day trailing average.
Based on providing a specific date of 2024-05-01, the returned SRR is 2.85%.
Platforms that want to provide a custom discount to their users can use the Discounting APIs to offload the calculations of a discounted net Protocol Service Fee to Alluvial's enterprise accounting service. Platforms can then use the API to call ETH network rewards, Protocol Service Fees, and end user discounts for one or more wallets or user accounts.
First, you will set a new target fee rate. This represents the new gross Protocol Service Fee rate that will be calculated for an account and/or wallet.
The target fee rate is now set at 8%.
On each Oracle report, you can use the Reporting APIs to see discounted values for the associated account, including the discounted net Protocol Service Fee, ETH network rewards received adjusted for that Fee, and effective end user discounts.
In the response below, you can see that the rebate_eth
or rebate_lseth
will reflect the amount the wallet is attributed to be reimbursed in order to achieve the the 8% target fee rate (or conversely, a 2% discount on the gross Protocol Service Fee).
The discount data is exposed via the following API endpoints:
/rewards/wallets
/eth/v0/rewards/accounts/:idOrKey/summary
/eth/v0/rewards/wallets/:idOrKey/summary
Reporting API now exposes an endpoint for Platforms with segregated accounts to get rewards information. You can see examples in our .
Reporting API now exposes an endpoint for Platforms with an omnibus to get rewards information. You can see examples in our .
On-Platform list has been created to allow Platforms that only want to allow for a wallet to take part in secondary actions of LsETH to be accounted for. More information can be found in the guide or guide.
Introduction of . These replace the Depositor APIs that are now marked for .
This API can be used with the .
This API can be used with the .
To get SRR information, call the endpoint.
For example, the current gross rate is 10%. To offer a user an effective net rate of 8%, you'd set the target fee rate at 0.08, which effectively provides a 2% discount to the associated wallet(s) and/or account(s).
Set account's status to REMOVED
.
Integrators can call this method for their inactive users to be removed from the liquid staking protocol(s) allowlist.
Once account's status has been set to REMOVED
:
Removing a wallet from the protocol allowlist results in disabling the ability to stake or redeem ETH.
Note that removing a wallet from the allowlist is an asynchronous process that can take some time.
Account ID or Key of the account to remove
Set account's status to PAUSED
.
Once an account's status has been set to PAUSED
:
allowlisted
wallet they will be unable to stake nor redeem.Account ID or Key of the account to remove
Set a PAUSED
account's status to ACTIVE
.
This will enable the account to stake and redeem again (if the account has allowlisted
wallets), and enable you to add more wallets to the account.
This is only possible for PAUSED
accounts - DENIED
or REMOVED
accounts cannot be reactivated.
Account ID or Key of the account to reactivate
Get platform wallet
Wallet ID or Address of the wallet (address in hex format prefixed with 0x)
List platform wallets
Enable to only return allowlisted wallets
Enable to only return wallets on_platform
Offset for pagination
Limit for pagination
List all wallets for a platform account
Account ID or Key of the account
Enable to only return allowlisted wallets
Enable to only return wallets on_platform
Offset for pagination
Limit for pagination
Remove wallet from a given wallet from either the on_platform
, the allowlisted
, or both.
Account ID or Key of the account to remove wallet from
Wallet ID or Address of the wallet to be removed
Enable to remove wallet from the allowlist
Enable to remove wallet on_platform
Set wallet's status to PAUSED
.
Once a wallet's status has been set to PAUSED
:
allowlisted
, it will be unable to stake nor redeem.Depositor ID or Key of the depositor to pause wallet from
Wallet ID or Address of the wallet to be paused
Enable to pause wallet from the allowlist
Enable to pause wallet on_platform
Set a PAUSED
wallet's status to ACTIVE
.
This will enable the wallet to stake and redeem again.
This action can only be performed under these conditions:
PAUSED
status - DENIED
or REMOVED
wallets cannot be reactivated.ACTIVE
status.Depositor ID or Key of the depositor to reactivate wallet from
Wallet ID or Address of the wallet to be reactivated
Enable to reactivate wallet from the allowlist
Enable to reactivate wallet on_platform
List the redeem requests for one or multiple owners (addresses)
The address or addresses to query, separated by commas
List a redeem request by the request id, obtained from /eth/v0/redeems
The redeem request id to query
Get a redeem request time projection for becoming redeemable
The redeem request id to query
List latest available high-level protocol
Display LsETH values in units of wei or ether. If not present, LsETH is set by default.
List high-level protocol info for a given date
Date filter by year, month or day (eg. 2023, 2023-12, 2023-12-31). Note: times are UTC
Display LsETH values in units of wei or ether. If not present, LsETH is set by default.
List rewards for a wallet with daily reports (public info) for a specific time period.
Wallet address
Start date (default: 1 month ago)
End date (default: current time)
Currency unit
Export to CSV
Page size (default: 100)
Next cursor (default: empty)
List wallet transactions for a given time period and wallet address
Wallet address
Start date (default: 1 year ago)
End date (default: current time)
Output as CSV
Page size (default: 100)
Next cursor
List the redeem requests for one or multiple owners (addresses)
The address or addresses to query, separated by commas
Get global information about protocol token supplies at a given block number
Block height to query
List LsETH balance of several addresses It is possible to specify for which block and which addresses to get the balances.
Comma separated list of addresses (if empty returns all addresses)
Block number at which to get balance for (latest by default)
Limit of elements to return (100 by default)
Get LsETH balance of a given address It is possible to specify for which block to get the balance at.
Account address to query the balance for
Block number at which to get balance for (latest by default)
Get the LsETH balance history for a given address It returns a balance object for every block a balance change occurred It is possible to filter the history for a range only blocks
Address of the balance to query balance for
Starting block height to query (0 by default)
Ending block height to query (latest by default)
Limit of elements to return (100 by default)
Get the discounted fee rate for an account
Account ID or Key of the account to get discount rate for
List latest available high-level protocol
Display LsETH values in units of wei or ether. If not present, LsETH is set by default.
List high-level protocol info for a given date
Date filter by year, month or day (eg. 2023, 2023-12, 2023-12-31). Note: times are UTC
Display LsETH values in units of wei or ether. If not present, LsETH is set by default.
List wallet transactions for a given time period and wallet address
Wallet address
Start date (default: 1 year ago)
End date (default: current time)
Output as CSV
Page size (default: 100)
Next cursor
List all platform fees for a given date
Date filter by year, month or day (eg. 2023, 2023-12, 2023-12-31). Note: times are UTC
Display LsETH values in units of wei or ether. If not present, LsETH is set by default.
List platform operator fees for a given date
Date filter by year, month or day (eg. 2023, 2023-12, 2023-12-31). Note: times are UTC
Display LsETH values in units of wei or ether. If not present, LsETH is set by default.
Provides a summary of rewards with a breakdown of reward totals for each wallet within the account. NOTE: Daily breakdown for each wallet address within the account is NOT included in the response.
Account ID
Start date (default: 1 month ago)
End date (default: current time)
Export to CSV
Currency unit
List wallet rewards summary with daily reports for a specific time period
Wallet ID or address
Start date (default: 1 month ago)
End date (default: current time)
Export to CSV
Currency unit
Create a platform account with an ACTIVE
status.
WARNING Integrators should make sure that the depositor has gone through an effective KYC/KYB process and is deemed within appetite to stake their ETH through the Liquid Collective.
When creating the depositor an integrator should supply a non-empty, unique identifying key
(it can typically be an internal user identifier).
As long as a depositor remains in ACTIVE
status, it is possible to attach wallets to it that will automatically be submitted to the protocol on-chain allowlist.
Unique identifier of the account (supplied by the client)
depositor_example
Create a wallet attached to an account
If the account is in ACTIVE
status and if the wallet's has the allowlisted
property set to true Alluvial will add the wallet address to the protocol on-chain allowlist.
Adding a wallet to the protocol allowlist results in enabling the wallet to proceed to deposit and redeem ETH on the protocol.
Note that submitting the wallet to the on-chain allowlist is an asynchronous process that can take some time. The onboarding allows to check the submission status by getting the wallet.
Important Notes:
allowlisted
wallets if your organization is configured to do so.allowlist
wallets, it method will do so by default.Account ID or Key of the account to add wallets to
Wallet address in hexadecimal format with 0x prefix
0xd8dA6BF26964aF9D7eEd9e03E53415D37aA96045
Opt out of wallet being added to the OnPlatform list
true
The type of wallet - chain or protocol it exists on
ETH
Create a discounted fee rate for an account
Account ID or Key of the account to add discounted fee rate to
0.05
Update a discounted fee rate for an account. If the discount rate already exists, it will be updated. A TargetFeeRate of 0.1 means no discount.
Account ID or Key of the account to update discounted fee rate for
0.05
List rewards by account for a specific date
["bb3e6114-af15-43a6-9b36-a5ec011d9d95"]
2024-04-11
lseth
Get rewards for a list of wallets on a specific date
["0x5d811a9d059dDAB0C18B385ad3b752f734f011cB"]
2024-04-11
eth
This guide shows you how to Stake ETH using the Liquid Collective Protocol and receive LsETH in return.
By the end of this guide you will:
Have an understanding of the Liquid Collective protocol
Stake ETH
Review the Authentication Guide for the Alluvial API.
Ethers.js: You will use Ether.js, a popular library for interacting with Smart Contracts to query blockchain information and send transactions.
Application Binary Interface (ABI): The ABI for the River smart contract will be used to READ /WRITE to the contract.
Holesky ETH: You will be accessing the Holesky testnet, ensure you have a wallet with HoleskyETH. If you don't have Holesky ETH you can use a faucet.
After reading both guides, come back to continue the implementation of the staking workflow.
In the backend (Node.js) application you will need to import one library.
The Liquid Collective uses a TUPProxy architecture. Below are the details about the Proxy address and implementation contract.
Holesky
0x1d8b30cC38Dba8aBce1ac29Ea27d9cFd05379A09
0x6edbde63319df1c54ee94075191c3d2ac5a1bf81
Mainnet
0x8c1BEd5b9a0928467c9B1341Da1D7BD5e10b6549
0x48D93d8C45Fb25125F13cdd40529BbeaA97A6565
You can save the ABI to a local .json file and import into your javascript file like below:
Contract.json
app.js
Next, add information about the RPC node Provider being used. This example uses this example you are using Chainstack, but there are several options for platforms to host an RPC node.
You can now create the transaction object. In the example below, you are transferring 0.0000001
from address 0xbe79ff177a8F6a0D9656cF47D8687f43666a4d1e to the River address.
Gas Limit uses a built in Ethers.js function. You may need to manually adjust when network has high demand. The code uses the estimateGas function.
Great job! You have successfully staked ETH. You can use the transaction hash returned to view more details. Or, you can view the transaction on Etherscan. Here is an example.
Congratulations! You've now staked ETH, received LsETH, and retrieved your balance. As a next step, review the guide on how to implement redemptions.
Full code for making deposit transaction
Code is meant for testing purposes and should not be used directly for production workloads.
Fireblocks is an easy-to-use platform to create new blockchain-based products and manage day-to-day digital asset operations. Fireblocks provides an MPC-based wallet allowing users to store their digital assets.
This guide will provide Platforms with a step-by-step process for using Fireblocks to interact with the Liquid Collective protocol.
Below is an example architecture for a Platform that uses Fireblocks as a custodian when the Platform's users stake ETH and/or redeem LsETH.
Create a new Javascript file, such as index.js
.
Define dependencies at top of file.
Create or use an existing Fireblocks vault.
Response:
Next, put digital assets into the Fireblocks account.
Define the contract address for Goerli:
Call the supported digital assets in Fireblocks.
Response:
Add the LsETH to the newly created vault.
Request:
Response:
Now that you have an address associated with your Fireblocks account, you will create a Depositor object and add the wallet address to the Liquid Collective protocol Allowlist via the Alluvial API.
Now that you have successfully added your wallets to the Allowlist you can continue with staking.
In order to interact with the Liquid Collective protocol you will need to invoke Smart Contract functions.
In the index.js
file add new dependencies
The Liquid Collective uses a TUPProxy architecture. Below are the details about the Proxy address and implementation contract.
Create a separate file called Contract.json
.
In the file, add the ABI for the Liquid Collective protocol:
Define the ABI address for Goerli.
Define the EIP-1193 Provider.
Create a function that calls the deposit function.
Request:
Submitting transactions will involve the Fireblocks TAP policy.
You've successfully staked ETH and should see LsETH returned in your Fireblocks vault.
Response:
Now you can implement the LsETH redemption flow, providing your users the ability to redeem their LsETH for ETH.
The next step is to allow the allowlisted wallets the ability to redeem their LsETH for ETH, thereby burning their LsETH.
There is a two-step process to receive the redeemed ETH. First, you will create a RedeemRequest that results in a redemption ID being returned. Once the redemption has been satisfied (full or partial) you can make a claimRedeemRequest call.
Request:
A request redeem ID will be generated.
The example below will show invoking the resolveRedeemRequest function.
Request:
Response:
111
You can proceed to make a claim now that you have both your redemption ID (53) and Withdrawal Event ID (111).
To make a claim request provide both the redemption request ID and the Withdrawal Event ID in an array.
Submitting transactions will involve the Fireblocks TAP policy.
Request:
After the claim is processed you will see the the deposit of ETH into your Fireblocks wallet.
Congratulations! You have now implemented the ETH staking and LsETH redemption flow using a Fireblocks account and SDK.
Appendix:
Below is the full code from the snippets above:
Liquid Collective Platforms can use to custody digital assets.
Fireblocks' offerings are third-party products that are not offered by or in partnership or affiliation with Alluvial. Products and services offered by Fireblocks and other third parties are subject to separate terms and conditions. Please visit for more information. Any links provided are for your convenience and informational purposes only. Inclusion of any link does not constitute an endorsement or an approval of any such third-party products by Alluvial or any other Liquid Collective protocol service provider.
This implementation will take advantage of the Fireblocks SDK, specifically the Javascript SDK. Please follow the to install the appropriate dependencies.
Admin permissions are needed to add ERC-20 digital assets to a Fireblocks account. Please review this information on .
The steps for onboarding and adding to the Allowlist can be found in the . Come back to this guide once your wallet(s) have been successfully added to the Allowlist.
This guide will use the Fireblocks Ethers.js provider, created by the Fireblocks team. To install read the .
Goerli can be found
Mainnet can be found
For more information check out .
The first function you will call is the .
You can retrieve the request redeem ID either via the Alluvial API or by listening the the Redeem Manager contract events. More information can be found .
In order to get the Withdrawal Event ID, invoke the or call via the .
The resolveRedeemRequest function returns the status of the Withdrawal Event ID. More can be found .
To see the status of the claim you can either listen for events or use the Alluvial API. More information on the claiming process can be found .
Goerli
0x3ecCAdA3e11c1Cc3e9B5a53176A67cc3ABDD3E46
0xF32fC26C9604a380c311e7eC0c5E545917e7934f
Mainnet
0x8c1BEd5b9a0928467c9B1341Da1D7BD5e10b6549
0x48D93d8C45Fb25125F13cdd40529BbeaA97A6565