Skip to main content
Polymarket is rolling out deposit wallets as a new account type alongside the existing Gnosis Safe proxy wallets. This guide explains what changed, whether you need to do anything, and how to handle both wallet types in your integration.

What are deposit wallets?

Deposit wallets are Polymarket’s new smart contract wallet type for new accounts. They replace the Gnosis Safe proxy that existing accounts use. From your perspective as a builder, the key differences are:
Safe proxy (existing)Deposit wallet (new)
Signature type2 (POLY_GNOSIS_SAFE)3 (POLY_1271)
How addresses are derivedCREATE2 from Safe factoryCREATE2 from deposit wallet factory
Order signingStandard EIP-712EIP-712 with ERC-7739 TypedDataSign wrapper
ApprovalsGasless via Safe multisendGasless via WALLET batch
V1 supportYesNo (V2 only)
The good news: if you use the polynode SDK, all of this is handled automatically. ensureReady() detects the wallet type, order() signs correctly, and the /resolve endpoint resolves both wallet types.

Do I need to change anything?

If you use the polynode SDK for trading: update to the latest version. Everything else is automatic.
# Minimum versions for deposit wallet support
cargo add polynode@0.13.3        # Rust
npm install polynode-sdk@0.10.6  # TypeScript
pip install polynode==0.10.3     # Python
If you only read data (positions, trades, /resolve): the response shape is unchanged. /resolve still returns safe, eoa, username, and type, but safe now means the preferred Polymarket trading wallet. If an EOA has a deployed deposit wallet, safe returns the deposit wallet. If you call the Polymarket CLOB directly (without our SDK): you need to implement ERC-7739 TypedDataSign wrapping for signatureType: 3 orders. See the signing details below.

How to detect wallet type

For trading, the SDK auto-detects wallet type when you call ensureReady() or detectWalletType(). Detection checks on-chain in this order:
  1. Safe deployed? -> POLY_GNOSIS_SAFE (type 2)
  2. Proxy deployed? -> POLY_PROXY (type 1)
  3. Deposit wallet deployed? -> POLY_1271 (type 3)
  4. Nothing deployed -> defaults to POLY_GNOSIS_SAFE (new Safe will be created)
This controls how the SDK signs orders. The /resolve identity endpoint uses a different rule for compatibility with Polymarket’s deposit-wallet migration: if a resolved EOA has a deployed deposit wallet, /resolve returns that deposit wallet in safe.
import { PolyNodeTrader } from 'polynode-sdk';

const trader = new PolyNodeTrader({ polynodeKey: 'pn_live_...' });
const status = await trader.ensureReady(privateKey);

console.log(status.signatureType);
// 2 = Safe (existing user)
// 3 = Deposit wallet (new user)

// Either way, order() works the same:
await trader.order({ tokenId: '...', side: 'BUY', price: 0.55, size: 100 });

How to handle mixed user bases

If your platform has existing Safe users and new deposit wallet users, no special handling is needed. The SDK stores each user’s wallet type in the local database and uses the correct signing method automatically.
// User A: existing Safe wallet
await trader.ensureReady(userA_privateKey);
// status.signatureType === 2, uses standard EIP-712

await trader.order({ tokenId: '...', side: 'BUY', price: 0.5, size: 50 });

// User B: new deposit wallet
await trader.ensureReady(userB_privateKey);
// status.signatureType === 3, uses ERC-7739 TypedDataSign wrapper

await trader.order({ tokenId: '...', side: 'BUY', price: 0.5, size: 50 });
// Same API, different signing under the hood

Wallet address derivation

You can derive any user’s deposit wallet address from their EOA without any network calls:
const { deriveDepositWalletAddress, deriveSafeAddress } = require('polynode-sdk');

const eoa = '0xA60601A4d903af91855C52BFB3814f6bA342f201';
const depositWallet = await deriveDepositWalletAddress(eoa);
const safe = await deriveSafeAddress(eoa);

// Both are deterministic — same EOA always produces the same addresses

The /resolve endpoint

The Resolve Wallet endpoint keeps the same fields and prioritizes deposit wallets in the existing safe field. The field name remains safe for backward compatibility.
curl "https://api.polynode.dev/v1/resolve/0x8b60bf0f650bf7a0d93f10d72375b37de18f8c40" \
  -H "x-api-key: YOUR_KEY"
{
  "safe": "0x8b60bf0f650bf7a0d93f10d72375b37de18f8c40",
  "eoa": "0xa60601a4d903af91855c52bfb3814f6ba342f201",
  "username": null,
  "type": "deposit_wallet"
}
The safe field contains the preferred trading wallet address. If the controlling EOA has a deployed deposit wallet, safe is the deposit wallet. Otherwise it is the existing Safe/proxy wallet. The type field tells you which kind of wallet was returned.
type valueDescription
"safe"Gnosis Safe proxy (most existing accounts)
"deposit_wallet"Deposit wallet (newer accounts)
"proxy"Legacy proxy wallet
Both forward resolution (EOA -> wallet) and reverse resolution (wallet -> EOA) work for all wallet types. When both a Safe and deposit wallet exist, deposit wallet wins.

How POLY_1271 signing works

You only need this section if you’re implementing signing without the polynode SDK. If you use signV2Order() or trader.order(), this is handled automatically.
For deposit wallet users, V2 orders use signatureType: 3 (POLY_1271). The signing process wraps the standard Order EIP-712 hash in a Solady TypedDataSign struct:
  1. Build the standard V2 Order struct (same fields as type 0/1/2)
  2. Set maker and signer to the deposit wallet address (not the EOA)
  3. Sign using TypedDataSign as the EIP-712 primary type under the exchange domain, with the deposit wallet parameters as struct fields
  4. Wrap the resulting signature: innerSig (65 bytes) + appDomainSeparator (32 bytes) + contentsHash (32 bytes) + orderTypeString (186 bytes) + typeStringLength (2 bytes)
The final wrapped signature is 317 bytes (634 hex chars + “0x” prefix = 636 total). The TypedDataSign struct includes the deposit wallet’s identity:
  • name: "DepositWallet"
  • version: "1"
  • verifyingContract: the deposit wallet address
  • salt: bytes32(0)

FAQ

No. Safe wallets continue to work exactly as before. The SDK detects your wallet type automatically. All existing integrations are backward compatible.
No field names changed. The endpoint still returns safe, eoa, username, and type. The important behavior change is that safe now returns the deposit wallet when the resolved EOA has one deployed.
Yes. The SDK defaults to creating Safe wallets for new users when no wallet is detected on-chain. Deposit wallets are created through Polymarket’s frontend or relayer, not through the polynode SDK’s default flow.
Deposit wallet address derivation and detection: >= 0.10.0. POLY_1271 order signing: >= 0.10.5. Full onboarding flow (deploy + approve): >= 0.10.6. Latest recommended: Rust 0.13.3, TypeScript 0.10.6, Python 0.10.3.
For /resolve, the deposit wallet is returned in safe when it is deployed. The field name stays safe so existing clients do not need to parse a new field.
No. POLY_1271 (signature type 3) is V2 only. V1 orders only support types 0 (EOA), 1 (POLY_PROXY), and 2 (POLY_GNOSIS_SAFE). Since Polymarket’s V2 exchange is now live, all new activity uses V2.
Register followers with their actual sig_type (2 for Safe, 3 for deposit wallet). The copy trading engine automatically generates the correct typed data for each follower’s wallet type. Your signing code should use signV2Order() from the SDK, which handles both types transparently.
Deposit wallet signatures are larger (317 bytes vs 65 bytes) due to the ERC-7739 wrapping, but this has no meaningful impact on order submission speed or gas costs. The CLOB processes both types identically.