> ## Documentation Index
> Fetch the complete documentation index at: https://docs.arc.io/llms.txt
> Use this file to discover all available pages before exploring further.

# How to: Detect and Process Deposits

> Implement deposit detection on Arc by monitoring native USDC Transfer events from the system emitter, confirming with deterministic finality, and sweeping funds.

Detect USDC deposits on Arc by generating addresses, monitoring the native USDC
`Transfer` event from the system emitter `0xffff…fffe` (Arc's
[EIP-7708](https://eips.ethereum.org/EIPS/eip-7708) implementation), crediting
after a single block confirmation (deterministic finality guarantees no reorgs),
and sweeping funds into a hot wallet.

## Prerequisites

Before you begin:

* Access to an Arc RPC endpoint (`https://rpc.testnet.arc.network`) or WebSocket
  (`wss://rpc.testnet.arc.network`)
* An HD wallet library for generating deposit addresses (for example, `ethers`
  or `viem`)
* Familiarity with Ethereum JSON-RPC methods and event log filtering
* A database to track processed deposits and prevent double-crediting

## Steps

### Step 1. Generate deposit addresses

Arc uses standard Ethereum addresses (0x-prefixed, 20 bytes, EIP-55 checksum).
Derive deposit addresses using the same HD wallet approach as Ethereum—one
unique address per user.

```typescript theme={null}
import { HDNodeWallet, Mnemonic } from "ethers";

// Derive a deposit address for a given user index
function getDepositAddress(mnemonic: string, userIndex: number): string {
  const hdNode = HDNodeWallet.fromMnemonic(
    Mnemonic.fromPhrase(mnemonic),
    `m/44'/60'/0'/0/${userIndex}`,
  );
  return hdNode.address;
}
```

Store the mapping between user IDs and their derived address index. Never expose
the mnemonic or private keys in client-side code.

### Step 2. Subscribe to new blocks

Use `eth_subscribe("newHeads")` over WebSocket for real-time block
notifications, or poll `eth_blockNumber` over HTTP as a fallback.

```typescript theme={null}
import { WebSocketProvider, JsonRpcProvider } from "ethers";

// Option A: WebSocket subscription (recommended)
const wsProvider = new WebSocketProvider("wss://rpc.testnet.arc.network");

wsProvider.on("block", async (blockNumber: number) => {
  console.log(`New block: ${blockNumber}`);
  await processBlock(blockNumber);
});

// Option B: HTTP polling fallback
const httpProvider = new JsonRpcProvider("https://rpc.testnet.arc.network");

let lastProcessedBlock = await httpProvider.getBlockNumber();

setInterval(async () => {
  const currentBlock = await httpProvider.getBlockNumber();
  for (let block = lastProcessedBlock + 1; block <= currentBlock; block++) {
    await processBlock(block);
  }
  lastProcessedBlock = currentBlock;
}, 2000);
```

### Step 3. Detect incoming transfers with the native USDC Transfer event

Every native USDC movement emits a standard ERC-20 `Transfer` log from the
system address `0xfffffffffffffffffffffffffffffffffffffffe` (Arc's
[EIP-7708](https://eips.ethereum.org/EIPS/eip-7708) implementation). This single
stream covers plain native sends and the native leg of ERC-20 transfers, with
values in **18 decimals**. Filter it to catch every deposit, including native
sends that emit no event on the ERC-20 contract.

<Warning>
  Do not filter the ERC-20 USDC contract (`0x3600…0000`) for deposit detection.
  Its `Transfer` events cover only ERC-20-interface activity, so a plain native
  send produces no log there and the deposit is missed. Filter the system
  emitter (`0xffff…fffe`) instead. See [USDC system
  events](/arc/references/usdc-system-events) for the full event matrix.
</Warning>

**Event signature:**

```solidity theme={null}
Transfer(address indexed from, address indexed to, uint256 value)
```

**Topic0:** `0xddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef`

**Native USDC system emitter:** `0xfffffffffffffffffffffffffffffffffffffffe`

Use `eth_getLogs` to filter for transfers to your deposit addresses:

```typescript theme={null}
import { Interface, Log, JsonRpcProvider } from "ethers";

// Native USDC system emitter (EIP-7708 Transfer logs, 18 decimals)
const NATIVE_USDC_EMITTER = "0xfffffffffffffffffffffffffffffffffffffffe";
const TRANSFER_TOPIC =
  "0xddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef";

const provider = new JsonRpcProvider("https://rpc.testnet.arc.network");
const erc20Interface = new Interface([
  "event Transfer(address indexed from, address indexed to, uint256 value)",
]);

async function processBlock(blockNumber: number): Promise<void> {
  const logs = await provider.getLogs({
    address: NATIVE_USDC_EMITTER,
    topics: [
      TRANSFER_TOPIC,
      null, // any sender
      null, // any recipient—filter client-side for your addresses
    ],
    fromBlock: blockNumber,
    toBlock: blockNumber,
  });

  for (const log of logs) {
    const parsed = erc20Interface.parseLog({
      topics: log.topics as string[],
      data: log.data,
    });

    if (!parsed) continue;

    const to = parsed.args.to as string;
    const value = parsed.args.value as bigint; // 18 decimals (native)

    // Check if the recipient is one of your deposit addresses
    if (isDepositAddress(to)) {
      await creditDeposit({
        txHash: log.transactionHash,
        logIndex: log.index,
        to,
        amount: value, // 18 decimals—convert to 6 for crediting (see Step 5)
        blockNumber,
      });
    }
  }
}
```

<Warning>
  Do not credit the same deposit twice. An ERC-20 `transfer()` emits a log from
  both the system emitter (`0xffff…fffe`, 18 decimals) and the ERC-20 contract
  (`0x3600…0000`, 6 decimals); filter only the system emitter. Likewise, do not
  reconcile `eth_getBalance` against Transfer events—they represent the same
  balance.
</Warning>

### Step 4. Confirm the deposit

Arc provides deterministic finality—once a transaction is included in a block,
it is final with no possibility of reorg. You can safely credit deposits after 1
confirmation.

```typescript theme={null}
import { JsonRpcProvider } from "ethers";

const provider = new JsonRpcProvider("https://rpc.testnet.arc.network");

async function isConfirmed(txHash: string): Promise<boolean> {
  const receipt = await provider.getTransactionReceipt(txHash);

  if (!receipt || receipt.status === 0) {
    return false; // Transaction failed or not yet mined
  }

  // On Arc, 1 confirmation = final. No reorgs possible.
  const currentBlock = await provider.getBlockNumber();
  return currentBlock >= receipt.blockNumber;
}
```

Since Arc has deterministic finality, you do not need to wait for multiple
confirmations. Credit the user once the block containing their transaction is
produced.

### Step 5. Handle decimal conversion

The native system `Transfer` event emits values with 18 decimals. Convert to
6-decimal USDC (divide by 10^12) before crediting user balances.

```typescript theme={null}
// Native system Transfer values use 18 decimals—convert to 6 for crediting
function formatDeposit(eventValue: bigint): string {
  const sixDecimals = eventValue / 10n ** 12n; // 18-decimal native → 6-decimal USDC
  const whole = sixDecimals / 1_000_000n;
  const fractional = (sixDecimals % 1_000_000n).toString().padStart(6, "0");
  return `${whole}.${fractional}`;
}
```

<Warning>
  The native system `Transfer` event and `eth_getBalance` both use 18 decimals;
  the ERC-20 contract's `Transfer` and `balanceOf` use 6. Never mix
  representations—convert 18-decimal values by dividing by 10^12 before
  displaying or crediting.
</Warning>

### Step 6. Sweep deposits to a hot wallet

Consolidate deposited funds from individual user addresses into your hot wallet.
Use EIP-1559 transactions with Arc's minimum base fee of 20 Gwei.

```typescript theme={null}
import { Wallet, JsonRpcProvider, parseUnits } from "ethers";

const provider = new JsonRpcProvider("https://rpc.testnet.arc.network");

async function sweepDeposit(
  depositPrivateKey: string,
  hotWalletAddress: string,
  amount: bigint,
): Promise<string> {
  const wallet = new Wallet(depositPrivateKey, provider);

  const feeData = await provider.getFeeData();
  const maxFeePerGas = feeData.maxFeePerGas ?? parseUnits("30", "gwei");
  const maxPriorityFeePerGas =
    feeData.maxPriorityFeePerGas ?? parseUnits("1", "gwei");

  // Estimate gas for an ERC-20 transfer (approximately 21,000 for native sends)
  const gasLimit = 65_000n;

  const tx = await wallet.sendTransaction({
    to: hotWalletAddress,
    value: amount * 10n ** 12n, // Convert 6-decimal amount to 18-decimal native value
    type: 2, // EIP-1559
    maxFeePerGas,
    maxPriorityFeePerGas,
    gasLimit,
  });

  const receipt = await tx.wait();
  return receipt!.hash;
}
```

<Note>
  When sweeping via a native USDC send, convert from 6-decimal amounts to
  18-decimal `value` fields. Alternatively, call the ERC-20 `transfer` function
  on the USDC contract, which accepts 6-decimal amounts directly.
</Note>

## Common mistakes

<Warning>
  **Avoid these pitfalls when implementing deposits:**

  * **Filtering the wrong emitter:** Plain native USDC sends emit no `Transfer` on
    the ERC-20 contract (`0x3600…0000`). Filter the system emitter (`0xffff…fffe`)
    so you do not miss native deposits.
  * **Double-counting:** An ERC-20 `transfer()` logs from both emitters. Filter
    only the system emitter, and do not reconcile `eth_getBalance` against
    Transfer events for the same address.
  * **Decimal mismatch:** The system emitter's values use 18 decimals. Treating
    them as 6-decimal amounts inflates credited balances by 10^12—divide by 10^12
    first.
  * **Waiting for multiple confirmations:** Arc has deterministic finality.
    Waiting for 6+ confirmations adds unnecessary latency with no security
    benefit.
  * **Hardcoded keys:** Never embed private keys in source code. Use environment
    variables or a secrets manager for sweep wallet credentials.
</Warning>
