Circle Wallet Service
The WalletService
class in MessageKit provides a way to create agent wallets that can perform gasless payments and transfers on Base.
Requirements
Visit the Circle Developer Portal to create an API key and entity secret.
@circle-fin/developer-controlled-wallets
: Already included in MessageKit- Environment variables:
CIRCLE_API_KEY
= # Circle API KeyCIRCLE_ENTITY_SECRET
= # Circle Entity SecretCIRCLE_WALLET_SET_ID
= # Circle Wallet Set ID
Code
plugins/circle.ts
import { initiateDeveloperControlledWalletsClient } from "@circle-fin/developer-controlled-wallets";
import { type Context } from "../lib/core";
import { AgentWallet, AgentWalletData } from "../helpers/types";
import { LocalStorage } from "./storage";
const apiKey = process.env.CIRCLE_API_KEY;
const entitySecret = process.env.CIRCLE_ENTITY_SECRET;
const walletSetId = process.env.CIRCLE_WALLET_SET_ID;
const client =
apiKey && entitySecret && walletSetId
? initiateDeveloperControlledWalletsClient({
apiKey,
entitySecret,
})
: undefined;
interface TransactionStatus {
id: string;
state: string;
txHash?: string;
createDate?: string;
updateDate?: string;
}
interface TokenBalance {
token: {
id: string;
blockchain: string;
name: string;
symbol: string;
decimals: number;
isNative: boolean;
tokenAddress?: string;
};
amount: string;
updateDate: string;
}
interface TokenBalanceResponse {
data: {
tokenBalances: TokenBalance[];
};
}
export class WalletService implements AgentWallet {
private walletStorage!: LocalStorage;
private senderAddress: string;
constructor(context: Context) {
this.senderAddress = context.message.sender.address;
this.walletStorage = new LocalStorage();
}
async getWallet(userAddress: string): Promise<AgentWalletData> {
const normalizedAddress = userAddress.toLowerCase();
const walletData = await this.walletStorage.get(
`wallet:${normalizedAddress}`,
);
if (walletData) {
const wallet = JSON.parse(walletData);
return wallet as AgentWalletData;
}
return this.createWallet(normalizedAddress);
}
async createWallet(identifier: string): Promise<AgentWalletData> {
const normalizedIdentifier = identifier.toLowerCase();
console.log(`Creating new wallet with id ${normalizedIdentifier}...`);
const response = await client?.createWallets({
accountType: "SCA",
blockchains: ["ETH-SEPOLIA"],
count: 1,
walletSetId: process.env.CIRCLE_WALLET_SET_ID as string,
metadata: [
{
name: "user",
refId: normalizedIdentifier,
},
{
name: "senderAddress",
refId: this.senderAddress,
},
],
});
if (!response?.data?.wallets?.[0]) {
throw new Error("Failed to create wallet");
}
const wallet = response?.data?.wallets?.[0];
// Store in LocalStorage
await this.walletStorage.set(
`wallet:${normalizedIdentifier}`,
JSON.stringify({
id: wallet.id,
address: wallet.address.toLowerCase(),
}),
);
console.log(`Created wallet with id ${normalizedIdentifier}`);
return {
id: wallet.id,
blockchain: wallet.blockchain,
state: wallet.state,
wallet: client?.getWallet,
address: this.senderAddress,
agent_address: wallet.address,
key: wallet.id,
};
}
async transfer(
fromAddress: string,
toAddress: string,
amount: number,
): Promise<any> {
try {
const normalizedFromAddress = fromAddress.toLowerCase();
const normalizedToAddress = toAddress.toLowerCase();
// Get token balances to find the requested token
const fromWallet = await this.getWallet(normalizedFromAddress);
const { balance, id } = await this.checkBalance(normalizedFromAddress);
// Check if balance is sufficient
if (balance < amount) {
throw new Error(`Insufficient USDC balance. Available: ${balance}`);
}
console.log("Initiating transfer:", {
from: fromWallet.id,
to: normalizedToAddress,
amount,
});
const response = await client?.createTransaction({
walletId: fromWallet.id,
tokenId: id as string,
destinationAddress: normalizedToAddress,
amount: [amount.toString()],
fee: {
type: "level",
config: {
feeLevel: "MEDIUM",
},
},
});
if (!response?.data?.id) {
throw new Error("Failed to create transaction");
}
// Start monitoring transaction status
const status = await this.checkTransactionStatus(response?.data?.id);
return status;
} catch (error) {
console.error("Error in transfer:", error);
throw error;
}
}
// Helper method to get available tokens
private async getAvailableTokens(
walletData: AgentWalletData,
): Promise<string[]> {
const balances = await this.getBalance(walletData);
return balances.map((b) => b.token.symbol);
}
private async getBalance(
walletData: AgentWalletData,
): Promise<TokenBalance[]> {
try {
console.log(`Checking balance for wallet ${walletData.id}...`);
const response = (await client?.getWalletTokenBalance({
id: walletData.id,
})) as TokenBalanceResponse;
//console.log("Wallet balance retrieved:", response.data.tokenBalances);
if (!response.data?.tokenBalances) {
throw new Error("Failed to get wallet balance");
}
return response.data.tokenBalances;
} catch (error) {
console.error("Error checking balance:", error);
throw error;
}
}
// Helper method to get specific token balance
async checkBalance(
humanAddress: string,
tokenSymbol: string = "USDC",
): Promise<{
address: string | undefined;
balance: number;
id: string | undefined;
}> {
const userWallet = await this.getWallet(humanAddress);
if (!userWallet) {
return { address: undefined, balance: 0, id: undefined };
}
const balances = await this.getBalance(userWallet);
const token = balances.find(
(b) => b.token.symbol.toUpperCase() === tokenSymbol.toUpperCase(),
);
return {
address: token?.token.tokenAddress,
id: token?.token.id,
balance: token ? parseFloat(token.amount) : 0,
};
}
async checkTransactionStatus(
transactionId: string,
): Promise<TransactionStatus> {
try {
const response = await client?.getTransaction({
id: transactionId,
});
if (!response?.data?.transaction) {
throw new Error("Failed to get transaction status");
}
const tx = response.data.transaction;
return {
id: tx.id,
state: tx.state,
txHash: tx.txHash,
createDate: tx.createDate,
updateDate: tx.updateDate,
};
} catch (error) {
console.error("Error checking transaction status:", error);
throw error;
}
}
async onRampURL(amount: number, address: string) {
return undefined;
}
}