Skip to content

Commit 8605c36

Browse files
committed
review
1 parent a29c5fc commit 8605c36

File tree

6 files changed

+97
-460
lines changed

6 files changed

+97
-460
lines changed

examples/coinbase-langchain/src/cdp.ts

+59-16
Original file line numberDiff line numberDiff line change
@@ -2,12 +2,13 @@ import {
22
Coinbase,
33
TimeoutError,
44
Wallet,
5+
type Trade,
56
type Transfer,
67
type WalletData,
78
} from "@coinbase/coinbase-sdk";
89
import { isAddress } from "viem";
910
import { getWalletData, saveWalletData } from "./storage";
10-
import type { XMTPUser } from "./types";
11+
import type { AgentWalletData, XMTPUser } from "./types";
1112

1213
const coinbaseApiKeyName = process.env.CDP_API_KEY_NAME;
1314
let coinbaseApiKeyPrivateKey = process.env.CDP_API_KEY_PRIVATE_KEY;
@@ -74,18 +75,6 @@ function initializeCoinbaseSDK(): boolean {
7475
}
7576
}
7677

77-
// Agent wallet data
78-
export type AgentWalletData = {
79-
id: string;
80-
wallet: Wallet;
81-
data: WalletData;
82-
human_address: string;
83-
agent_address: string;
84-
blockchain?: string;
85-
state?: string;
86-
inboxId: string;
87-
};
88-
8978
// Wallet service class based on cointoss implementation
9079
export class WalletService {
9180
private walletStorage: WalletStorage;
@@ -285,12 +274,12 @@ export class WalletService {
285274
}
286275

287276
async transfer(
288-
inboxId: string,
289-
humanAddress: string,
277+
xmtpUser: XMTPUser,
290278
toAddress: string,
291279
amount: number,
292280
): Promise<Transfer | undefined> {
293-
humanAddress = humanAddress.toLowerCase();
281+
const humanAddress = xmtpUser.address.toLowerCase();
282+
const inboxId = xmtpUser.inboxId;
294283
toAddress = toAddress.toLowerCase();
295284

296285
console.log("📤 TRANSFER INITIATED");
@@ -390,4 +379,58 @@ export class WalletService {
390379
throw error;
391380
}
392381
}
382+
383+
async swap(
384+
xmtpUser: XMTPUser,
385+
fromAssetId: string,
386+
toAssetId: string,
387+
amount: number,
388+
): Promise<Trade | undefined> {
389+
const inboxId = xmtpUser.inboxId;
390+
const walletData = await this.getWallet(inboxId);
391+
if (!walletData) return undefined;
392+
console.log(`Retrieved wallet data for ${inboxId}`);
393+
394+
console.log(
395+
`Initiating swap from ${fromAssetId} to ${toAssetId} for amount: ${amount}`,
396+
);
397+
const trade = await walletData.wallet.createTrade({
398+
amount,
399+
fromAssetId,
400+
toAssetId,
401+
});
402+
403+
try {
404+
await trade.wait();
405+
} catch (err) {
406+
if (err instanceof TimeoutError) {
407+
console.log("Waiting for trade timed out");
408+
} else {
409+
console.error("Error while waiting for trade to complete: ", err);
410+
}
411+
}
412+
413+
return trade;
414+
}
415+
416+
async deleteWallet(xmtpUser: XMTPUser): Promise<boolean> {
417+
const inboxId = xmtpUser.inboxId;
418+
console.log(`Deleting wallet for key ${inboxId}`);
419+
const encryptedKey = `wallet:${inboxId}`;
420+
421+
// Create an empty wallet object to effectively delete the wallet
422+
const emptyWallet: AgentWalletData = {
423+
id: encryptedKey,
424+
wallet: {} as Wallet,
425+
data: {} as WalletData,
426+
human_address: this.humanAddress,
427+
agent_address: this.inboxId,
428+
inboxId: this.inboxId,
429+
};
430+
431+
await this.walletStorage.set(this.inboxId, JSON.stringify(emptyWallet));
432+
433+
console.log(`Wallet deleted for key ${inboxId}`);
434+
return true;
435+
}
393436
}

examples/coinbase-langchain/src/langchain.ts

+1-2
Original file line numberDiff line numberDiff line change
@@ -59,7 +59,6 @@ function createWalletTools(xmtpUser: XMTPUser) {
5959

6060
const result = await walletService.transfer(
6161
xmtpUser.inboxId,
62-
xmtpUser.address,
6362
recipientAddress,
6463
numericAmount,
6564
);
@@ -101,7 +100,7 @@ export async function initializeAgent(
101100
// Check if we already have an agent for this user
102101
if (xmtpUser.inboxId in agentStore) {
103102
console.log(`Using existing agent for user: ${xmtpUser.inboxId}`);
104-
const agentConfig = {
103+
const agentConfig: AgentConfig = {
105104
configurable: { thread_id: xmtpUser.inboxId },
106105
};
107106
return { agent: agentStore[xmtpUser.inboxId], config: agentConfig };

examples/coinbase-langchain/src/storage.ts

+13-27
Original file line numberDiff line numberDiff line change
@@ -1,34 +1,23 @@
11
import * as fs from "fs";
22
import { createClient, type RedisClientType } from "redis";
33

4-
// Storage constants
4+
// Storage constantsf
55
export const WALLET_KEY_PREFIX = "wallet_data:";
6-
export const LOCAL_STORAGE_DIR = ".data/wallet_data";
6+
export const WALLET_STORAGE_DIR = ".data/wallet_data";
7+
export const XMTP_STORAGE_DIR = ".data/xmtp";
78
export let redisClient: RedisClientType | null = null;
89

9-
if (!process.env.REDIS_URL) {
10-
console.warn(
11-
"Warning: REDIS_URL not set, using local file storage for wallet data",
12-
);
13-
}
1410
/**
1511
* Initialize Redis client and handle fallback to local storage
1612
*/
1713
export async function initializeStorage() {
1814
if (process.env.REDIS_URL) {
19-
try {
20-
redisClient = createClient({
21-
url: process.env.REDIS_URL,
22-
});
15+
redisClient = createClient({
16+
url: process.env.REDIS_URL,
17+
});
2318

24-
await redisClient.connect();
25-
console.log("Connected to Redis");
26-
} catch (error: unknown) {
27-
console.error("Failed to connect to Redis:", error);
28-
console.log("Falling back to local file storage");
29-
redisClient = null;
30-
ensureLocalStorage();
31-
}
19+
await redisClient.connect();
20+
console.log("Connected to Redis");
3221
} else {
3322
console.log("Using local file storage for wallet data");
3423
ensureLocalStorage();
@@ -39,8 +28,8 @@ export async function initializeStorage() {
3928
* Ensure local storage directory exists
4029
*/
4130
export function ensureLocalStorage() {
42-
if (!fs.existsSync(LOCAL_STORAGE_DIR)) {
43-
fs.mkdirSync(LOCAL_STORAGE_DIR, { recursive: true });
31+
if (!fs.existsSync(WALLET_STORAGE_DIR)) {
32+
fs.mkdirSync(WALLET_STORAGE_DIR, { recursive: true });
4433
}
4534
}
4635

@@ -59,7 +48,7 @@ export async function saveWalletData(
5948
} else {
6049
// Save to local file
6150
try {
62-
fs.writeFileSync(LOCAL_STORAGE_DIR + "/" + key + ".json", walletData);
51+
fs.writeFileSync(WALLET_STORAGE_DIR + "/" + key + ".json", walletData);
6352
} catch (error: unknown) {
6453
const errorMessage =
6554
error instanceof Error ? error.message : String(error);
@@ -81,11 +70,8 @@ export async function getWalletData(
8170
return await redisClient.get(userKey);
8271
} else {
8372
try {
84-
if (fs.existsSync(LOCAL_STORAGE_DIR + "/" + userKey + ".json")) {
85-
return fs.readFileSync(
86-
LOCAL_STORAGE_DIR + "/" + userKey + ".json",
87-
"utf8",
88-
);
73+
if (fs.existsSync(WALLET_STORAGE_DIR + "/" + userKey)) {
74+
return fs.readFileSync(WALLET_STORAGE_DIR + "/" + userKey, "utf8");
8975
}
9076
} catch (error: unknown) {
9177
const errorMessage =

examples/coinbase-langchain/src/types.ts

+12
Original file line numberDiff line numberDiff line change
@@ -1,3 +1,4 @@
1+
import type { Wallet, WalletData } from "@coinbase/coinbase-sdk";
12
import type { MemorySaver } from "@langchain/langgraph";
23
import type { createReactAgent } from "@langchain/langgraph/prebuilt";
34

@@ -13,6 +14,17 @@ export interface AgentConfig {
1314
thread_id: string;
1415
};
1516
}
17+
// Agent wallet data
18+
export type AgentWalletData = {
19+
id: string;
20+
wallet: Wallet;
21+
data: WalletData;
22+
human_address: string;
23+
agent_address: string;
24+
blockchain?: string;
25+
state?: string;
26+
inboxId: string;
27+
};
1628

1729
export type Agent = ReturnType<typeof createReactAgent>;
1830

examples/coinbase-langchain/src/xmtp.ts

+7-4
Original file line numberDiff line numberDiff line change
@@ -5,6 +5,7 @@ import {
55
type XmtpEnv,
66
} from "@xmtp/node-sdk";
77
import { createSigner, getEncryptionKeyFromHex } from "@/helpers";
8+
import { XMTP_STORAGE_DIR } from "./storage";
89

910
/**
1011
* Initialize the XMTP client
@@ -21,18 +22,20 @@ export async function initializeXmtpClient() {
2122
const signer = createSigner(WALLET_KEY as `0x${string}`);
2223
const encryptionKey = getEncryptionKeyFromHex(ENCRYPTION_KEY);
2324

25+
const identifier = await signer.getIdentifier();
26+
const address = identifier.identifier;
2427
// Set the environment to dev or production
2528
const env: XmtpEnv = XMTP_ENV as XmtpEnv;
2629

2730
console.log(`Creating XMTP client on the '${env}' network...`);
28-
const client = await Client.create(signer, encryptionKey, { env });
31+
const client = await Client.create(signer, encryptionKey, {
32+
env,
33+
dbPath: XMTP_STORAGE_DIR + `/${env}-${address}`,
34+
});
2935

3036
console.log("Syncing conversations...");
3137
await client.conversations.sync();
3238

33-
const identifier = await signer.getIdentifier();
34-
const address = identifier.identifier;
35-
3639
console.log(
3740
`Agent initialized on ${address}\nSend a message on http://xmtp.chat/dm/${address}?env=${env}`,
3841
);

0 commit comments

Comments
 (0)