Deposits
L1 → L2 deposits for ETH and ERC-20 tokens with quote, prepare, create, status, and wait helpers.
At a Glance
- Resource:
sdk.deposits - Typical flow:
quote → create → wait({ for: 'l2' }) - Auto-routing: ETH vs ERC-20 and base-token vs non-base handled automatically
- Error style: Throwing methods (
quote,prepare,create,wait) + safe variants (tryQuote,tryPrepare,tryCreate,tryWait)
Import
import { JsonRpcProvider, Wallet, parseEther } from 'ethers';
import { createEthersClient, createEthersSdk } from '@matterlabs/zksync-js/ethers';
const l1 = new JsonRpcProvider(process.env.ETH_RPC!);
const l2 = new JsonRpcProvider(process.env.ZKSYNC_RPC!);
const signer = new Wallet(process.env.PRIVATE_KEY!, l1);
const client = createEthersClient({ l1, l2, signer });
const sdk = createEthersSdk(client);
// sdk.deposits → DepositsResource
Quick Start
Deposit 0.1 ETH from L1 → L2 and wait for L2 execution:
const handle = await sdk.deposits.create({
token: ETH_ADDRESS, // 0x…00 for ETH
amount: parseEther('0.1'),
to: await signer.getAddress(),
});
const l2Receipt = await sdk.deposits.wait(handle, { for: 'l2' }); // null only if no L1 hash
[!TIP] For UX that never throws, use the
try*variants and branch onok.
Route Selection (Automatic)
| Route | Meaning |
|---|---|
eth-base | ETH when L2 base token is ETH |
eth-nonbase | ETH when L2 base token ≠ ETH |
erc20-base | ERC-20 that is the L2 base token |
erc20-nonbase | ERC-20 that is not the L2 base token |
You don’t pass a route manually; it’s derived from network metadata and the token.
Method Reference
quote(p: DepositParams) → Promise<DepositQuote>
Estimate the operation (route, approvals, gas hints). Does not send transactions.
Parameters
| Name | Type | Required | Description |
|---|---|---|---|
token | Address | ✅ | L1 token address. Use 0x…00 for ETH. |
amount | bigint | ✅ | Amount in wei to deposit. |
to | Address | ❌ | L2 recipient address. Defaults to the signer’s address if omitted. |
refundRecipient | Address | ❌ | Optional address on L1 to receive refunds for unspent gas. |
l2GasLimit | bigint | ❌ | Optional manual L2 gas limit override. |
gasPerPubdata | bigint | ❌ | Optional custom gas-per-pubdata value. |
operatorTip | bigint | ❌ | Optional operator tip (in wei) for priority execution. |
l1TxOverrides | Eip1559GasOverrides | ❌ | Optional EIP-1559 gas settings for the L1 transaction. |
Returns: DepositQuote
const q = await sdk.deposits.quote({
token: ETH_L1,
amount: parseEther('0.25'),
to: await signer.getAddress(),
});
/*
{
route: "eth-base" | "eth-nonbase" | "erc20-base" | "erc20-nonbase",
approvalsNeeded: [{ token, spender, amount }],
baseCost?: bigint,
mintValue?: bigint,
suggestedL2GasLimit?: bigint,
gasPerPubdata?: bigint
}
*/
[!TIP] If
approvalsNeededis non-empty (ERC-20),create()will include those approval steps automatically.
tryQuote(p) → Promise<{ ok: true; value: DepositQuote } | { ok: false; error }>
Result-style version of quote.
prepare(p: DepositParams) → Promise<DepositPlan<TransactionRequest>>
Build the plan (ordered steps + unsigned transactions) without sending.
Returns: DepositPlan
const plan = await sdk.deposits.prepare({ token: ETH_L1, amount: parseEther('0.05'), to });
/*
{
route,
summary: DepositQuote,
steps: [
{ key: "approve:USDC", kind: "approve", tx: TransactionRequest },
{ key: "bridge", kind: "bridge", tx: TransactionRequest }
]
}
*/
tryPrepare(p) → Promise<{ ok: true; value: DepositPlan } | { ok: false; error }>
Result-style prepare.
create(p: DepositParams) → Promise<DepositHandle<TransactionRequest>>
Prepares and executes all required L1 steps. Returns a handle with the L1 transaction hash and per-step hashes.
Returns: DepositHandle
const handle = await sdk.deposits.create({ token, amount, to });
/*
{
kind: "deposit",
l1TxHash: Hex,
stepHashes: Record<string, Hex>,
plan: DepositPlan
}
*/
[!WARNING] If any step reverts,
create()throws a typed error. PrefertryCreate()to avoid exceptions.
tryCreate(p) → Promise<{ ok: true; value: DepositHandle } | { ok: false; error }>
Result-style create.
status(handleOrHash) → Promise<DepositStatus>
Resolve the current phase for a deposit.
Accepts either the DepositHandle from create() or a raw L1 transaction hash.
Phases
| Phase | Meaning |
|---|---|
UNKNOWN | No L1 hash provided |
L1_PENDING | L1 receipt not yet found |
L1_INCLUDED | Included on L1; L2 hash not derivable yet |
L2_PENDING | L2 hash known; waiting for L2 receipt |
L2_EXECUTED | L2 receipt found with status === 1 |
L2_FAILED | L2 receipt found with status !== 1 |
const s = await sdk.deposits.status(handle);
// { phase, l1TxHash, l2TxHash? }
wait(handleOrHash, { for: 'l1' | 'l2' }) → Promise<TransactionReceipt | null>
Block until the specified checkpoint.
{ for: 'l1' }→ L1 receipt (ornullif no L1 hash){ for: 'l2' }→ L2 receipt after canonical execution (ornullif no L1 hash)
const l1Receipt = await sdk.deposits.wait(handle, { for: 'l1' });
const l2Receipt = await sdk.deposits.wait(handle, { for: 'l2' });
tryWait(handleOrHash, opts) → Result<TransactionReceipt>
Result-style wait.
End-to-End Examples
ETH Deposit (Typical)
const handle = await sdk.deposits.create({
token: ETH_ADDRESS,
amount: parseEther('0.001'),
to: await signer.getAddress(),
});
await sdk.deposits.wait(handle, { for: 'l2' });
ERC-20 Deposit
const handle = await sdk.deposits.create({
token: '0xA0b86991c6218b36c1d19D4a2e9Eb0cE3606eB48', // Example: USDC
amount: 1_000_000n, // 1.0 USDC (6 decimals)
to: await signer.getAddress(),
});
const l1Receipt = await sdk.deposits.wait(handle, { for: 'l1' });
Types (Overview)
type DepositParams = {
token: Address; // 0x…00 for ETH
amount: bigint; // wei
to?: Address; // L2 recipient
refundRecipient?: Address;
l2GasLimit?: bigint;
gasPerPubdata?: bigint;
operatorTip?: bigint;
l1TxOverrides?: Eip1559GasOverrides;
};
type Eip1559GasOverrides = {
gasLimit?: bigint;
maxFeePerGas?: bigint;
maxPriorityFeePerGas?: bigint;
};
type DepositQuote = {
route: 'eth-base' | 'eth-nonbase' | 'erc20-base' | 'erc20-nonbase';
approvalsNeeded: Array<{ token: Address; spender: Address; amount: bigint }>;
baseCost?: bigint;
mintValue?: bigint;
suggestedL2GasLimit?: bigint;
gasPerPubdata?: bigint;
};
type DepositPlan<TTx = TransactionRequest> = {
route: DepositQuote['route'];
summary: DepositQuote;
steps: Array<{ key: string; kind: string; tx: TTx }>;
};
type DepositHandle<TTx = TransactionRequest> = {
kind: 'deposit';
l1TxHash: Hex;
stepHashes: Record<string, Hex>;
plan: DepositPlan<TTx>;
};
type DepositStatus =
| { phase: 'UNKNOWN'; l1TxHash: Hex }
| { phase: 'L1_PENDING'; l1TxHash: Hex }
| { phase: 'L1_INCLUDED'; l1TxHash: Hex }
| { phase: 'L2_PENDING'; l1TxHash: Hex; l2TxHash: Hex }
| { phase: 'L2_EXECUTED'; l1TxHash: Hex; l2TxHash: Hex }
| { phase: 'L2_FAILED'; l1TxHash: Hex; l2TxHash: Hex };
[!TIP] Prefer the
try*variants if you want to avoid exceptions and work with structured result objects.
Notes & Pitfalls
- ETH sentinel: Use the canonical
0x…00address when passing ETH astoken. - Receipt timing:
wait({ for: 'l2' })resolves only after canonical L2 execution — it can take longer than L1 inclusion.