From a connected chain
Setting up Flows
In the previous step we showed how the flow process looks like by submitting an flow on Intento. You can do this with the TriggerPortal interface, a IntentoJS front-end integration or locally through the CLI.
In addition, you can also submit an flow from another chain using the ICS20 standard.
Interchain Accounts
Users and entities on Cosmos SDK chains may be able set up interchain account to Intento and submit the flows using MsgSubmitFlow
. It is also easy to deploy using the ICS20 standard.
ICS20 Standard
With Intento’s ICS20 transfer middleware, you send a transfer token memo on a chain, and Intento will convert the token transfer to an flow submission. ICS20 is an interchain standard that enables the transfer of fungible tokens between independent blockchains. It is a protocol that defines a standard interface for token transfers across different blockchains that implement the Inter-Blockchain Communication (IBC) protocol.
Using ICS20, accounts on connected chains can create flows. This can be done by specifying flow details in the memo field of an ICS20 transfer message. Upon receiving this message, Intento's IBC hooks transforms this into a submit flow message.
This is useful for DAOs and other decentralized organizations on any connected chain. They can safely and reliably execute on Intento's connected chains. For DAOs, this gives certainty to stakeholders, whilst also reducing manual work on governance proposals.
For DAOs
Setting up an flow on a connected chain can be particularly useful for DAOs. Using this middleware, DAOs can now automate tasks not only on their chain but also on any chain connected to Intento. What can DAO's do with this? DAOs can orchestrate periodic token swaps, payroll, payment in installments amongst other scheduled flows. These flows can be performed in one proposal, which normally require periodically voting on individual proposals. This normally requires manual flow from the proposer and DAO participants.
ICS20 Middleware
A MsgRegisterAccountAndSubmitFlow or a MsgSubmitFlow can be derrived from the memo field in the ICS20 transfer message.
Our custom middleware is based on the wasmhooks implementation on Osmosis.
The mechanism enabling this is a memo
field on every ICS20 transfer packet as of IBC v3.4.0.
ics_middleware.go is IBC middleware that parses an ICS20 transfer, and if the memo
field is of a particular form, it creates an flow by parsing and handling a SubmitFlow message.
These are the fields for flow
that are derived from the ICS20 message:
- Owner: This field is directly obtained from the ICS20 packet metadata and equals the ICS20 recipient. If unspecified, a placeholder is made from the ICS20 sender and channel.
- Msg: This field should be directly obtained from the ICS20 packet metadata.
- FeeFunds: This field is set to the amount of funds being sent over in the ICS20 packet. One detail here is that the denom in the packet is the source chains representation of the denom, this will be translated into INTO on Intento.
The constructed message for MsgSubmitFlow under the hood will look like:
msg := MsgSubmitFlow{
// If let unspecified, owner is the actor that submitted the ICS20 message and a placeholder only
Owner: "into1-hash-of-channel-and-sender" OR packet.data.memo["flow"]["owner"],
// Array of Msg json encoded, then transformed into a proto.message
Msgs: packet.data.memo["flow"]["msgs"],
// Funds coins that are transferred to the owner
FeeFunds: sdk.NewCoin{Denom: ibc.ConvertSenderDenomToLocalDenom(packet.data.Denom), Amount: packet.data.Amount}
// other fields
}
ICS20 packet structure
So given the details above, we propogate the implied ICS20 packet data structure. ICS20 is JSON native, so we use JSON for the memo format.
{
//... other ibc fields that we don't care about
"data": {
"denom": "INTO denom on counterparty chain (e.g. ibc/abc...)",
"amount": "1000", //for execution fees
"sender": "...",
"receiver": "A INTO addr prefixed with into1",
"memo": {
"flow": {
"owner": "into1address", //owner is optional
"msgs": [
{
"@type": "/cosmos.somemodule.v1beta1.sometype"
//message values in JSON format
}
],
"duration": "111h",
"start_at": "11h",
"label": "my_label",
"interval": "11h", //optional
"cid": "connection-0", //connection ID is optional, omit or leave blank in case local INTO message.
"cp_cid": "connection-0", //counterparty connection ID is optional and is only needed to register ICA.
"register_ica": "false", //optional, set to true to register interchain account
//////configuration,optional
"save_responses": "true", //save message responses of Cosmos SDK v0.46+ chain output, defaults to false
"update_disabled": "true", //optional, disables the owner's ability to update the config, defaults to false
"stop_on_success": "true", //optional, defaults to false
"stop_on_fail": "true" //optional, defaults to false
}
}
}
}
An ICS20 packet is formatted correctly for submitting an flow if the following all hold:
memo
is not blankmemo
is valid JSONmemo
has at least one key,"flow"
memo["flow"]["msgs"]
is an array with valid JSON SDK message objects with a key "@type" and sdk message valuesreceiver == memo["flow"]["owner"]
. Optional, an owner can be specifed and is the address that receives remaining fee balance after execution ends.memo["flow"]["cid"]
is a valid connection ID on INTO -> Destination chain, omit it for local INTO execution of the message.memo["flow"]["register_ica"]
can be added, and true to register an ICA.
Fees are paid with a newly generated flow fee account.
If an ICS20 packet does not contain a memo containing "flow", a regular MsgTransfer takes place. If an ICS20 packet is directed towards flow, and is formated incorrectly, then it returns an error. Here’s an improved version of the example with clearer language, structure, and formatting:
Example: DAO Integration
Overview
Using ICS20 to set up Intent-based flows is for users familiar with ICS20 transfers and Authz permissions. Our ICS20 middleware is designed to allow DAOs to submit flows, which can be done locally on Intento or on a destination chain. This can also be the source chain.
There are several caveats when setting up flows with ICS20. When automating on a destination chain for the first time, two messages are required to activate the flow. One flow sets up the flow and creates an Interchain Account on Intento, while the other sets permissions and sends funds to the Interchain Account on the destination chain.
In this example, we will demonstrate how a DAO can integrate with Intento by automating the payment process to a service provider.
Scenario: DAO Payment to Service Provider
The DAO
wishes to pay Service Provider ABC
monthly for their services in TOKEN1
. The DAO holds TOKEN2
and NTRN
.
Service Provider ABC invoice example
In this case, the DAO triggers a recurring swap of TOKEN2
for TOKEN1
on the decentralized exchange "DEX" and automatically sends the tokens to Service Provider ABC
.
This setup is an ideal use case for Intento, as it involves asset movement between chains and accounts, while the DAO maintains control of the tokens. Since the flows are on-chain, the process is fully decentralized and does not require third-party trust.
The DAO can appoint an owner to manage the flow or use a placeholder account on Intento to remain in full control.
Proposal Details
In this example, the DAO submits a proposal with the following name and description:
Proposal Name:
[Trigger] Pay Service Provider ABC in TOKEN1
Proposal Description:
Submit a flow to send TOKEN1 to "Service Provider ABC."
This flow will swap TOKEN2 for TOKEN1 on DEX "DEX" on the Destination Chain and automatically send these tokens to "Service Provider ABC." By performing this swap on a recurring basis, we achieve:
- Gradual selling pressure on TOKEN2
- Maintaining positive cash flow
This trigger on Intento automates asset workflows, ensuring liquidity and financial stability for the DAO.
Having sufficient liquidity ensures smooth operations and financial stability, allowing the DAO to meet its obligations and grow sustainably. Proper liquidity management helps to pay expenses, invest in growth, and strengthen the DAO's reputation in the market.
1. Submitting the Flow
To submit the flow, the DAO creates a proposal with a custom message to execute. For a CosmWasm-based DAO like DAO DAO on Neutron, this message contains the ICS20 MsgTransfer
. In the memo
, the DAO provides flow details such as automated messages, time parameters, and a custom label.
Here’s an example of how the flow proposal message might look for a CosmWasm-based DAO:
[
{
"stargate": {
"typeUrl": "/ibc.applications.transfer.v1.MsgTransfer",
"value": {
"source_port": "transfer",
"source_channel": "channel-to-intento",
"token": { "denom": "ibc/....", "amount": "10" },
"sender": "neutron1validbech32address",
"receiver": "into1address",
"timeout_height": "0",
"timeout_timestamp": "0",
"memo": {
"flow": {
"owner": "into1address",
"msgs": [
{
"@type": "/someprefix.somemodule.someversion.sometype"
}
],
"duration": "111h",
"interval": "11h",
"start_at": "1677841601",
"label": "my_label",
"cid": "connection-0",
"register_ica": "true"
}
}
}
}
}
]
Important Notes:
- Ensure that
timeout_timestamp
ortimeout_height
reflects the proposal's end time, or set both to0
if no timeout is needed. - When creating the flow for the first time, make sure the
register_ica
field is set to"true"
, which registers an Interchain Account on the destination chain.
A message within the flow["msgs"]
array might look like this MsgSend
:
{
"@type": "/cosmos.bank.v1beta1.MsgSend",
"amount": [
{
"amount": "70",
"denom": "stake"
}
],
"from_address": "ICA_ADDR",
"to_address": "some_destination_chain_address"
}
Alternatively, the DAO can use a custom smart contract message like SwapAndSendTo in a MsgExecuteContract message to swap TOKEN2
for TOKEN1
.
{
"@type": "/cosmos.authz.v1beta1.MsgExec",
"msgs": [
{
"@type": "/cosmwasm.wasm.v1.MsgExecuteContract",
"msg": {
"swap_and_send_to": {
"input_token": "TOKEN2",
"min_token": "500",
"recipient": "neutron1_address"
}
},
"sender": "neutron1_address_dao",
"contract": "neutron1_address_swap_contract",
"funds": []
}
],
"grantee": "ICA_ADDR"
}
Write ICA_ADDR
as a grantee
or in other fields within the message, and Intento will parse the to-be-defined Interchain Account address.
You can also use MsgSwapExactAmountOut to swap tokens on decentralized exchanges like Osmosis or Dymension.
2. Setting Up Permissions and Funds
The Interchain Account (ICA) must be properly funded and authorized before it can execute the flow.
Paying for Fees
Flow fees on Intento are automatically paid from the funds sent via ICS20. For flows executed on the destination chain, the Flow Account
on the destination chain should be funded with the destination chain's fee token.
If needed, a separate proposal can be submitted to send tokens (via MsgSend
or MsgTransfer
) to the destination chain's ICA address.
If the source and destination chains are the same, you can set up a FeeGrant for the ICA on the destination chain.
Setting Permissions
For flows on Cosmos chains, set up an AuthZ grant using MsgGrant
. For EVM chains, you can give the flow account allowance
for ERC20 tokens or approval
for NFTs.
Once funds and permissions are set up, only one flow proposal is required to trigger the flow.
3. Managing Flows
You can manage your flows on TriggerPortal, where you can create, view, update, and control your flows.