Skip to content
Alchemy Logo

Same-chain swaps (Alpha)

Swaps let you convert any token to any other token onchain. They're built natively into Wallet APIs and you can integrate in minutes.

You can also add actions after the swap completes. For example, you can deposit your newly swapped tokens into a DeFi protocol.

Swaps work like any other Smart Wallet transaction, so you can sponsor gas for gasless swaps, or pay for gas in an ERC-20 token.

Cross-chain swaps are live! Send your first cross-chain swap now!

Swaps are in alpha. There may be changes in the future to simplify the endpoint/SDK.

  1. Request a swap quote
  2. Sign the prepared swap calls (including any post swap action)
  3. Send prepared calls
  4. Wait for onchain confirmation

When requesting a swap quote, specify either an amount in, or a minimum amount out.

// Mode 1: Swap exact input amount
{
  fromAmount: "0x2710";
} // Swap exactly 0.01 USDC (10000 in hex, 6 decimals)
 
// Mode 2: Get minimum output amount
{
  minimumToAmount: "0x5AF3107A4000";
} // Get at least 0.0001 ETH (18 decimals). We calculate how much USDC you need to spend to get at least your desired ETH amount.

Before you begin, ensure you have:

  • An API key
  • If you're sponsoring gas, a Gas Manager policy
  • A small amount of USDC for testing (~$1 worth is enough)
    • Important: Send these tokens to your smart wallet address before swapping.
  • An owner to sign messages
@alchemy/wallet-apis (v5.x.x) is currently in beta but is the recommended replacement for @account-kit/wallet-client (v4.x.x). If you run into any issues, please reach out.

You need the following environment variables:

  • ALCHEMY_API_KEY: An API key
  • ALCHEMY_POLICY_ID: A Gas Manager policy ID
  • PRIVATE_KEY: A private key for an owner
requestQuote.ts
import { swapActions } from "@alchemy/wallet-apis/experimental";
import { client } from "./client";
 
// Add the swap actions to the client
const swapClient = client.extend(swapActions);
 
// Request the swap quote
const { quote, ...calls } = await swapClient.requestQuoteV0({
  fromToken: "0x...",
  toToken: "0x...",
  minimumToAmount: "0x...",
});
 
// Display the swap quote, including the minimum amount to receive and the expiry
console.log(quote);
 
// Assert that the calls are not raw calls.
// This will always be the case when requestQuoteV0 is used without the `returnRawCalls` option,
// the assertion is just needed for Typescript to recognize the result type.
if (calls.rawCalls) {
  throw new Error("Expected user operation calls");
}
 
// Sign the quote using the signer on the client
const signedCalls = await swapClient.signPreparedCalls(calls);
 
// Send the prepared calls
const { id } = await swapClient.sendPreparedCalls(signedCalls);
 
// Wait for the call to resolve
const callStatusResult = await swapClient.waitForCallsStatus({
  id,
});
 
// Filter through success or failure cases
if (
  callStatusResult.status !== "success" ||
  !callStatusResult.receipts ||
  !callStatusResult.receipts[0]
) {
  throw new Error(
    `Transaction failed with status ${callStatusResult.status}, full receipt:\n ${JSON.stringify(callStatusResult, null, 2)}`,
  );
}
 
console.log("Swap confirmed!");
console.log(
  `Transaction hash: ${callStatusResult.receipts[0].transactionHash}`,
);
Using @account-kit/wallet-client (v4.x.x)?

The examples on this page use @alchemy/wallet-apis (v5.x.x). If you're using @account-kit/wallet-client (v4.x.x), the client setup looks like this:

client.ts (v4.x.x)
import { LocalAccountSigner } from "@aa-sdk/core";
import { createSmartWalletClient } from "@account-kit/wallet-client";
import { alchemy, sepolia } from "@account-kit/infra";
 
const signer = LocalAccountSigner.privateKeyToAccountSigner("0xYOUR_PRIVATE_KEY" as const);
 
export const client = createSmartWalletClient({
  transport: alchemy({ apiKey: "YOUR_API_KEY" }),
  chain: sepolia,
  signer,
  account: signer.address, // can also be passed per action as `from` or `account`
  // Optional: sponsor gas for your users (see "Sponsor gas" guide)
  policyId: "YOUR_POLICY_ID", 
});

Key v4.x.x differences:

  • Account address must be specified on the client or per action (from or account). In v5.x.x, the client automatically uses the owner's address as the account address via EIP-7702.
  • Chain imports come directly from @account-kit/infra instead of viem/chains.
  • Numeric values use hex strings: value: "0x0" instead of value: BigInt(0).
  • In v4.x.x, the paymaster capability on prepareCalls or sendCalls is called paymasterService instead of paymaster, or you can set the policyId directly on the client.
  • Owners use LocalAccountSigner / WalletClientSigner from @aa-sdk/core. In v5.x.x, a viem LocalAccount or WalletClient is used directly.

See the full migration guide for a complete cheat sheet.

Chains supported (for now) are: Ethereum, Arbitrum, Base, Berachain, BSC/BNB, Ink, Monad, Optimism, Polygon, Unichain and World Chain mainnets.

Currently, the Swap API supports only single-chain swaps. Cross-chain swaps are coming soon!

Values are passed as hexadecimal strings. The Swap API does not add complexity to consider decimals, so 0x01 is always the smallest amount of a given asset. 1 ETH, or DAI (18 decimals) is 0xDE0B6B3A7640000 1 USDC (6 decimals) is 0xF4240 This removes any ambiguity— if it’s numerical, it’s a hex.

The expiry is an informational indicator of when you can expect to be able to process the swap request. If you’re at/near the expiry, it might be a good time to request a new quote.

Was this page helpful?