Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat: ibc transfer on cosmos blockchains #2358

Merged
merged 15 commits into from
Jan 17, 2025
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
30 changes: 30 additions & 0 deletions packages/plugin-cosmos/README.md
Original file line number Diff line number Diff line change
Expand Up @@ -102,6 +102,36 @@ Yes

4. Action executed.

### Token IBC Transfer

This plugin supports a token transfer action, which allows users to transfer tokens between addresses on Cosmos-compatible blockchains between different chains.

#### Example Prompts

Below are examples of how the ibc transfer action can be initiated and confirmed:

**Example**

1. User input:

```
Make an IBC transfer 0.0001 OSMO to neutron1nk3uuw6zt5t5aqw5fvujkd54sa4uws9xg2nk82 from osmosistestnet to neutrontestnet
```

2. Plugin response:

```
Before making the IBC transfer, I would like to confirm the details. You would like to transfer 0.0001 OSMO from osmosistestnet to neutrontestnet, specifically to the address neutron1nk3uuw6zt5t5aqw5fvujkd54sa4uws9xg2nk82, is that correct?
```

3. User confirmation:

```
Yes
```

4. Action executed.

---

## Contribution
Expand Down
2 changes: 2 additions & 0 deletions packages/plugin-cosmos/package.json
Original file line number Diff line number Diff line change
Expand Up @@ -10,6 +10,8 @@
"@cosmjs/cosmwasm-stargate": "^0.32.4",
"@cosmjs/proto-signing": "^0.32.4",
"@cosmjs/stargate": "^0.32.4",
"@skip-go/client": "^0.16.3",
"axios": "^1.7.9",
"bignumber.js": "9.1.2",
"chain-registry": "^1.69.68",
"tsup": "8.3.5",
Expand Down
226 changes: 226 additions & 0 deletions packages/plugin-cosmos/src/actions/ibc-transfer/index.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,226 @@
import {
composeContext,
generateObjectDeprecated,
HandlerCallback,
IAgentRuntime,
Memory,
ModelClass,
State,
} from "@elizaos/core";
import { initWalletChainsData } from "../../providers/wallet/utils";
import {
cosmosIBCTransferTemplate,
cosmosTransferTemplate,
} from "../../templates";
import type {
ICosmosPluginOptions,
ICosmosWalletChains,
} from "../../shared/interfaces";
import { IBCTransferActionParams } from "./types";
import { IBCTransferAction } from "./services/ibc-transfer-action-service";
import { bridgeDenomProvider } from "./services/bridge-denom-provider";

export const createIBCTransferAction = (
pluginOptions: ICosmosPluginOptions
) => ({
name: "COSMOS_IBC_TRANSFER",
description: "Transfer tokens between addresses on cosmos chains",
handler: async (
_runtime: IAgentRuntime,
_message: Memory,
state: State,
_options: { [key: string]: unknown },
_callback?: HandlerCallback
) => {
const cosmosIBCTransferContext = composeContext({
state: state,
template: cosmosIBCTransferTemplate,
templatingEngine: "handlebars",
});

const cosmosIBCTransferContent = await generateObjectDeprecated({
runtime: _runtime,
context: cosmosIBCTransferContext,
modelClass: ModelClass.SMALL,
});

const paramOptions: IBCTransferActionParams = {
chainName: cosmosIBCTransferContent.chainName,
symbol: cosmosIBCTransferContent.symbol,
amount: cosmosIBCTransferContent.amount,
toAddress: cosmosIBCTransferContent.toAddress,
targetChainName: cosmosIBCTransferContent.targetChainName,
};

try {
const walletProvider: ICosmosWalletChains =
await initWalletChainsData(_runtime);

const action = new IBCTransferAction(walletProvider);

const customAssets = (pluginOptions?.customChainData ?? []).map(
(chainData) => chainData.assets
);

const transferResp = await action.execute(
paramOptions,
bridgeDenomProvider,
customAssets
);

if (_callback) {
await _callback({
text: `Successfully transferred ${paramOptions.amount} tokens from ${paramOptions.chainName} to ${paramOptions.toAddress} on ${paramOptions.targetChainName}\nTransaction Hash: ${transferResp.txHash}`,
content: {
success: true,
hash: transferResp.txHash,
amount: paramOptions.amount,
recipient: transferResp.to,
fromChain: paramOptions.chainName,
toChain: paramOptions.targetChainName,
},
});

const newMemory: Memory = {
userId: _message.agentId,
agentId: _message.agentId,
roomId: _message.roomId,
content: {
text: `Transaction ${paramOptions.amount} ${paramOptions.symbol} to address ${paramOptions.toAddress} from chain ${paramOptions.chainName} to ${paramOptions.targetChainName} was successfully transferred. Tx hash: ${transferResp.txHash}`,
},
};

await _runtime.messageManager.createMemory(newMemory);
}
return true;
} catch (error) {
console.error("Error during ibc token transfer:", error);

if (_callback) {
await _callback({
text: `Error ibc transferring tokens: ${error.message}`,
content: { error: error.message },
});
}

const newMemory: Memory = {
userId: _message.agentId,
agentId: _message.agentId,
roomId: _message.roomId,
content: {
text: `Transaction ${paramOptions.amount} ${paramOptions.symbol} to address ${paramOptions.toAddress} on chain ${paramOptions.chainName} to ${paramOptions.targetChainName} was unsuccessful.`,
},
};

await _runtime.messageManager.createMemory(newMemory);

return false;
}
},
template: cosmosTransferTemplate,
validate: async (runtime: IAgentRuntime) => {
const mnemonic = runtime.getSetting("COSMOS_RECOVERY_PHRASE");
const availableChains = runtime.getSetting("COSMOS_AVAILABLE_CHAINS");
const availableChainsArray = availableChains?.split(",");

return !!(mnemonic && availableChains && availableChainsArray.length);
},
examples: [
[
{
user: "{{user1}}",
content: {
text: "Make an IBC transfer {{0.0001 ATOM}} to {{osmosis1pcnw46km8m5amvf7jlk2ks5std75k73aralhcf}} from {{cosmoshub}} to {{osmosis}}",
action: "COSMOS_IBC_TRANSFER",
},
},
{
user: "{{user2}}",
content: {
text: "Do you confirm the IBC transfer action?",
action: "COSMOS_IBC_TRANSFER",
},
},
{
user: "{{user1}}",
content: {
text: "Yes",
action: "COSMOS_IBC_TRANSFER",
},
},
{
user: "{{user2}}",
content: {
text: "",
action: "COSMOS_IBC_TRANSFER",
},
},
],
[
{
user: "{{user1}}",
content: {
text: "Send {{50 OSMO}} to {{juno13248w8dtnn07sxc3gq4l3ts4rvfyat6f4qkdd6}} from {{osmosis}} to {{juno}}",
action: "COSMOS_IBC_TRANSFER",
},
},
{
user: "{{user2}}",
content: {
text: "Do you confirm the IBC transfer action?",
action: "COSMOS_IBC_TRANSFER",
},
},
{
user: "{{user1}}",
content: {
text: "Yes",
action: "COSMOS_IBC_TRANSFER",
},
},
{
user: "{{user2}}",
content: {
text: "",
action: "COSMOS_IBC_TRANSFER",
},
},
],
[
{
user: "{{user1}}",
content: {
text: "Transfer {{0.005 JUNO}} from {{juno}} to {{cosmos1n0xv7z2pkl4eppnm7g2rqhe2q8q6v69h7w93fc}} on {{cosmoshub}}",
action: "COSMOS_IBC_TRANSFER",
},
},
{
user: "{{user2}}",
content: {
text: "Do you confirm the IBC transfer action?",
action: "COSMOS_IBC_TRANSFER",
},
},
{
user: "{{user1}}",
content: {
text: "Yes",
action: "COSMOS_IBC_TRANSFER",
},
},
{
user: "{{user2}}",
content: {
text: "",
action: "COSMOS_IBC_TRANSFER",
},
},
],
],
similes: [
"COSMOS_BRIDGE_TOKEN",
"COSMOS_IBC_SEND_TOKEN",
"COSMOS_TOKEN_IBC_TRANSFER",
"COSMOS_MOVE_IBC_TOKENS",
],
});
9 changes: 9 additions & 0 deletions packages/plugin-cosmos/src/actions/ibc-transfer/schema.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,9 @@
import { z } from "zod";

export const IBCTransferParamsSchema = z.object({
chainName: z.string(),
symbol: z.string(),
amount: z.string().regex(/^\d+$/, "Amount must be a numeric string"),
toAddress: z.string().regex(/^[a-z0-9]+$/, "Invalid bech32 address format"),
targetChainName: z.string(),
});
Original file line number Diff line number Diff line change
@@ -0,0 +1,37 @@
import { IDenomProvider } from "../../../shared/interfaces";
import { SkipApiAssetsFromSourceFetcher } from "../../../shared/services/skip-api/assets-from-source-fetcher/skip-api-assets-from-source-fetcher";

export const bridgeDenomProvider: IDenomProvider = async (
sourceAssetDenom: string,
sourceAssetChainId: string,
destChainId: string
) => {
const skipApiAssetsFromSourceFetcher =
SkipApiAssetsFromSourceFetcher.getInstance();
const bridgeData = await skipApiAssetsFromSourceFetcher.fetch(
sourceAssetDenom,
sourceAssetChainId
);

const destAssets = bridgeData.dest_assets[destChainId];

if (!destAssets?.assets) {
throw new Error(`No assets found for chain ${destChainId}`);
}

const ibcAssetData = destAssets.assets?.find(
({ origin_denom }) => origin_denom === sourceAssetDenom
);

if (!ibcAssetData) {
throw new Error(`No matching asset found for denom ${sourceAssetDenom}`);
}

if (!ibcAssetData.denom) {
throw new Error("No IBC asset data");
}
coderabbitai[bot] marked this conversation as resolved.
Show resolved Hide resolved

return {
denom: ibcAssetData.denom,
};
};
Loading
Loading