/Cycles.Finance

An ICP/Cycles Market

Primary LanguageMotokoGNU General Public License v3.0GPL-3.0

Cycles.Finance

Website: http://cycles.finance
Canister Id: 6nmrm-laaaa-aaaak-aacfq-cai
Module hash: 070a8e3cfb65204e29241278f19c5c440dabfbf6913e080441e647759102847e
Version: 0.6

Disclaimers.

This project is in beta and may have defects. It‘s a Dapp, please be knowledgeable and participate voluntarily at your own risk.

Overview

Cycles.Finance is an ICP/Cycles marketplace that supports bidirectional exchange of ICP, Cycles, using a multiplicative constant K model (A*B=K), similar to UniSwapV2.

Restrictions on Swapping

The project is still in beta and limits have been placed on a swap in order to control risk.

  • ICP: max 10 icp each swap, min 10,000 e8s each swap.
  • Cycles: max 3*10^14 cycles each swap, min 10^8 cycles each swap.
  • Swap volatility limit: a swap causing price fluctuations more than 20% will be rejected.

Swap fees

Swapping fee: 1%, on a post-fee model, charged in ICPs or Cycles.
ICP transfer fee: 10,000 e8s per transfer, deducted by ic network.
Swapping fee usage: All ICP charged is moved to the liquidity reward pool, 80% of Cycles charged is moved to the liquidity reward pool (the other 20% is used for Canister gas).

Liquidity (AMM)

Market making model: automatic market making model(AMM). Using a multiplicative constant K model (AB=K).

Liquidity reward pool

  • Liquidity providers can claim liquidity rewards in proportion to the share they hold.
  • [Plan] Participate in the ICLighthouse liquidity mining program and receive ICL token rewards.

Note: To participate in AMM, you need to add both ICP and Cycles assets to the liquidity pool. The amount of these two assets will change as users trade. For example, if you add 1 ICP and 30 TCycles to the liquidity pool, after a period of time you may withdraw 0.9 ICP and 33.3 TCycles or other possible amounts.

How does it work?

CyclesFinance is designed to facilitate the swap of ICPs and Cycles in a different way to traditional exchanges.

It does this by using Canister smart contracts, which allow users (called liquidity providers) to deposit ICPs/Cycles into pool. These smart contracts allow traders to buy and sell these assets. Users who trade these assets pay a fee which is then distributed to all liquidity providers proportionally (based on their contribution to the asset pool).

image

Liquidity pool

The liquidity pool holds ICPs and Cycles, which together represent a trading pair for those assets.

CyclesFinance uses a pricing mechanism called the ‘multiplicative constant k market maker model’. The formula (A * B = k) is used to determine the pricing for the pair.
A and B represent the pool balance of each asset, and k is the total constant price of said pool.
In the liquidity pool, the first liquidity provider sets the initial price of the assets in the pool by supplying an equal value to both tokens.

Swap

Buyers can then swap ICP/Cycles within the pool, based on the formula. Smart contracts running the rule use the above formula to take the amount of one asset from the buyer and send an equivalent amount of another asset purchased back to the buyer, keeping the total pool constant stable (k).

Example:

Let us imagine the ICP/Cycles liquidity pool contains 10 ICP(A) and 300 TCycles (B), therefore making the pool constant value 3000 (k).
This implies that the pool’s starting price is 30 TCycles per ICP.

Now, let us imagine a trader comes in and wants to buy 0.1 ICP.

The ICP/Cycles pool will now have:

New A: 9.9 ICP (10 - 0.1)
k: 3000 (stays constant)
New B: 303.03 TCycles (3000 / 9.9)
Therefore, the pool would imply a price of 30.303 TCycles per ICP to keep k constant since you had to add 3.03 TCycles (303.03 TCycles – 300 TCycles) to the pool to buy 0.1 ICP.

In the above example, the next implied rate for ICP will be around 30.609 TCycles per ICP (303.03 TCycles/9.9 ICP).

When the price of an asset starts to trade away from market prices, arbitragers see this as an opportunity to make risk-free returns. Therefore, they come in to trade the price back to market rates. This is a vital part of the ICP/Cycles ecosystem.

Technical features

Achieving eventually consistency

CyclesFinance faces atomicity problems mainly with ICP internal transfer failures and Cycles send failures. We use a Best Effort Commit strategy together with an error handling mechanism to ensure ultimate consistency.

  • Before updating state variables, canister will throw an exception when an error is encountered.
  • It ensures that all internal state variables are successfully saved when some of them have already been updated, and external calls take a Best Effort Commit strategy, but to prevent duplicate transactions, so an error handling mechanism is added.
  • With regard to the error handling mechanism, it requires the administrator or governance contract to trigger a retransaction. For accounts that are unable to receive Cycles, the manager or governance contract can modify the receiving account.

Enhanced idempotency

The idempotency prevents repeated executions when an external canister is submitted repeatedly in the event of an error in the call to CyclesFinance.

  • The transaction txid is computable and globally unique.
  • The nonce mechanism for accounts is supported.

Oracle quotes support

ICP/Cycles swap prices could be used as native Oracle quotes on the IC network. CyclesFinance offers two forms of Oracle quotes, including

  • Latest prices: The liquidity(null) method of the CyclesFinance canister is used to query the number of ICPs and Cycles in the liquidity pool. cycles/icp.e8s means the ICP/Cycles price.
  • Time-weighted prices: The liquidity(null) method of the CyclesFinance canister is used to query the time-weighted cumulative value of ICPs and Cycles. image

liquidity mining support

CyclesFinance can provide a data resource for liquidity mining contracts as well as a space for innovation in community governance and economic models. The liquidity() method of the CyclesFinance canister is used to query the time-weighted cumulative value of the LP's liquidity pool share. image

Trading mining support

CyclesFinance can provide data resources for trading mining. It can query the cumulative value of trading volume on a global or account basis.

Scalable storage of transaction records

To ensure that CyclesFinance can support large-scale application scenarios, the CyclesFinance canister stores only recent transactions, which are stored persistently via an external scalable canisters.

Usage (Command line interface)

Notes

  • UI interaction interface: http://cycles.finance
  • The basic unit of ICP in canister is e8s, 1 icp = 10^8 e8s;
  • The ICP/Cycles rate on IC network changes dynamically and is pegged to the XDR value, 1 XDR = 10^12 cycles (value approx. 1.4 USD).
  • The ICP/Cycles rate on this canister is automatically formed by the market and may deviate from other markets.
  • Your ICP account principal and Cycles wallet account principal are used to interact with this canister, please note the difference between them.

Query ICP/Cycles price

dfx canister --network ic call 6nmrm-laaaa-aaaak-aacfq-cai liquidity '(null)'

The cycles (or 2_190_693_645) field in the return value is divided by the icpE8s (or 1_180_746_538) field to indicate how many cycles can be exchanged for 1 e8s. This value is multiplied by 10^8 to indicate how many cycles can be exchanged for 1 icp. this is an estimate.

(
  record {
    icpE8s = 48_521_783 : nat;
    vol = record {
      swapIcpVol = 1_740_878 : nat;
      swapCyclesVol = 573_069_740_022 : nat;
    };
    shareWeighted = record {
      updateTime = 1_638_592_854 : nat;
      shareTimeWeighted = 3_894_326_391_123 : nat;
    };
    unitValue = record { 329155.999121 : float64; 0.972376 : float64 };
    shares = 809_508_285 : nat;
    cycles = 266_454_525_225_963 : nat;
    priceWeighted = record {
      updateTime = 1_638_592_854 : nat;
      icpTimeWeighted = 3_800_565_037_457 : nat;
      cyclesTimeWeighted = 1_277_301_377_584_917_917 : nat;
    };
    swapCount = 0 : nat64;
  },
)

ICP to Cycles

Step1: Get your dedicated ICP deposit account-id (DepositAccountId)

dfx canister --network ic call 6nmrm-laaaa-aaaak-aacfq-cai getAccountId '("<your_icp_principal_or_accountid>")'

Return DepositAccountId(example)

("f2d1945ebc293bdc2cc6ef**************e84cf61f51ce6798fc4283") 

Step2: Send ICP to DepositAccountId

dfx ledger --network ic transfer <your_DepositAccountId> --memo 0 --e8s <icp_e8s_amount>

Step3: Converting to Cycles. Parameters icp_e8s_amount is the amount sent in Step2 and your_cycles_wallet_principal is the principal of your cycles wallet (note: not your ICP account principal).

dfx canister --network ic call 6nmrm-laaaa-aaaak-aacfq-cai icpToCycles '(<icp_e8s_amount>:nat, principal "<your_cycles_wallet_principal>", null, null, null)'

Check your wallet balance

dfx wallet --network ic balance

Cycles to ICP

Step1: Use the didc tool to encode the parameters. Note, didc tool resources: https://github.com/dfinity/candid/tree/master/tools/didc

didc encode '("<your_icp_principal_or_accountid>",null,null)' -t '(text,opt nat,opt blob)' -f blob

Return CallArgs(example)

blob "DIDL\02n\01m{\02h\00\01\**************\88\01\e1\18\fd6G\02\00"

Step2: Converting to ICP. The parameter cycles_amount is the amount of cycles you want to convert, and the parameter call_args is the CallArgs got from Step1.

dfx canister --network ic call <your_cycles_wallet_principal> wallet_call '(record {canister=principal "6nmrm-laaaa-aaaak-aacfq-cai"; method_name="cyclesToIcp"; cycles=<cycles_amount>:nat64; args=<call_args>})'

Check your account balance

dfx ledger --network ic balance

Add liquidity

To add liquidity, both ICP and Cycles to be added to the liquidity pool, the proportion is calculated based on the current price and the excess will refunded.

Step1: Get your dedicated ICP deposit account-id(DepositAccountId

dfx canister --network ic call 6nmrm-laaaa-aaaak-aacfq-cai getAccountId '("<your_icp_principal_or_accountid>")'

Return (example)

("f2d1945ebc293bdc2cc6ef**************e84cf61f51ce6798fc4283") 

Step2: Send ICP to DepositAccountId

dfx ledger --network ic transfer <your_DepositAccountId> --memo 0 --e8s <icp_e8s_amount>

Step3: Use the didc tool to encode the parameters. Note, didc tool resources: https://github.com/dfinity/candid/tree/master/tools/didc

didc encode '("<your_icp_principal_or_accountid>",null,null)' -t '(text,opt nat,opt blob)' -f blob

Return CallArgs(example)

blob "DIDL\02n\01m{\02h\00\01\**************\88\01\e1\18\fd6G\02\00"

Step4: Send Cycles, add liquidity. The parameter cycles_amount is the amount of cycles you want to add, and the parameter call_args is the CallArgs got from Step3.

dfx canister --network ic call <your_cycles_wallet_principal> wallet_call '(record {canister=principal "6nmrm-laaaa-aaaak-aacfq-cai"; method_name="add"; cycles=<cycles_amount>:nat64; args=<call_args>})'

Step5: Enquire about liquidity shares

dfx canister --network ic call 6nmrm-laaaa-aaaak-aacfq-cai liquidity '(opt "<your_icp_principal_or_accountid>")'

Return (example). The shares (or 489_381_556) field is the share of liquidity you hold.

(
  record {
    icpE8s = 48_521_783 : nat;
    vol = record {
      swapIcpVol = 1_648_218 : nat;
      swapCyclesVol = 541_650_948_359 : nat;
    };
    shareWeighted = record {
      updateTime = 1_638_528_867 : nat;
      shareTimeWeighted = 695_045_889_662 : nat;
    };
    unitValue = record { 329748.544469 : float64; 0.970629 : float64 };
    shares = 49_990_000 : nat;   
    cycles = 16_484_143_085_896 : nat;
    priceWeighted = record {
      updateTime = 1_638_528_867 : nat;
      icpTimeWeighted = 689_683_291_306 : nat;
      cyclesTimeWeighted = 224_229_508_922_468_505 : nat;
    };
    swapCount = 0 : nat64;
  },
)

Remove liquidity

Step1: Query your liquidity share, the shares (or 489_381_556) field in the return is your share held.

dfx canister --network ic call 6nmrm-laaaa-aaaak-aacfq-cai liquidity '(opt "<your_icp_principal_or_accountid>")'

Step2: Remove liquidity. The parameter share_amount must be equal to or less than the value queried by Step1, and the parameter your_cycles_wallet_principal is wallet principal used to receive the cycles, and caller principal will receive icp.

dfx canister --network ic call 6nmrm-laaaa-aaaak-aacfq-cai remove '(opt <share_amount>:opt nat, principal "<your_cycles_wallet_principal>", null, null, null)'

Check your account balance

dfx ledger --network ic balance
dfx wallet --network ic balance

Claim Rewards

Claim Rewards. The parameter your_cycles_wallet_principal is wallet principal used to receive cycles, and caller principal will receive icp.

dfx canister --network ic call 6nmrm-laaaa-aaaak-aacfq-cai claim '(principal "<your_cycles_wallet_principal>", null, null, null)'

Check your account balance

dfx ledger --network ic balance
dfx wallet --network ic balance

DID

type definite_canister_settings = 
 record {
   compute_allocation: nat;
   controllers: vec principal;
   freezing_threshold: nat;
   memory_allocation: nat;
 };
type canister_status = 
 record {
   cycles: nat;
   memory_size: nat;
   module_hash: opt vec nat8;
   settings: definite_canister_settings;
   status: variant {
             running;
             stopped;
             stopping;
           };
 };
type Vol = 
 record {
   swapCyclesVol: CyclesAmount;
   swapIcpVol: IcpE8s;
 };
type TxnResult = 
 variant {
   err:
    record {
      code:
       variant {
         IcpTransferException;
         InsufficientShares;
         InvalidCyclesAmout;
         InvalidIcpAmout;
         NonceError;
         PoolIsEmpty;
         UnacceptableVolatility;
         UndefinedError;
       };
      message: text;
    };
   ok:
    record {
      cycles: BalanceChange;
      icpE8s: BalanceChange;
      shares: ShareChange;
      txid: Txid;
    };
 };
type TxnRecord = 
 record {
   account: AccountId;
   caller: AccountId;
   cyclesWallet: opt CyclesWallet;
   data: opt Data;
   details:
    vec
     record {
       counterparty: Txid;
       token0Value: BalanceChange;
       token1Value: BalanceChange;
     };
   fee: record {
          token0Fee: nat;
          token1Fee: nat;
        };
   index: nat;
   msgCaller: opt principal;
   nonce: Nonce;
   operation: OperationType;
   orderType: variant {
                AMM;
                OrderBook;
              };
   shares: ShareChange;
   time: Time;
   token0: TokenType;
   token0Value: BalanceChange;
   token1: TokenType;
   token1Value: BalanceChange;
   txid: Txid;
 };
type Txid = blob;
type TransferError = 
 variant {
   BadFee: record {expected_fee: ICP;};
   InsufficientFunds: record {balance: ICP;};
   TxCreatedInFuture;
   TxDuplicate: record {duplicate_of: BlockIndex;};
   TxTooOld: record {allowed_window_nanos: nat64;};
 };
type TransStatus = 
 variant {
   Failure;
   Fallback;
   Processing;
   Success;
 };
type TokenType = 
 variant {
   Cycles;
   Icp;
   Token: principal;
 };
type Timestamp = nat;
type Time = int;
type Shares = nat;
type ShareWeighted = 
 record {
   shareTimeWeighted: nat;
   updateTime: Timestamp;
 };
type ShareChange = 
 variant {
   Burn: Shares;
   Mint: Shares;
   NoChange;
 };
type Sa = vec nat8;
type PriceWeighted = 
 record {
   cyclesTimeWeighted: nat;
   icpTimeWeighted: nat;
   updateTime: Timestamp;
 };
type OperationType = 
 variant {
   AddLiquidity;
   Claim;
   RemoveLiquidity;
   Swap;
 };
type Nonce = nat;
type Liquidity = 
 record {
   cycles: nat;
   icpE8s: IcpE8s;
   priceWeighted: PriceWeighted;
   shareWeighted: ShareWeighted;
   shares: Shares;
   swapCount: nat64;
   unitValue: record {
                float64;
                float64;
              };
   vol: Vol;
 };
type IcpTransferLog = 
 record {
   fee: IcpE8s;
   from: AccountId;
   status: TransStatus;
   to: AccountId;
   updateTime: Timestamp;
   value: IcpE8s;
 };
type IcpE8s = nat;
type ICP = record {e8s: nat64;};
type FeeStatus = 
 record {
   cumulFee: record {
               cyclesBalance: CyclesAmount;
               icpBalance: IcpE8s;
             };
   fee: float64;
   totalFee: record {
               cyclesBalance: CyclesAmount;
               icpBalance: IcpE8s;
             };
 };
type ErrorLog = 
 variant {
   IcpSaToMain:
    record {
      debit: record {
               nat64;
               AccountId;
               IcpE8s;
             };
      errMsg: TransferError;
      time: Timestamp;
      user: AccountId;
    };
   Withdraw:
    record {
      credit: record {
                CyclesWallet;
                CyclesAmount;
                AccountId;
                IcpE8s;
              };
      cyclesErrMsg: text;
      icpErrMsg: opt TransferError;
      time: Timestamp;
      user: AccountId;
    };
 };
type ErrorAction = 
 variant {
   delete;
   fallback;
   resendCycles;
   resendIcp;
   resendIcpCycles;
 };
type Data = blob;
type DRC207Support = 
 record {
   cycles_receivable: bool;
   monitorable_by_blackhole:
    record {
      allowed: bool;
      canister_id: opt principal;
    };
   monitorable_by_self: bool;
   timer: record {
            enable: bool;
            interval_seconds: opt nat;
          };
 };
type CyclesWallet = principal;
type CyclesTransferLog = 
 record {
   from: principal;
   status: TransStatus;
   to: principal;
   updateTime: Timestamp;
   value: CyclesAmount;
 };
type CyclesAmount = nat;
type Config = 
 record {
   CYCLES_LIMIT: opt nat;
   FEE: opt nat;
   ICP_FEE: opt nat64;
   ICP_LIMIT: opt nat;
   MAX_CACHE_NUMBER_PER: opt nat;
   MAX_CACHE_TIME: opt nat;
   MAX_STORAGE_TRIES: opt nat;
   MIN_CYCLES: opt nat;
   MIN_ICP_E8S: opt nat;
   STORAGE_CANISTER: opt text;
   CYCLESFEE_RETENTION_RATE: opt nat;
 };
type BlockIndex = nat64;
type BalanceChange = 
 variant {
   CreditRecord: nat;
   DebitRecord: nat;
   NoChange;
 };
type Address = text;
type AccountId = blob;
type CyclesMarket = service {
   getAccountId: (Address) -> (text) query;
   add: (Address, opt Nonce, opt Data) -> (TxnResult);
   remove: (opt Shares, CyclesWallet, opt Nonce, opt Sa, opt Data) -> (TxnResult);
   cyclesToIcp: (Address, opt Nonce, opt Data) -> (TxnResult);
   icpToCycles: (IcpE8s, CyclesWallet, opt Nonce, opt Sa, opt Data) -> (TxnResult);
   claim: (CyclesWallet, opt Nonce, opt Sa, opt Data) -> (TxnResult);
   count: (opt Address) -> (nat) query;
   canister_status: () -> (canister_status);
   feeStatus: () -> (FeeStatus) query;
   getConfig: () -> (Config) query;
   getEvents: (opt Address) -> (vec TxnRecord) query;
   lastTxids: (opt Address) -> (vec Txid) query;
   liquidity: (opt Address) -> (Liquidity) query;
   lpRewards: (Address) -> (record { cycles: nat; icp: nat;}) query;
   txnRecord: (Txid) -> (opt TxnRecord) query;
   txnRecord2: (Txid) -> (opt TxnRecord);
   yield: () -> (record { apyCycles: float64; apyIcp: float64;}, record {apyCycles: float64; apyIcp: float64;}) query;
   version: () -> (text) query;
};
service : () -> CyclesMarket

Roadmap

Development of UI interface, open source contract code.

(doing) Upgrade to v1.0.

Opening up liquidity mining.

Community

Web: http://cycles.finance/

Github: https://github.com/iclighthouse/Cycles.Finance

Twitter: https://twitter.com/ICLighthouse

Medium: https://medium.com/@ICLighthouse

Discord: https://discord.gg/FQZFGGq7zv