Skip to content

Commit e9d343d

Browse files
committed
feat: transfer on Conflux (conflux-plugin)
1 parent 0ffa45c commit e9d343d

File tree

10 files changed

+235
-0
lines changed

10 files changed

+235
-0
lines changed

.env.example

+6
Original file line numberDiff line numberDiff line change
@@ -91,6 +91,12 @@ STARKNET_ADDRESS=
9191
STARKNET_PRIVATE_KEY=
9292
STARKNET_RPC_URL=
9393

94+
# Conflux Configuration
95+
CONFLUX_CORE_PRIVATE_KEY=
96+
CONFLUX_CORE_SPACE_RPC_URL=
97+
CONFLUX_ESPACE_PRIVATE_KEY=
98+
CONFLUX_ESPACE_RPC_URL=
99+
94100
# Coinbase
95101
COINBASE_COMMERCE_KEY= # from coinbase developer portal
96102
COINBASE_API_KEY= # from coinbase developer portal

agent/package.json

+1
Original file line numberDiff line numberDiff line change
@@ -21,6 +21,7 @@
2121
"@ai16z/client-twitter": "workspace:*",
2222
"@ai16z/eliza": "workspace:*",
2323
"@ai16z/plugin-bootstrap": "workspace:*",
24+
"@ai16z/plugin-conflux": "workspace:*",
2425
"@ai16z/plugin-image-generation": "workspace:*",
2526
"@ai16z/plugin-node": "workspace:*",
2627
"@ai16z/plugin-solana": "workspace:*",

agent/src/index.ts

+4
Original file line numberDiff line numberDiff line change
@@ -23,6 +23,7 @@ import {
2323
validateCharacterConfig,
2424
} from "@ai16z/eliza";
2525
import { bootstrapPlugin } from "@ai16z/plugin-bootstrap";
26+
import { confluxPlugin } from "@ai16z/plugin-conflux";
2627
import { solanaPlugin } from "@ai16z/plugin-solana";
2728
import { nodePlugin } from "@ai16z/plugin-node";
2829
import {
@@ -252,6 +253,9 @@ export function createAgent(
252253
character,
253254
plugins: [
254255
bootstrapPlugin,
256+
character.settings.secrets?.CONFLUX_CORE_PRIVATE_KEY
257+
? confluxPlugin
258+
: null,
255259
nodePlugin,
256260
character.settings.secrets?.WALLET_PUBLIC_KEY ? solanaPlugin : null,
257261
character.settings.secrets?.COINBASE_COMMERCE_KEY ||

packages/plugin-conflux/README.md

+25
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,25 @@
1+
# @ai16z/plugin-conflux
2+
3+
This plugin provides actions and providers for interacting with the [Conflux network](https://www.confluxdocs.com/docs/general).
4+
5+
## Actions
6+
7+
### ConfiPump
8+
9+
Buy and sell tokens on Conflux's implementation of pump.fun (ConfiPump).
10+
11+
### Transfer
12+
13+
Transfer tokens from one address to another within Conflux core space.
14+
15+
### Bridge Transfer
16+
17+
Transfer tokens from one address to Conflux eSpace.
18+
19+
### Sponsor
20+
21+
Provide gas for Conflux core space contracts so they can be called without the need to have Conflux in user's wallet.
22+
23+
### Swap
24+
25+
Swap tokens on Conflux DEXs.

packages/plugin-conflux/package.json

+15
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,15 @@
1+
{
2+
"name": "@ai16z/plugin-conflux",
3+
"version": "0.0.1",
4+
"main": "dist/index.js",
5+
"type": "module",
6+
"types": "dist/index.d.ts",
7+
"dependencies": {
8+
"cive": "^0.7.1",
9+
"@ai16z/eliza": "workspace:*"
10+
},
11+
"scripts": {
12+
"build": "tsup --format esm --dts",
13+
"dev": "tsup --watch"
14+
}
15+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,147 @@
1+
import {
2+
Action,
3+
IAgentRuntime,
4+
Memory,
5+
State,
6+
HandlerCallback,
7+
} from "@ai16z/eliza";
8+
import { z } from "zod";
9+
import { generateObjectV2, composeContext, ModelClass, Content } from "@ai16z/eliza";
10+
import { createPublicClient, createWalletClient, http, parseCFX } from "cive";
11+
import { privateKeyToAccount } from "cive/accounts";
12+
import { testnet } from "cive/chains";
13+
import { confluxTransferTemplate } from "../templates/transfer";
14+
15+
const TransferSchema = z.object({
16+
to: z.string(),
17+
amount: z.number(), // use number ignoring decimals issue
18+
});
19+
20+
interface TransferContent extends Content {
21+
to: string;
22+
amount: number;
23+
}
24+
25+
const isTransferContent = (object: any): object is TransferContent => {
26+
if (TransferSchema.safeParse(object).success) {
27+
return true;
28+
}
29+
console.error("Invalid content: ", object);
30+
return false;
31+
};
32+
33+
const sendCFX = async (
34+
secretKey: `0x${string}`,
35+
rpcUrl: string,
36+
to: string,
37+
amount: string
38+
) => {
39+
const client = createPublicClient({
40+
transport: http(rpcUrl),
41+
});
42+
const networkId = await client.getChainId();
43+
const account = privateKeyToAccount(secretKey, { networkId });
44+
45+
const walletClient = createWalletClient({
46+
transport: http(rpcUrl),
47+
chain: testnet,
48+
});
49+
50+
const hash = await walletClient.sendTransaction({
51+
account,
52+
to,
53+
value: parseCFX(amount),
54+
chain: testnet,
55+
});
56+
57+
await client.waitForTransactionReceipt({
58+
hash,
59+
});
60+
return hash;
61+
};
62+
63+
export const transfer: Action = {
64+
name: "SEND_CFX",
65+
description:
66+
"Transfer CFX from to another in Conflux Core Space",
67+
similes: ["SEND_CONFLUX", "SEND_CFX_CORE_SPACE", "TRANSFER_CFX"],
68+
examples: [
69+
[
70+
{
71+
user: "{{user1}}",
72+
content: {
73+
text: "Send 1 CFX to 0x1234567890abcdef",
74+
},
75+
},
76+
{
77+
user: "{{user2}}",
78+
content: {
79+
text: "1 CFX sent to 0x1234567890abcdef: 0x1234567890abcdef",
80+
content: {
81+
to: "0x1234567890abcdef",
82+
amount: "1",
83+
},
84+
},
85+
},
86+
],
87+
],
88+
validate: async (runtime: IAgentRuntime, message: Memory) => {
89+
// no extra validation needed
90+
return true;
91+
},
92+
handler: async (
93+
runtime: IAgentRuntime,
94+
message: Memory,
95+
state?: State,
96+
options?: { [key: string]: unknown },
97+
callback?: HandlerCallback
98+
) => {
99+
if (!state) {
100+
state = (await runtime.composeState(message)) as State;
101+
} else {
102+
state = await runtime.updateRecentMessageState(state);
103+
}
104+
105+
const context = composeContext({
106+
state,
107+
template: confluxTransferTemplate,
108+
});
109+
110+
const content = await generateObjectV2({
111+
runtime,
112+
context,
113+
modelClass: ModelClass.SMALL,
114+
schema: TransferSchema,
115+
});
116+
117+
if (!isTransferContent(content.object)) {
118+
throw new Error("Invalid content");
119+
}
120+
121+
const secretKey = runtime.getSetting("CONFLUX_CORE_PRIVATE_KEY") as `0x${string}`;
122+
const rpcUrl = runtime.getSetting("CONFLUX_CORE_SPACE_RPC_URL");
123+
124+
let success = false;
125+
126+
try {
127+
const hash = await sendCFX(secretKey, rpcUrl, content.object.to, content.object.amount.toString());
128+
success = true;
129+
if (!callback) {
130+
return success;
131+
}
132+
callback({
133+
text: `${content.object.amount} CFX sent to ${content.object.to}: ${hash}`,
134+
content: content.object,
135+
});
136+
} catch (error) {
137+
console.error(`Error sending CFX: ${error}`);
138+
if (!callback) {
139+
return success;
140+
}
141+
callback({
142+
text: `Failed to send ${content.object.amount} CFX to ${content.object.to}: ${error}`,
143+
});
144+
}
145+
return success;
146+
},
147+
};

packages/plugin-conflux/src/index.ts

+9
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,9 @@
1+
import { Plugin } from "@ai16z/eliza";
2+
import { transfer } from "./actions/transfer";
3+
4+
export const confluxPlugin: Plugin = {
5+
name: "conflux",
6+
description: "Conflux Plugin for Eliza",
7+
actions: [transfer],
8+
providers: [],
9+
};
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,7 @@
1+
export const confluxTransferTemplate = `
2+
Extract Conflux Core Space Transfer Parameters from the latest message:
3+
4+
{{recentMessages}}
5+
6+
The to address should be the Conflux Core Space address, starting with "cfx:" or "cfxtest:".
7+
`;

packages/plugin-conflux/tsconfig.json

+8
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,8 @@
1+
{
2+
"extends": "../../tsconfig.json",
3+
"compilerOptions": {
4+
"outDir": "dist",
5+
"rootDir": "./src"
6+
},
7+
"include": ["src"]
8+
}
+13
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,13 @@
1+
import { defineConfig } from "tsup";
2+
3+
export default defineConfig({
4+
entry: ["src/index.ts"],
5+
outDir: "dist",
6+
sourcemap: true,
7+
clean: true,
8+
format: ["esm"], // Ensure you're targeting CommonJS
9+
external: [
10+
"cive",
11+
// Add other modules you want to externalize
12+
],
13+
});

0 commit comments

Comments
 (0)