Skip to content

Zeto - Zero Knowledge Proof based tokens

Zeto is a UTXO based privacy-preserving token toolkit for EVM, using Zero Knowledge Proofs, implemented via Circom.

The architecture documentations for Zeto is being managed in a separate Github repository here: https://github.com/kaleido-io/zeto

Zeto is a growing collection of token implementations that enforce a wide variety of token transaction policies including, but not limited to, mass conservation (for fungible tokens), preservation of asset properties during ownership transfer (for non-fungible tokens), KYC with privacy, and non-repudiation compliance. Each policy is expressed in zeto knowledge proof circuits using Circom. The list of policies and their corresponding token implementations will continue to grow to meet the needs of enterprise use cases.

Paladin Support

Zeto tokens are natively supported by Paladin, as a domain implementation called "Zeto". The foundational operations of Zeto tokens, mint, transfer are supported in the initial Paladin release. Support for other operations such as deposit, withdraw will be added later.

As a client to Zeto tokens, Paladin has the following features built into the single runtime that runs alongside an Ethereum node:

  • Tokens indexer: by the nature of a UTXO based design, an account's balance is not known by querying the smart contract, as is the case with ERC-20 tokens. Instead, the UTXOs must be indexed from confirmed onchain transactions in order for an account to know the balance, by adding together all the tokens that belong to that account.
  • Token selector: when sending a transaction that transfers certain amount to another account, the transaction input is made up of a collection of UTXOs that will be spent. The UTXOs must be selected from the unspent UTXOs that the account holds. The selection process needs to take into account the intend transfer amount and the values of the available tokens.
  • ZK proof generator: each Zeto transaction must be accompanied by a ZK proof to demonstrate the validity of the transaction proposal. This is accomplished by a proof generator that is able to use the secrets known only to the Paladin runtime hosting the account key, as private input to the ZKP circuit.

Private ABI

The private ABI of Zeto is implemented in Go, and can be accessed by calling ptx_sendTransaction with "type": "private".

constructor

Creates a new Zeto token contract, with a new address on the base ledger.

{
  "name": "",
  "type": "constructor",
  "inputs": [{ "name": "tokenName", "type": "string" }]
}

Inputs:

mint

Mint new value. New UTXO state(s) will automatically be created to fulfill the requested mint.

{
  "name": "mint",
  "type": "function",
  "inputs": [
    {
      "name": "mints",
      "type": "tuple[]",
      "components": [
        {
          "name": "to",
          "type": "string",
          "internalType": "string"
        },
        {
          "name": "amount",
          "type": "uint256",
          "internalType": "uint256"
        }
      ]
    }
  ]
}

Inputs:

  • mints - list of mints, each with a receiver name and amount
  • to - lookup string for the identity that will receive minted value
  • amount - amount of new value to create

transfer

Transfer value from the sender to another recipient. Available UTXO states will be selected for spending, and new UTXO states will be created, in order to facilitate the requested transfer of value.

{
  "type": "function",
  "name": "transfer",
  "inputs": [
    {
      "name": "transfers",
      "type": "tuple[]",
      "components": [
        {
          "name": "to",
          "type": "string",
          "internalType": "string"
        },
        {
          "name": "amount",
          "type": "uint256",
          "internalType": "uint256"
        }
      ]
    }
  ],
  "outputs": null
}

Inputs:

  • transfers - list of transfers, each with a receiver name and amount
  • to - lookup string for the identity that will receive transferred value
  • amount - amount of value to transfer

deposit

The Zeto token implementations support interaction with an ERC20 token, to control the value supply publicly. With this paradigm, the token issuer, such as a central bank for digital currencies, can control the total supply in the ERC20 contract. This makes the supply of the tokens public.

The Zeto token contract can be configured to allow balances from a designated ERC20 contract to be "swapped" for Zeto tokens, by calling the deposit API. This allows any accounts that have a balance in the ERC20 contract to swap them for Zeto tokens. The exchange rate between the ERC20 and Zeto tokens is 1:1. On successful deposit, the ERC20 balance is transferred to the Zeto contract.

Typically in this paradigm, the mint API on the Zeto domain should be locked down (disabled) so that the only way to mint Zeto tokens is by depositing.

{
  "type": "function",
  "name": "deposit",
  "inputs": [
    {
      "name": "amount",
      "type": "uint256",
      "internalType": "uint256"
    }
  ],
  "outputs": null
}

Inputs:

  • amount - amount of value to deposit

withdraw

Opposite to the "deposit" operation, users can swap Zeto tokens back to ERC20 balances.

On successful withdrawal, the ERC20 balance is released by the Zeto contract and transferred back to the user account.

{
  "type": "function",
  "name": "withdraw",
  "inputs": [
    {
      "name": "amount",
      "type": "uint256",
      "internalType": "uint256"
    }
  ],
  "outputs": null
}

Inputs:

  • amount - amount of value to withdraw

lockProof

This is a special purpose function used in coordinating multi-party transactions, such as Delivery-vs-Payment (DvP) contracts. When a party commits to the trade first by uploading the ZK proof to the orchestration contract, they must be protected from a malicious party seeing the proof and using it to unilaterally execute the token transfer. The lockProof() function allows an account, which can be a smart contract address, to designate the finaly submitter of the proof, thus protecting anybody else from abusing the proof outside of the atomic settlement of the multi-leg trade.

{
  "type": "function",
  "name": "lockProof",
  "inputs": [
    {
      "name": "delegate",
      "type": "address"
    },
    {
      "name": "call",
      "type": "bytes"
    }
  ],
  "outputs": null
}

Inputs:

  • delegate - set to the Ethereum account, which can be an externally owned account or a smart contract address, that is allowed to submit the transaction to use the locked proof to execute the Zeto token transfer
  • call - this is an abi encoded bytes from a call to the transfer() function of the target Zeto token smart contract. Refer to the PvP test case for an example of how to construct the encode call bytes