10519 lines
514 KiB
Python
10519 lines
514 KiB
Python
# -*- coding: utf-8 -*-
|
||
|
||
# PLEASE DO NOT EDIT THIS FILE, IT IS GENERATED AND WILL BE OVERWRITTEN:
|
||
# https://github.com/ccxt/ccxt/blob/master/CONTRIBUTING.md#how-to-contribute-code
|
||
|
||
from ccxt.async_support.base.exchange import Exchange
|
||
from ccxt.abstract.bitget import ImplicitAPI
|
||
import asyncio
|
||
import hashlib
|
||
import json
|
||
from ccxt.base.types import Any, Balances, BorrowInterest, Conversion, CrossBorrowRate, Currencies, Currency, DepositAddress, FundingHistory, Int, IsolatedBorrowRate, LedgerEntry, Leverage, LeverageTier, Liquidation, LongShortRatio, MarginMode, MarginModification, Market, Num, Order, OrderBook, OrderRequest, OrderSide, OrderType, Position, Str, Strings, Ticker, Tickers, FundingRate, FundingRates, Trade, TradingFeeInterface, TradingFees, Transaction, TransferEntry
|
||
from typing import List
|
||
from ccxt.base.errors import ExchangeError
|
||
from ccxt.base.errors import AuthenticationError
|
||
from ccxt.base.errors import PermissionDenied
|
||
from ccxt.base.errors import AccountSuspended
|
||
from ccxt.base.errors import ArgumentsRequired
|
||
from ccxt.base.errors import BadRequest
|
||
from ccxt.base.errors import BadSymbol
|
||
from ccxt.base.errors import InsufficientFunds
|
||
from ccxt.base.errors import InvalidAddress
|
||
from ccxt.base.errors import InvalidOrder
|
||
from ccxt.base.errors import OrderNotFound
|
||
from ccxt.base.errors import NotSupported
|
||
from ccxt.base.errors import DDoSProtection
|
||
from ccxt.base.errors import RateLimitExceeded
|
||
from ccxt.base.errors import ExchangeNotAvailable
|
||
from ccxt.base.errors import OnMaintenance
|
||
from ccxt.base.errors import InvalidNonce
|
||
from ccxt.base.errors import RequestTimeout
|
||
from ccxt.base.errors import CancelPending
|
||
from ccxt.base.decimal_to_precision import TICK_SIZE
|
||
from ccxt.base.precise import Precise
|
||
|
||
|
||
class bitget(Exchange, ImplicitAPI):
|
||
|
||
def describe(self) -> Any:
|
||
return self.deep_extend(super(bitget, self).describe(), {
|
||
'id': 'bitget',
|
||
'name': 'Bitget',
|
||
'countries': ['SG'],
|
||
'version': 'v2',
|
||
'rateLimit': 50, # up to 3000 requests per 5 minutes ≈ 600 requests per minute ≈ 10 requests per second ≈ 100 ms
|
||
'certified': True,
|
||
'pro': True,
|
||
'has': {
|
||
'CORS': None,
|
||
'spot': True,
|
||
'margin': True,
|
||
'swap': True,
|
||
'future': True,
|
||
'option': False,
|
||
'addMargin': True,
|
||
'borrowCrossMargin': True,
|
||
'borrowIsolatedMargin': True,
|
||
'cancelAllOrders': True,
|
||
'cancelOrder': True,
|
||
'cancelOrders': True,
|
||
'closeAllPositions': True,
|
||
'closePosition': True,
|
||
'createConvertTrade': True,
|
||
'createDepositAddress': False,
|
||
'createMarketBuyOrderWithCost': True,
|
||
'createMarketOrderWithCost': False,
|
||
'createMarketSellOrderWithCost': False,
|
||
'createOrder': True,
|
||
'createOrders': True,
|
||
'createOrderWithTakeProfitAndStopLoss': True,
|
||
'createPostOnlyOrder': True,
|
||
'createReduceOnlyOrder': False,
|
||
'createStopLimitOrder': True,
|
||
'createStopLossOrder': True,
|
||
'createStopMarketOrder': True,
|
||
'createStopOrder': True,
|
||
'createTakeProfitOrder': True,
|
||
'createTrailingAmountOrder': False,
|
||
'createTrailingPercentOrder': True,
|
||
'createTriggerOrder': True,
|
||
'editOrder': True,
|
||
'fetchAccounts': False,
|
||
'fetchBalance': True,
|
||
'fetchBorrowInterest': True,
|
||
'fetchBorrowRateHistories': False,
|
||
'fetchBorrowRateHistory': False,
|
||
'fetchCanceledAndClosedOrders': True,
|
||
'fetchCanceledOrders': True,
|
||
'fetchClosedOrders': True,
|
||
'fetchConvertCurrencies': True,
|
||
'fetchConvertQuote': True,
|
||
'fetchConvertTrade': False,
|
||
'fetchConvertTradeHistory': True,
|
||
'fetchCrossBorrowRate': True,
|
||
'fetchCrossBorrowRates': False,
|
||
'fetchCurrencies': True,
|
||
'fetchDeposit': False,
|
||
'fetchDepositAddress': True,
|
||
'fetchDepositAddresses': False,
|
||
'fetchDepositAddressesByNetwork': False,
|
||
'fetchDeposits': True,
|
||
'fetchDepositsWithdrawals': False,
|
||
'fetchDepositWithdrawFee': 'emulated',
|
||
'fetchDepositWithdrawFees': True,
|
||
'fetchFundingHistory': True,
|
||
'fetchFundingInterval': True,
|
||
'fetchFundingIntervals': True,
|
||
'fetchFundingRate': True,
|
||
'fetchFundingRateHistory': True,
|
||
'fetchFundingRates': True,
|
||
'fetchIndexOHLCV': True,
|
||
'fetchIsolatedBorrowRate': True,
|
||
'fetchIsolatedBorrowRates': False,
|
||
'fetchLedger': True,
|
||
'fetchLeverage': True,
|
||
'fetchLeverageTiers': False,
|
||
'fetchLiquidations': False,
|
||
'fetchLongShortRatio': False,
|
||
'fetchLongShortRatioHistory': True,
|
||
'fetchMarginAdjustmentHistory': False,
|
||
'fetchMarginMode': True,
|
||
'fetchMarketLeverageTiers': True,
|
||
'fetchMarkets': True,
|
||
'fetchMarkOHLCV': True,
|
||
'fetchMarkPrice': True,
|
||
'fetchMyLiquidations': True,
|
||
'fetchMyTrades': True,
|
||
'fetchOHLCV': True,
|
||
'fetchOpenInterest': True,
|
||
'fetchOpenInterestHistory': False,
|
||
'fetchOpenOrders': True,
|
||
'fetchOrder': True,
|
||
'fetchOrderBook': True,
|
||
'fetchOrderBooks': False,
|
||
'fetchOrders': False,
|
||
'fetchOrderTrades': False,
|
||
'fetchPosition': True,
|
||
'fetchPositionHistory': 'emulated',
|
||
'fetchPositionMode': False,
|
||
'fetchPositions': True,
|
||
'fetchPositionsHistory': True,
|
||
'fetchPositionsRisk': False,
|
||
'fetchPremiumIndexOHLCV': False,
|
||
'fetchStatus': False,
|
||
'fetchTicker': True,
|
||
'fetchTickers': True,
|
||
'fetchTime': True,
|
||
'fetchTrades': True,
|
||
'fetchTradingFee': True,
|
||
'fetchTradingFees': True,
|
||
'fetchTransactions': False,
|
||
'fetchTransfer': False,
|
||
'fetchTransfers': True,
|
||
'fetchWithdrawAddresses': False,
|
||
'fetchWithdrawal': False,
|
||
'fetchWithdrawals': True,
|
||
'reduceMargin': True,
|
||
'repayCrossMargin': True,
|
||
'repayIsolatedMargin': True,
|
||
'setLeverage': True,
|
||
'setMargin': False,
|
||
'setMarginMode': True,
|
||
'setPositionMode': True,
|
||
'signIn': False,
|
||
'transfer': True,
|
||
'withdraw': True,
|
||
},
|
||
'timeframes': {
|
||
'1m': '1m',
|
||
'3m': '3m',
|
||
'5m': '5m',
|
||
'15m': '15m',
|
||
'30m': '30m',
|
||
'1h': '1h',
|
||
'2h': '2h',
|
||
'4h': '4h',
|
||
'6h': '6h',
|
||
'12h': '12h',
|
||
'1d': '1d',
|
||
'3d': '3d',
|
||
'1w': '1w',
|
||
'1M': '1m',
|
||
},
|
||
'hostname': 'bitget.com',
|
||
'urls': {
|
||
'logo': 'https://github.com/user-attachments/assets/fbaa10cc-a277-441d-a5b7-997dd9a87658',
|
||
'api': {
|
||
'spot': 'https://api.{hostname}',
|
||
'mix': 'https://api.{hostname}',
|
||
'user': 'https://api.{hostname}',
|
||
'p2p': 'https://api.{hostname}',
|
||
'broker': 'https://api.{hostname}',
|
||
'margin': 'https://api.{hostname}',
|
||
'common': 'https://api.{hostname}',
|
||
'tax': 'https://api.{hostname}',
|
||
'convert': 'https://api.{hostname}',
|
||
'copy': 'https://api.{hostname}',
|
||
'earn': 'https://api.{hostname}',
|
||
'uta': 'https://api.{hostname}',
|
||
},
|
||
'www': 'https://www.bitget.com',
|
||
'doc': [
|
||
'https://www.bitget.com/api-doc/common/intro',
|
||
'https://www.bitget.com/api-doc/spot/intro',
|
||
'https://www.bitget.com/api-doc/contract/intro',
|
||
'https://www.bitget.com/api-doc/broker/intro',
|
||
'https://www.bitget.com/api-doc/margin/intro',
|
||
'https://www.bitget.com/api-doc/copytrading/intro',
|
||
'https://www.bitget.com/api-doc/earn/intro',
|
||
'https://bitgetlimited.github.io/apidoc/en/mix',
|
||
'https://bitgetlimited.github.io/apidoc/en/spot',
|
||
'https://bitgetlimited.github.io/apidoc/en/broker',
|
||
'https://bitgetlimited.github.io/apidoc/en/margin',
|
||
],
|
||
'fees': 'https://www.bitget.cc/zh-CN/rate?tab=1',
|
||
'referral': 'https://www.bitget.com/expressly?languageType=0&channelCode=ccxt&vipCode=tg9j',
|
||
},
|
||
'api': {
|
||
'public': {
|
||
'common': {
|
||
'get': {
|
||
'v2/public/annoucements': 1,
|
||
'v2/public/time': 1,
|
||
},
|
||
},
|
||
'spot': {
|
||
'get': {
|
||
'spot/v1/notice/queryAllNotices': 1, # 20 times/1s(IP) => 20/20 = 1
|
||
'spot/v1/public/time': 1,
|
||
'spot/v1/public/currencies': 6.6667, # 3 times/1s(IP) => 20/3 = 6.6667
|
||
'spot/v1/public/products': 1,
|
||
'spot/v1/public/product': 1,
|
||
'spot/v1/market/ticker': 1,
|
||
'spot/v1/market/tickers': 1,
|
||
'spot/v1/market/fills': 2, # 10 times/1s(IP) => 20/10 = 2
|
||
'spot/v1/market/fills-history': 2,
|
||
'spot/v1/market/candles': 1,
|
||
'spot/v1/market/depth': 1,
|
||
'spot/v1/market/spot-vip-level': 2,
|
||
'spot/v1/market/merge-depth': 1,
|
||
'spot/v1/market/history-candles': 1,
|
||
'spot/v1/public/loan/coinInfos': 2, # 10 times/1s(IP) => 20/10 = 2
|
||
'spot/v1/public/loan/hour-interest': 2, # 10 times/1s(IP) => 20/10 = 2
|
||
'v2/spot/public/coins': 6.6667,
|
||
'v2/spot/public/symbols': 1,
|
||
'v2/spot/market/vip-fee-rate': 2,
|
||
'v2/spot/market/tickers': 1,
|
||
'v2/spot/market/merge-depth': 1,
|
||
'v2/spot/market/orderbook': 1,
|
||
'v2/spot/market/candles': 1,
|
||
'v2/spot/market/history-candles': 1,
|
||
'v2/spot/market/fills': 2,
|
||
'v2/spot/market/fills-history': 2,
|
||
},
|
||
},
|
||
'mix': {
|
||
'get': {
|
||
'mix/v1/market/contracts': 1,
|
||
'mix/v1/market/depth': 1,
|
||
'mix/v1/market/ticker': 1,
|
||
'mix/v1/market/tickers': 1,
|
||
'mix/v1/market/contract-vip-level': 2,
|
||
'mix/v1/market/fills': 1,
|
||
'mix/v1/market/fills-history': 2,
|
||
'mix/v1/market/candles': 1,
|
||
'mix/v1/market/index': 1,
|
||
'mix/v1/market/funding-time': 1,
|
||
'mix/v1/market/history-fundRate': 1,
|
||
'mix/v1/market/current-fundRate': 1,
|
||
'mix/v1/market/open-interest': 1,
|
||
'mix/v1/market/mark-price': 1,
|
||
'mix/v1/market/symbol-leverage': 1,
|
||
'mix/v1/market/queryPositionLever': 1,
|
||
'mix/v1/market/open-limit': 1,
|
||
'mix/v1/market/history-candles': 1,
|
||
'mix/v1/market/history-index-candles': 1,
|
||
'mix/v1/market/history-mark-candles': 1,
|
||
'mix/v1/market/merge-depth': 1,
|
||
'v2/mix/market/vip-fee-rate': 2,
|
||
'v2/mix/market/merge-depth': 1,
|
||
'v2/mix/market/ticker': 1,
|
||
'v2/mix/market/tickers': 1,
|
||
'v2/mix/market/fills': 1,
|
||
'v2/mix/market/fills-history': 2,
|
||
'v2/mix/market/candles': 1,
|
||
'v2/mix/market/history-candles': 1,
|
||
'v2/mix/market/history-index-candles': 1,
|
||
'v2/mix/market/history-mark-candles': 1,
|
||
'v2/mix/market/open-interest': 1,
|
||
'v2/mix/market/funding-time': 1,
|
||
'v2/mix/market/symbol-price': 1,
|
||
'v2/mix/market/history-fund-rate': 1,
|
||
'v2/mix/market/current-fund-rate': 1,
|
||
'v2/mix/market/contracts': 1,
|
||
'v2/mix/market/query-position-lever': 2,
|
||
'v2/mix/market/account-long-short': 20,
|
||
},
|
||
},
|
||
'margin': {
|
||
'get': {
|
||
'margin/v1/cross/public/interestRateAndLimit': 2, # 10 times/1s(IP) => 20/10 = 2
|
||
'margin/v1/isolated/public/interestRateAndLimit': 2, # 10 times/1s(IP) => 20/10 = 2
|
||
'margin/v1/cross/public/tierData': 2, # 10 times/1s(IP) => 20/10 = 2
|
||
'margin/v1/isolated/public/tierData': 2, # 10 times/1s(IP) => 20/10 = 2
|
||
'margin/v1/public/currencies': 1, # 20 times/1s(IP) => 20/20 = 1
|
||
'v2/margin/currencies': 2,
|
||
'v2/margin/market/long-short-ratio': 20,
|
||
},
|
||
},
|
||
'earn': {
|
||
'get': {
|
||
'v2/earn/loan/public/coinInfos': 2,
|
||
'v2/earn/loan/public/hour-interest': 2,
|
||
},
|
||
},
|
||
'uta': {
|
||
'get': {
|
||
'v3/market/instruments': 1,
|
||
'v3/market/tickers': 1,
|
||
'v3/market/orderbook': 1,
|
||
'v3/market/fills': 1,
|
||
'v3/market/open-interest': 1,
|
||
'v3/market/candles': 1,
|
||
'v3/market/history-candles': 1,
|
||
'v3/market/current-fund-rate': 1,
|
||
'v3/market/history-fund-rate': 1,
|
||
'v3/market/risk-reserve': 1,
|
||
'v3/market/discount-rate': 1,
|
||
'v3/market/margin-loans': 1,
|
||
'v3/market/position-tier': 1,
|
||
'v3/market/oi-limit': 2,
|
||
},
|
||
},
|
||
},
|
||
'private': {
|
||
'spot': {
|
||
'get': {
|
||
'spot/v1/wallet/deposit-address': 4,
|
||
'spot/v1/wallet/withdrawal-list': 1,
|
||
'spot/v1/wallet/deposit-list': 1,
|
||
'spot/v1/account/getInfo': 20,
|
||
'spot/v1/account/assets': 2,
|
||
'spot/v1/account/assets-lite': 2, # 10 times/1s(UID) => 20/10 = 2
|
||
'spot/v1/account/transferRecords': 1, # 20 times/1s(UID) => 20/20 = 1
|
||
'spot/v1/convert/currencies': 2,
|
||
'spot/v1/convert/convert-record': 2,
|
||
'spot/v1/loan/ongoing-orders': 2, # 10 times/1s(UID) => 20/10 = 2
|
||
'spot/v1/loan/repay-history': 2, # 10 times/1s(UID) => 20/10 = 2
|
||
'spot/v1/loan/revise-history': 2, # 10 times/1s(UID) => 20/10 = 2
|
||
'spot/v1/loan/borrow-history': 2, # 10 times/1s(UID) => 20/10 = 2
|
||
'spot/v1/loan/debts': 2, # 10 times/1s(UID) => 20/10 = 2
|
||
'v2/spot/trade/orderInfo': 1,
|
||
'v2/spot/trade/unfilled-orders': 1,
|
||
'v2/spot/trade/history-orders': 1,
|
||
'v2/spot/trade/fills': 2,
|
||
'v2/spot/trade/current-plan-order': 1,
|
||
'v2/spot/trade/history-plan-order': 1,
|
||
'v2/spot/account/info': 20,
|
||
'v2/spot/account/assets': 2,
|
||
'v2/spot/account/subaccount-assets': 2,
|
||
'v2/spot/account/bills': 2,
|
||
'v2/spot/account/transferRecords': 1,
|
||
'v2/account/funding-assets': 2,
|
||
'v2/account/bot-assets': 2,
|
||
'v2/account/all-account-balance': 20,
|
||
'v2/spot/wallet/deposit-address': 2,
|
||
'v2/spot/wallet/deposit-records': 2,
|
||
'v2/spot/wallet/withdrawal-records': 2,
|
||
},
|
||
'post': {
|
||
'spot/v1/wallet/transfer': 4,
|
||
'spot/v1/wallet/transfer-v2': 4,
|
||
'spot/v1/wallet/subTransfer': 10,
|
||
'spot/v1/wallet/withdrawal': 4,
|
||
'spot/v1/wallet/withdrawal-v2': 4,
|
||
'spot/v1/wallet/withdrawal-inner': 4,
|
||
'spot/v1/wallet/withdrawal-inner-v2': 4,
|
||
'spot/v1/account/sub-account-spot-assets': 200,
|
||
'spot/v1/account/bills': 2,
|
||
'spot/v1/trade/orders': 2,
|
||
'spot/v1/trade/batch-orders': 4,
|
||
'spot/v1/trade/cancel-order': 2,
|
||
'spot/v1/trade/cancel-order-v2': 2,
|
||
'spot/v1/trade/cancel-symbol-order': 2,
|
||
'spot/v1/trade/cancel-batch-orders': 4,
|
||
'spot/v1/trade/cancel-batch-orders-v2': 4,
|
||
'spot/v1/trade/orderInfo': 1,
|
||
'spot/v1/trade/open-orders': 1,
|
||
'spot/v1/trade/history': 1,
|
||
'spot/v1/trade/fills': 1,
|
||
'spot/v1/plan/placePlan': 1,
|
||
'spot/v1/plan/modifyPlan': 1,
|
||
'spot/v1/plan/cancelPlan': 1,
|
||
'spot/v1/plan/currentPlan': 1,
|
||
'spot/v1/plan/historyPlan': 1,
|
||
'spot/v1/plan/batchCancelPlan': 2, # 10 times/1s(UID) => 20/10 = 2
|
||
'spot/v1/convert/quoted-price': 4,
|
||
'spot/v1/convert/trade': 4,
|
||
'spot/v1/loan/borrow': 2, # 10 times/1s(UID) => 20/10 = 2
|
||
'spot/v1/loan/repay': 2, # 10 times/1s(UID) => 20/10 = 2
|
||
'spot/v1/loan/revise-pledge': 2, # 10 times/1s(UID) => 20/10 = 2
|
||
'spot/v1/trace/order/orderCurrentList': 2, # 10 times/1s(UID) => 20/10 = 2
|
||
'spot/v1/trace/order/orderHistoryList': 2, # 10 times/1s(UID) => 20/10 = 2
|
||
'spot/v1/trace/order/closeTrackingOrder': 2, # 10 times/1s(UID) => 20/10 = 2
|
||
'spot/v1/trace/order/updateTpsl': 2, # 10 times/1s(UID) => 20/10 = 2
|
||
'spot/v1/trace/order/followerEndOrder': 2, # 10 times/1s(UID) => 20/10 = 2
|
||
'spot/v1/trace/order/spotInfoList': 2, # 10 times/1s(UID) => 20/10 = 2
|
||
'spot/v1/trace/config/getTraderSettings': 2, # 10 times/1s(UID) => 20/10 = 2
|
||
'spot/v1/trace/config/getFollowerSettings': 2, # 10 times/1s(UID) => 20/10 = 2
|
||
'spot/v1/trace/user/myTraders': 2, # 10 times/1s(UID) => 20/10 = 2
|
||
'spot/v1/trace/config/setFollowerConfig': 2, # 10 times/1s(UID) => 20/10 = 2
|
||
'spot/v1/trace/user/myFollowers': 2, # 10 times/1s(UID) => 20/10 = 2
|
||
'spot/v1/trace/config/setProductCode': 2, # 10 times/1s(UID) => 20/10 = 2
|
||
'spot/v1/trace/user/removeTrader': 2, # 10 times/1s(UID) => 20/10 = 2
|
||
'spot/v1/trace/getRemovableFollower': 2,
|
||
'spot/v1/trace/user/removeFollower': 2,
|
||
'spot/v1/trace/profit/totalProfitInfo': 2, # 10 times/1s(UID) => 20/10 = 2
|
||
'spot/v1/trace/profit/totalProfitList': 2, # 10 times/1s(UID) => 20/10 = 2
|
||
'spot/v1/trace/profit/profitHisList': 2, # 10 times/1s(UID) => 20/10 = 2
|
||
'spot/v1/trace/profit/profitHisDetailList': 2, # 10 times/1s(UID) => 20/10 = 2
|
||
'spot/v1/trace/profit/waitProfitDetailList': 2, # 10 times/1s(UID) => 20/10 = 2
|
||
'spot/v1/trace/user/getTraderInfo': 2, # 10 times/1s(UID) => 20/10 = 2
|
||
'v2/spot/trade/place-order': 2,
|
||
'v2/spot/trade/cancel-order': 2,
|
||
'v2/spot/trade/batch-orders': 20,
|
||
'v2/spot/trade/batch-cancel-order': 2,
|
||
'v2/spot/trade/cancel-symbol-order': 4,
|
||
'v2/spot/trade/place-plan-order': 1,
|
||
'v2/spot/trade/modify-plan-order': 1,
|
||
'v2/spot/trade/cancel-plan-order': 1,
|
||
'v2/spot/trade/cancel-replace-order': 2,
|
||
'v2/spot/trade/batch-cancel-plan-order': 2,
|
||
'v2/spot/wallet/transfer': 2,
|
||
'v2/spot/wallet/subaccount-transfer': 2,
|
||
'v2/spot/wallet/withdrawal': 2,
|
||
'v2/spot/wallet/cancel-withdrawal': 2,
|
||
'v2/spot/wallet/modify-deposit-account': 2,
|
||
},
|
||
},
|
||
'mix': {
|
||
'get': {
|
||
'mix/v1/account/account': 2,
|
||
'mix/v1/account/accounts': 2,
|
||
'mix/v1/position/singlePosition': 2,
|
||
'mix/v1/position/singlePosition-v2': 2,
|
||
'mix/v1/position/allPosition': 4, # 5 times/1s(UID) => 20/5 = 4
|
||
'mix/v1/position/allPosition-v2': 4, # 5 times/1s(UID) => 20/5 = 4
|
||
'mix/v1/position/history-position': 1,
|
||
'mix/v1/account/accountBill': 2,
|
||
'mix/v1/account/accountBusinessBill': 4,
|
||
'mix/v1/order/current': 1, # 20 times/1s(UID) => 20/20 = 1
|
||
'mix/v1/order/marginCoinCurrent': 1, # 20 times/1s(UID) => 20/20 = 1
|
||
'mix/v1/order/history': 2,
|
||
'mix/v1/order/historyProductType': 4, # 5 times/1s(UID) => 20/5 = 4
|
||
'mix/v1/order/detail': 2,
|
||
'mix/v1/order/fills': 2,
|
||
'mix/v1/order/allFills': 2,
|
||
'mix/v1/plan/currentPlan': 1, # 20 times/1s(UID) => 20/20 = 1
|
||
'mix/v1/plan/historyPlan': 2,
|
||
'mix/v1/trace/currentTrack': 2,
|
||
'mix/v1/trace/followerOrder': 2,
|
||
'mix/v1/trace/followerHistoryOrders': 2,
|
||
'mix/v1/trace/historyTrack': 2,
|
||
'mix/v1/trace/summary': 1, # 20 times/1s(UID) => 20/20 = 1
|
||
'mix/v1/trace/profitSettleTokenIdGroup': 1, # 20 times/1s(UID) => 20/20 = 1
|
||
'mix/v1/trace/profitDateGroupList': 1, # 20 times/1s(UID) => 20/20 = 1
|
||
'mix/v1/trade/profitDateList': 2,
|
||
'mix/v1/trace/waitProfitDateList': 1, # 20 times/1s(UID) => 20/20 = 1
|
||
'mix/v1/trace/traderSymbols': 1, # 20 times/1s(UID) => 20/20 = 1
|
||
'mix/v1/trace/traderList': 2,
|
||
'mix/v1/trace/traderDetail': 2, # 10 times/1s(UID) => 20/10 = 2
|
||
'mix/v1/trace/queryTraceConfig': 2,
|
||
'v2/mix/account/account': 2,
|
||
'v2/mix/account/accounts': 2,
|
||
'v2/mix/account/sub-account-assets': 200,
|
||
'v2/mix/account/open-count': 2,
|
||
'v2/mix/account/bill': 2,
|
||
'v2/mix/market/query-position-lever': 2,
|
||
'v2/mix/position/single-position': 2,
|
||
'v2/mix/position/all-position': 4,
|
||
'v2/mix/position/history-position': 1,
|
||
'v2/mix/order/detail': 2,
|
||
'v2/mix/order/fills': 2,
|
||
'v2/mix/order/fill-history': 2,
|
||
'v2/mix/order/orders-pending': 2,
|
||
'v2/mix/order/orders-history': 2,
|
||
'v2/mix/order/orders-plan-pending': 2,
|
||
'v2/mix/order/orders-plan-history': 2,
|
||
'v2/mix/market/position-long-short': 20,
|
||
},
|
||
'post': {
|
||
'mix/v1/account/sub-account-contract-assets': 200, # 0.1 times/1s(UID) => 20/0.1 = 200
|
||
'mix/v1/account/open-count': 1,
|
||
'mix/v1/account/setLeverage': 4, # 5 times/1s(UID) => 20/5 = 4
|
||
'mix/v1/account/setMargin': 4, # 5 times/1s(UID) => 20/5 = 4
|
||
'mix/v1/account/setMarginMode': 4, # 5 times/1s(UID) => 20/5 = 4
|
||
'mix/v1/account/setPositionMode': 4, # 5 times/1s(UID) => 20/5 = 4
|
||
'mix/v1/order/placeOrder': 2,
|
||
'mix/v1/order/batch-orders': 2,
|
||
'mix/v1/order/cancel-order': 2,
|
||
'mix/v1/order/cancel-batch-orders': 2,
|
||
'mix/v1/order/modifyOrder': 2, # 10 times/1s(UID) => 20/10 = 2
|
||
'mix/v1/order/cancel-symbol-orders': 2,
|
||
'mix/v1/order/cancel-all-orders': 2,
|
||
'mix/v1/order/close-all-positions': 20,
|
||
'mix/v1/plan/placePlan': 2,
|
||
'mix/v1/plan/modifyPlan': 2,
|
||
'mix/v1/plan/modifyPlanPreset': 2,
|
||
'mix/v1/plan/placeTPSL': 2,
|
||
'mix/v1/plan/placeTrailStop': 2,
|
||
'mix/v1/plan/placePositionsTPSL': 2,
|
||
'mix/v1/plan/modifyTPSLPlan': 2,
|
||
'mix/v1/plan/cancelPlan': 2,
|
||
'mix/v1/plan/cancelSymbolPlan': 2,
|
||
'mix/v1/plan/cancelAllPlan': 2,
|
||
'mix/v1/trace/closeTrackOrder': 2,
|
||
'mix/v1/trace/modifyTPSL': 2, # 10 times/1s(UID) => 20/10 = 2
|
||
'mix/v1/trace/closeTrackOrderBySymbol': 2,
|
||
'mix/v1/trace/setUpCopySymbols': 2,
|
||
'mix/v1/trace/followerSetBatchTraceConfig': 2,
|
||
'mix/v1/trace/followerCloseByTrackingNo': 2,
|
||
'mix/v1/trace/followerCloseByAll': 2,
|
||
'mix/v1/trace/followerSetTpsl': 2,
|
||
'mix/v1/trace/cancelCopyTrader': 4, # 5 times/1s(UID) => 20/5 = 4
|
||
'mix/v1/trace/traderUpdateConfig': 2, # 10 times/1s(UID) => 20/10 = 2
|
||
'mix/v1/trace/myTraderList': 2, # 10 times/1s(UID) => 20/10 = 2
|
||
'mix/v1/trace/myFollowerList': 2, # 10 times/1s(UID) => 20/10 = 2
|
||
'mix/v1/trace/removeFollower': 2, # 10 times/1s(UID) => 20/10 = 2
|
||
'mix/v1/trace/public/getFollowerConfig': 2, # 10 times/1s(UID) => 20/10 = 2
|
||
'mix/v1/trace/report/order/historyList': 2, # 10 times/1s(IP) => 20/10 = 2
|
||
'mix/v1/trace/report/order/currentList': 2, # 10 times/1s(IP) => 20/10 = 2
|
||
'mix/v1/trace/queryTraderTpslRatioConfig': 2, # 10 times/1s(UID) => 20/10 = 2
|
||
'mix/v1/trace/traderUpdateTpslRatioConfig': 2, # 10 times/1s(UID) => 20/10 = 2
|
||
'v2/mix/account/set-leverage': 4,
|
||
'v2/mix/account/set-margin': 4,
|
||
'v2/mix/account/set-margin-mode': 4,
|
||
'v2/mix/account/set-position-mode': 4,
|
||
'v2/mix/order/place-order': 2,
|
||
'v2/mix/order/click-backhand': 20,
|
||
'v2/mix/order/batch-place-order': 20,
|
||
'v2/mix/order/modify-order': 2,
|
||
'v2/mix/order/cancel-order': 2,
|
||
'v2/mix/order/batch-cancel-orders': 2,
|
||
'v2/mix/order/close-positions': 20,
|
||
'v2/mix/order/place-tpsl-order': 2,
|
||
'v2/mix/order/place-plan-order': 2,
|
||
'v2/mix/order/modify-tpsl-order': 2,
|
||
'v2/mix/order/modify-plan-order': 2,
|
||
'v2/mix/order/cancel-plan-order': 2,
|
||
},
|
||
},
|
||
'user': {
|
||
'get': {
|
||
'user/v1/fee/query': 2,
|
||
'user/v1/sub/virtual-list': 2,
|
||
'user/v1/sub/virtual-api-list': 2,
|
||
'user/v1/tax/spot-record': 1,
|
||
'user/v1/tax/future-record': 1,
|
||
'user/v1/tax/margin-record': 1,
|
||
'user/v1/tax/p2p-record': 1,
|
||
'v2/user/virtual-subaccount-list': 2,
|
||
'v2/user/virtual-subaccount-apikey-list': 2,
|
||
},
|
||
'post': {
|
||
'user/v1/sub/virtual-create': 4,
|
||
'user/v1/sub/virtual-modify': 4,
|
||
'user/v1/sub/virtual-api-batch-create': 20, # 1 times/1s(UID) => 20/1 = 20
|
||
'user/v1/sub/virtual-api-create': 4,
|
||
'user/v1/sub/virtual-api-modify': 4,
|
||
'v2/user/create-virtual-subaccount': 4,
|
||
'v2/user/modify-virtual-subaccount': 4,
|
||
'v2/user/batch-create-subaccount-and-apikey': 20,
|
||
'v2/user/create-virtual-subaccount-apikey': 4,
|
||
'v2/user/modify-virtual-subaccount-apikey': 4,
|
||
},
|
||
},
|
||
'p2p': {
|
||
'get': {
|
||
'p2p/v1/merchant/merchantList': 2, # 10 times/1s(UID) => 20/10 = 2
|
||
'p2p/v1/merchant/merchantInfo': 2, # 10 times/1s(UID) => 20/10 = 2
|
||
'p2p/v1/merchant/advList': 2, # 10 times/1s(UID) => 20/10 = 2
|
||
'p2p/v1/merchant/orderList': 2, # 10 times/1s(UID) => 20/10 = 2
|
||
'v2/p2p/merchantList': 2,
|
||
'v2/p2p/merchantInfo': 2,
|
||
'v2/p2p/orderList': 2,
|
||
'v2/p2p/advList': 2,
|
||
},
|
||
},
|
||
'broker': {
|
||
'get': {
|
||
'broker/v1/account/info': 2, # 10 times/1s(UID) => 20/10 = 2
|
||
'broker/v1/account/sub-list': 20, # 1 times/1s(UID) => 20/1 = 20
|
||
'broker/v1/account/sub-email': 20, # 1 times/1s(UID) => 20/1 = 20
|
||
'broker/v1/account/sub-spot-assets': 2, # 10 times/1s(UID) => 20/10 = 2
|
||
'broker/v1/account/sub-future-assets': 2, # 10 times/1s(UID) => 20/10 = 2
|
||
'broker/v1/account/subaccount-transfer': 1, # unknown
|
||
'broker/v1/account/subaccount-deposit': 1, # unknown
|
||
'broker/v1/account/subaccount-withdrawal': 1, # unknown
|
||
'broker/v1/account/sub-api-list': 2, # 10 times/1s(UID) => 20/10 = 2
|
||
'v2/broker/account/info': 2,
|
||
'v2/broker/account/subaccount-list': 20,
|
||
'v2/broker/account/subaccount-email': 2,
|
||
'v2/broker/account/subaccount-spot-assets': 2,
|
||
'v2/broker/account/subaccount-future-assets': 2,
|
||
'v2/broker/manage/subaccount-apikey-list': 2,
|
||
},
|
||
'post': {
|
||
'broker/v1/account/sub-create': 20, # 1 times/1s(UID) => 20/1 = 20
|
||
'broker/v1/account/sub-modify': 20, # 1 times/1s(UID) => 20/1 = 20
|
||
'broker/v1/account/sub-modify-email': 20, # 1 times/1s(UID) => 20/1 = 20
|
||
'broker/v1/account/sub-address': 2, # 10 times/1s(UID) => 20/10 = 2
|
||
'broker/v1/account/sub-withdrawal': 2, # 10 times/1s(UID) => 20/10 = 2
|
||
'broker/v1/account/sub-auto-transfer': 4, # 5 times/1s(UID) => 20/5 = 4
|
||
'broker/v1/account/sub-api-create': 2, # 10 times/1s(UID) => 20/10 = 2
|
||
'broker/v1/account/sub-api-modify': 2, # 10 times/1s(UID) => 20/10 = 2
|
||
'v2/broker/account/modify-subaccount-email': 2,
|
||
'v2/broker/account/create-subaccount': 20,
|
||
'v2/broker/account/modify-subaccount': 20,
|
||
'v2/broker/account/subaccount-address': 2,
|
||
'v2/broker/account/subaccount-withdrawal': 2,
|
||
'v2/broker/account/set-subaccount-autotransfer': 2,
|
||
'v2/broker/manage/create-subaccount-apikey': 2,
|
||
'v2/broker/manage/modify-subaccount-apikey': 2,
|
||
},
|
||
},
|
||
'margin': {
|
||
'get': {
|
||
'margin/v1/cross/account/riskRate': 2, # 10 times/1s(UID) => 20/10 = 2
|
||
'margin/v1/cross/account/maxTransferOutAmount': 2, # 10 times/1s(UID) => 20/10 = 2
|
||
'margin/v1/isolated/account/maxTransferOutAmount': 2, # 10 times/1s(UID) => 20/10 = 2
|
||
'margin/v1/isolated/order/openOrders': 2, # 10 times/1s(UID) => 20/10 = 2
|
||
'margin/v1/isolated/order/history': 2, # 10 times/1s(UID) => 20/10 = 2
|
||
'margin/v1/isolated/order/fills': 2, # 10 times/1s(UID) => 20/10 = 2
|
||
'margin/v1/isolated/loan/list': 2, # 10 times/1s(UID) => 20/10 = 2
|
||
'margin/v1/isolated/repay/list': 2, # 10 times/1s(UID) => 20/10 = 2
|
||
'margin/v1/isolated/interest/list': 2, # 10 times/1s(UID) => 20/10 = 2
|
||
'margin/v1/isolated/liquidation/list': 2, # 10 times/1s(UID) => 20/10 = 2
|
||
'margin/v1/isolated/fin/list': 2, # 10 times/1s(UID) => 20/10 = 2
|
||
'margin/v1/cross/order/openOrders': 2, # 10 times/1s(UID) => 20/10 = 2
|
||
'margin/v1/cross/order/history': 2, # 10 times/1s(UID) => 20/10 = 2
|
||
'margin/v1/cross/order/fills': 2, # 10 times/1s(UID) => 20/10 = 2
|
||
'margin/v1/cross/loan/list': 2, # 10 times/1s(UID) => 20/10 = 2
|
||
'margin/v1/cross/repay/list': 2, # 10 times/1s(UID) => 20/10 = 2
|
||
'margin/v1/cross/interest/list': 2, # 10 times/1s(UID) => 20/10 = 2
|
||
'margin/v1/cross/liquidation/list': 2, # 10 times/1s(UID) => 20/10 = 2
|
||
'margin/v1/cross/fin/list': 2, # 10 times/1s(UID) => 20/10 = 2
|
||
'margin/v1/cross/account/assets': 2, # 10 times/1s(IP) => 20/10 = 2
|
||
'margin/v1/isolated/account/assets': 2, # 10 times/1s(IP) => 20/10 = 2
|
||
'v2/margin/crossed/borrow-history': 2,
|
||
'v2/margin/crossed/repay-history': 2,
|
||
'v2/margin/crossed/interest-history': 2,
|
||
'v2/margin/crossed/liquidation-history': 2,
|
||
'v2/margin/crossed/financial-records': 2,
|
||
'v2/margin/crossed/account/assets': 2,
|
||
'v2/margin/crossed/account/risk-rate': 2,
|
||
'v2/margin/crossed/account/max-borrowable-amount': 2,
|
||
'v2/margin/crossed/account/max-transfer-out-amount': 2,
|
||
'v2/margin/crossed/interest-rate-and-limit': 2,
|
||
'v2/margin/crossed/tier-data': 2,
|
||
'v2/margin/crossed/open-orders': 2,
|
||
'v2/margin/crossed/history-orders': 2,
|
||
'v2/margin/crossed/fills': 2,
|
||
'v2/margin/isolated/borrow-history': 2,
|
||
'v2/margin/isolated/repay-history': 2,
|
||
'v2/margin/isolated/interest-history': 2,
|
||
'v2/margin/isolated/liquidation-history': 2,
|
||
'v2/margin/isolated/financial-records': 2,
|
||
'v2/margin/isolated/account/assets': 2,
|
||
'v2/margin/isolated/account/risk-rate': 2,
|
||
'v2/margin/isolated/account/max-borrowable-amount': 2,
|
||
'v2/margin/isolated/account/max-transfer-out-amount': 2,
|
||
'v2/margin/isolated/interest-rate-and-limit': 2,
|
||
'v2/margin/isolated/tier-data': 2,
|
||
'v2/margin/isolated/open-orders': 2,
|
||
'v2/margin/isolated/history-orders': 2,
|
||
'v2/margin/isolated/fills': 2,
|
||
},
|
||
'post': {
|
||
'margin/v1/cross/account/borrow': 2, # 10 times/1s(UID) => 20/10 = 2
|
||
'margin/v1/isolated/account/borrow': 2, # 10 times/1s(UID) => 20/10 = 2
|
||
'margin/v1/cross/account/repay': 2, # 10 times/1s(UID) => 20/10 = 2
|
||
'margin/v1/isolated/account/repay': 2, # 10 times/1s(UID) => 20/10 = 2
|
||
'margin/v1/isolated/account/riskRate': 2, # 10 times/1s(UID) => 20/10 = 2
|
||
'margin/v1/cross/account/maxBorrowableAmount': 2, # 10 times/1s(UID) => 20/10 = 2
|
||
'margin/v1/isolated/account/maxBorrowableAmount': 2, # 10 times/1s(UID) => 20/10 = 2
|
||
'margin/v1/isolated/account/flashRepay': 2, # 10 times/1s(UID) => 20/10 = 2
|
||
'margin/v1/isolated/account/queryFlashRepayStatus': 2, # 10 times/1s(UID) => 20/10 = 2
|
||
'margin/v1/cross/account/flashRepay': 2, # 10 times/1s(UID) => 20/10 = 2
|
||
'margin/v1/cross/account/queryFlashRepayStatus': 2, # 10 times/1s(UID) => 20/10 = 2
|
||
'margin/v1/isolated/order/placeOrder': 4, # 5 times/1s(UID) => 20/5 = 4
|
||
'margin/v1/isolated/order/batchPlaceOrder': 4, # 5 times/1s(UID) => 20/5 = 4
|
||
'margin/v1/isolated/order/cancelOrder': 2, # 10 times/1s(UID) => 20/10 = 2
|
||
'margin/v1/isolated/order/batchCancelOrder': 2, # 10 times/1s(UID) => 20/10 = 2
|
||
'margin/v1/cross/order/placeOrder': 2, # 10 times/1s(UID) => 20/10 = 2
|
||
'margin/v1/cross/order/batchPlaceOrder': 2, # 10 times/1s(UID) => 20/10 = 2
|
||
'margin/v1/cross/order/cancelOrder': 2, # 10 times/1s(UID) => 20/10 = 2
|
||
'margin/v1/cross/order/batchCancelOrder': 2, # 10 times/1s(UID) => 20/10 = 2
|
||
'v2/margin/crossed/account/borrow': 2,
|
||
'v2/margin/crossed/account/repay': 2,
|
||
'v2/margin/crossed/account/flash-repay': 2,
|
||
'v2/margin/crossed/account/query-flash-repay-status': 2,
|
||
'v2/margin/crossed/place-order': 2,
|
||
'v2/margin/crossed/batch-place-order': 2,
|
||
'v2/margin/crossed/cancel-order': 2,
|
||
'v2/margin/crossed/batch-cancel-order': 2,
|
||
'v2/margin/isolated/account/borrow': 2,
|
||
'v2/margin/isolated/account/repay': 2,
|
||
'v2/margin/isolated/account/flash-repay': 2,
|
||
'v2/margin/isolated/account/query-flash-repay-status': 2,
|
||
'v2/margin/isolated/place-order': 2,
|
||
'v2/margin/isolated/batch-place-order': 2,
|
||
'v2/margin/isolated/cancel-order': 2,
|
||
'v2/margin/isolated/batch-cancel-order': 2,
|
||
},
|
||
},
|
||
'copy': {
|
||
'get': {
|
||
'v2/copy/mix-trader/order-current-track': 2,
|
||
'v2/copy/mix-trader/order-history-track': 2,
|
||
'v2/copy/mix-trader/order-total-detail': 2,
|
||
'v2/copy/mix-trader/profit-history-summarys': 1,
|
||
'v2/copy/mix-trader/profit-history-details': 1,
|
||
'v2/copy/mix-trader/profit-details': 1,
|
||
'v2/copy/mix-trader/profits-group-coin-date': 1,
|
||
'v2/copy/mix-trader/config-query-symbols': 1,
|
||
'v2/copy/mix-trader/config-query-followers': 2,
|
||
'v2/copy/mix-follower/query-current-orders': 2,
|
||
'v2/copy/mix-follower/query-history-orders': 1,
|
||
'v2/copy/mix-follower/query-settings': 2,
|
||
'v2/copy/mix-follower/query-traders': 2,
|
||
'v2/copy/mix-follower/query-quantity-limit': 2,
|
||
'v2/copy/mix-broker/query-traders': 2,
|
||
'v2/copy/mix-broker/query-history-traces': 2,
|
||
'v2/copy/mix-broker/query-current-traces': 2,
|
||
'v2/copy/spot-trader/profit-summarys': 2,
|
||
'v2/copy/spot-trader/profit-history-details': 2,
|
||
'v2/copy/spot-trader/profit-details': 2,
|
||
'v2/copy/spot-trader/order-total-detail': 2,
|
||
'v2/copy/spot-trader/order-history-track': 2,
|
||
'v2/copy/spot-trader/order-current-track': 2,
|
||
'v2/copy/spot-trader/config-query-settings': 2,
|
||
'v2/copy/spot-trader/config-query-followers': 2,
|
||
'v2/copy/spot-follower/query-traders': 2,
|
||
'v2/copy/spot-follower/query-trader-symbols': 2,
|
||
'v2/copy/spot-follower/query-settings': 2,
|
||
'v2/copy/spot-follower/query-history-orders': 2,
|
||
'v2/copy/spot-follower/query-current-orders': 2,
|
||
},
|
||
'post': {
|
||
'v2/copy/mix-trader/order-modify-tpsl': 2,
|
||
'v2/copy/mix-trader/order-close-positions': 2,
|
||
'v2/copy/mix-trader/config-setting-symbols': 2,
|
||
'v2/copy/mix-trader/config-setting-base': 2,
|
||
'v2/copy/mix-trader/config-remove-follower': 2,
|
||
'v2/copy/mix-follower/setting-tpsl': 1,
|
||
'v2/copy/mix-follower/settings': 2,
|
||
'v2/copy/mix-follower/close-positions': 2,
|
||
'v2/copy/mix-follower/cancel-trader': 4,
|
||
'v2/copy/spot-trader/order-modify-tpsl': 2,
|
||
'v2/copy/spot-trader/order-close-tracking': 2,
|
||
'v2/copy/spot-trader/config-setting-symbols': 2,
|
||
'v2/copy/spot-trader/config-remove-follower': 2,
|
||
'v2/copy/spot-follower/stop-order': 2,
|
||
'v2/copy/spot-follower/settings': 2,
|
||
'v2/copy/spot-follower/setting-tpsl': 2,
|
||
'v2/copy/spot-follower/order-close-tracking': 2,
|
||
'v2/copy/spot-follower/cancel-trader': 2,
|
||
},
|
||
},
|
||
'tax': {
|
||
'get': {
|
||
'v2/tax/spot-record': 20,
|
||
'v2/tax/future-record': 20,
|
||
'v2/tax/margin-record': 20,
|
||
'v2/tax/p2p-record': 20,
|
||
},
|
||
},
|
||
'convert': {
|
||
'get': {
|
||
'v2/convert/currencies': 2,
|
||
'v2/convert/quoted-price': 2,
|
||
'v2/convert/convert-record': 2,
|
||
'v2/convert/bgb-convert-coin-list': 2,
|
||
'v2/convert/bgb-convert-records': 2,
|
||
},
|
||
'post': {
|
||
'v2/convert/trade': 2,
|
||
'v2/convert/bgb-convert': 2,
|
||
},
|
||
},
|
||
'earn': {
|
||
'get': {
|
||
'v2/earn/savings/product': 2,
|
||
'v2/earn/savings/account': 2,
|
||
'v2/earn/savings/assets': 2,
|
||
'v2/earn/savings/records': 2,
|
||
'v2/earn/savings/subscribe-info': 2,
|
||
'v2/earn/savings/subscribe-result': 2,
|
||
'v2/earn/savings/redeem-result': 2,
|
||
'v2/earn/sharkfin/product': 2,
|
||
'v2/earn/sharkfin/account': 2,
|
||
'v2/earn/sharkfin/assets': 2,
|
||
'v2/earn/sharkfin/records': 2,
|
||
'v2/earn/sharkfin/subscribe-info': 2,
|
||
'v2/earn/sharkfin/subscribe-result': 4,
|
||
'v2/earn/loan/ongoing-orders': 2,
|
||
'v2/earn/loan/repay-history': 2,
|
||
'v2/earn/loan/revise-history': 2,
|
||
'v2/earn/loan/borrow-history': 2,
|
||
'v2/earn/loan/debts': 2,
|
||
'v2/earn/loan/reduces': 2,
|
||
'v2/earn/account/assets': 2,
|
||
},
|
||
'post': {
|
||
'v2/earn/savings/subscribe': 2,
|
||
'v2/earn/savings/redeem': 2,
|
||
'v2/earn/sharkfin/subscribe': 2,
|
||
'v2/earn/loan/borrow': 2,
|
||
'v2/earn/loan/repay': 2,
|
||
'v2/earn/loan/revise-pledge': 2,
|
||
},
|
||
},
|
||
'common': {
|
||
'get': {
|
||
'v2/common/trade-rate': 2,
|
||
},
|
||
},
|
||
'uta': {
|
||
'get': {
|
||
'v3/account/assets': 1,
|
||
'v3/account/settings': 1,
|
||
'v3/account/deposit-records': 2,
|
||
'v3/account/financial-records': 1,
|
||
'v3/account/repayable-coins': 2,
|
||
'v3/account/payment-coins': 2,
|
||
'v3/account/convert-records': 1,
|
||
'v3/account/transferable-coins': 2,
|
||
'v3/account/sub-transfer-record': 4,
|
||
'v3/ins-loan/transfered': 6.6667,
|
||
'v3/ins-loan/symbols': 6.6667,
|
||
'v3/ins-loan/risk-unit': 6.6667,
|
||
'v3/ins-loan/repaid-history': 6.6667,
|
||
'v3/ins-loan/product-infos': 6.6667,
|
||
'v3/ins-loan/loan-order': 6.6667,
|
||
'v3/ins-loan/ltv-convert': 6.6667,
|
||
'v3/ins-loan/ensure-coins-convert': 6.6667,
|
||
'v3/position/current-position': 1,
|
||
'v3/position/history-position': 1,
|
||
'v3/trade/order-info': 1,
|
||
'v3/trade/unfilled-orders': 1,
|
||
'v3/trade/unfilled-strategy-orders': 1,
|
||
'v3/trade/history-orders': 1,
|
||
'v3/trade/history-strategy-orders': 1,
|
||
'v3/trade/fills': 1,
|
||
'v3/user/sub-list': 2,
|
||
'v3/user/sub-api-list': 2,
|
||
},
|
||
'post': {
|
||
'v3/account/set-leverage': 2,
|
||
'v3/account/set-hold-mode': 2,
|
||
'v3/account/repay': 4,
|
||
'v3/account/transfer': 4,
|
||
'v3/account/sub-transfer': 4,
|
||
'v3/account/max-open-available': 4,
|
||
'v3/ins-loan/bind-uid': 6.6667,
|
||
'v3/trade/place-order': 2,
|
||
'v3/trade/place-strategy-order': 2,
|
||
'v3/trade/modify-order': 2,
|
||
'v3/trade/modify-strategy-order': 2,
|
||
'v3/trade/cancel-order': 2,
|
||
'v3/trade/cancel-strategy-order': 2,
|
||
'v3/trade/place-batch': 4,
|
||
'v3/trade/batch-modify-order': 2,
|
||
'v3/trade/cancel-batch': 4,
|
||
'v3/trade/cancel-symbol-order': 4,
|
||
'v3/trade/close-positions': 4,
|
||
'v3/user/create-sub': 2,
|
||
'v3/user/freeze-sub': 2,
|
||
'v3/user/create-sub-api': 2,
|
||
'v3/user/update-sub-api': 2,
|
||
'v3/user/delete-sub-api': 2,
|
||
},
|
||
},
|
||
},
|
||
},
|
||
'fees': {
|
||
'spot': {
|
||
'taker': self.parse_number('0.002'),
|
||
'maker': self.parse_number('0.002'),
|
||
},
|
||
'swap': {
|
||
'taker': self.parse_number('0.0006'),
|
||
'maker': self.parse_number('0.0004'),
|
||
},
|
||
},
|
||
'requiredCredentials': {
|
||
'apiKey': True,
|
||
'secret': True,
|
||
'password': True,
|
||
},
|
||
'exceptions': {
|
||
# http error codes
|
||
# 400 Bad Request — Invalid request format
|
||
# 401 Unauthorized — Invalid API Key
|
||
# 403 Forbidden — You do not have access to the requested resource
|
||
# 404 Not Found
|
||
# 500 Internal Server Error — We had a problem with our server
|
||
'exact': {
|
||
'1': ExchangeError, # {"code": 1, "message": "System error"}
|
||
# undocumented
|
||
'failure to get a peer from the ring-balancer': ExchangeNotAvailable, # {"message": "failure to get a peer from the ring-balancer"}
|
||
'4010': PermissionDenied, # {"code": 4010, "message": "For the security of your funds, withdrawals are not permitted within 24 hours after changing fund password / mobile number / Google Authenticator settings "}
|
||
# common
|
||
# '0': ExchangeError, # 200 successful,when the order placement / cancellation / operation is successful
|
||
'4001': ExchangeError, # no data received in 30s
|
||
'4002': ExchangeError, # Buffer full. cannot write data
|
||
'40020': BadRequest, # {"code":"40020","msg":"Parameter orderId error","requestTime":1754305078588,"data":null}
|
||
# --------------------------------------------------------
|
||
'30001': AuthenticationError, # {"code": 30001, "message": 'request header "OK_ACCESS_KEY" cannot be blank'}
|
||
'30002': AuthenticationError, # {"code": 30002, "message": 'request header "OK_ACCESS_SIGN" cannot be blank'}
|
||
'30003': AuthenticationError, # {"code": 30003, "message": 'request header "OK_ACCESS_TIMESTAMP" cannot be blank'}
|
||
'30004': AuthenticationError, # {"code": 30004, "message": 'request header "OK_ACCESS_PASSPHRASE" cannot be blank'}
|
||
'30005': InvalidNonce, # {"code": 30005, "message": "invalid OK_ACCESS_TIMESTAMP"}
|
||
'30006': AuthenticationError, # {"code": 30006, "message": "invalid OK_ACCESS_KEY"}
|
||
'30007': BadRequest, # {"code": 30007, "message": 'invalid Content_Type, please use "application/json" format'}
|
||
'30008': RequestTimeout, # {"code": 30008, "message": "timestamp request expired"}
|
||
'30009': ExchangeError, # {"code": 30009, "message": "system error"}
|
||
'30010': AuthenticationError, # {"code": 30010, "message": "API validation failed"}
|
||
'30011': PermissionDenied, # {"code": 30011, "message": "invalid IP"}
|
||
'30012': AuthenticationError, # {"code": 30012, "message": "invalid authorization"}
|
||
'30013': AuthenticationError, # {"code": 30013, "message": "invalid sign"}
|
||
'30014': DDoSProtection, # {"code": 30014, "message": "request too frequent"}
|
||
'30015': AuthenticationError, # {"code": 30015, "message": 'request header "OK_ACCESS_PASSPHRASE" incorrect'}
|
||
'30016': ExchangeError, # {"code": 30015, "message": "you are using v1 apiKey, please use v1 endpoint. If you would like to use v3 endpoint, please subscribe to v3 apiKey"}
|
||
'30017': ExchangeError, # {"code": 30017, "message": "apikey's broker id does not match"}
|
||
'30018': ExchangeError, # {"code": 30018, "message": "apikey's domain does not match"}
|
||
'30019': ExchangeNotAvailable, # {"code": 30019, "message": "Api is offline or unavailable"}
|
||
'30020': BadRequest, # {"code": 30020, "message": "body cannot be blank"}
|
||
'30021': BadRequest, # {"code": 30021, "message": "Json data format error"}, {"code": 30021, "message": "json data format error"}
|
||
'30022': PermissionDenied, # {"code": 30022, "message": "Api has been frozen"}
|
||
'30023': BadRequest, # {"code": 30023, "message": "{0} parameter cannot be blank"}
|
||
'30024': BadSymbol, # {"code":30024,"message":"\"instrument_id\" is an invalid parameter"}
|
||
'30025': BadRequest, # {"code": 30025, "message": "{0} parameter category error"}
|
||
'30026': DDoSProtection, # {"code": 30026, "message": "requested too frequent"}
|
||
'30027': AuthenticationError, # {"code": 30027, "message": "login failure"}
|
||
'30028': PermissionDenied, # {"code": 30028, "message": "unauthorized execution"}
|
||
'30029': AccountSuspended, # {"code": 30029, "message": "account suspended"}
|
||
'30030': ExchangeError, # {"code": 30030, "message": "endpoint request failed. Please try again"}
|
||
'30031': BadRequest, # {"code": 30031, "message": "token does not exist"}
|
||
'30032': BadSymbol, # {"code": 30032, "message": "pair does not exist"}
|
||
'30033': BadRequest, # {"code": 30033, "message": "exchange domain does not exist"}
|
||
'30034': ExchangeError, # {"code": 30034, "message": "exchange ID does not exist"}
|
||
'30035': ExchangeError, # {"code": 30035, "message": "trading is not hasattr(self, supported) website"}
|
||
'30036': ExchangeError, # {"code": 30036, "message": "no relevant data"}
|
||
'30037': ExchangeNotAvailable, # {"code": 30037, "message": "endpoint is offline or unavailable"}
|
||
# '30038': AuthenticationError, # {"code": 30038, "message": "user does not exist"}
|
||
'30038': OnMaintenance, # {"client_oid":"","code":"30038","error_code":"30038","error_message":"Matching engine is being upgraded. Please try in about 1 minute.","message":"Matching engine is being upgraded. Please try in about 1 minute.","order_id":"-1","result":false}
|
||
# futures
|
||
'32001': AccountSuspended, # {"code": 32001, "message": "futures account suspended"}
|
||
'32002': PermissionDenied, # {"code": 32002, "message": "futures account does not exist"}
|
||
'32003': CancelPending, # {"code": 32003, "message": "canceling, please wait"}
|
||
'32004': ExchangeError, # {"code": 32004, "message": "you have no unfilled orders"}
|
||
'32005': InvalidOrder, # {"code": 32005, "message": "max order quantity"}
|
||
'32006': InvalidOrder, # {"code": 32006, "message": "the order price or trigger price exceeds USD 1 million"}
|
||
'32007': InvalidOrder, # {"code": 32007, "message": "leverage level must be the same for orders on the same side of the contract"}
|
||
'32008': InvalidOrder, # {"code": 32008, "message": "Max. positions to open(cross margin)"}
|
||
'32009': InvalidOrder, # {"code": 32009, "message": "Max. positions to open(fixed margin)"}
|
||
'32010': ExchangeError, # {"code": 32010, "message": "leverage cannot be changed with open positions"}
|
||
'32011': ExchangeError, # {"code": 32011, "message": "futures status error"}
|
||
'32012': ExchangeError, # {"code": 32012, "message": "futures order update error"}
|
||
'32013': ExchangeError, # {"code": 32013, "message": "token type is blank"}
|
||
'32014': ExchangeError, # {"code": 32014, "message": "your number of contracts closing is larger than the number of contracts available"}
|
||
'32015': ExchangeError, # {"code": 32015, "message": "margin ratio is lower than 100% before opening positions"}
|
||
'32016': ExchangeError, # {"code": 32016, "message": "margin ratio is lower than 100% after opening position"}
|
||
'32017': ExchangeError, # {"code": 32017, "message": "no BBO"}
|
||
'32018': ExchangeError, # {"code": 32018, "message": "the order quantity is less than 1, please try again"}
|
||
'32019': ExchangeError, # {"code": 32019, "message": "the order price deviates from the price of the previous minute by more than 3%"}
|
||
'32020': ExchangeError, # {"code": 32020, "message": "the price is not in the range of the price limit"}
|
||
'32021': ExchangeError, # {"code": 32021, "message": "leverage error"}
|
||
'32022': ExchangeError, # {"code": 32022, "message": "self function is not supported in your country or region according to the regulations"}
|
||
'32023': ExchangeError, # {"code": 32023, "message": "self account has outstanding loan"}
|
||
'32024': ExchangeError, # {"code": 32024, "message": "order cannot be placed during delivery"}
|
||
'32025': ExchangeError, # {"code": 32025, "message": "order cannot be placed during settlement"}
|
||
'32026': ExchangeError, # {"code": 32026, "message": "your account is restricted from opening positions"}
|
||
'32027': ExchangeError, # {"code": 32027, "message": "cancelled over 20 orders"}
|
||
'32028': AccountSuspended, # {"code": 32028, "message": "account is suspended and liquidated"}
|
||
'32029': ExchangeError, # {"code": 32029, "message": "order info does not exist"}
|
||
'32030': InvalidOrder, # The order cannot be cancelled
|
||
'32031': ArgumentsRequired, # client_oid or order_id is required.
|
||
'32038': AuthenticationError, # User does not exist
|
||
'32040': ExchangeError, # User have open contract orders or position
|
||
'32044': ExchangeError, # {"code": 32044, "message": "The margin ratio after submitting self order is lower than the minimum requirement({0}) for your tier."}
|
||
'32045': ExchangeError, # str of commission over 1 million
|
||
'32046': ExchangeError, # Each user can hold up to 10 trade plans at the same time
|
||
'32047': ExchangeError, # system error
|
||
'32048': InvalidOrder, # Order strategy track range error
|
||
'32049': ExchangeError, # Each user can hold up to 10 track plans at the same time
|
||
'32050': InvalidOrder, # Order strategy rang error
|
||
'32051': InvalidOrder, # Order strategy ice depth error
|
||
'32052': ExchangeError, # str of commission over 100 thousand
|
||
'32053': ExchangeError, # Each user can hold up to 6 ice plans at the same time
|
||
'32057': ExchangeError, # The order price is zero. Market-close-all function cannot be executed
|
||
'32054': ExchangeError, # Trade not allow
|
||
'32055': InvalidOrder, # cancel order error
|
||
'32056': ExchangeError, # iceberg per order average should between {0}-{1} contracts
|
||
'32058': ExchangeError, # Each user can hold up to 6 initiative plans at the same time
|
||
'32059': InvalidOrder, # Total amount should exceed per order amount
|
||
'32060': InvalidOrder, # Order strategy type error
|
||
'32061': InvalidOrder, # Order strategy initiative limit error
|
||
'32062': InvalidOrder, # Order strategy initiative range error
|
||
'32063': InvalidOrder, # Order strategy initiative rate error
|
||
'32064': ExchangeError, # Time Stringerval of orders should set between 5-120s
|
||
'32065': ExchangeError, # Close amount exceeds the limit of Market-close-all(999 for BTC, and 9999 for the rest tokens)
|
||
'32066': ExchangeError, # You have open orders. Please cancel all open orders before changing your leverage level.
|
||
'32067': ExchangeError, # Account equity < required hasattr(self, margin) setting. Please adjust your leverage level again.
|
||
'32068': ExchangeError, # The margin for self position will fall short of the required hasattr(self, margin) setting. Please adjust your leverage level or increase your margin to proceed.
|
||
'32069': ExchangeError, # Target leverage level too low. Your account balance is insufficient to cover the margin required. Please adjust the leverage level again.
|
||
'32070': ExchangeError, # Please check open position or unfilled order
|
||
'32071': ExchangeError, # Your current liquidation mode does not support self action.
|
||
'32072': ExchangeError, # The highest available margin for your order’s tier is {0}. Please edit your margin and place a new order.
|
||
'32073': ExchangeError, # The action does not apply to the token
|
||
'32074': ExchangeError, # The number of contracts of your position, open orders, and the current order has exceeded the maximum order limit of self asset.
|
||
'32075': ExchangeError, # Account risk rate breach
|
||
'32076': ExchangeError, # Liquidation of the holding position(s) at market price will require cancellation of all pending close orders of the contracts.
|
||
'32077': ExchangeError, # Your margin for self asset in futures account is insufficient and the position has been taken over for liquidation.(You will not be able to place orders, close positions, transfer funds, or add margin during self period of time. Your account will be restored after the liquidation is complete.)
|
||
'32078': ExchangeError, # Please cancel all open orders before switching the liquidation mode(Please cancel all open orders before switching the liquidation mode)
|
||
'32079': ExchangeError, # Your open positions are at high risk.(Please add margin or reduce positions before switching the mode)
|
||
'32080': ExchangeError, # Funds cannot be transferred out within 30 minutes after futures settlement
|
||
'32083': ExchangeError, # The number of contracts should be a positive multiple of %%. Please place your order again
|
||
# token and margin trading
|
||
'33001': PermissionDenied, # {"code": 33001, "message": "margin account for self pair is not enabled yet"}
|
||
'33002': AccountSuspended, # {"code": 33002, "message": "margin account for self pair is suspended"}
|
||
'33003': InsufficientFunds, # {"code": 33003, "message": "no loan balance"}
|
||
'33004': ExchangeError, # {"code": 33004, "message": "loan amount cannot be smaller than the minimum limit"}
|
||
'33005': ExchangeError, # {"code": 33005, "message": "repayment amount must exceed 0"}
|
||
'33006': ExchangeError, # {"code": 33006, "message": "loan order not found"}
|
||
'33007': ExchangeError, # {"code": 33007, "message": "status not found"}
|
||
'33008': InsufficientFunds, # {"code": 33008, "message": "loan amount cannot exceed the maximum limit"}
|
||
'33009': ExchangeError, # {"code": 33009, "message": "user ID is blank"}
|
||
'33010': ExchangeError, # {"code": 33010, "message": "you cannot cancel an order during session 2 of call auction"}
|
||
'33011': ExchangeError, # {"code": 33011, "message": "no new market data"}
|
||
'33012': ExchangeError, # {"code": 33012, "message": "order cancellation failed"}
|
||
'33013': InvalidOrder, # {"code": 33013, "message": "order placement failed"}
|
||
'33014': OrderNotFound, # {"code": 33014, "message": "order does not exist"}
|
||
'33015': InvalidOrder, # {"code": 33015, "message": "exceeded maximum limit"}
|
||
'33016': ExchangeError, # {"code": 33016, "message": "margin trading is not open for self token"}
|
||
'33017': InsufficientFunds, # {"code": 33017, "message": "insufficient balance"}
|
||
'33018': ExchangeError, # {"code": 33018, "message": "self parameter must be smaller than 1"}
|
||
'33020': ExchangeError, # {"code": 33020, "message": "request not supported"}
|
||
'33021': BadRequest, # {"code": 33021, "message": "token and the pair do not match"}
|
||
'33022': InvalidOrder, # {"code": 33022, "message": "pair and the order do not match"}
|
||
'33023': ExchangeError, # {"code": 33023, "message": "you can only place market orders during call auction"}
|
||
'33024': InvalidOrder, # {"code": 33024, "message": "trading amount too small"}
|
||
'33025': InvalidOrder, # {"code": 33025, "message": "base token amount is blank"}
|
||
'33026': ExchangeError, # {"code": 33026, "message": "transaction completed"}
|
||
'33027': InvalidOrder, # {"code": 33027, "message": "cancelled order or order cancelling"}
|
||
'33028': InvalidOrder, # {"code": 33028, "message": "the decimal places of the trading price exceeded the limit"}
|
||
'33029': InvalidOrder, # {"code": 33029, "message": "the decimal places of the trading size exceeded the limit"}
|
||
'33034': ExchangeError, # {"code": 33034, "message": "You can only place limit order after Call Auction has started"}
|
||
'33035': ExchangeError, # This type of order cannot be canceled(This type of order cannot be canceled)
|
||
'33036': ExchangeError, # Exceeding the limit of entrust order
|
||
'33037': ExchangeError, # The buy order price should be lower than 130% of the trigger price
|
||
'33038': ExchangeError, # The sell order price should be higher than 70% of the trigger price
|
||
'33039': ExchangeError, # The limit of callback rate is 0 < x <= 5%
|
||
'33040': ExchangeError, # The trigger price of a buy order should be lower than the latest transaction price
|
||
'33041': ExchangeError, # The trigger price of a sell order should be higher than the latest transaction price
|
||
'33042': ExchangeError, # The limit of price variance is 0 < x <= 1%
|
||
'33043': ExchangeError, # The total amount must be larger than 0
|
||
'33044': ExchangeError, # The average amount should be 1/1000 * total amount <= x <= total amount
|
||
'33045': ExchangeError, # The price should not be 0, including trigger price, order price, and price limit
|
||
'33046': ExchangeError, # Price variance should be 0 < x <= 1%
|
||
'33047': ExchangeError, # Sweep ratio should be 0 < x <= 100%
|
||
'33048': ExchangeError, # Per order limit: Total amount/1000 < x <= Total amount
|
||
'33049': ExchangeError, # Total amount should be X > 0
|
||
'33050': ExchangeError, # Time interval should be 5 <= x <= 120s
|
||
'33051': ExchangeError, # cancel order number not higher limit: plan and track entrust no more than 10, ice and time entrust no more than 6
|
||
'33059': BadRequest, # {"code": 33059, "message": "client_oid or order_id is required"}
|
||
'33060': BadRequest, # {"code": 33060, "message": "Only fill in either parameter client_oid or order_id"}
|
||
'33061': ExchangeError, # Value of a single market price order cannot exceed 100,000 USD
|
||
'33062': ExchangeError, # The leverage ratio is too high. The borrowed position has exceeded the maximum position of self leverage ratio. Please readjust the leverage ratio
|
||
'33063': ExchangeError, # Leverage multiple is too low, there is insufficient margin in the account, please readjust the leverage ratio
|
||
'33064': ExchangeError, # The setting of the leverage ratio cannot be less than 2, please readjust the leverage ratio
|
||
'33065': ExchangeError, # Leverage ratio exceeds maximum leverage ratio, please readjust leverage ratio
|
||
# account
|
||
'21009': ExchangeError, # Funds cannot be transferred out within 30 minutes after swap settlement(Funds cannot be transferred out within 30 minutes after swap settlement)
|
||
'34001': PermissionDenied, # {"code": 34001, "message": "withdrawal suspended"}
|
||
'34002': InvalidAddress, # {"code": 34002, "message": "please add a withdrawal address"}
|
||
'34003': ExchangeError, # {"code": 34003, "message": "sorry, self token cannot be withdrawn to xx at the moment"}
|
||
'34004': ExchangeError, # {"code": 34004, "message": "withdrawal fee is smaller than minimum limit"}
|
||
'34005': ExchangeError, # {"code": 34005, "message": "withdrawal fee exceeds the maximum limit"}
|
||
'34006': ExchangeError, # {"code": 34006, "message": "withdrawal amount is lower than the minimum limit"}
|
||
'34007': ExchangeError, # {"code": 34007, "message": "withdrawal amount exceeds the maximum limit"}
|
||
'34008': InsufficientFunds, # {"code": 34008, "message": "insufficient balance"}
|
||
'34009': ExchangeError, # {"code": 34009, "message": "your withdrawal amount exceeds the daily limit"}
|
||
'34010': ExchangeError, # {"code": 34010, "message": "transfer amount must be larger than 0"}
|
||
'34011': ExchangeError, # {"code": 34011, "message": "conditions not met"}
|
||
'34012': ExchangeError, # {"code": 34012, "message": "the minimum withdrawal amount for NEO is 1, and the amount must be an integer"}
|
||
'34013': ExchangeError, # {"code": 34013, "message": "please transfer"}
|
||
'34014': ExchangeError, # {"code": 34014, "message": "transfer limited"}
|
||
'34015': ExchangeError, # {"code": 34015, "message": "subaccount does not exist"}
|
||
'34016': PermissionDenied, # {"code": 34016, "message": "transfer suspended"}
|
||
'34017': AccountSuspended, # {"code": 34017, "message": "account suspended"}
|
||
'34018': AuthenticationError, # {"code": 34018, "message": "incorrect trades password"}
|
||
'34019': PermissionDenied, # {"code": 34019, "message": "please bind your email before withdrawal"}
|
||
'34020': PermissionDenied, # {"code": 34020, "message": "please bind your funds password before withdrawal"}
|
||
'34021': InvalidAddress, # {"code": 34021, "message": "Not verified address"}
|
||
'34022': ExchangeError, # {"code": 34022, "message": "Withdrawals are not available for sub accounts"}
|
||
'34023': PermissionDenied, # {"code": 34023, "message": "Please enable futures trading before transferring your funds"}
|
||
'34026': ExchangeError, # transfer too frequently(transfer too frequently)
|
||
'34036': ExchangeError, # Parameter is incorrect, please refer to API documentation
|
||
'34037': ExchangeError, # Get the sub-account balance interface, account type is not supported
|
||
'34038': ExchangeError, # Since your C2C transaction is unusual, you are restricted from fund transfer. Please contact our customer support to cancel the restriction
|
||
'34039': ExchangeError, # You are now restricted from transferring out your funds due to abnormal trades on C2C Market. Please transfer your fund on our website or app instead to verify your identity
|
||
# swap
|
||
'35001': ExchangeError, # {"code": 35001, "message": "Contract does not exist"}
|
||
'35002': ExchangeError, # {"code": 35002, "message": "Contract settling"}
|
||
'35003': ExchangeError, # {"code": 35003, "message": "Contract paused"}
|
||
'35004': ExchangeError, # {"code": 35004, "message": "Contract pending settlement"}
|
||
'35005': AuthenticationError, # {"code": 35005, "message": "User does not exist"}
|
||
'35008': InvalidOrder, # {"code": 35008, "message": "Risk ratio too high"}
|
||
'35010': InvalidOrder, # {"code": 35010, "message": "Position closing too large"}
|
||
'35012': InvalidOrder, # {"code": 35012, "message": "Incorrect order size"}
|
||
'35014': InvalidOrder, # {"code": 35014, "message": "Order price is not within limit"}
|
||
'35015': InvalidOrder, # {"code": 35015, "message": "Invalid leverage level"}
|
||
'35017': ExchangeError, # {"code": 35017, "message": "Open orders exist"}
|
||
'35019': InvalidOrder, # {"code": 35019, "message": "Order size too large"}
|
||
'35020': InvalidOrder, # {"code": 35020, "message": "Order price too high"}
|
||
'35021': InvalidOrder, # {"code": 35021, "message": "Order size exceeded current tier limit"}
|
||
'35022': ExchangeError, # {"code": 35022, "message": "Contract status error"}
|
||
'35024': ExchangeError, # {"code": 35024, "message": "Contract not initialized"}
|
||
'35025': InsufficientFunds, # {"code": 35025, "message": "No account balance"}
|
||
'35026': ExchangeError, # {"code": 35026, "message": "Contract settings not initialized"}
|
||
'35029': OrderNotFound, # {"code": 35029, "message": "Order does not exist"}
|
||
'35030': InvalidOrder, # {"code": 35030, "message": "Order size too large"}
|
||
'35031': InvalidOrder, # {"code": 35031, "message": "Cancel order size too large"}
|
||
'35032': ExchangeError, # {"code": 35032, "message": "Invalid user status"}
|
||
'35037': ExchangeError, # No last traded price in cache
|
||
'35039': ExchangeError, # {"code": 35039, "message": "Open order quantity exceeds limit"}
|
||
'35040': InvalidOrder, # {"error_message":"Invalid order type","result":"true","error_code":"35040","order_id":"-1"}
|
||
'35044': ExchangeError, # {"code": 35044, "message": "Invalid order status"}
|
||
'35046': InsufficientFunds, # {"code": 35046, "message": "Negative account balance"}
|
||
'35047': InsufficientFunds, # {"code": 35047, "message": "Insufficient account balance"}
|
||
'35048': ExchangeError, # {"code": 35048, "message": "User contract is frozen and liquidating"}
|
||
'35049': InvalidOrder, # {"code": 35049, "message": "Invalid order type"}
|
||
'35050': InvalidOrder, # {"code": 35050, "message": "Position settings are blank"}
|
||
'35052': InsufficientFunds, # {"code": 35052, "message": "Insufficient cross margin"}
|
||
'35053': ExchangeError, # {"code": 35053, "message": "Account risk too high"}
|
||
'35055': InsufficientFunds, # {"code": 35055, "message": "Insufficient account balance"}
|
||
'35057': ExchangeError, # {"code": 35057, "message": "No last traded price"}
|
||
'35058': ExchangeError, # {"code": 35058, "message": "No limit"}
|
||
'35059': BadRequest, # {"code": 35059, "message": "client_oid or order_id is required"}
|
||
'35060': BadRequest, # {"code": 35060, "message": "Only fill in either parameter client_oid or order_id"}
|
||
'35061': BadRequest, # {"code": 35061, "message": "Invalid instrument_id"}
|
||
'35062': InvalidOrder, # {"code": 35062, "message": "Invalid match_price"}
|
||
'35063': InvalidOrder, # {"code": 35063, "message": "Invalid order_size"}
|
||
'35064': InvalidOrder, # {"code": 35064, "message": "Invalid client_oid"}
|
||
'35066': InvalidOrder, # Order interval error
|
||
'35067': InvalidOrder, # Time-weighted order ratio error
|
||
'35068': InvalidOrder, # Time-weighted order range error
|
||
'35069': InvalidOrder, # Time-weighted single transaction limit error
|
||
'35070': InvalidOrder, # Algo order type error
|
||
'35071': InvalidOrder, # Order total must be larger than single order limit
|
||
'35072': InvalidOrder, # Maximum 6 unfulfilled time-weighted orders can be held at the same time
|
||
'35073': InvalidOrder, # Order price is 0. Market-close-all not available
|
||
'35074': InvalidOrder, # Iceberg order single transaction average error
|
||
'35075': InvalidOrder, # Failed to cancel order
|
||
'35076': InvalidOrder, # LTC 20x leverage. Not allowed to open position
|
||
'35077': InvalidOrder, # Maximum 6 unfulfilled iceberg orders can be held at the same time
|
||
'35078': InvalidOrder, # Order amount exceeded 100,000
|
||
'35079': InvalidOrder, # Iceberg order price variance error
|
||
'35080': InvalidOrder, # Callback rate error
|
||
'35081': InvalidOrder, # Maximum 10 unfulfilled trail orders can be held at the same time
|
||
'35082': InvalidOrder, # Trail order callback rate error
|
||
'35083': InvalidOrder, # Each user can only hold a maximum of 10 unfulfilled stop-limit orders at the same time
|
||
'35084': InvalidOrder, # Order amount exceeded 1 million
|
||
'35085': InvalidOrder, # Order amount is not in the correct range
|
||
'35086': InvalidOrder, # Price exceeds 100 thousand
|
||
'35087': InvalidOrder, # Price exceeds 100 thousand
|
||
'35088': InvalidOrder, # Average amount error
|
||
'35089': InvalidOrder, # Price exceeds 100 thousand
|
||
'35090': ExchangeError, # No stop-limit orders available for cancelation
|
||
'35091': ExchangeError, # No trail orders available for cancellation
|
||
'35092': ExchangeError, # No iceberg orders available for cancellation
|
||
'35093': ExchangeError, # No trail orders available for cancellation
|
||
'35094': ExchangeError, # Stop-limit order last traded price error
|
||
'35095': BadRequest, # Instrument_id error
|
||
'35096': ExchangeError, # Algo order status error
|
||
'35097': ExchangeError, # Order status and order ID cannot exist at the same time
|
||
'35098': ExchangeError, # An order status or order ID must exist
|
||
'35099': ExchangeError, # Algo order ID error
|
||
# option
|
||
'36001': BadRequest, # Invalid underlying index.
|
||
'36002': BadRequest, # Instrument does not exist.
|
||
'36005': ExchangeError, # Instrument status is invalid.
|
||
'36101': AuthenticationError, # Account does not exist.
|
||
'36102': PermissionDenied, # Account status is invalid.
|
||
'36103': AccountSuspended, # Account is suspended due to ongoing liquidation.
|
||
'36104': PermissionDenied, # Account is not enabled for options trading.
|
||
'36105': PermissionDenied, # Please enable the account for option contract.
|
||
'36106': AccountSuspended, # Funds cannot be transferred in or out, is suspended.
|
||
'36107': PermissionDenied, # Funds cannot be transferred out within 30 minutes after option exercising or settlement.
|
||
'36108': InsufficientFunds, # Funds cannot be transferred in or out, of the account is less than zero.
|
||
'36109': PermissionDenied, # Funds cannot be transferred in or out during option exercising or settlement.
|
||
'36201': PermissionDenied, # New order function is blocked.
|
||
'36202': PermissionDenied, # Account does not have permission to short option.
|
||
'36203': InvalidOrder, # Invalid format for client_oid.
|
||
'36204': ExchangeError, # Invalid format for request_id.
|
||
'36205': BadRequest, # Instrument id does not match underlying index.
|
||
'36206': BadRequest, # Order_id and client_oid can not be used at the same time.
|
||
'36207': InvalidOrder, # Either order price or fartouch price must be present.
|
||
'36208': InvalidOrder, # Either order price or size must be present.
|
||
'36209': InvalidOrder, # Either order_id or client_oid must be present.
|
||
'36210': InvalidOrder, # Either order_ids or client_oids must be present.
|
||
'36211': InvalidOrder, # Exceeding max batch size for order submission.
|
||
'36212': InvalidOrder, # Exceeding max batch size for oder cancellation.
|
||
'36213': InvalidOrder, # Exceeding max batch size for order amendment.
|
||
'36214': ExchangeError, # Instrument does not have valid bid/ask quote.
|
||
'36216': OrderNotFound, # Order does not exist.
|
||
'36217': InvalidOrder, # Order submission failed.
|
||
'36218': InvalidOrder, # Order cancellation failed.
|
||
'36219': InvalidOrder, # Order amendment failed.
|
||
'36220': InvalidOrder, # Order is pending cancel.
|
||
'36221': InvalidOrder, # Order qty is not valid multiple of lot size.
|
||
'36222': InvalidOrder, # Order price is breaching highest buy limit.
|
||
'36223': InvalidOrder, # Order price is breaching lowest sell limit.
|
||
'36224': InvalidOrder, # Exceeding max order size.
|
||
'36225': InvalidOrder, # Exceeding max open order count for instrument.
|
||
'36226': InvalidOrder, # Exceeding max open order count for underlying.
|
||
'36227': InvalidOrder, # Exceeding max open size across all orders for underlying
|
||
'36228': InvalidOrder, # Exceeding max available qty for instrument.
|
||
'36229': InvalidOrder, # Exceeding max available qty for underlying.
|
||
'36230': InvalidOrder, # Exceeding max position limit for underlying.
|
||
# --------------------------------------------------------
|
||
# swap
|
||
'400': BadRequest, # Bad Request
|
||
'401': AuthenticationError, # Unauthorized access
|
||
'403': PermissionDenied, # Access prohibited
|
||
'404': BadRequest, # Request address does not exist
|
||
'405': BadRequest, # The HTTP Method is not supported
|
||
'415': BadRequest, # The current media type is not supported
|
||
'429': DDoSProtection, # Too many requests
|
||
'500': ExchangeNotAvailable, # System busy
|
||
'1001': RateLimitExceeded, # The request is too frequent and has been throttled
|
||
'1002': ExchangeError, # {0} verifications within 24 hours
|
||
'1003': ExchangeError, # You failed more than {0} times today, the current operation is locked, please try again in 24 hours
|
||
# '00000': ExchangeError, # success
|
||
'40001': AuthenticationError, # ACCESS_KEY cannot be empty
|
||
'40002': AuthenticationError, # SECRET_KEY cannot be empty
|
||
'40003': AuthenticationError, # Signature cannot be empty
|
||
'40004': InvalidNonce, # Request timestamp expired
|
||
'40005': InvalidNonce, # Invalid ACCESS_TIMESTAMP
|
||
'40006': AuthenticationError, # Invalid ACCESS_KEY
|
||
'40007': BadRequest, # Invalid Content_Type
|
||
'40008': InvalidNonce, # Request timestamp expired
|
||
'40009': AuthenticationError, # sign signature error
|
||
'40010': AuthenticationError, # sign signature error
|
||
'40011': AuthenticationError, # ACCESS_PASSPHRASE cannot be empty
|
||
'40012': AuthenticationError, # apikey/password is incorrect
|
||
'40013': ExchangeError, # User status is abnormal
|
||
'40014': PermissionDenied, # Incorrect permissions
|
||
'40015': ExchangeError, # System is abnormal, please try again later
|
||
'40016': PermissionDenied, # The user must bind the phone or Google
|
||
'40017': ExchangeError, # Parameter verification failed
|
||
'40018': PermissionDenied, # Invalid IP
|
||
'40019': BadRequest, # {"code":"40019","msg":"Parameter QLCUSDT_SPBL cannot be empty","requestTime":1679196063659,"data":null}
|
||
'40031': AccountSuspended, # The account has been cancelled and cannot be used again
|
||
'40037': AuthenticationError, # Apikey does not exist
|
||
'40102': BadRequest, # Contract configuration does not exist, please check the parameters
|
||
'40103': BadRequest, # Request method cannot be empty
|
||
'40104': ExchangeError, # Lever adjustment failure
|
||
'40105': ExchangeError, # Abnormal access to current price limit data
|
||
'40106': ExchangeError, # Abnormal get next settlement time
|
||
'40107': ExchangeError, # Abnormal access to index price data
|
||
'40108': InvalidOrder, # Wrong order quantity
|
||
'40109': OrderNotFound, # The data of the order cannot be found, please confirm the order number
|
||
'40200': OnMaintenance, # Server upgrade, please try again later
|
||
'40201': InvalidOrder, # Order number cannot be empty
|
||
'40202': ExchangeError, # User information cannot be empty
|
||
'40203': BadRequest, # The amount of adjustment margin cannot be empty or negative
|
||
'40204': BadRequest, # Adjustment margin type cannot be empty
|
||
'40205': BadRequest, # Adjusted margin type data is wrong
|
||
'40206': BadRequest, # The direction of the adjustment margin cannot be empty
|
||
'40207': BadRequest, # The adjustment margin data is wrong
|
||
'40208': BadRequest, # The accuracy of the adjustment margin amount is incorrect
|
||
'40209': BadRequest, # The current page number is wrong, please confirm
|
||
'40300': ExchangeError, # User does not exist
|
||
'40301': PermissionDenied, # Permission has not been obtained yet. If you need to use it, please contact customer service
|
||
'40302': BadRequest, # Parameter abnormality
|
||
'40303': BadRequest, # Can only query up to 20,000 data
|
||
'40304': BadRequest, # Parameter type is abnormal
|
||
'40305': BadRequest, # Client_oid length is not greater than 50, and cannot be Martian characters
|
||
'40306': ExchangeError, # Batch processing orders can only process up to 20
|
||
'40308': OnMaintenance, # The contract is being temporarily maintained
|
||
'40309': BadSymbol, # The contract has been removed
|
||
'40400': ExchangeError, # Status check abnormal
|
||
'40401': ExchangeError, # The operation cannot be performed
|
||
'40402': BadRequest, # The opening direction cannot be empty
|
||
'40403': BadRequest, # Wrong opening direction format
|
||
'40404': BadRequest, # Whether to enable automatic margin call parameters cannot be empty
|
||
'40405': BadRequest, # Whether to enable the automatic margin call parameter type is wrong
|
||
'40406': BadRequest, # Whether to enable automatic margin call parameters is of unknown type
|
||
'40407': ExchangeError, # The query direction is not the direction entrusted by the plan
|
||
'40408': ExchangeError, # Wrong time range
|
||
'40409': ExchangeError, # Time format error
|
||
'40500': InvalidOrder, # Client_oid check error
|
||
'40501': ExchangeError, # Channel name error
|
||
'40502': ExchangeError, # If it is a copy user, you must pass the copy to whom
|
||
'40503': ExchangeError, # With the single type
|
||
'40504': ExchangeError, # Platform code must pass
|
||
'40505': ExchangeError, # Not the same type
|
||
'40506': AuthenticationError, # Platform signature error
|
||
'40507': AuthenticationError, # Api signature error
|
||
'40508': ExchangeError, # KOL is not authorized
|
||
'40509': ExchangeError, # Abnormal copy end
|
||
'40600': ExchangeError, # Copy function suspended
|
||
'40601': ExchangeError, # Followers cannot be KOL
|
||
'40602': ExchangeError, # The number of copies has reached the limit and cannot process the request
|
||
'40603': ExchangeError, # Abnormal copy end
|
||
'40604': ExchangeNotAvailable, # Server is busy, please try again later
|
||
'40605': ExchangeError, # Copy type, the copy number must be passed
|
||
'40606': ExchangeError, # The type of document number is wrong
|
||
'40607': ExchangeError, # Document number must be passed
|
||
'40608': ExchangeError, # No documented products currently supported
|
||
'40609': ExchangeError, # The contract product does not support copying
|
||
'40700': BadRequest, # Cursor parameters are incorrect
|
||
'40701': ExchangeError, # KOL is not authorized
|
||
'40702': ExchangeError, # Unauthorized copying user
|
||
'40703': ExchangeError, # Bill inquiry start and end time cannot be empty
|
||
'40704': ExchangeError, # Can only check the data of the last three months
|
||
'40705': BadRequest, # The start and end time cannot exceed 90 days
|
||
'40706': InvalidOrder, # Wrong order price
|
||
'40707': BadRequest, # Start time is greater than end time
|
||
'40708': BadRequest, # Parameter verification is abnormal
|
||
'40709': ExchangeError, # There is no hasattr(self, position) position, and no automatic margin call can be set
|
||
'40710': ExchangeError, # Abnormal account status
|
||
'40711': InsufficientFunds, # Insufficient contract account balance
|
||
'40712': InsufficientFunds, # Insufficient margin
|
||
'40713': ExchangeError, # Cannot exceed the maximum transferable margin amount
|
||
'40714': ExchangeError, # No direct margin call is allowed
|
||
'40762': InsufficientFunds, # {"code":"40762","msg":"The order amount exceeds the balance","requestTime":1716572156622,"data":null}
|
||
'40768': OrderNotFound, # Order does not exist
|
||
'40808': InvalidOrder, # {"code":"40808","msg":"Parameter verification exception size checkBDScale error value=2293.577 checkScale=2","requestTime":1725638500052,"data":null}
|
||
'41103': InvalidOrder, # {"code":"41103","msg":"param price scale error error","requestTime":1725635883561,"data":null}
|
||
'41114': OnMaintenance, # {"code":"41114","msg":"The current trading pair is under maintenance, please refer to the official announcement for the opening time","requestTime":1679196062544,"data":null}
|
||
'43011': InvalidOrder, # The parameter does not meet the specification executePrice <= 0
|
||
'43001': OrderNotFound,
|
||
'43012': InsufficientFunds, # {"code":"43012","msg":"Insufficient balance","requestTime":1711648951774,"data":null}
|
||
'43025': InvalidOrder, # Plan order does not exist
|
||
'43115': OnMaintenance, # {"code":"43115","msg":"The current trading pair is opening soon, please refer to the official announcement for the opening time","requestTime":1688907202434,"data":null}
|
||
'45110': InvalidOrder, # {"code":"45110","msg":"less than the minimum amount 5 USDT","requestTime":1669911118932,"data":null}
|
||
'40774': InvalidOrder, # {"code":"40774","msg":"The order type for unilateral position must also be the unilateral position type.","requestTime":1758709764409,"data":null}
|
||
'45122': InvalidOrder, # {"code":"45122","msg":"Short position stop loss price please > mark price 106.86","requestTime":1758709970499,"data":null}
|
||
# spot
|
||
'invalid sign': AuthenticationError,
|
||
'invalid currency': BadSymbol, # invalid trading pair
|
||
'invalid symbol': BadSymbol,
|
||
'invalid period': BadRequest, # invalid Kline type
|
||
'invalid user': ExchangeError,
|
||
'invalid amount': InvalidOrder,
|
||
'invalid type': InvalidOrder, # {"status":"error","ts":1595700344504,"err_code":"invalid-parameter","err_msg":"invalid type"}
|
||
'invalid orderId': InvalidOrder,
|
||
'invalid record': ExchangeError,
|
||
'invalid accountId': BadRequest,
|
||
'invalid address': BadRequest,
|
||
'accesskey not None': AuthenticationError, # {"status":"error","ts":1595704360508,"err_code":"invalid-parameter","err_msg":"accesskey not null"}
|
||
'illegal accesskey': AuthenticationError,
|
||
'sign not null': AuthenticationError,
|
||
'req_time is too much difference from server time': InvalidNonce,
|
||
'permissions not right': PermissionDenied, # {"status":"error","ts":1595704490084,"err_code":"invalid-parameter","err_msg":"permissions not right"}
|
||
'illegal sign invalid': AuthenticationError, # {"status":"error","ts":1595684716042,"err_code":"invalid-parameter","err_msg":"illegal sign invalid"}
|
||
'user locked': AccountSuspended,
|
||
'Request Frequency Is Too High': RateLimitExceeded,
|
||
'more than a daily rate of cash': BadRequest,
|
||
'more than the maximum daily withdrawal amount': BadRequest,
|
||
'need to bind email or mobile': ExchangeError,
|
||
'user forbid': PermissionDenied,
|
||
'User Prohibited Cash Withdrawal': PermissionDenied,
|
||
'Cash Withdrawal Is Less Than The Minimum Value': BadRequest,
|
||
'Cash Withdrawal Is More Than The Maximum Value': BadRequest,
|
||
'the account with in 24 hours ban coin': PermissionDenied,
|
||
'order cancel fail': BadRequest, # {"status":"error","ts":1595703343035,"err_code":"bad-request","err_msg":"order cancel fail"}
|
||
'base symbol error': BadSymbol,
|
||
'base date error': ExchangeError,
|
||
'api signature not valid': AuthenticationError,
|
||
'gateway internal error': ExchangeError,
|
||
'audit failed': ExchangeError,
|
||
'order queryorder invalid': BadRequest,
|
||
'market no need price': InvalidOrder,
|
||
'limit need price': InvalidOrder,
|
||
'userid not equal to account_id': ExchangeError,
|
||
'your balance is low': InsufficientFunds, # {"status":"error","ts":1595594160149,"err_code":"invalid-parameter","err_msg":"invalid size, valid range: [1,2000]"}
|
||
'address invalid cointype': ExchangeError,
|
||
'system exception': ExchangeError, # {"status":"error","ts":1595711862763,"err_code":"system exception","err_msg":"system exception"}
|
||
'50003': ExchangeError, # No record
|
||
'50004': BadSymbol, # The transaction pair is currently not supported or has been suspended
|
||
'50006': PermissionDenied, # The account is forbidden to withdraw. If you have any questions, please contact customer service.
|
||
'50007': PermissionDenied, # The account is forbidden to withdraw within 24 hours. If you have any questions, please contact customer service.
|
||
'50008': RequestTimeout, # network timeout
|
||
'50009': RateLimitExceeded, # The operation is too frequent, please try again later
|
||
'50010': ExchangeError, # The account is abnormally frozen. If you have any questions, please contact customer service.
|
||
'50014': InvalidOrder, # The transaction amount under minimum limits
|
||
'50015': InvalidOrder, # The transaction amount exceed maximum limits
|
||
'50016': InvalidOrder, # The price can't be higher than the current price
|
||
'50017': InvalidOrder, # Price under minimum limits
|
||
'50018': InvalidOrder, # The price exceed maximum limits
|
||
'50019': InvalidOrder, # The amount under minimum limits
|
||
'50020': InsufficientFunds, # Insufficient balance
|
||
'50021': InvalidOrder, # Price is under minimum limits
|
||
'50026': InvalidOrder, # Market price parameter error
|
||
'invalid order query time': ExchangeError, # start time is greater than end time; or the time interval between start time and end time is greater than 48 hours
|
||
'invalid start time': BadRequest, # start time is a date 30 days ago; or start time is a date in the future
|
||
'invalid end time': BadRequest, # end time is a date 30 days ago; or end time is a date in the future
|
||
'20003': ExchangeError, # operation failed, {"status":"error","ts":1595730308979,"err_code":"bad-request","err_msg":"20003"}
|
||
'01001': ExchangeError, # order failed, {"status":"fail","err_code":"01001","err_msg":"系统异常,请稍后重试"}
|
||
'43111': PermissionDenied, # {"code":"43111","msg":"参数错误 address not in address book","requestTime":1665394201164,"data":null}
|
||
},
|
||
'broad': {
|
||
'invalid size, valid range': ExchangeError,
|
||
},
|
||
},
|
||
'precisionMode': TICK_SIZE,
|
||
'commonCurrencies': {
|
||
'APX': 'AstroPepeX',
|
||
'DEGEN': 'DegenReborn',
|
||
'EVA': 'Evadore', # conflict with EverValue Coin
|
||
'JADE': 'Jade Protocol',
|
||
'OMNI': 'omni', # conflict with Omni Network
|
||
'TONCOIN': 'TON',
|
||
},
|
||
'options': {
|
||
'uta': False,
|
||
'timeDifference': 0, # the difference between system clock and Binance clock
|
||
'adjustForTimeDifference': False, # controls the adjustment logic upon instantiation
|
||
'timeframes': {
|
||
'spot': {
|
||
'1m': '1min',
|
||
'5m': '5min',
|
||
'3m': '3min',
|
||
'15m': '15min',
|
||
'30m': '30min',
|
||
'1h': '1h',
|
||
'4h': '4h',
|
||
'6h': '6Hutc',
|
||
'12h': '12Hutc',
|
||
'1d': '1Dutc',
|
||
'3d': '3Dutc',
|
||
'1w': '1Wutc',
|
||
'1M': '1Mutc',
|
||
},
|
||
'swap': {
|
||
'1m': '1m',
|
||
'3m': '3m',
|
||
'5m': '5m',
|
||
'15m': '15m',
|
||
'30m': '30m',
|
||
'1h': '1H',
|
||
'2h': '2H',
|
||
'4h': '4H',
|
||
'6h': '6Hutc',
|
||
'12h': '12Hutc',
|
||
'1d': '1Dutc',
|
||
'3d': '3Dutc',
|
||
'1w': '1Wutc',
|
||
'1M': '1Mutc',
|
||
},
|
||
'uta': {
|
||
'1m': '1m',
|
||
'3m': '3m',
|
||
'5m': '5m',
|
||
'15m': '15m',
|
||
'30m': '30m',
|
||
'1h': '1H',
|
||
'2h': '2H',
|
||
'4h': '4H',
|
||
'6h': '6H',
|
||
'12h': '12H',
|
||
'1d': '1D',
|
||
},
|
||
},
|
||
'fetchMarkets': {
|
||
'types': ['spot', 'swap'], # there is future markets but they use the same endpoints
|
||
},
|
||
'defaultType': 'spot', # 'spot', 'swap', 'future'
|
||
'defaultSubType': 'linear', # 'linear', 'inverse'
|
||
'createMarketBuyOrderRequiresPrice': True,
|
||
'broker': 'p4sve',
|
||
'withdraw': {
|
||
'fillResponseFromRequest': True,
|
||
},
|
||
'fetchOHLCV': {
|
||
# ### Timeframe settings ###
|
||
# after testing, the below values are real ones, because the values provided by API DOCS are wrong
|
||
# so, start timestamp should be within these thresholds to be able to call "recent" candles endpoint
|
||
'maxRecentDaysPerTimeframe': {
|
||
'1m': 30,
|
||
'3m': 30,
|
||
'5m': 30,
|
||
'15m': 30,
|
||
'30m': 30,
|
||
'1h': 60,
|
||
'2h': 120,
|
||
'4h': 240,
|
||
'6h': 360,
|
||
'12h': 720,
|
||
'1d': 1440,
|
||
'3d': 1440 * 3,
|
||
'1w': 1440 * 7,
|
||
'1M': 1440 * 30,
|
||
},
|
||
'spot': {
|
||
'maxLimitPerTimeframe': {
|
||
'1d': 300,
|
||
'3d': 100,
|
||
'1w': 100,
|
||
'1M': 100,
|
||
},
|
||
'method': 'publicSpotGetV2SpotMarketCandles', # publicSpotGetV2SpotMarketCandles or publicSpotGetV2SpotMarketHistoryCandles
|
||
},
|
||
'swap': {
|
||
'maxLimitPerTimeframe': {
|
||
'4h': 540,
|
||
'6h': 360,
|
||
'12h': 180,
|
||
'1d': 90,
|
||
'3d': 30,
|
||
'1w': 13,
|
||
'1M': 4,
|
||
},
|
||
'method': 'publicMixGetV2MixMarketCandles', # publicMixGetV2MixMarketCandles or publicMixGetV2MixMarketHistoryCandles or publicMixGetV2MixMarketHistoryIndexCandles or publicMixGetV2MixMarketHistoryMarkCandles
|
||
},
|
||
},
|
||
'fetchTrades': {
|
||
'spot': {
|
||
'method': 'publicSpotGetV2SpotMarketFillsHistory', # or publicSpotGetV2SpotMarketFills
|
||
},
|
||
'swap': {
|
||
'method': 'publicMixGetV2MixMarketFillsHistory', # or publicMixGetV2MixMarketFills
|
||
},
|
||
},
|
||
'fetchFundingRate': {
|
||
'method': 'publicMixGetV2MixMarketCurrentFundRate', # or publicMixGetV2MixMarketFundingTime
|
||
},
|
||
'accountsByType': {
|
||
'spot': 'spot',
|
||
'cross': 'crossed_margin',
|
||
'isolated': 'isolated_margin',
|
||
'swap': 'usdt_futures',
|
||
'usdc_swap': 'usdc_futures',
|
||
'future': 'coin_futures',
|
||
'p2p': 'p2p',
|
||
},
|
||
'accountsById': {
|
||
'spot': 'spot',
|
||
'crossed_margin': 'cross',
|
||
'isolated_margin': 'isolated',
|
||
'usdt_futures': 'swap',
|
||
'usdc_futures': 'usdc_swap',
|
||
'coin_futures': 'future',
|
||
'p2p': 'p2p',
|
||
},
|
||
'sandboxMode': False,
|
||
'networks': {
|
||
# 'TRX': 'TRX', # different code for mainnet
|
||
'TRC20': 'TRC20',
|
||
# 'ETH': 'ETH', # different code for mainnet
|
||
'ERC20': 'ERC20',
|
||
'BEP20': 'BSC',
|
||
# 'BEP20': 'BEP20', # different for BEP20
|
||
'ATOM': 'ATOM',
|
||
'ACA': 'AcalaToken',
|
||
'APT': 'Aptos',
|
||
'ARBONE': 'ArbitrumOne',
|
||
'ARBNOVA': 'ArbitrumNova',
|
||
'AVAXC': 'C-Chain',
|
||
'AVAXX': 'X-Chain',
|
||
'AR': 'Arweave',
|
||
'BCH': 'BCH',
|
||
'BCHA': 'BCHA',
|
||
'BITCI': 'BITCI',
|
||
'BTC': 'BTC',
|
||
'CELO': 'CELO',
|
||
'CSPR': 'CSPR',
|
||
'ADA': 'Cardano',
|
||
'CHZ': 'ChilizChain',
|
||
'CRC20': 'CronosChain',
|
||
'DOGE': 'DOGE',
|
||
'DOT': 'DOT',
|
||
'EOS': 'EOS',
|
||
'ETHF': 'ETHFAIR',
|
||
'ETHW': 'ETHW',
|
||
'ETC': 'ETC',
|
||
'EGLD': 'Elrond',
|
||
'FIL': 'FIL',
|
||
'FIO': 'FIO',
|
||
'FTM': 'Fantom',
|
||
'HRC20': 'HECO',
|
||
'ONE': 'Harmony',
|
||
'HNT': 'Helium',
|
||
'ICP': 'ICP',
|
||
'IOTX': 'IoTeX',
|
||
'KARDIA': 'KAI',
|
||
'KAVA': 'KAVA',
|
||
'KDA': 'KDA',
|
||
'KLAY': 'Klaytn',
|
||
'KSM': 'Kusama',
|
||
'LAT': 'LAT',
|
||
'LTC': 'LTC',
|
||
'MINA': 'MINA',
|
||
'MOVR': 'MOVR',
|
||
'METIS': 'MetisToken',
|
||
'GLMR': 'Moonbeam',
|
||
'NEAR': 'NEARProtocol',
|
||
'NULS': 'NULS',
|
||
'OASYS': 'OASYS',
|
||
'OASIS': 'ROSE',
|
||
'OMNI': 'OMNI',
|
||
'ONT': 'Ontology',
|
||
'OPTIMISM': 'Optimism',
|
||
'OSMO': 'Osmosis',
|
||
'POKT': 'PocketNetwork',
|
||
'MATIC': 'Polygon',
|
||
'QTUM': 'QTUM',
|
||
'REEF': 'REEF',
|
||
'SOL': 'SOL',
|
||
'SYS': 'SYS', # SyscoinNEVM is different
|
||
'SXP': 'Solar',
|
||
'XYM': 'Symbol',
|
||
'TON': 'TON',
|
||
'TT': 'TT',
|
||
'TLOS': 'Telos',
|
||
'THETA': 'ThetaToken',
|
||
'VITE': 'VITE',
|
||
'WAVES': 'WAVES',
|
||
'WAX': 'WAXP',
|
||
'WEMIX': 'WEMIXMainnet',
|
||
'XDC': 'XDCNetworkXDC',
|
||
'XRP': 'XRP',
|
||
'FET': 'FETCH',
|
||
'NEM': 'NEM',
|
||
'REI': 'REINetwork',
|
||
'ZIL': 'ZIL',
|
||
'ABBC': 'ABBCCoin',
|
||
'RSK': 'RSK',
|
||
'AZERO': 'AZERO',
|
||
'TRC10': 'TRC10',
|
||
'JUNO': 'JUNO',
|
||
# undetected: USDSP, more info at https://www.bitget.com/v1/spot/public/coinChainList
|
||
# todo: uncomment below after unification
|
||
# 'TERRACLASSIC': 'Terra', # tbd, that network id is also assigned to TERRANEW network
|
||
# 'CUBENETWORK': 'CUBE',
|
||
# 'CADUCEUS': 'CMP',
|
||
# 'CONFLUX': 'CFX', # CFXeSpace is different
|
||
# 'CERE': 'CERE',
|
||
# 'CANTO': 'CANTO',
|
||
'ZKSYNC': 'zkSyncEra',
|
||
'STARKNET': 'Starknet',
|
||
'VIC': 'VICTION',
|
||
},
|
||
'networksById': {
|
||
},
|
||
'fetchPositions': {
|
||
'method': 'privateMixGetV2MixPositionAllPosition', # or privateMixGetV2MixPositionHistoryPosition
|
||
},
|
||
'defaultTimeInForce': 'GTC', # 'GTC' = Good To Cancel(default), 'IOC' = Immediate Or Cancel
|
||
# fiat currencies on deposit page
|
||
'fiatCurrencies': ['EUR', 'VND', 'PLN', 'CZK', 'HUF', 'DKK', 'AUD', 'CAD', 'NOK', 'SEK', 'CHF', 'MXN', 'COP', 'ARS', 'GBP', 'BRL', 'UAH', 'ZAR'],
|
||
},
|
||
'features': {
|
||
'spot': {
|
||
'sandbox': True,
|
||
'createOrder': {
|
||
'marginMode': True,
|
||
'triggerPrice': True,
|
||
'triggerPriceType': {
|
||
'last': True,
|
||
'mark': True,
|
||
'index': False, # not on spot
|
||
},
|
||
'triggerDirection': False,
|
||
'stopLossPrice': True, # todo: not yet implemented in spot
|
||
'takeProfitPrice': True, # todo: not yet implemented in spot
|
||
'attachedStopLossTakeProfit': {
|
||
'triggerPriceType': {
|
||
'last': False,
|
||
'mark': False,
|
||
'index': False,
|
||
},
|
||
'price': True,
|
||
},
|
||
'timeInForce': {
|
||
'IOC': True,
|
||
'FOK': True,
|
||
'PO': True,
|
||
'GTD': False,
|
||
},
|
||
'hedged': False,
|
||
'trailing': False,
|
||
'marketBuyRequiresPrice': True,
|
||
'marketBuyByCost': True,
|
||
# exchange-supported features
|
||
# 'selfTradePrevention': True,
|
||
# 'twap': False,
|
||
# 'iceberg': False,
|
||
# 'oco': False,
|
||
},
|
||
'createOrders': {
|
||
'max': 50,
|
||
},
|
||
'fetchMyTrades': {
|
||
'marginMode': True,
|
||
'limit': 100,
|
||
'daysBack': None,
|
||
'untilDays': 90,
|
||
'symbolRequired': True,
|
||
},
|
||
'fetchOrder': {
|
||
'marginMode': False,
|
||
'trigger': False,
|
||
'trailing': False,
|
||
'symbolRequired': True,
|
||
},
|
||
'fetchOpenOrders': {
|
||
'marginMode': True,
|
||
'limit': 100,
|
||
'trigger': True,
|
||
'trailing': False,
|
||
'symbolRequired': False,
|
||
},
|
||
'fetchOrders': None,
|
||
'fetchClosedOrders': {
|
||
'marginMode': True,
|
||
'limit': 100,
|
||
'daysBack': None,
|
||
'daysBackCanceled': None,
|
||
'untilDays': 90,
|
||
'trigger': True,
|
||
'trailing': False,
|
||
'symbolRequired': False,
|
||
},
|
||
'fetchOHLCV': {
|
||
'limit': 200, # variable timespans for recent endpoint, 200 for historical
|
||
},
|
||
},
|
||
'forPerps': {
|
||
'extends': 'spot',
|
||
'createOrder': {
|
||
'triggerPrice': True,
|
||
'triggerPriceType': {
|
||
'last': True,
|
||
'mark': True,
|
||
'index': False, # not on spot
|
||
},
|
||
'triggerDirection': False,
|
||
'stopLossPrice': True,
|
||
'takeProfitPrice': True,
|
||
'attachedStopLossTakeProfit': {
|
||
'triggerPriceType': {
|
||
'last': True,
|
||
'mark': True,
|
||
'index': True,
|
||
},
|
||
'price': False,
|
||
},
|
||
'timeInForce': {
|
||
'IOC': True,
|
||
'FOK': True,
|
||
'PO': True,
|
||
'GTD': False,
|
||
},
|
||
'hedged': True,
|
||
'trailing': True,
|
||
'marketBuyRequiresPrice': False,
|
||
'marketBuyByCost': False,
|
||
# exchange-supported features
|
||
# 'selfTradePrevention': True,
|
||
# 'trailing': True,
|
||
# 'twap': False,
|
||
# 'iceberg': False,
|
||
# 'oco': False,
|
||
},
|
||
'fetchMyTrades': {
|
||
'untilDays': 7,
|
||
},
|
||
'fetchClosedOrders': {
|
||
'trailing': True,
|
||
},
|
||
},
|
||
'swap': {
|
||
'linear': {
|
||
'extends': 'forPerps',
|
||
},
|
||
'inverse': {
|
||
'extends': 'forPerps',
|
||
},
|
||
},
|
||
'future': {
|
||
'linear': {
|
||
'extends': 'forPerps',
|
||
},
|
||
'inverse': {
|
||
'extends': 'forPerps',
|
||
},
|
||
},
|
||
},
|
||
})
|
||
|
||
def set_sandbox_mode(self, enabled: bool):
|
||
"""
|
||
enables or disables demo trading mode, if enabled will send PAPTRADING=1 in headers
|
||
@param enabled
|
||
"""
|
||
self.options['sandboxMode'] = enabled
|
||
|
||
def enable_demo_trading(self, enabled: bool):
|
||
"""
|
||
enables or disables demo trading mode, if enabled will send PAPTRADING=1 in headers
|
||
@param enabled
|
||
"""
|
||
self.set_sandbox_mode(enabled)
|
||
|
||
def handle_product_type_and_params(self, market=None, params={}):
|
||
subType = None
|
||
subType, params = self.handle_sub_type_and_params('handleProductTypeAndParams', None, params)
|
||
defaultProductType = None
|
||
if (subType is not None) and (market is None):
|
||
# set default only if subType is defined and market is not defined, since there is also USDC productTypes which are also linear
|
||
# sandboxMode = self.safe_bool(self.options, 'sandboxMode', False)
|
||
# if sandboxMode:
|
||
# defaultProductType = 'SUSDT-FUTURES' if (subType == 'linear') else 'SCOIN-FUTURES'
|
||
# else:
|
||
defaultProductType = 'USDT-FUTURES' if (subType == 'linear') else 'COIN-FUTURES'
|
||
# }
|
||
productType = self.safe_string_2(params, 'productType', 'category', defaultProductType)
|
||
if (productType is None) and (market is not None):
|
||
settle = market['settle']
|
||
if market['spot']:
|
||
marginMode = None
|
||
marginMode, params = self.handle_margin_mode_and_params('handleProductTypeAndParams', params)
|
||
if marginMode is not None:
|
||
productType = 'MARGIN'
|
||
else:
|
||
productType = 'SPOT'
|
||
elif settle == 'USDT':
|
||
productType = 'USDT-FUTURES'
|
||
elif settle == 'USDC':
|
||
productType = 'USDC-FUTURES'
|
||
elif settle == 'SUSDT':
|
||
productType = 'SUSDT-FUTURES'
|
||
elif settle == 'SUSDC':
|
||
productType = 'SUSDC-FUTURES'
|
||
elif (settle == 'SBTC') or (settle == 'SETH') or (settle == 'SEOS'):
|
||
productType = 'SCOIN-FUTURES'
|
||
else:
|
||
productType = 'COIN-FUTURES'
|
||
if productType is None:
|
||
raise ArgumentsRequired(self.id + ' requires a productType param, one of "USDT-FUTURES", "USDC-FUTURES", "COIN-FUTURES", "SUSDT-FUTURES", "SUSDC-FUTURES", "SCOIN-FUTURES" or for uta only "SPOT"')
|
||
params = self.omit(params, ['productType', 'category'])
|
||
return [productType, params]
|
||
|
||
async def fetch_time(self, params={}) -> Int:
|
||
"""
|
||
fetches the current integer timestamp in milliseconds from the exchange server
|
||
|
||
https://www.bitget.com/api-doc/common/public/Get-Server-Time
|
||
|
||
:param dict [params]: extra parameters specific to the exchange API endpoint
|
||
:returns int: the current integer timestamp in milliseconds from the exchange server
|
||
"""
|
||
response = await self.publicCommonGetV2PublicTime(params)
|
||
#
|
||
# {
|
||
# "code": "00000",
|
||
# "msg": "success",
|
||
# "requestTime": 1700111073740,
|
||
# "data": {
|
||
# "serverTime": "1700111073740"
|
||
# }
|
||
# }
|
||
#
|
||
data = self.safe_value(response, 'data', {})
|
||
return self.safe_integer(data, 'serverTime')
|
||
|
||
async def fetch_markets(self, params={}) -> List[Market]:
|
||
"""
|
||
retrieves data on all markets for bitget
|
||
|
||
https://www.bitget.com/api-doc/spot/market/Get-Symbols
|
||
https://www.bitget.com/api-doc/contract/market/Get-All-Symbols-Contracts
|
||
https://www.bitget.com/api-doc/margin/common/support-currencies
|
||
https://www.bitget.com/api-doc/uta/public/Instruments
|
||
|
||
:param dict [params]: extra parameters specific to the exchange API endpoint
|
||
:param boolean [params.uta]: set to True for the unified trading account(uta), defaults to False
|
||
:returns dict[]: an array of objects representing market data
|
||
"""
|
||
if self.options['adjustForTimeDifference']:
|
||
await self.load_time_difference()
|
||
uta = None
|
||
uta, params = self.handle_option_and_params(params, 'fetchMarkets', 'uta', False)
|
||
if uta:
|
||
return await self.fetch_uta_markets(params)
|
||
else:
|
||
return await self.fetch_default_markets(params)
|
||
|
||
async def fetch_default_markets(self, params) -> List[Market]:
|
||
types = None
|
||
fetchMarketsOptions = self.safe_dict(self.options, 'fetchMarkets')
|
||
defaultMarkets = ['spot', 'swap']
|
||
if fetchMarketsOptions is not None:
|
||
types = self.safe_list(fetchMarketsOptions, 'types', defaultMarkets)
|
||
else:
|
||
# for backward-compatibility
|
||
types = self.safe_list(self.options, 'fetchMarkets', defaultMarkets)
|
||
promises = []
|
||
fetchMargins = False
|
||
for i in range(0, len(types)):
|
||
type = types[i]
|
||
if (type == 'swap') or (type == 'future'):
|
||
subTypes = ['USDT-FUTURES', 'COIN-FUTURES', 'USDC-FUTURES', 'SUSDT-FUTURES', 'SCOIN-FUTURES', 'SUSDC-FUTURES']
|
||
for j in range(0, len(subTypes)):
|
||
promises.append(self.publicMixGetV2MixMarketContracts(self.extend(params, {
|
||
'productType': subTypes[j],
|
||
})))
|
||
elif type == 'spot':
|
||
promises.append(self.publicSpotGetV2SpotPublicSymbols(params))
|
||
fetchMargins = True
|
||
promises.append(self.publicMarginGetV2MarginCurrencies(params))
|
||
else:
|
||
raise NotSupported(self.id + ' does not support ' + type + ' market')
|
||
results = await asyncio.gather(*promises)
|
||
markets = []
|
||
self.options['crossMarginPairsData'] = []
|
||
self.options['isolatedMarginPairsData'] = []
|
||
for i in range(0, len(results)):
|
||
res = self.safe_dict(results, i)
|
||
data = self.safe_list(res, 'data', [])
|
||
firstData = self.safe_dict(data, 0, {})
|
||
isBorrowable = self.safe_bool(firstData, 'isBorrowable')
|
||
if fetchMargins and isBorrowable is not None:
|
||
keysList = list(self.index_by(data, 'symbol').keys())
|
||
self.options['crossMarginPairsData'] = keysList
|
||
self.options['isolatedMarginPairsData'] = keysList
|
||
else:
|
||
markets = self.array_concat(markets, data)
|
||
#
|
||
# spot
|
||
#
|
||
# {
|
||
# "symbol": "TRXUSDT",
|
||
# "baseCoin": "TRX",
|
||
# "quoteCoin": "USDT",
|
||
# "minTradeAmount": "0",
|
||
# "maxTradeAmount": "10000000000",
|
||
# "takerFeeRate": "0.002",
|
||
# "makerFeeRate": "0.002",
|
||
# "pricePrecision": "6",
|
||
# "quantityPrecision": "4",
|
||
# "quotePrecision": "6",
|
||
# "status": "online",
|
||
# "minTradeUSDT": "5",
|
||
# "buyLimitPriceRatio": "0.05",
|
||
# "sellLimitPriceRatio": "0.05"
|
||
# }
|
||
#
|
||
# swap and future
|
||
#
|
||
# {
|
||
# "symbol": "BTCUSDT",
|
||
# "baseCoin": "BTC",
|
||
# "quoteCoin": "USDT",
|
||
# "buyLimitPriceRatio": "0.01",
|
||
# "sellLimitPriceRatio": "0.01",
|
||
# "feeRateUpRatio": "0.005",
|
||
# "makerFeeRate": "0.0002",
|
||
# "takerFeeRate": "0.0006",
|
||
# "openCostUpRatio": "0.01",
|
||
# "supportMarginCoins": ["USDT"],
|
||
# "minTradeNum": "0.001",
|
||
# "priceEndStep": "1",
|
||
# "volumePlace": "3",
|
||
# "pricePlace": "1",
|
||
# "sizeMultiplier": "0.001",
|
||
# "symbolType": "perpetual",
|
||
# "minTradeUSDT": "5",
|
||
# "maxSymbolOrderNum": "200",
|
||
# "maxProductOrderNum": "400",
|
||
# "maxPositionNum": "150",
|
||
# "symbolStatus": "normal",
|
||
# "offTime": "-1",
|
||
# "limitOpenTime": "-1",
|
||
# "deliveryTime": "",
|
||
# "deliveryStartTime": "",
|
||
# "deliveryPeriod": "",
|
||
# "launchTime": "",
|
||
# "fundInterval": "8",
|
||
# "minLever": "1",
|
||
# "maxLever": "125",
|
||
# "posLimit": "0.05",
|
||
# "maintainTime": ""
|
||
# }
|
||
#
|
||
result = []
|
||
for i in range(0, len(markets)):
|
||
market = markets[i]
|
||
marketId = self.safe_string(market, 'symbol')
|
||
quoteId = self.safe_string(market, 'quoteCoin')
|
||
baseId = self.safe_string(market, 'baseCoin')
|
||
quote = self.safe_currency_code(quoteId)
|
||
base = self.safe_currency_code(baseId)
|
||
supportMarginCoins = self.safe_value(market, 'supportMarginCoins', [])
|
||
settleId = None
|
||
if self.in_array(baseId, supportMarginCoins):
|
||
settleId = baseId
|
||
elif self.in_array(quoteId, supportMarginCoins):
|
||
settleId = quoteId
|
||
else:
|
||
settleId = self.safe_string(supportMarginCoins, 0)
|
||
settle = self.safe_currency_code(settleId)
|
||
symbol = base + '/' + quote
|
||
type = None
|
||
swap = False
|
||
spot = False
|
||
future = False
|
||
contract = False
|
||
pricePrecision = None
|
||
amountPrecision = None
|
||
linear = None
|
||
inverse = None
|
||
expiry = None
|
||
expiryDatetime = None
|
||
symbolType = self.safe_string(market, 'symbolType')
|
||
marginModes = None
|
||
isMarginTradingAllowed = False
|
||
if symbolType is None:
|
||
type = 'spot'
|
||
spot = True
|
||
pricePrecision = self.parse_number(self.parse_precision(self.safe_string(market, 'pricePrecision')))
|
||
amountPrecision = self.parse_number(self.parse_precision(self.safe_string(market, 'quantityPrecision')))
|
||
hasCrossMargin = self.in_array(marketId, self.options['crossMarginPairsData'])
|
||
hasIsolatedMargin = self.in_array(marketId, self.options['isolatedMarginPairsData'])
|
||
marginModes = {
|
||
'cross': hasCrossMargin,
|
||
'isolated': hasIsolatedMargin,
|
||
}
|
||
isMarginTradingAllowed = hasCrossMargin or hasIsolatedMargin
|
||
else:
|
||
if symbolType == 'perpetual':
|
||
type = 'swap'
|
||
swap = True
|
||
symbol = symbol + ':' + settle
|
||
elif symbolType == 'delivery':
|
||
expiry = self.safe_integer(market, 'deliveryTime')
|
||
expiryDatetime = self.iso8601(expiry)
|
||
expiryParts = expiryDatetime.split('-')
|
||
yearPart = self.safe_string(expiryParts, 0)
|
||
dayPart = self.safe_string(expiryParts, 2)
|
||
year = yearPart[2:4]
|
||
month = self.safe_string(expiryParts, 1)
|
||
day = dayPart[0:2]
|
||
expiryString = year + month + day
|
||
type = 'future'
|
||
future = True
|
||
symbol = symbol + ':' + settle + '-' + expiryString
|
||
contract = True
|
||
inverse = (base == settle)
|
||
linear = not inverse
|
||
priceDecimals = self.safe_integer(market, 'pricePlace')
|
||
amountDecimals = self.safe_integer(market, 'volumePlace')
|
||
priceStep = self.safe_string(market, 'priceEndStep')
|
||
amountStep = self.safe_string(market, 'sizeMultiplier')
|
||
precise = Precise(priceStep)
|
||
precise.decimals = max(precise.decimals, priceDecimals)
|
||
precise.reduce()
|
||
priceString = str(precise)
|
||
pricePrecision = self.parse_number(priceString)
|
||
preciseAmount = Precise(amountStep)
|
||
preciseAmount.decimals = max(preciseAmount.decimals, amountDecimals)
|
||
preciseAmount.reduce()
|
||
amountString = str(preciseAmount)
|
||
amountPrecision = self.parse_number(amountString)
|
||
marginModes = {
|
||
'cross': True,
|
||
'isolated': True,
|
||
}
|
||
status = self.safe_string_2(market, 'status', 'symbolStatus')
|
||
active = None
|
||
if status is not None:
|
||
active = ((status == 'online') or (status == 'normal'))
|
||
minCost = None
|
||
if quote == 'USDT':
|
||
minCost = self.safe_number(market, 'minTradeUSDT')
|
||
contractSize = 1 if contract else None
|
||
result.append(self.safe_market_structure({
|
||
'id': marketId,
|
||
'symbol': symbol,
|
||
'base': base,
|
||
'quote': quote,
|
||
'settle': settle,
|
||
'baseId': baseId,
|
||
'quoteId': quoteId,
|
||
'settleId': settleId,
|
||
'type': type,
|
||
'spot': spot,
|
||
'margin': spot and isMarginTradingAllowed,
|
||
'marginModes': marginModes,
|
||
'swap': swap,
|
||
'future': future,
|
||
'option': False,
|
||
'active': active,
|
||
'contract': contract,
|
||
'linear': linear,
|
||
'inverse': inverse,
|
||
'taker': self.safe_number(market, 'takerFeeRate'),
|
||
'maker': self.safe_number(market, 'makerFeeRate'),
|
||
'contractSize': contractSize,
|
||
'expiry': expiry,
|
||
'expiryDatetime': expiryDatetime,
|
||
'strike': None,
|
||
'optionType': None,
|
||
'precision': {
|
||
'amount': amountPrecision,
|
||
'price': pricePrecision,
|
||
},
|
||
'limits': {
|
||
'leverage': {
|
||
'min': self.safe_number(market, 'minLever'),
|
||
'max': self.safe_number(market, 'maxLever'),
|
||
},
|
||
'amount': {
|
||
'min': self.safe_number_2(market, 'minTradeNum', 'minTradeAmount'),
|
||
'max': self.safe_number(market, 'maxTradeAmount'),
|
||
},
|
||
'price': {
|
||
'min': None,
|
||
'max': None,
|
||
},
|
||
'cost': {
|
||
'min': minCost,
|
||
'max': None,
|
||
},
|
||
},
|
||
'created': self.safe_integer(market, 'launchTime'),
|
||
'info': market,
|
||
}))
|
||
return result
|
||
|
||
async def fetch_uta_markets(self, params) -> List[Market]:
|
||
subTypes = ['SPOT', 'USDT-FUTURES', 'COIN-FUTURES', 'USDC-FUTURES']
|
||
promises = []
|
||
for i in range(0, len(subTypes)):
|
||
req = self.extend(params, {
|
||
'category': subTypes[i],
|
||
})
|
||
promises.append(self.publicUtaGetV3MarketInstruments(req))
|
||
results = await asyncio.gather(*promises)
|
||
markets = []
|
||
for i in range(0, len(results)):
|
||
res = self.safe_dict(results, i)
|
||
data = self.safe_list(res, 'data', [])
|
||
markets = self.array_concat(markets, data)
|
||
#
|
||
# spot uta
|
||
#
|
||
# {
|
||
# "symbol": "BTCUSDT",
|
||
# "category": "SPOT",
|
||
# "baseCoin": "BTC",
|
||
# "quoteCoin": "USDT",
|
||
# "buyLimitPriceRatio": "0.05",
|
||
# "sellLimitPriceRatio": "0.05",
|
||
# "minOrderQty": "0.000001",
|
||
# "maxOrderQty": "0",
|
||
# "pricePrecision": "2",
|
||
# "quantityPrecision": "6",
|
||
# "quotePrecision": "8",
|
||
# "minOrderAmount": "1",
|
||
# "maxSymbolOrderNum": "400",
|
||
# "maxProductOrderNum": "400",
|
||
# "status": "online",
|
||
# "maintainTime": ""
|
||
# }
|
||
#
|
||
# margin uta
|
||
#
|
||
# {
|
||
# "symbol": "BTCUSDC",
|
||
# "category": "MARGIN",
|
||
# "baseCoin": "BTC",
|
||
# "quoteCoin": "USDC",
|
||
# "buyLimitPriceRatio": "0.05",
|
||
# "sellLimitPriceRatio": "0.05",
|
||
# "minOrderQty": "0.00001",
|
||
# "maxOrderQty": "0",
|
||
# "pricePrecision": "2",
|
||
# "quantityPrecision": "5",
|
||
# "quotePrecision": "7",
|
||
# "minOrderAmount": "1",
|
||
# "maxSymbolOrderNum": "400",
|
||
# "maxProductOrderNum": "400",
|
||
# "status": "online",
|
||
# "maintainTime": "",
|
||
# "isIsolatedBaseBorrowable": "NO",
|
||
# "isIsolatedQuotedBorrowable": "NO",
|
||
# "warningRiskRatio": "0.8",
|
||
# "liquidationRiskRatio": "1",
|
||
# "maxCrossedLeverage": "3",
|
||
# "maxIsolatedLeverage": "0",
|
||
# "userMinBorrow": "0.00000001",
|
||
# "areaSymbol": "no"
|
||
# }
|
||
#
|
||
# swap and future uta
|
||
#
|
||
# {
|
||
# "symbol": "BTCPERP",
|
||
# "category": "USDC-FUTURES",
|
||
# "baseCoin": "BTC",
|
||
# "quoteCoin": "USDC",
|
||
# "buyLimitPriceRatio": "0.02",
|
||
# "sellLimitPriceRatio": "0.02",
|
||
# "feeRateUpRatio": "0.005",
|
||
# "makerFeeRate": "0.0002",
|
||
# "takerFeeRate": "0.0006",
|
||
# "openCostUpRatio": "0.01",
|
||
# "minOrderQty": "0.0001",
|
||
# "maxOrderQty": "",
|
||
# "pricePrecision": "1",
|
||
# "quantityPrecision": "4",
|
||
# "quotePrecision": null,
|
||
# "priceMultiplier": "0.5",
|
||
# "quantityMultiplier": "0.0001",
|
||
# "type": "perpetual",
|
||
# "minOrderAmount": "5",
|
||
# "maxSymbolOrderNum": "200",
|
||
# "maxProductOrderNum": "1000",
|
||
# "maxPositionNum": "150",
|
||
# "status": "online",
|
||
# "offTime": "-1",
|
||
# "limitOpenTime": "-1",
|
||
# "deliveryTime": "",
|
||
# "deliveryStartTime": "",
|
||
# "deliveryPeriod": "",
|
||
# "launchTime": "",
|
||
# "fundInterval": "8",
|
||
# "minLeverage": "1",
|
||
# "maxLeverage": "125",
|
||
# "maintainTime": ""
|
||
# }
|
||
#
|
||
result = []
|
||
for i in range(0, len(markets)):
|
||
market = markets[i]
|
||
category = self.safe_string(market, 'category')
|
||
marketId = self.safe_string(market, 'symbol')
|
||
quoteId = self.safe_string(market, 'quoteCoin')
|
||
baseId = self.safe_string(market, 'baseCoin')
|
||
quote = self.safe_currency_code(quoteId)
|
||
base = self.safe_currency_code(baseId)
|
||
settleId = None
|
||
settle = None
|
||
if category == 'USDT-FUTURES':
|
||
settleId = 'USDT'
|
||
elif category == 'USDC-FUTURES':
|
||
settleId = 'USDC'
|
||
elif category == 'COIN-FUTURES':
|
||
settleId = base
|
||
if settleId is not None:
|
||
settle = self.safe_currency_code(settleId)
|
||
symbol = base + '/' + quote
|
||
type = None
|
||
swap = False
|
||
spot = False
|
||
future = False
|
||
contract = False
|
||
pricePrecision = None
|
||
amountPrecision = None
|
||
linear = None
|
||
inverse = None
|
||
expiry = None
|
||
expiryDatetime = None
|
||
symbolType = self.safe_string(market, 'type')
|
||
marginModes = None
|
||
isMarginTradingAllowed = False
|
||
isUtaMargin = (category == 'MARGIN')
|
||
if isUtaMargin or (category == 'SPOT'):
|
||
type = 'spot'
|
||
spot = True
|
||
if isUtaMargin:
|
||
isolatedBase = self.safe_string(market, 'isIsolatedBaseBorrowable')
|
||
isolatedQuote = self.safe_string(market, 'isIsolatedQuotedBorrowable')
|
||
isolated = (isolatedBase == 'YES') or (isolatedQuote == 'YES')
|
||
maxCrossLeverage = self.safe_string(market, 'maxCrossedLeverage')
|
||
cross = (maxCrossLeverage != '0')
|
||
marginModes = {
|
||
'cross': cross,
|
||
'isolated': isolated,
|
||
}
|
||
isMarginTradingAllowed = True
|
||
else:
|
||
if symbolType == 'perpetual':
|
||
type = 'swap'
|
||
swap = True
|
||
symbol = symbol + ':' + settle
|
||
elif symbolType == 'delivery':
|
||
expiry = self.safe_integer(market, 'deliveryTime')
|
||
expiryDatetime = self.iso8601(expiry)
|
||
expiryParts = expiryDatetime.split('-')
|
||
yearPart = self.safe_string(expiryParts, 0)
|
||
dayPart = self.safe_string(expiryParts, 2)
|
||
year = yearPart[2:4]
|
||
month = self.safe_string(expiryParts, 1)
|
||
day = dayPart[0:2]
|
||
expiryString = year + month + day
|
||
type = 'future'
|
||
future = True
|
||
symbol = symbol + ':' + settle + '-' + expiryString
|
||
contract = True
|
||
inverse = (base == settle)
|
||
linear = not inverse
|
||
marginModes = {
|
||
'cross': True,
|
||
'isolated': True,
|
||
}
|
||
pricePrecision = self.parse_number(self.parse_precision(self.safe_string(market, 'pricePrecision')))
|
||
amountPrecision = self.parse_number(self.parse_precision(self.safe_string(market, 'quantityPrecision')))
|
||
status = self.safe_string(market, 'status')
|
||
active = None
|
||
if status is not None:
|
||
active = ((status == 'online') or (status == 'normal'))
|
||
contractSize = 1 if contract else None
|
||
result.append(self.safe_market_structure({
|
||
'id': marketId,
|
||
'symbol': symbol,
|
||
'base': base,
|
||
'quote': quote,
|
||
'settle': settle,
|
||
'baseId': baseId,
|
||
'quoteId': quoteId,
|
||
'settleId': settleId,
|
||
'type': type,
|
||
'spot': spot,
|
||
'margin': spot and isMarginTradingAllowed,
|
||
'marginModes': marginModes,
|
||
'swap': swap,
|
||
'future': future,
|
||
'option': False,
|
||
'active': active,
|
||
'contract': contract,
|
||
'linear': linear,
|
||
'inverse': inverse,
|
||
'taker': self.safe_number(market, 'takerFeeRate'),
|
||
'maker': self.safe_number(market, 'makerFeeRate'),
|
||
'contractSize': contractSize,
|
||
'expiry': expiry,
|
||
'expiryDatetime': expiryDatetime,
|
||
'strike': None,
|
||
'optionType': None,
|
||
'precision': {
|
||
'amount': amountPrecision,
|
||
'price': pricePrecision,
|
||
},
|
||
'limits': {
|
||
'leverage': {
|
||
'min': self.safe_number(market, 'minLeverage'),
|
||
'max': self.safe_number(market, 'maxLeverage'),
|
||
},
|
||
'amount': {
|
||
'min': self.safe_number(market, 'minOrderQty'),
|
||
'max': self.safe_number(market, 'maxOrderQty'),
|
||
},
|
||
'price': {
|
||
'min': None,
|
||
'max': None,
|
||
},
|
||
'cost': {
|
||
'min': None,
|
||
'max': None,
|
||
},
|
||
},
|
||
'created': self.safe_integer(market, 'launchTime'),
|
||
'info': market,
|
||
}))
|
||
return result
|
||
|
||
async def fetch_currencies(self, params={}) -> Currencies:
|
||
"""
|
||
fetches all available currencies on an exchange
|
||
|
||
https://www.bitget.com/api-doc/spot/market/Get-Coin-List
|
||
|
||
:param dict [params]: extra parameters specific to the exchange API endpoint
|
||
:returns dict: an associative dictionary of currencies
|
||
"""
|
||
response = await self.publicSpotGetV2SpotPublicCoins(params)
|
||
#
|
||
# {
|
||
# "code": "00000",
|
||
# "msg": "success",
|
||
# "requestTime": "1746195617812",
|
||
# "data": [
|
||
# {
|
||
# "coinId": "1456",
|
||
# "coin": "NEIROETH",
|
||
# "transfer": "false",
|
||
# "chains": [
|
||
# {
|
||
# "chain": "ERC20",
|
||
# "needTag": "false",
|
||
# "withdrawable": "true",
|
||
# "rechargeable": "true",
|
||
# "withdrawFee": "44.91017965",
|
||
# "extraWithdrawFee": "0",
|
||
# "depositConfirm": "12",
|
||
# "withdrawConfirm": "64",
|
||
# "minDepositAmount": "0.06",
|
||
# "minWithdrawAmount": "60",
|
||
# "browserUrl": "https://etherscan.io/tx/",
|
||
# "contractAddress": "0xee2a03aa6dacf51c18679c516ad5283d8e7c2637",
|
||
# "withdrawStep": "0",
|
||
# "withdrawMinScale": "8",
|
||
# "congestion": "normal"
|
||
# }
|
||
# ],
|
||
# "areaCoin": "no"
|
||
# },
|
||
# ...
|
||
#
|
||
result: dict = {}
|
||
data = self.safe_value(response, 'data', [])
|
||
fiatCurrencies = self.safe_list(self.options, 'fiatCurrencies', [])
|
||
for i in range(0, len(data)):
|
||
entry = data[i]
|
||
id = self.safe_string(entry, 'coin') # we don't use 'coinId' has no use. it is 'coin' field that needs to be used in currency related endpoints(deposit, withdraw, etc..)
|
||
code = self.safe_currency_code(id)
|
||
chains = self.safe_value(entry, 'chains', [])
|
||
networks: dict = {}
|
||
withdraw = None
|
||
deposit = None
|
||
chainsLength = len(chains)
|
||
if chainsLength == 0:
|
||
withdraw = False
|
||
deposit = False
|
||
for j in range(0, chainsLength):
|
||
chain = chains[j]
|
||
networkId = self.safe_string(chain, 'chain')
|
||
network = self.network_id_to_code(networkId, code)
|
||
network = network.upper()
|
||
withdrawable = (self.safe_string(chain, 'withdrawable') == 'true')
|
||
rechargeable = (self.safe_string(chain, 'rechargeable') == 'true')
|
||
withdraw = withdrawable if (withdraw is None) else (withdraw or withdrawable)
|
||
deposit = rechargeable if (deposit is None) else (deposit or rechargeable)
|
||
networks[network] = {
|
||
'info': chain,
|
||
'id': networkId,
|
||
'network': network,
|
||
'limits': {
|
||
'withdraw': {
|
||
'min': self.safe_number(chain, 'minWithdrawAmount'),
|
||
'max': None,
|
||
},
|
||
'deposit': {
|
||
'min': self.safe_number(chain, 'minDepositAmount'),
|
||
'max': None,
|
||
},
|
||
},
|
||
'active': None,
|
||
'withdraw': withdrawable,
|
||
'deposit': rechargeable,
|
||
'fee': self.safe_number(chain, 'withdrawFee'),
|
||
'precision': self.parse_number(self.parse_precision(self.safe_string(chain, 'withdrawMinScale'))),
|
||
}
|
||
active = withdraw and deposit
|
||
isFiat = self.in_array(code, fiatCurrencies)
|
||
result[code] = self.safe_currency_structure({
|
||
'info': entry,
|
||
'id': id,
|
||
'code': code,
|
||
'networks': networks,
|
||
'type': 'fiat' if isFiat else 'crypto',
|
||
'name': None,
|
||
'active': active,
|
||
'deposit': deposit,
|
||
'withdraw': withdraw,
|
||
'fee': None,
|
||
'precision': None,
|
||
'limits': {
|
||
'amount': {
|
||
'min': None,
|
||
'max': None,
|
||
},
|
||
'withdraw': {
|
||
'min': None,
|
||
'max': None,
|
||
},
|
||
'deposit': {
|
||
'min': None,
|
||
'max': None,
|
||
},
|
||
},
|
||
'created': None,
|
||
})
|
||
return result
|
||
|
||
async def fetch_market_leverage_tiers(self, symbol: str, params={}) -> List[LeverageTier]:
|
||
"""
|
||
retrieve information on the maximum leverage, and maintenance margin for trades of varying trade sizes for a single market
|
||
|
||
https://www.bitget.com/api-doc/contract/position/Get-Query-Position-Lever
|
||
https://www.bitget.com/api-doc/margin/cross/account/Cross-Tier-Data
|
||
https://www.bitget.com/api-doc/margin/isolated/account/Isolated-Tier-Data
|
||
https://www.bitget.com/api-doc/uta/public/Get-Position-Tier-Data
|
||
|
||
:param str symbol: unified market symbol
|
||
:param dict [params]: extra parameters specific to the exchange API endpoint
|
||
:param str [params.marginMode]: for spot margin 'cross' or 'isolated', default is 'isolated'
|
||
:param str [params.code]: required for cross spot margin
|
||
:param str [params.productType]: *contract and uta only* 'USDT-FUTURES', 'USDC-FUTURES', 'COIN-FUTURES', 'SUSDT-FUTURES', 'SUSDC-FUTURES' or 'SCOIN-FUTURES'
|
||
:param boolean [params.uta]: set to True for the unified trading account(uta), defaults to False
|
||
:returns dict: a `leverage tiers structure <https://docs.ccxt.com/#/?id=leverage-tiers-structure>`
|
||
"""
|
||
await self.load_markets()
|
||
market = self.market(symbol)
|
||
request: dict = {}
|
||
response = None
|
||
marginMode = None
|
||
productType = None
|
||
uta = None
|
||
marginMode, params = self.handle_margin_mode_and_params('fetchMarketLeverageTiers', params, 'isolated')
|
||
productType, params = self.handle_product_type_and_params(market, params)
|
||
uta, params = self.handle_option_and_params(params, 'fetchMarketLeverageTiers', 'uta', False)
|
||
if uta:
|
||
if productType == 'SPOT':
|
||
if marginMode is not None:
|
||
productType = 'MARGIN'
|
||
request['symbol'] = market['id']
|
||
request['category'] = productType
|
||
response = await self.publicUtaGetV3MarketPositionTier(self.extend(request, params))
|
||
elif (market['swap']) or (market['future']):
|
||
request['productType'] = productType
|
||
request['symbol'] = market['id']
|
||
response = await self.publicMixGetV2MixMarketQueryPositionLever(self.extend(request, params))
|
||
elif marginMode == 'isolated':
|
||
request['symbol'] = market['id']
|
||
response = await self.privateMarginGetV2MarginIsolatedTierData(self.extend(request, params))
|
||
elif marginMode == 'cross':
|
||
code = self.safe_string(params, 'code')
|
||
if code is None:
|
||
raise ArgumentsRequired(self.id + ' fetchMarketLeverageTiers() requires a code argument')
|
||
params = self.omit(params, 'code')
|
||
currency = self.currency(code)
|
||
request['coin'] = currency['id']
|
||
response = await self.privateMarginGetV2MarginCrossedTierData(self.extend(request, params))
|
||
else:
|
||
raise BadRequest(self.id + ' fetchMarketLeverageTiers() symbol does not support market ' + market['symbol'])
|
||
#
|
||
# swap and future
|
||
#
|
||
# {
|
||
# "code": "00000",
|
||
# "msg": "success",
|
||
# "requestTime": 1700290724614,
|
||
# "data": [
|
||
# {
|
||
# "symbol": "BTCUSDT",
|
||
# "level": "1",
|
||
# "startUnit": "0",
|
||
# "endUnit": "150000",
|
||
# "leverage": "125",
|
||
# "keepMarginRate": "0.004"
|
||
# },
|
||
# ]
|
||
# }
|
||
#
|
||
# isolated
|
||
#
|
||
# {
|
||
# "code": "00000",
|
||
# "msg": "success",
|
||
# "requestTime": 1700291531894,
|
||
# "data": [
|
||
# {
|
||
# "tier": "1",
|
||
# "symbol": "BTCUSDT",
|
||
# "leverage": "10",
|
||
# "baseCoin": "BTC",
|
||
# "quoteCoin": "USDT",
|
||
# "baseMaxBorrowableAmount": "2",
|
||
# "quoteMaxBorrowableAmount": "24000",
|
||
# "maintainMarginRate": "0.05",
|
||
# "initRate": "0.1111"
|
||
# },
|
||
# ]
|
||
# }
|
||
#
|
||
# cross
|
||
#
|
||
# {
|
||
# "code": "00000",
|
||
# "msg": "success",
|
||
# "requestTime": 1700291818831,
|
||
# "data": [
|
||
# {
|
||
# "tier": "1",
|
||
# "leverage": "3",
|
||
# "coin": "BTC",
|
||
# "maxBorrowableAmount": "26",
|
||
# "maintainMarginRate": "0.1"
|
||
# }
|
||
# ]
|
||
# }
|
||
#
|
||
# uta
|
||
#
|
||
# {
|
||
# "code": "00000",
|
||
# "msg": "success",
|
||
# "requestTime": 1752735673127,
|
||
# "data": [
|
||
# {
|
||
# "tier": "1",
|
||
# "minTierValue": "0",
|
||
# "maxTierValue": "150000",
|
||
# "leverage": "125",
|
||
# "mmr": "0.004"
|
||
# },
|
||
# ]
|
||
# }
|
||
#
|
||
result = self.safe_value(response, 'data', [])
|
||
return self.parse_market_leverage_tiers(result, market)
|
||
|
||
def parse_market_leverage_tiers(self, info, market: Market = None) -> List[LeverageTier]:
|
||
#
|
||
# swap and future
|
||
#
|
||
# {
|
||
# "symbol": "BTCUSDT",
|
||
# "level": "1",
|
||
# "startUnit": "0",
|
||
# "endUnit": "150000",
|
||
# "leverage": "125",
|
||
# "keepMarginRate": "0.004"
|
||
# }
|
||
#
|
||
# isolated
|
||
#
|
||
# {
|
||
# "tier": "1",
|
||
# "symbol": "BTCUSDT",
|
||
# "leverage": "10",
|
||
# "baseCoin": "BTC",
|
||
# "quoteCoin": "USDT",
|
||
# "baseMaxBorrowableAmount": "2",
|
||
# "quoteMaxBorrowableAmount": "24000",
|
||
# "maintainMarginRate": "0.05",
|
||
# "initRate": "0.1111"
|
||
# }
|
||
#
|
||
# cross
|
||
#
|
||
# {
|
||
# "tier": "1",
|
||
# "leverage": "3",
|
||
# "coin": "BTC",
|
||
# "maxBorrowableAmount": "26",
|
||
# "maintainMarginRate": "0.1"
|
||
# }
|
||
#
|
||
# uta
|
||
#
|
||
# {
|
||
# "tier": "1",
|
||
# "minTierValue": "0",
|
||
# "maxTierValue": "150000",
|
||
# "leverage": "125",
|
||
# "mmr": "0.004"
|
||
# }
|
||
#
|
||
tiers = []
|
||
minNotional = 0
|
||
for i in range(0, len(info)):
|
||
item = info[i]
|
||
minimumNotional = self.safe_number_2(item, 'startUnit', 'minTierValue')
|
||
if minimumNotional is not None:
|
||
minNotional = minimumNotional
|
||
maxNotional = self.safe_number_n(item, ['endUnit', 'maxBorrowableAmount', 'baseMaxBorrowableAmount', 'maxTierValue'])
|
||
marginCurrency = self.safe_string_2(item, 'coin', 'baseCoin')
|
||
currencyId = marginCurrency if (marginCurrency is not None) else market['base']
|
||
marketId = self.safe_string(item, 'symbol')
|
||
tiers.append({
|
||
'tier': self.safe_integer_2(item, 'level', 'tier'),
|
||
'symbol': self.safe_symbol(marketId, market),
|
||
'currency': self.safe_currency_code(currencyId),
|
||
'minNotional': minNotional,
|
||
'maxNotional': maxNotional,
|
||
'maintenanceMarginRate': self.safe_number_n(item, ['keepMarginRate', 'maintainMarginRate', 'mmr']),
|
||
'maxLeverage': self.safe_number(item, 'leverage'),
|
||
'info': item,
|
||
})
|
||
minNotional = maxNotional
|
||
return tiers
|
||
|
||
async def fetch_deposits(self, code: Str = None, since: Int = None, limit: Int = None, params={}) -> List[Transaction]:
|
||
"""
|
||
fetch all deposits made to an account
|
||
|
||
https://www.bitget.com/api-doc/spot/account/Get-Deposit-Record
|
||
|
||
:param str code: unified currency code
|
||
:param int [since]: the earliest time in ms to fetch deposits for
|
||
:param int [limit]: the maximum number of deposits structures to retrieve
|
||
:param dict [params]: extra parameters specific to the exchange API endpoint
|
||
:param int [params.until]: end time in milliseconds
|
||
:param str [params.idLessThan]: return records with id less than the provided value
|
||
:param boolean [params.paginate]: default False, when True will automatically paginate by calling self endpoint multiple times. See in the docs all the [available parameters](https://github.com/ccxt/ccxt/wiki/Manual#pagination-params)
|
||
:returns dict[]: a list of `transaction structures <https://docs.ccxt.com/#/?id=transaction-structure>`
|
||
"""
|
||
await self.load_markets()
|
||
paginate = False
|
||
paginate, params = self.handle_option_and_params(params, 'fetchDeposits', 'paginate')
|
||
if paginate:
|
||
return await self.fetch_paginated_call_cursor('fetchDeposits', None, since, limit, params, 'idLessThan', 'idLessThan', None, 100)
|
||
if since is None:
|
||
since = self.milliseconds() - 7776000000 # 90 days
|
||
request: dict = {
|
||
'startTime': since,
|
||
'endTime': self.milliseconds(),
|
||
}
|
||
currency = None
|
||
if code is not None:
|
||
currency = self.currency(code)
|
||
request['coin'] = currency['id']
|
||
if limit is not None:
|
||
request['limit'] = limit
|
||
request, params = self.handle_until_option('endTime', request, params)
|
||
response = await self.privateSpotGetV2SpotWalletDepositRecords(self.extend(request, params))
|
||
#
|
||
# {
|
||
# "code": "00000",
|
||
# "msg": "success",
|
||
# "requestTime": 1700528340608,
|
||
# "data": [
|
||
# {
|
||
# "orderId": "1083832260799930368",
|
||
# "tradeId": "35bf0e588a42b25c71a9d45abe7308cabdeec6b7b423910b9bd4743d3a9a9efa",
|
||
# "coin": "BTC",
|
||
# "type": "deposit",
|
||
# "size": "0.00030000",
|
||
# "status": "success",
|
||
# "toAddress": "1BfZh7JESJGBUszCGeZnzxbVVvBycbJSbA",
|
||
# "dest": "on_chain",
|
||
# "chain": "BTC",
|
||
# "fromAddress": null,
|
||
# "cTime": "1694131668281",
|
||
# "uTime": "1694131680247"
|
||
# }
|
||
# ]
|
||
# }
|
||
#
|
||
rawTransactions = self.safe_list(response, 'data', [])
|
||
return self.parse_transactions(rawTransactions, None, since, limit)
|
||
|
||
async def withdraw(self, code: str, amount: float, address: str, tag: Str = None, params={}) -> Transaction:
|
||
"""
|
||
make a withdrawal
|
||
|
||
https://www.bitget.com/api-doc/spot/account/Wallet-Withdrawal
|
||
|
||
:param str code: unified currency code
|
||
:param float amount: the amount to withdraw
|
||
:param str address: the address to withdraw to
|
||
:param str tag:
|
||
:param dict [params]: extra parameters specific to the exchange API endpoint
|
||
:param str [params.chain]: the blockchain network the withdrawal is taking place on
|
||
:returns dict: a `transaction structure <https://docs.ccxt.com/#/?id=transaction-structure>`
|
||
"""
|
||
self.check_address(address)
|
||
networkCode = None
|
||
networkCode, params = self.handle_network_code_and_params(params)
|
||
if networkCode is None:
|
||
raise ArgumentsRequired(self.id + ' withdraw() requires a "network" parameter')
|
||
await self.load_markets()
|
||
currency = self.currency(code)
|
||
networkId = self.network_code_to_id(networkCode)
|
||
request: dict = {
|
||
'coin': currency['id'],
|
||
'address': address,
|
||
'chain': networkId,
|
||
'size': self.currency_to_precision(code, amount, networkCode),
|
||
'transferType': 'on_chain',
|
||
}
|
||
if tag is not None:
|
||
request['tag'] = tag
|
||
response = await self.privateSpotPostV2SpotWalletWithdrawal(self.extend(request, params))
|
||
#
|
||
# {
|
||
# "code":"00000",
|
||
# "msg":"success",
|
||
# "requestTime":1696784219602,
|
||
# "data": {
|
||
# "orderId":"1094957867615789056",
|
||
# "clientOid":"64f1e4ce842041d296b4517df1b5c2d7"
|
||
# }
|
||
# }
|
||
#
|
||
data = self.safe_value(response, 'data', {})
|
||
result = self.parse_transaction(data, currency)
|
||
result['type'] = 'withdrawal'
|
||
withdrawOptions = self.safe_value(self.options, 'withdraw', {})
|
||
fillResponseFromRequest = self.safe_bool(withdrawOptions, 'fillResponseFromRequest', True)
|
||
if fillResponseFromRequest:
|
||
result['currency'] = code
|
||
result['amount'] = amount
|
||
result['tag'] = tag
|
||
result['address'] = address
|
||
result['addressTo'] = address
|
||
result['network'] = networkCode
|
||
return result
|
||
|
||
async def fetch_withdrawals(self, code: Str = None, since: Int = None, limit: Int = None, params={}) -> List[Transaction]:
|
||
"""
|
||
fetch all withdrawals made from an account
|
||
|
||
https://www.bitget.com/api-doc/spot/account/Get-Withdraw-Record
|
||
|
||
:param str code: unified currency code
|
||
:param int [since]: the earliest time in ms to fetch withdrawals for
|
||
:param int [limit]: the maximum number of withdrawals structures to retrieve
|
||
:param dict [params]: extra parameters specific to the exchange API endpoint
|
||
:param int [params.until]: end time in milliseconds
|
||
:param str [params.idLessThan]: return records with id less than the provided value
|
||
:param boolean [params.paginate]: default False, when True will automatically paginate by calling self endpoint multiple times. See in the docs all the [available parameters](https://github.com/ccxt/ccxt/wiki/Manual#pagination-params)
|
||
:returns dict[]: a list of `transaction structures <https://docs.ccxt.com/#/?id=transaction-structure>`
|
||
"""
|
||
await self.load_markets()
|
||
paginate = False
|
||
paginate, params = self.handle_option_and_params(params, 'fetchWithdrawals', 'paginate')
|
||
if paginate:
|
||
return await self.fetch_paginated_call_cursor('fetchWithdrawals', None, since, limit, params, 'idLessThan', 'idLessThan', None, 100)
|
||
currency = None
|
||
if code is not None:
|
||
currency = self.currency(code)
|
||
if since is None:
|
||
since = self.milliseconds() - 7776000000 # 90 days
|
||
request: dict = {
|
||
'startTime': since,
|
||
'endTime': self.milliseconds(),
|
||
}
|
||
if currency is not None:
|
||
request['coin'] = currency['id']
|
||
request, params = self.handle_until_option('endTime', request, params)
|
||
if limit is not None:
|
||
request['limit'] = limit
|
||
response = await self.privateSpotGetV2SpotWalletWithdrawalRecords(self.extend(request, params))
|
||
#
|
||
# {
|
||
# "code": "00000",
|
||
# "msg": "success",
|
||
# "requestTime": 1700528340608,
|
||
# "data": [
|
||
# {
|
||
# "orderId": "1083832260799930368",
|
||
# "tradeId": "35bf0e588a42b25c71a9d45abe7308cabdeec6b7b423910b9bd4743d3a9a9efa",
|
||
# "clientOid": "123",
|
||
# "coin": "BTC",
|
||
# "type": "withdraw",
|
||
# "size": "0.00030000",
|
||
# "fee": "-1.0000000",
|
||
# "status": "success",
|
||
# "toAddress": "1BfZh7JESJGBUszCGeZnzxbVVvBycbJSbA",
|
||
# "dest": "on_chain",
|
||
# "chain": "BTC",
|
||
# "confirm": "100",
|
||
# "fromAddress": null,
|
||
# "cTime": "1694131668281",
|
||
# "uTime": "1694131680247"
|
||
# }
|
||
# ]
|
||
# }
|
||
#
|
||
rawTransactions = self.safe_list(response, 'data', [])
|
||
return self.parse_transactions(rawTransactions, currency, since, limit)
|
||
|
||
def parse_transaction(self, transaction: dict, currency: Currency = None) -> Transaction:
|
||
#
|
||
# fetchDeposits
|
||
#
|
||
# {
|
||
# "orderId": "1083832260799930368",
|
||
# "tradeId": "35bf0e588a42b25c71a9d45abe7308cabdeec6b7b423910b9bd4743d3a9a9efa",
|
||
# "coin": "BTC",
|
||
# "type": "deposit",
|
||
# "size": "0.00030000",
|
||
# "status": "success",
|
||
# "toAddress": "1BfZh7JESJGBUszCGeZnzxbVVvBycbJSbA",
|
||
# "dest": "on_chain",
|
||
# "chain": "BTC",
|
||
# "fromAddress": null,
|
||
# "cTime": "1694131668281",
|
||
# "uTime": "1694131680247"
|
||
# }
|
||
#
|
||
# fetchWithdrawals
|
||
#
|
||
# {
|
||
# "orderId": "1083832260799930368",
|
||
# "tradeId": "35bf0e588a42b25c71a9d45abe7308cabdeec6b7b423910b9bd4743d3a9a9efa",
|
||
# "clientOid": "123",
|
||
# "coin": "BTC",
|
||
# "type": "withdraw",
|
||
# "size": "0.00030000",
|
||
# "fee": "-1.0000000",
|
||
# "status": "success",
|
||
# "toAddress": "1BfZh7JESJGBUszCGeZnzxbVVvBycbJSbA",
|
||
# "dest": "on_chain",
|
||
# "chain": "BTC",
|
||
# "confirm": "100",
|
||
# "fromAddress": null,
|
||
# "cTime": "1694131668281",
|
||
# "uTime": "1694131680247"
|
||
# }
|
||
#
|
||
currencyId = self.safe_string(transaction, 'coin')
|
||
code = self.safe_currency_code(currencyId, currency)
|
||
timestamp = self.safe_integer(transaction, 'cTime')
|
||
networkId = self.safe_string(transaction, 'chain')
|
||
status = self.safe_string(transaction, 'status')
|
||
tag = self.safe_string(transaction, 'tag')
|
||
feeCostString = self.safe_string(transaction, 'fee')
|
||
feeCostAbsString = None
|
||
if feeCostString is not None:
|
||
feeCostAbsString = Precise.string_abs(feeCostString)
|
||
fee = None
|
||
amountString = self.safe_string(transaction, 'size')
|
||
if feeCostAbsString is not None:
|
||
fee = {'currency': code, 'cost': self.parse_number(feeCostAbsString)}
|
||
amountString = Precise.string_sub(amountString, feeCostAbsString)
|
||
return {
|
||
'id': self.safe_string(transaction, 'orderId'),
|
||
'info': transaction,
|
||
'txid': self.safe_string(transaction, 'tradeId'),
|
||
'timestamp': timestamp,
|
||
'datetime': self.iso8601(timestamp),
|
||
'network': self.network_id_to_code(networkId),
|
||
'addressFrom': self.safe_string(transaction, 'fromAddress'),
|
||
'address': self.safe_string(transaction, 'toAddress'),
|
||
'addressTo': self.safe_string(transaction, 'toAddress'),
|
||
'amount': self.parse_number(amountString),
|
||
'type': self.safe_string(transaction, 'type'),
|
||
'currency': code,
|
||
'status': self.parse_transaction_status(status),
|
||
'updated': self.safe_integer(transaction, 'uTime'),
|
||
'tagFrom': None,
|
||
'tag': tag,
|
||
'tagTo': tag,
|
||
'comment': None,
|
||
'internal': None,
|
||
'fee': fee,
|
||
}
|
||
|
||
def parse_transaction_status(self, status: Str):
|
||
statuses: dict = {
|
||
'success': 'ok',
|
||
'Pending': 'pending',
|
||
'pending_review': 'pending',
|
||
'pending_review_fail': 'failed',
|
||
'reject': 'failed',
|
||
}
|
||
return self.safe_string(statuses, status, status)
|
||
|
||
async def fetch_deposit_address(self, code: str, params={}) -> DepositAddress:
|
||
"""
|
||
fetch the deposit address for a currency associated with self account
|
||
|
||
https://www.bitget.com/api-doc/spot/account/Get-Deposit-Address
|
||
|
||
:param str code: unified currency code
|
||
:param dict [params]: extra parameters specific to the exchange API endpoint
|
||
:returns dict: an `address structure <https://docs.ccxt.com/#/?id=address-structure>`
|
||
"""
|
||
await self.load_markets()
|
||
networkCode = None
|
||
networkCode, params = self.handle_network_code_and_params(params)
|
||
currency = self.currency(code)
|
||
request: dict = {
|
||
'coin': currency['id'],
|
||
}
|
||
if networkCode is not None:
|
||
request['chain'] = self.network_code_to_id(networkCode, code)
|
||
response = await self.privateSpotGetV2SpotWalletDepositAddress(self.extend(request, params))
|
||
#
|
||
# {
|
||
# "code": "00000",
|
||
# "msg": "success",
|
||
# "requestTime": 1700532244807,
|
||
# "data": {
|
||
# "coin": "BTC",
|
||
# "address": "1BfZh7JESJGBUszCGeZnzxbVVvBycbJSbA",
|
||
# "chain": "",
|
||
# "tag": null,
|
||
# "url": "https://blockchair.com/bitcoin/transaction/"
|
||
# }
|
||
# }
|
||
#
|
||
data = self.safe_dict(response, 'data', {})
|
||
return self.parse_deposit_address(data, currency)
|
||
|
||
def parse_deposit_address(self, depositAddress, currency: Currency = None) -> DepositAddress:
|
||
#
|
||
# {
|
||
# "coin": "BTC",
|
||
# "address": "1BfZh7JESJGBUszCGeZnzxbVVvBycbJSbA",
|
||
# "chain": "",
|
||
# "tag": null,
|
||
# "url": "https://blockchair.com/bitcoin/transaction/"
|
||
# }
|
||
#
|
||
currencyId = self.safe_string(depositAddress, 'coin')
|
||
networkId = self.safe_string(depositAddress, 'chain')
|
||
parsedCurrency = self.safe_currency_code(currencyId, currency)
|
||
network = None
|
||
if networkId is not None:
|
||
network = self.network_id_to_code(networkId, parsedCurrency)
|
||
return {
|
||
'info': depositAddress,
|
||
'currency': parsedCurrency,
|
||
'network': network,
|
||
'address': self.safe_string(depositAddress, 'address'),
|
||
'tag': self.safe_string(depositAddress, 'tag'),
|
||
}
|
||
|
||
async def fetch_order_book(self, symbol: str, limit: Int = None, params={}) -> OrderBook:
|
||
"""
|
||
fetches information on open orders with bid(buy) and ask(sell) prices, volumes and other data
|
||
|
||
https://www.bitget.com/api-doc/spot/market/Get-Orderbook
|
||
https://www.bitget.com/api-doc/contract/market/Get-Merge-Depth
|
||
https://www.bitget.com/api-doc/uta/public/OrderBook
|
||
|
||
:param str symbol: unified symbol of the market to fetch the order book for
|
||
:param int [limit]: the maximum amount of order book entries to return
|
||
:param dict [params]: extra parameters specific to the exchange API endpoint
|
||
:param boolean [params.uta]: set to True for the unified trading account(uta), defaults to False
|
||
:returns dict: A dictionary of `order book structures <https://docs.ccxt.com/#/?id=order-book-structure>` indexed by market symbols
|
||
"""
|
||
await self.load_markets()
|
||
market = self.market(symbol)
|
||
request: dict = {
|
||
'symbol': market['id'],
|
||
}
|
||
if limit is not None:
|
||
request['limit'] = limit
|
||
productType = None
|
||
productType, params = self.handle_product_type_and_params(market, params)
|
||
response = None
|
||
uta = None
|
||
uta, params = self.handle_option_and_params(params, 'fetchOrderBook', 'uta', False)
|
||
if uta:
|
||
request['category'] = productType
|
||
response = await self.publicUtaGetV3MarketOrderbook(self.extend(request, params))
|
||
elif market['spot']:
|
||
response = await self.publicSpotGetV2SpotMarketOrderbook(self.extend(request, params))
|
||
else:
|
||
request['productType'] = productType
|
||
response = await self.publicMixGetV2MixMarketMergeDepth(self.extend(request, params))
|
||
#
|
||
# {
|
||
# "code": "00000",
|
||
# "msg": "success",
|
||
# "requestTime": 1645854610294,
|
||
# "data": {
|
||
# "asks": [["39102", "11.026"]],
|
||
# "bids": [['39100.5', "1.773"]],
|
||
# "ts": "1645854610294"
|
||
# }
|
||
# }
|
||
#
|
||
# uta
|
||
#
|
||
# {
|
||
# "code": "00000",
|
||
# "msg": "success",
|
||
# "requestTime": 1750329437753,
|
||
# "data": {
|
||
# "a": [[104992.60, 0.018411]],
|
||
# "b":[[104927.40, 0.229914]],
|
||
# "ts": "1750329437763"
|
||
# }
|
||
# }
|
||
#
|
||
data = self.safe_value(response, 'data', {})
|
||
bidsKey = 'b' if uta else 'bids'
|
||
asksKey = 'a' if uta else 'asks'
|
||
timestamp = self.safe_integer(data, 'ts')
|
||
return self.parse_order_book(data, market['symbol'], timestamp, bidsKey, asksKey)
|
||
|
||
def parse_ticker(self, ticker: dict, market: Market = None) -> Ticker:
|
||
#
|
||
# {
|
||
# "symbol": "BTCUSDT",
|
||
# "price": "26242",
|
||
# "indexPrice": "34867",
|
||
# "markPrice": "25555",
|
||
# "ts": "1695793390482"
|
||
# }
|
||
#
|
||
# spot
|
||
#
|
||
# {
|
||
# "open": "37202.46",
|
||
# "symbol": "BTCUSDT",
|
||
# "high24h": "37744.75",
|
||
# "low24h": "36666",
|
||
# "lastPr": "37583.69",
|
||
# "quoteVolume": "519127705.303",
|
||
# "baseVolume": "13907.0386",
|
||
# "usdtVolume": "519127705.302908",
|
||
# "ts": "1700532903261",
|
||
# "bidPr": "37583.68",
|
||
# "askPr": "37583.69",
|
||
# "bidSz": "0.0007",
|
||
# "askSz": "0.0829",
|
||
# "openUtc": "37449.4",
|
||
# "changeUtc24h": "0.00359",
|
||
# "change24h": "0.00321"
|
||
# }
|
||
#
|
||
# swap and future
|
||
#
|
||
# {
|
||
# "symbol": "BTCUSDT",
|
||
# "lastPr": "104823.8",
|
||
# "askPr": "104823.8",
|
||
# "bidPr": "104823.5",
|
||
# "bidSz": "0.703",
|
||
# "askSz": "13.894",
|
||
# "high24h": "105289.3",
|
||
# "low24h": "103447.9",
|
||
# "ts": "1750332210370",
|
||
# "change24h": "0.00471",
|
||
# "baseVolume": "79089.5675",
|
||
# "quoteVolume": "8274870921.80485",
|
||
# "usdtVolume": "8274870921.80485",
|
||
# "openUtc": "104833",
|
||
# "changeUtc24h": "-0.00009",
|
||
# "indexPrice": "104881.953125",
|
||
# "fundingRate": "-0.000014",
|
||
# "holdingAmount": "7452.6421",
|
||
# "deliveryStartTime": null,
|
||
# "deliveryTime": null,
|
||
# "deliveryStatus": "",
|
||
# "open24h": "104332.3",
|
||
# "markPrice": "104824.2"
|
||
# }
|
||
#
|
||
# spot uta
|
||
#
|
||
# {
|
||
# "category": "SPOT",
|
||
# "symbol": "BTCUSDT",
|
||
# "ts": "1750330651972",
|
||
# "lastPrice": "104900.2",
|
||
# "openPrice24h": "104321.2",
|
||
# "highPrice24h": "107956.8",
|
||
# "lowPrice24h": "103600.1",
|
||
# "ask1Price": "104945.8",
|
||
# "bid1Price": "104880.6",
|
||
# "bid1Size": "0.266534",
|
||
# "ask1Size": "0.014001",
|
||
# "price24hPcnt": "0.00555",
|
||
# "volume24h": "355.941109",
|
||
# "turnover24h": "37302936.008134"
|
||
# }
|
||
#
|
||
# swap and future uta
|
||
#
|
||
# {
|
||
# "category": "USDT-FUTURES",
|
||
# "symbol": "BTCUSDT",
|
||
# "ts": "1750332730472",
|
||
# "lastPrice": "104738",
|
||
# "openPrice24h": "104374",
|
||
# "highPrice24h": "105289.3",
|
||
# "lowPrice24h": "103447.9",
|
||
# "ask1Price": "104738",
|
||
# "bid1Price": "104737.7",
|
||
# "bid1Size": "2.036",
|
||
# "ask1Size": "8.094",
|
||
# "price24hPcnt": "0.00349",
|
||
# "volume24h": "79101.6477",
|
||
# "turnover24h": "8276293391.45973",
|
||
# "indexPrice": "104785.956168",
|
||
# "markPrice": "104738",
|
||
# "fundingRate": "-0.000007",
|
||
# "openInterest": "7465.5938",
|
||
# "deliveryStartTime": "",
|
||
# "deliveryTime": "",
|
||
# "deliveryStatus": ""
|
||
# }
|
||
#
|
||
marketId = self.safe_string(ticker, 'symbol')
|
||
close = self.safe_string_2(ticker, 'lastPr', 'lastPrice')
|
||
timestamp = self.safe_integer_omit_zero(ticker, 'ts') # exchange bitget provided 0
|
||
change = self.safe_string(ticker, 'change24h')
|
||
category = self.safe_string(ticker, 'category')
|
||
markPrice = self.safe_string(ticker, 'markPrice')
|
||
marketType: str
|
||
if (markPrice is not None) and (category != 'SPOT'):
|
||
marketType = 'contract'
|
||
else:
|
||
marketType = 'spot'
|
||
percentage = self.safe_string(ticker, 'price24hPcnt')
|
||
if percentage is None:
|
||
percentage = Precise.string_mul(change, '100')
|
||
return self.safe_ticker({
|
||
'symbol': self.safe_symbol(marketId, market, None, marketType),
|
||
'timestamp': timestamp,
|
||
'datetime': self.iso8601(timestamp),
|
||
'high': self.safe_string_2(ticker, 'high24h', 'highPrice24h'),
|
||
'low': self.safe_string_2(ticker, 'low24h', 'lowPrice24h'),
|
||
'bid': self.safe_string_2(ticker, 'bidPr', 'bid1Price'),
|
||
'bidVolume': self.safe_string_2(ticker, 'bidSz', 'bid1Size'),
|
||
'ask': self.safe_string_2(ticker, 'askPr', 'ask1Price'),
|
||
'askVolume': self.safe_string_2(ticker, 'askSz', 'ask1Size'),
|
||
'vwap': None,
|
||
'open': self.safe_string_n(ticker, ['open', 'open24h', 'openPrice24h']),
|
||
'close': close,
|
||
'last': close,
|
||
'previousClose': None,
|
||
'change': change,
|
||
'percentage': percentage,
|
||
'average': None,
|
||
'baseVolume': self.safe_string_2(ticker, 'baseVolume', 'volume24h'),
|
||
'quoteVolume': self.safe_string_2(ticker, 'quoteVolume', 'turnover24h'),
|
||
'indexPrice': self.safe_string(ticker, 'indexPrice'),
|
||
'markPrice': markPrice,
|
||
'info': ticker,
|
||
}, market)
|
||
|
||
async def fetch_ticker(self, symbol: str, params={}) -> Ticker:
|
||
"""
|
||
fetches a price ticker, a statistical calculation with the information calculated over the past 24 hours for a specific market
|
||
|
||
https://www.bitget.com/api-doc/spot/market/Get-Tickers
|
||
https://www.bitget.com/api-doc/contract/market/Get-Ticker
|
||
https://www.bitget.com/api-doc/uta/public/Tickers
|
||
|
||
:param str symbol: unified symbol of the market to fetch the ticker for
|
||
:param dict [params]: extra parameters specific to the exchange API endpoint
|
||
:param boolean [params.uta]: set to True for the unified trading account(uta), defaults to False
|
||
:returns dict: a `ticker structure <https://docs.ccxt.com/#/?id=ticker-structure>`
|
||
"""
|
||
await self.load_markets()
|
||
market = self.market(symbol)
|
||
request: dict = {
|
||
'symbol': market['id'],
|
||
}
|
||
productType = None
|
||
productType, params = self.handle_product_type_and_params(market, params)
|
||
response = None
|
||
uta = None
|
||
uta, params = self.handle_option_and_params(params, 'fetchTicker', 'uta', False)
|
||
if uta:
|
||
request['category'] = productType
|
||
response = await self.publicUtaGetV3MarketTickers(self.extend(request, params))
|
||
elif market['spot']:
|
||
response = await self.publicSpotGetV2SpotMarketTickers(self.extend(request, params))
|
||
else:
|
||
request['productType'] = productType
|
||
response = await self.publicMixGetV2MixMarketTicker(self.extend(request, params))
|
||
#
|
||
# spot
|
||
#
|
||
# {
|
||
# "code": "00000",
|
||
# "msg": "success",
|
||
# "requestTime": 1700532903782,
|
||
# "data": [
|
||
# {
|
||
# "open": "37202.46",
|
||
# "symbol": "BTCUSDT",
|
||
# "high24h": "37744.75",
|
||
# "low24h": "36666",
|
||
# "lastPr": "37583.69",
|
||
# "quoteVolume": "519127705.303",
|
||
# "baseVolume": "13907.0386",
|
||
# "usdtVolume": "519127705.302908",
|
||
# "ts": "1700532903261",
|
||
# "bidPr": "37583.68",
|
||
# "askPr": "37583.69",
|
||
# "bidSz": "0.0007",
|
||
# "askSz": "0.0829",
|
||
# "openUtc": "37449.4",
|
||
# "changeUtc24h": "0.00359",
|
||
# "change24h": "0.00321"
|
||
# }
|
||
# ]
|
||
# }
|
||
#
|
||
# swap and future
|
||
#
|
||
# {
|
||
# "code": "00000",
|
||
# "msg": "success",
|
||
# "requestTime": 1750332210369,
|
||
# "data": [
|
||
# {
|
||
# "symbol": "BTCUSDT",
|
||
# "lastPr": "104823.8",
|
||
# "askPr": "104823.8",
|
||
# "bidPr": "104823.5",
|
||
# "bidSz": "0.703",
|
||
# "askSz": "13.894",
|
||
# "high24h": "105289.3",
|
||
# "low24h": "103447.9",
|
||
# "ts": "1750332210370",
|
||
# "change24h": "0.00471",
|
||
# "baseVolume": "79089.5675",
|
||
# "quoteVolume": "8274870921.80485",
|
||
# "usdtVolume": "8274870921.80485",
|
||
# "openUtc": "104833",
|
||
# "changeUtc24h": "-0.00009",
|
||
# "indexPrice": "104881.953125",
|
||
# "fundingRate": "-0.000014",
|
||
# "holdingAmount": "7452.6421",
|
||
# "deliveryStartTime": null,
|
||
# "deliveryTime": null,
|
||
# "deliveryStatus": "",
|
||
# "open24h": "104332.3",
|
||
# "markPrice": "104824.2"
|
||
# }
|
||
# ]
|
||
# }
|
||
#
|
||
# spot uta
|
||
#
|
||
# {
|
||
# "code": "00000",
|
||
# "msg": "success",
|
||
# "requestTime": 1750330653575,
|
||
# "data": [
|
||
# {
|
||
# "category": "SPOT",
|
||
# "symbol": "BTCUSDT",
|
||
# "ts": "1750330651972",
|
||
# "lastPrice": "104900.2",
|
||
# "openPrice24h": "104321.2",
|
||
# "highPrice24h": "107956.8",
|
||
# "lowPrice24h": "103600.1",
|
||
# "ask1Price": "104945.8",
|
||
# "bid1Price": "104880.6",
|
||
# "bid1Size": "0.266534",
|
||
# "ask1Size": "0.014001",
|
||
# "price24hPcnt": "0.00555",
|
||
# "volume24h": "355.941109",
|
||
# "turnover24h": "37302936.008134"
|
||
# }
|
||
# ]
|
||
# }
|
||
#
|
||
# swap and future uta
|
||
#
|
||
# {
|
||
# "code": "00000",
|
||
# "msg": "success",
|
||
# "requestTime": 1750332731203,
|
||
# "data": [
|
||
# {
|
||
# "category": "USDT-FUTURES",
|
||
# "symbol": "BTCUSDT",
|
||
# "ts": "1750332730472",
|
||
# "lastPrice": "104738",
|
||
# "openPrice24h": "104374",
|
||
# "highPrice24h": "105289.3",
|
||
# "lowPrice24h": "103447.9",
|
||
# "ask1Price": "104738",
|
||
# "bid1Price": "104737.7",
|
||
# "bid1Size": "2.036",
|
||
# "ask1Size": "8.094",
|
||
# "price24hPcnt": "0.00349",
|
||
# "volume24h": "79101.6477",
|
||
# "turnover24h": "8276293391.45973",
|
||
# "indexPrice": "104785.956168",
|
||
# "markPrice": "104738",
|
||
# "fundingRate": "-0.000007",
|
||
# "openInterest": "7465.5938",
|
||
# "deliveryStartTime": "",
|
||
# "deliveryTime": "",
|
||
# "deliveryStatus": ""
|
||
# }
|
||
# ]
|
||
# }
|
||
#
|
||
data = self.safe_list(response, 'data', [])
|
||
return self.parse_ticker(data[0], market)
|
||
|
||
async def fetch_mark_price(self, symbol: str, params={}) -> Ticker:
|
||
"""
|
||
fetches the mark price for a specific market
|
||
|
||
https://www.bitget.com/api-doc/contract/market/Get-Symbol-Price
|
||
|
||
:param str symbol: unified symbol of the market to fetch the ticker for
|
||
:param dict [params]: extra parameters specific to the exchange API endpoint
|
||
:returns dict: a `ticker structure <https://docs.ccxt.com/#/?id=ticker-structure>`
|
||
"""
|
||
await self.load_markets()
|
||
market = self.market(symbol)
|
||
request: dict = {
|
||
'symbol': market['id'],
|
||
}
|
||
response = None
|
||
if market['spot']:
|
||
raise NotSupported(self.id + ' fetchMarkPrice() is not supported for spot markets')
|
||
else:
|
||
productType = None
|
||
productType, params = self.handle_product_type_and_params(market, params)
|
||
request['productType'] = productType
|
||
response = await self.publicMixGetV2MixMarketSymbolPrice(self.extend(request, params))
|
||
data = self.safe_list(response, 'data', [])
|
||
return self.parse_ticker(data[0], market)
|
||
|
||
async def fetch_tickers(self, symbols: Strings = None, params={}) -> Tickers:
|
||
"""
|
||
fetches price tickers for multiple markets, statistical information calculated over the past 24 hours for each market
|
||
|
||
https://www.bitget.com/api-doc/spot/market/Get-Tickers
|
||
https://www.bitget.com/api-doc/contract/market/Get-All-Symbol-Ticker
|
||
https://www.bitget.com/api-doc/uta/public/Tickers
|
||
|
||
:param str[]|None symbols: unified symbols of the markets to fetch the ticker for, all market tickers are returned if not assigned
|
||
:param dict [params]: extra parameters specific to the exchange API endpoint
|
||
:param boolean [params.uta]: set to True for the unified trading account(uta), defaults to False
|
||
:param str [params.subType]: *contract only* 'linear', 'inverse'
|
||
:param str [params.productType]: *contract only* 'USDT-FUTURES', 'USDC-FUTURES', 'COIN-FUTURES', 'SUSDT-FUTURES', 'SUSDC-FUTURES' or 'SCOIN-FUTURES'
|
||
:returns dict: a dictionary of `ticker structures <https://docs.ccxt.com/#/?id=ticker-structure>`
|
||
"""
|
||
await self.load_markets()
|
||
market = None
|
||
if symbols is not None:
|
||
symbol = self.safe_value(symbols, 0)
|
||
market = self.market(symbol)
|
||
response = None
|
||
request: dict = {}
|
||
type = None
|
||
type, params = self.handle_market_type_and_params('fetchTickers', market, params)
|
||
# Calls like `.fetchTickers(None, {subType:'inverse'})` should be supported for self exchange, so
|
||
# as "options.defaultSubType" is also set in exchange options, we should consider `params.subType`
|
||
# with higher priority and only default to spot, if `subType` is not set in params
|
||
passedSubType = self.safe_string(params, 'subType')
|
||
productType = None
|
||
productType, params = self.handle_product_type_and_params(market, params)
|
||
# only if passedSubType and productType is None, then use spot
|
||
uta = None
|
||
uta, params = self.handle_option_and_params(params, 'fetchTickers', 'uta', False)
|
||
if uta:
|
||
symbolsLength = len(symbols)
|
||
if (symbols is not None) and (symbolsLength == 1):
|
||
request['symbol'] = market['id']
|
||
request['category'] = productType
|
||
response = await self.publicUtaGetV3MarketTickers(self.extend(request, params))
|
||
elif type == 'spot' and passedSubType is None:
|
||
response = await self.publicSpotGetV2SpotMarketTickers(self.extend(request, params))
|
||
else:
|
||
request['productType'] = productType
|
||
response = await self.publicMixGetV2MixMarketTickers(self.extend(request, params))
|
||
#
|
||
# spot
|
||
#
|
||
# {
|
||
# "code": "00000",
|
||
# "msg": "success",
|
||
# "requestTime": 1700532903782,
|
||
# "data": [
|
||
# {
|
||
# "open": "37202.46",
|
||
# "symbol": "BTCUSDT",
|
||
# "high24h": "37744.75",
|
||
# "low24h": "36666",
|
||
# "lastPr": "37583.69",
|
||
# "quoteVolume": "519127705.303",
|
||
# "baseVolume": "13907.0386",
|
||
# "usdtVolume": "519127705.302908",
|
||
# "ts": "1700532903261",
|
||
# "bidPr": "37583.68",
|
||
# "askPr": "37583.69",
|
||
# "bidSz": "0.0007",
|
||
# "askSz": "0.0829",
|
||
# "openUtc": "37449.4",
|
||
# "changeUtc24h": "0.00359",
|
||
# "change24h": "0.00321"
|
||
# }
|
||
# ]
|
||
# }
|
||
#
|
||
# swap and future
|
||
#
|
||
# {
|
||
# "code": "00000",
|
||
# "msg": "success",
|
||
# "requestTime": 1700533773477,
|
||
# "data": [
|
||
# {
|
||
# "open": "14.9776",
|
||
# "symbol": "LINKUSDT",
|
||
# "high24h": "15.3942",
|
||
# "low24h": "14.3457",
|
||
# "lastPr": "14.3748",
|
||
# "quoteVolume": "7008612.4299",
|
||
# "baseVolume": "469908.8523",
|
||
# "usdtVolume": "7008612.42986561",
|
||
# "ts": "1700533772309",
|
||
# "bidPr": "14.375",
|
||
# "askPr": "14.3769",
|
||
# "bidSz": "50.004",
|
||
# "askSz": "0.7647",
|
||
# "openUtc": "14.478",
|
||
# "changeUtc24h": "-0.00713",
|
||
# "change24h": "-0.04978"
|
||
# },
|
||
# ]
|
||
# }
|
||
#
|
||
# spot uta
|
||
#
|
||
# {
|
||
# "code": "00000",
|
||
# "msg": "success",
|
||
# "requestTime": 1750330653575,
|
||
# "data": [
|
||
# {
|
||
# "category": "SPOT",
|
||
# "symbol": "BTCUSDT",
|
||
# "ts": "1750330651972",
|
||
# "lastPrice": "104900.2",
|
||
# "openPrice24h": "104321.2",
|
||
# "highPrice24h": "107956.8",
|
||
# "lowPrice24h": "103600.1",
|
||
# "ask1Price": "104945.8",
|
||
# "bid1Price": "104880.6",
|
||
# "bid1Size": "0.266534",
|
||
# "ask1Size": "0.014001",
|
||
# "price24hPcnt": "0.00555",
|
||
# "volume24h": "355.941109",
|
||
# "turnover24h": "37302936.008134"
|
||
# }
|
||
# ]
|
||
# }
|
||
#
|
||
# swap and future uta
|
||
#
|
||
# {
|
||
# "code": "00000",
|
||
# "msg": "success",
|
||
# "requestTime": 1750332731203,
|
||
# "data": [
|
||
# {
|
||
# "category": "USDT-FUTURES",
|
||
# "symbol": "BTCUSDT",
|
||
# "ts": "1750332730472",
|
||
# "lastPrice": "104738",
|
||
# "openPrice24h": "104374",
|
||
# "highPrice24h": "105289.3",
|
||
# "lowPrice24h": "103447.9",
|
||
# "ask1Price": "104738",
|
||
# "bid1Price": "104737.7",
|
||
# "bid1Size": "2.036",
|
||
# "ask1Size": "8.094",
|
||
# "price24hPcnt": "0.00349",
|
||
# "volume24h": "79101.6477",
|
||
# "turnover24h": "8276293391.45973",
|
||
# "indexPrice": "104785.956168",
|
||
# "markPrice": "104738",
|
||
# "fundingRate": "-0.000007",
|
||
# "openInterest": "7465.5938",
|
||
# "deliveryStartTime": "",
|
||
# "deliveryTime": "",
|
||
# "deliveryStatus": ""
|
||
# }
|
||
# ]
|
||
# }
|
||
#
|
||
data = self.safe_list(response, 'data', [])
|
||
return self.parse_tickers(data, symbols)
|
||
|
||
def parse_trade(self, trade: dict, market: Market = None) -> Trade:
|
||
#
|
||
# spot, swap and future: fetchTrades
|
||
#
|
||
# {
|
||
# "tradeId": "1075199767891652609",
|
||
# "price": "29376.5",
|
||
# "size": "6.035",
|
||
# "side": "Buy",
|
||
# "ts": "1692073521000",
|
||
# "symbol": "BTCUSDT"
|
||
# }
|
||
#
|
||
# spot: fetchMyTrades
|
||
#
|
||
# {
|
||
# "userId": "7264631750",
|
||
# "symbol": "BTCUSDT",
|
||
# "orderId": "1098394344925597696",
|
||
# "tradeId": "1098394344974925824",
|
||
# "orderType": "market",
|
||
# "side": "sell",
|
||
# "priceAvg": "28467.68",
|
||
# "size": "0.0002",
|
||
# "amount": "5.693536",
|
||
# "feeDetail": {
|
||
# "deduction": "no",
|
||
# "feeCoin": "USDT",
|
||
# "totalDeductionFee": "",
|
||
# "totalFee": "-0.005693536"
|
||
# },
|
||
# "tradeScope": "taker",
|
||
# "cTime": "1697603539699",
|
||
# "uTime": "1697603539754"
|
||
# }
|
||
#
|
||
# spot margin: fetchMyTrades
|
||
#
|
||
# {
|
||
# "orderId": "1099353730455318528",
|
||
# "tradeId": "1099353730627092481",
|
||
# "orderType": "market",
|
||
# "side": "sell",
|
||
# "priceAvg": "29543.7",
|
||
# "size": "0.0001",
|
||
# "amount": "2.95437",
|
||
# "tradeScope": "taker",
|
||
# "feeDetail": {
|
||
# "deduction": "no",
|
||
# "feeCoin": "USDT",
|
||
# "totalDeductionFee": "0",
|
||
# "totalFee": "-0.00295437"
|
||
# },
|
||
# "cTime": "1697832275063",
|
||
# "uTime": "1697832275150"
|
||
# }
|
||
#
|
||
# swap and future: fetchMyTrades
|
||
#
|
||
# {
|
||
# "tradeId": "1111468664328269825",
|
||
# "symbol": "BTCUSDT",
|
||
# "orderId": "1111468664264753162",
|
||
# "price": "37271.4",
|
||
# "baseVolume": "0.001",
|
||
# "feeDetail": [
|
||
# {
|
||
# "deduction": "no",
|
||
# "feeCoin": "USDT",
|
||
# "totalDeductionFee": null,
|
||
# "totalFee": "-0.02236284"
|
||
# }
|
||
# ],
|
||
# "side": "buy",
|
||
# "quoteVolume": "37.2714",
|
||
# "profit": "-0.0007",
|
||
# "enterPointSource": "web",
|
||
# "tradeSide": "close",
|
||
# "posMode": "hedge_mode",
|
||
# "tradeScope": "taker",
|
||
# "cTime": "1700720700342"
|
||
# }
|
||
#
|
||
# uta fetchTrades
|
||
#
|
||
# {
|
||
# "execId": "1319896716324937729",
|
||
# "price": "105909.1",
|
||
# "size": "6.3090",
|
||
# "side": "sell",
|
||
# "ts": "1750413820344"
|
||
# }
|
||
#
|
||
# uta fetchMyTrades
|
||
#
|
||
# {
|
||
# "execId": "1322441401010528257",
|
||
# "orderId": "1322441400976261120",
|
||
# "category": "USDT-FUTURES",
|
||
# "symbol": "BTCUSDT",
|
||
# "orderType": "market",
|
||
# "side": "sell",
|
||
# "execPrice": "107005.4",
|
||
# "execQty": "0.0001",
|
||
# "execValue": "10.7005",
|
||
# "tradeScope": "taker",
|
||
# "feeDetail": [{
|
||
# "feeCoin": "USDT",
|
||
# "fee":"0.00642032"
|
||
# }],
|
||
# "createdTime": "1751020520451",
|
||
# "updatedTime": "1751020520458",
|
||
# "execPnl": "0.00017"
|
||
# }
|
||
#
|
||
marketId = self.safe_string(trade, 'symbol')
|
||
symbol = self.safe_symbol(marketId, market)
|
||
timestamp = self.safe_integer_n(trade, ['cTime', 'ts', 'createdTime'])
|
||
fee = None
|
||
feeDetail = self.safe_value(trade, 'feeDetail')
|
||
posMode = self.safe_string(trade, 'posMode')
|
||
category = self.safe_string(trade, 'category')
|
||
isFeeStructure = (posMode is not None) or (category is not None)
|
||
feeStructure = feeDetail[0] if isFeeStructure else feeDetail
|
||
if feeStructure is not None:
|
||
currencyCode = self.safe_currency_code(self.safe_string(feeStructure, 'feeCoin'))
|
||
fee = {
|
||
'currency': currencyCode,
|
||
}
|
||
feeCostString = self.safe_string_2(feeStructure, 'totalFee', 'fee')
|
||
deduction = self.safe_string(feeStructure, 'deduction') is True if 'yes' else False
|
||
if deduction:
|
||
fee['cost'] = feeCostString
|
||
else:
|
||
fee['cost'] = Precise.string_neg(feeCostString)
|
||
return self.safe_trade({
|
||
'info': trade,
|
||
'id': self.safe_string_2(trade, 'tradeId', 'execId'),
|
||
'order': self.safe_string(trade, 'orderId'),
|
||
'symbol': symbol,
|
||
'side': self.safe_string_lower(trade, 'side'),
|
||
'type': self.safe_string(trade, 'orderType'),
|
||
'takerOrMaker': self.safe_string(trade, 'tradeScope'),
|
||
'price': self.safe_string_n(trade, ['priceAvg', 'price', 'execPrice']),
|
||
'amount': self.safe_string_n(trade, ['baseVolume', 'size', 'execQty']),
|
||
'cost': self.safe_string_n(trade, ['quoteVolume', 'amount', 'execValue']),
|
||
'timestamp': timestamp,
|
||
'datetime': self.iso8601(timestamp),
|
||
'fee': fee,
|
||
}, market)
|
||
|
||
async def fetch_trades(self, symbol: str, since: Int = None, limit: Int = None, params={}) -> List[Trade]:
|
||
"""
|
||
get the list of most recent trades for a particular symbol
|
||
|
||
https://www.bitget.com/api-doc/spot/market/Get-Recent-Trades
|
||
https://www.bitget.com/api-doc/spot/market/Get-Market-Trades
|
||
https://www.bitget.com/api-doc/contract/market/Get-Recent-Fills
|
||
https://www.bitget.com/api-doc/contract/market/Get-Fills-History
|
||
https://www.bitget.com/api-doc/uta/public/Fills
|
||
|
||
:param str symbol: unified symbol of the market to fetch trades for
|
||
:param int [since]: timestamp in ms of the earliest trade to fetch
|
||
:param int [limit]: the maximum amount of trades to fetch
|
||
:param dict [params]: extra parameters specific to the exchange API endpoint
|
||
:param boolean [params.uta]: set to True for the unified trading account(uta), defaults to False
|
||
:param int [params.until]: *only applies to publicSpotGetV2SpotMarketFillsHistory and publicMixGetV2MixMarketFillsHistory* the latest time in ms to fetch trades for
|
||
:param boolean [params.paginate]: *only applies to publicSpotGetV2SpotMarketFillsHistory and publicMixGetV2MixMarketFillsHistory* default False, when True will automatically paginate by calling self endpoint multiple times
|
||
:returns Trade[]: a list of `trade structures <https://docs.ccxt.com/#/?id=public-trades>`
|
||
"""
|
||
await self.load_markets()
|
||
paginate = False
|
||
paginate, params = self.handle_option_and_params(params, 'fetchTrades', 'paginate')
|
||
if paginate:
|
||
return await self.fetch_paginated_call_cursor('fetchTrades', symbol, since, limit, params, 'idLessThan', 'idLessThan')
|
||
market = self.market(symbol)
|
||
request: dict = {
|
||
'symbol': market['id'],
|
||
}
|
||
uta = None
|
||
uta, params = self.handle_option_and_params(params, 'fetchTrades', 'uta', False)
|
||
if limit is not None:
|
||
if uta:
|
||
request['limit'] = min(limit, 100)
|
||
elif market['contract']:
|
||
request['limit'] = min(limit, 1000)
|
||
else:
|
||
request['limit'] = limit
|
||
options = self.safe_value(self.options, 'fetchTrades', {})
|
||
response = None
|
||
productType = None
|
||
productType, params = self.handle_product_type_and_params(market, params)
|
||
if uta:
|
||
if productType == 'SPOT':
|
||
marginMode = None
|
||
marginMode, params = self.handle_margin_mode_and_params('fetchTrades', params)
|
||
if marginMode is not None:
|
||
productType = 'MARGIN'
|
||
request['category'] = productType
|
||
response = await self.publicUtaGetV3MarketFills(self.extend(request, params))
|
||
elif market['spot']:
|
||
spotOptions = self.safe_value(options, 'spot', {})
|
||
defaultSpotMethod = self.safe_string(spotOptions, 'method', 'publicSpotGetV2SpotMarketFillsHistory')
|
||
spotMethod = self.safe_string(params, 'method', defaultSpotMethod)
|
||
params = self.omit(params, 'method')
|
||
if spotMethod == 'publicSpotGetV2SpotMarketFillsHistory':
|
||
request, params = self.handle_until_option('endTime', request, params)
|
||
if since is not None:
|
||
request['startTime'] = since
|
||
response = await self.publicSpotGetV2SpotMarketFillsHistory(self.extend(request, params))
|
||
elif spotMethod == 'publicSpotGetV2SpotMarketFills':
|
||
response = await self.publicSpotGetV2SpotMarketFills(self.extend(request, params))
|
||
else:
|
||
swapOptions = self.safe_value(options, 'swap', {})
|
||
defaultSwapMethod = self.safe_string(swapOptions, 'method', 'publicMixGetV2MixMarketFillsHistory')
|
||
swapMethod = self.safe_string(params, 'method', defaultSwapMethod)
|
||
params = self.omit(params, 'method')
|
||
request['productType'] = productType
|
||
if swapMethod == 'publicMixGetV2MixMarketFillsHistory':
|
||
request, params = self.handle_until_option('endTime', request, params)
|
||
if since is not None:
|
||
request['startTime'] = since
|
||
response = await self.publicMixGetV2MixMarketFillsHistory(self.extend(request, params))
|
||
elif swapMethod == 'publicMixGetV2MixMarketFills':
|
||
response = await self.publicMixGetV2MixMarketFills(self.extend(request, params))
|
||
#
|
||
# spot
|
||
#
|
||
# {
|
||
# "code": "00000",
|
||
# "msg": "success",
|
||
# "requestTime": 1692073693562,
|
||
# "data": [
|
||
# {
|
||
# "symbol": "BTCUSDT_SPBL",
|
||
# "tradeId": "1075200479040323585",
|
||
# "side": "Sell",
|
||
# "price": "29381.54",
|
||
# "size": "0.0056",
|
||
# "ts": "1692073691000"
|
||
# },
|
||
# ]
|
||
# }
|
||
#
|
||
# swap
|
||
#
|
||
# {
|
||
# "code": "00000",
|
||
# "msg": "success",
|
||
# "requestTime": 1692073522689,
|
||
# "data": [
|
||
# {
|
||
# "tradeId": "1075199767891652609",
|
||
# "price": "29376.5",
|
||
# "size": "6.035",
|
||
# "side": "Buy",
|
||
# "ts": "1692073521000",
|
||
# "symbol": "BTCUSDT_UMCBL"
|
||
# },
|
||
# ]
|
||
# }
|
||
#
|
||
# uta
|
||
#
|
||
# {
|
||
# "code": "00000",
|
||
# "msg": "success",
|
||
# "requestTime": 1750413823980,
|
||
# "data": [
|
||
# {
|
||
# "execId": "1319896716324937729",
|
||
# "price": "105909.1",
|
||
# "size": "6.3090",
|
||
# "side": "sell",
|
||
# "ts": "1750413820344"
|
||
# }
|
||
# ]
|
||
# }
|
||
#
|
||
data = self.safe_list(response, 'data', [])
|
||
return self.parse_trades(data, market, since, limit)
|
||
|
||
async def fetch_trading_fee(self, symbol: str, params={}) -> TradingFeeInterface:
|
||
"""
|
||
fetch the trading fees for a market
|
||
|
||
https://www.bitget.com/api-doc/common/public/Get-Trade-Rate
|
||
|
||
:param str symbol: unified market symbol
|
||
:param dict [params]: extra parameters specific to the exchange API endpoint
|
||
:param str [params.marginMode]: 'isolated' or 'cross', for finding the fee rate of spot margin trading pairs
|
||
:returns dict: a `fee structure <https://docs.ccxt.com/#/?id=fee-structure>`
|
||
"""
|
||
await self.load_markets()
|
||
market = self.market(symbol)
|
||
request: dict = {
|
||
'symbol': market['id'],
|
||
}
|
||
marginMode = None
|
||
marginMode, params = self.handle_margin_mode_and_params('fetchTradingFee', params)
|
||
if market['spot']:
|
||
if marginMode is not None:
|
||
request['businessType'] = 'margin'
|
||
else:
|
||
request['businessType'] = 'spot'
|
||
else:
|
||
request['businessType'] = 'mix'
|
||
response = await self.privateCommonGetV2CommonTradeRate(self.extend(request, params))
|
||
#
|
||
# {
|
||
# "code": "00000",
|
||
# "msg": "success",
|
||
# "requestTime": 1700549524887,
|
||
# "data": {
|
||
# "makerFeeRate": "0.001",
|
||
# "takerFeeRate": "0.001"
|
||
# }
|
||
# }
|
||
#
|
||
data = self.safe_value(response, 'data', {})
|
||
return self.parse_trading_fee(data, market)
|
||
|
||
async def fetch_trading_fees(self, params={}) -> TradingFees:
|
||
"""
|
||
fetch the trading fees for multiple markets
|
||
|
||
https://www.bitget.com/api-doc/spot/market/Get-Symbols
|
||
https://www.bitget.com/api-doc/contract/market/Get-All-Symbols-Contracts
|
||
https://www.bitget.com/api-doc/margin/common/support-currencies
|
||
|
||
:param dict [params]: extra parameters specific to the exchange API endpoint
|
||
:param str [params.productType]: *contract only* 'USDT-FUTURES', 'USDC-FUTURES', 'COIN-FUTURES', 'SUSDT-FUTURES', 'SUSDC-FUTURES' or 'SCOIN-FUTURES'
|
||
:param boolean [params.margin]: set to True for spot margin
|
||
:returns dict: a dictionary of `fee structures <https://docs.ccxt.com/#/?id=fee-structure>` indexed by market symbols
|
||
"""
|
||
await self.load_markets()
|
||
response = None
|
||
marginMode = None
|
||
marketType = None
|
||
marginMode, params = self.handle_margin_mode_and_params('fetchTradingFees', params)
|
||
marketType, params = self.handle_market_type_and_params('fetchTradingFees', None, params)
|
||
if marketType == 'spot':
|
||
margin = self.safe_bool(params, 'margin', False)
|
||
params = self.omit(params, 'margin')
|
||
if (marginMode is not None) or margin:
|
||
response = await self.publicMarginGetV2MarginCurrencies(params)
|
||
else:
|
||
response = await self.publicSpotGetV2SpotPublicSymbols(params)
|
||
elif (marketType == 'swap') or (marketType == 'future'):
|
||
productType = None
|
||
productType, params = self.handle_product_type_and_params(None, params)
|
||
params['productType'] = productType
|
||
response = await self.publicMixGetV2MixMarketContracts(params)
|
||
else:
|
||
raise NotSupported(self.id + ' does not support ' + marketType + ' market')
|
||
#
|
||
# spot and margin
|
||
#
|
||
# {
|
||
# "code": "00000",
|
||
# "msg": "success",
|
||
# "requestTime": 1700102364653,
|
||
# "data": [
|
||
# {
|
||
# "symbol": "TRXUSDT",
|
||
# "baseCoin": "TRX",
|
||
# "quoteCoin": "USDT",
|
||
# "minTradeAmount": "0",
|
||
# "maxTradeAmount": "10000000000",
|
||
# "takerFeeRate": "0.002",
|
||
# "makerFeeRate": "0.002",
|
||
# "pricePrecision": "6",
|
||
# "quantityPrecision": "4",
|
||
# "quotePrecision": "6",
|
||
# "status": "online",
|
||
# "minTradeUSDT": "5",
|
||
# "buyLimitPriceRatio": "0.05",
|
||
# "sellLimitPriceRatio": "0.05"
|
||
# },
|
||
# ]
|
||
# }
|
||
#
|
||
# swap and future
|
||
#
|
||
# {
|
||
# "code": "00000",
|
||
# "msg": "success",
|
||
# "requestTime": 1700102364709,
|
||
# "data": [
|
||
# {
|
||
# "symbol": "BTCUSDT",
|
||
# "baseCoin": "BTC",
|
||
# "quoteCoin": "USDT",
|
||
# "buyLimitPriceRatio": "0.01",
|
||
# "sellLimitPriceRatio": "0.01",
|
||
# "feeRateUpRatio": "0.005",
|
||
# "makerFeeRate": "0.0002",
|
||
# "takerFeeRate": "0.0006",
|
||
# "openCostUpRatio": "0.01",
|
||
# "supportMarginCoins": ["USDT"],
|
||
# "minTradeNum": "0.001",
|
||
# "priceEndStep": "1",
|
||
# "volumePlace": "3",
|
||
# "pricePlace": "1",
|
||
# "sizeMultiplier": "0.001",
|
||
# "symbolType": "perpetual",
|
||
# "minTradeUSDT": "5",
|
||
# "maxSymbolOrderNum": "200",
|
||
# "maxProductOrderNum": "400",
|
||
# "maxPositionNum": "150",
|
||
# "symbolStatus": "normal",
|
||
# "offTime": "-1",
|
||
# "limitOpenTime": "-1",
|
||
# "deliveryTime": "",
|
||
# "deliveryStartTime": "",
|
||
# "deliveryPeriod": "",
|
||
# "launchTime": "",
|
||
# "fundInterval": "8",
|
||
# "minLever": "1",
|
||
# "maxLever": "125",
|
||
# "posLimit": "0.05",
|
||
# "maintainTime": ""
|
||
# },
|
||
# ]
|
||
# }
|
||
#
|
||
data = self.safe_value(response, 'data', [])
|
||
result: dict = {}
|
||
for i in range(0, len(data)):
|
||
entry = data[i]
|
||
marketId = self.safe_string(entry, 'symbol')
|
||
symbol = self.safe_symbol(marketId, None, None, marketType)
|
||
market = self.market(symbol)
|
||
fee = self.parse_trading_fee(entry, market)
|
||
result[symbol] = fee
|
||
return result
|
||
|
||
def parse_trading_fee(self, data, market: Market = None):
|
||
marketId = self.safe_string(data, 'symbol')
|
||
return {
|
||
'info': data,
|
||
'symbol': self.safe_symbol(marketId, market),
|
||
'maker': self.safe_number(data, 'makerFeeRate'),
|
||
'taker': self.safe_number(data, 'takerFeeRate'),
|
||
'percentage': None,
|
||
'tierBased': None,
|
||
}
|
||
|
||
def parse_ohlcv(self, ohlcv, market: Market = None) -> list:
|
||
#
|
||
# [
|
||
# "1645911960000",
|
||
# "39406",
|
||
# "39407",
|
||
# "39374.5",
|
||
# "39379",
|
||
# "35.526",
|
||
# "1399132.341"
|
||
# ]
|
||
#
|
||
inverse = self.safe_bool(market, 'inverse')
|
||
volumeIndex = 6 if inverse else 5
|
||
return [
|
||
self.safe_integer(ohlcv, 0),
|
||
self.safe_number(ohlcv, 1),
|
||
self.safe_number(ohlcv, 2),
|
||
self.safe_number(ohlcv, 3),
|
||
self.safe_number(ohlcv, 4),
|
||
self.safe_number(ohlcv, volumeIndex),
|
||
]
|
||
|
||
async def fetch_ohlcv(self, symbol: str, timeframe: str = '1m', since: Int = None, limit: Int = None, params={}) -> List[list]:
|
||
"""
|
||
fetches historical candlestick data containing the open, high, low, and close price, and the volume of a market
|
||
|
||
https://www.bitget.com/api-doc/spot/market/Get-Candle-Data
|
||
https://www.bitget.com/api-doc/spot/market/Get-History-Candle-Data
|
||
https://www.bitget.com/api-doc/contract/market/Get-Candle-Data
|
||
https://www.bitget.com/api-doc/contract/market/Get-History-Candle-Data
|
||
https://www.bitget.com/api-doc/contract/market/Get-History-Index-Candle-Data
|
||
https://www.bitget.com/api-doc/contract/market/Get-History-Mark-Candle-Data
|
||
https://www.bitget.com/api-doc/uta/public/Get-Candle-Data
|
||
|
||
:param str symbol: unified symbol of the market to fetch OHLCV data for
|
||
:param str timeframe: the length of time each candle represents
|
||
:param int [since]: timestamp in ms of the earliest candle to fetch
|
||
:param int [limit]: the maximum amount of candles to fetch
|
||
:param dict [params]: extra parameters specific to the exchange API endpoint
|
||
:param boolean [params.uta]: set to True for the unified trading account(uta), defaults to False
|
||
:param int [params.until]: timestamp in ms of the latest candle to fetch
|
||
:param boolean [params.useHistoryEndpoint]: whether to force to use historical endpoint(it has max limit of 200)
|
||
:param boolean [params.useHistoryEndpointForPagination]: whether to force to use historical endpoint for pagination(default True)
|
||
:param boolean [params.paginate]: default False, when True will automatically paginate by calling self endpoint multiple times. See in the docs all the [available parameters](https://github.com/ccxt/ccxt/wiki/Manual#pagination-params)
|
||
:param str [params.price]: *swap only* "mark"(to fetch mark price candles) or "index"(to fetch index price candles)
|
||
:returns int[][]: A list of candles ordered, open, high, low, close, volume
|
||
"""
|
||
await self.load_markets()
|
||
defaultLimit = 100 # default 100, max 1000
|
||
maxLimitForRecentEndpoint = 1000
|
||
maxLimitForHistoryEndpoint = 200 # note, max 1000 bars are supported for "recent-candles" endpoint, but "historical-candles" support only max 200
|
||
useHistoryEndpoint = self.safe_bool(params, 'useHistoryEndpoint', False)
|
||
useHistoryEndpointForPagination = self.safe_bool(params, 'useHistoryEndpointForPagination', True)
|
||
paginate = False
|
||
paginate, params = self.handle_option_and_params(params, 'fetchOHLCV', 'paginate')
|
||
if paginate:
|
||
limitForPagination = maxLimitForHistoryEndpoint if useHistoryEndpointForPagination else maxLimitForRecentEndpoint
|
||
return await self.fetch_paginated_call_deterministic('fetchOHLCV', symbol, since, limit, timeframe, params, limitForPagination)
|
||
market = self.market(symbol)
|
||
request: dict = {
|
||
'symbol': market['id'],
|
||
}
|
||
marketType = None
|
||
timeframes = None
|
||
uta = None
|
||
uta, params = self.handle_option_and_params(params, 'fetchOHLCV', 'uta', False)
|
||
if uta:
|
||
timeframes = self.options['timeframes']['uta']
|
||
request['interval'] = self.safe_string(timeframes, timeframe, timeframe)
|
||
else:
|
||
marketType = 'spot' if market['spot'] else 'swap'
|
||
timeframes = self.options['timeframes'][marketType]
|
||
request['granularity'] = self.safe_string(timeframes, timeframe, timeframe)
|
||
msInDay = 86400000
|
||
now = self.milliseconds()
|
||
duration = self.parse_timeframe(timeframe) * 1000
|
||
until = self.safe_integer(params, 'until')
|
||
limitDefined = limit is not None
|
||
sinceDefined = since is not None
|
||
untilDefined = until is not None
|
||
params = self.omit(params, ['until'])
|
||
# retrievable periods listed here:
|
||
# - https://www.bitget.com/api-doc/spot/market/Get-Candle-Data#request-parameters
|
||
# - https://www.bitget.com/api-doc/contract/market/Get-Candle-Data#description
|
||
key = 'spot' if market['spot'] else 'swap'
|
||
ohlcOptions = self.safe_dict(self.options['fetchOHLCV'], key, {})
|
||
maxLimitPerTimeframe = self.safe_dict(ohlcOptions, 'maxLimitPerTimeframe', {})
|
||
maxLimitForThisTimeframe = self.safe_integer(maxLimitPerTimeframe, timeframe, limit)
|
||
recentEndpointDaysMap = self.safe_dict(self.options['fetchOHLCV'], 'maxRecentDaysPerTimeframe', {})
|
||
recentEndpointAvailableDays = self.safe_integer(recentEndpointDaysMap, timeframe)
|
||
recentEndpointBoundaryTs = now - (recentEndpointAvailableDays - 1) * msInDay
|
||
if limitDefined:
|
||
limit = min(limit, maxLimitForRecentEndpoint)
|
||
limit = min(limit, maxLimitForThisTimeframe)
|
||
else:
|
||
limit = defaultLimit
|
||
limitMultipliedDuration = limit * duration
|
||
# exchange aligns from endTime, so it's important, not startTime
|
||
# startTime is supported only on "recent" endpoint, not on "historical" endpoint
|
||
calculatedStartTime = None
|
||
calculatedEndTime = None
|
||
if sinceDefined:
|
||
calculatedStartTime = since
|
||
request['startTime'] = since
|
||
if not untilDefined:
|
||
calculatedEndTime = self.sum(calculatedStartTime, limitMultipliedDuration)
|
||
if calculatedEndTime > now:
|
||
calculatedEndTime = now
|
||
request['endTime'] = calculatedEndTime
|
||
if untilDefined:
|
||
calculatedEndTime = until
|
||
if calculatedEndTime > now:
|
||
calculatedEndTime = now
|
||
request['endTime'] = calculatedEndTime
|
||
if not sinceDefined:
|
||
calculatedStartTime = calculatedEndTime - limitMultipliedDuration
|
||
# we do not need to set "startTime" here
|
||
# if historical endpoint is needed, we should re-set the variables
|
||
historicalEndpointNeeded = False
|
||
if (calculatedStartTime is not None and calculatedStartTime <= recentEndpointBoundaryTs) or useHistoryEndpoint:
|
||
historicalEndpointNeeded = True
|
||
# only for "historical-candles" - ensure we use correct max limit
|
||
limit = min(limit, maxLimitForHistoryEndpoint)
|
||
limitMultipliedDuration = limit * duration
|
||
calculatedStartTime = calculatedEndTime - limitMultipliedDuration
|
||
request['startTime'] = calculatedStartTime
|
||
# for contract, maximum 90 days allowed between start-end times
|
||
if not market['spot']:
|
||
maxDistanceDaysForContracts = 90
|
||
# only correct if request is larger
|
||
if calculatedEndTime - calculatedStartTime > maxDistanceDaysForContracts * msInDay:
|
||
calculatedEndTime = self.sum(calculatedStartTime, maxDistanceDaysForContracts * msInDay)
|
||
request['endTime'] = calculatedEndTime
|
||
# we need to set limit to safely cover the period
|
||
request['limit'] = limit
|
||
# make request
|
||
response = None
|
||
productType = None
|
||
priceType = None
|
||
priceType, params = self.handle_param_string(params, 'price')
|
||
productType, params = self.handle_product_type_and_params(market, params)
|
||
if uta:
|
||
if priceType is not None:
|
||
if priceType == 'mark':
|
||
request['type'] = 'MARK'
|
||
elif priceType == 'index':
|
||
request['type'] = 'INDEX'
|
||
request['category'] = productType
|
||
response = await self.publicUtaGetV3MarketCandles(self.extend(request, params))
|
||
elif market['spot']:
|
||
# checks if we need history endpoint
|
||
if historicalEndpointNeeded:
|
||
response = await self.publicSpotGetV2SpotMarketHistoryCandles(self.extend(request, params))
|
||
else:
|
||
if not limitDefined:
|
||
request['limit'] = 1000
|
||
limit = 1000
|
||
response = await self.publicSpotGetV2SpotMarketCandles(self.extend(request, params))
|
||
else:
|
||
request['productType'] = productType
|
||
extended = self.extend(request, params)
|
||
if not historicalEndpointNeeded and (priceType == 'mark' or priceType == 'index'):
|
||
if not limitDefined:
|
||
extended['limit'] = 1000
|
||
limit = 1000
|
||
# Recent endpoint for mark/index prices
|
||
# https://www.bitget.com/api-doc/contract/market/Get-Candle-Data
|
||
response = await self.publicMixGetV2MixMarketCandles(self.extend({'kLineType': priceType}, extended))
|
||
elif priceType == 'mark':
|
||
response = await self.publicMixGetV2MixMarketHistoryMarkCandles(extended)
|
||
elif priceType == 'index':
|
||
response = await self.publicMixGetV2MixMarketHistoryIndexCandles(extended)
|
||
else:
|
||
if historicalEndpointNeeded:
|
||
response = await self.publicMixGetV2MixMarketHistoryCandles(extended)
|
||
else:
|
||
if not limitDefined:
|
||
extended['limit'] = 1000
|
||
limit = 1000
|
||
response = await self.publicMixGetV2MixMarketCandles(extended)
|
||
if response == '':
|
||
return [] # happens when a new token is listed
|
||
# [["1645911960000","39406","39407","39374.5","39379","35.526","1399132.341"]]
|
||
data = self.safe_list(response, 'data', response)
|
||
return self.parse_ohlcvs(data, market, timeframe, since, limit)
|
||
|
||
async def fetch_balance(self, params={}) -> Balances:
|
||
"""
|
||
query for balance and get the amount of funds available for trading or funds locked in orders
|
||
|
||
https://www.bitget.com/api-doc/spot/account/Get-Account-Assets
|
||
https://www.bitget.com/api-doc/contract/account/Get-Account-List
|
||
https://www.bitget.com/api-doc/margin/cross/account/Get-Cross-Assets
|
||
https://www.bitget.com/api-doc/margin/isolated/account/Get-Isolated-Assets
|
||
https://bitgetlimited.github.io/apidoc/en/margin/#get-cross-assets
|
||
https://bitgetlimited.github.io/apidoc/en/margin/#get-isolated-assets
|
||
https://www.bitget.com/api-doc/uta/account/Get-Account
|
||
|
||
:param dict [params]: extra parameters specific to the exchange API endpoint
|
||
:param str [params.productType]: *contract only* 'USDT-FUTURES', 'USDC-FUTURES', 'COIN-FUTURES', 'SUSDT-FUTURES', 'SUSDC-FUTURES' or 'SCOIN-FUTURES'
|
||
:param str [params.uta]: set to True for the unified trading account(uta), defaults to False
|
||
:returns dict: a `balance structure <https://docs.ccxt.com/#/?id=balance-structure>`
|
||
"""
|
||
await self.load_markets()
|
||
request: dict = {}
|
||
marketType = None
|
||
marginMode = None
|
||
response = None
|
||
uta = None
|
||
uta, params = self.handle_option_and_params(params, 'fetchBalance', 'uta', False)
|
||
marketType, params = self.handle_market_type_and_params('fetchBalance', None, params)
|
||
marginMode, params = self.handle_margin_mode_and_params('fetchBalance', params)
|
||
if uta:
|
||
response = await self.privateUtaGetV3AccountAssets(self.extend(request, params))
|
||
results = self.safe_dict(response, 'data', {})
|
||
assets = self.safe_list(results, 'assets', [])
|
||
return self.parse_uta_balance(assets)
|
||
elif (marketType == 'swap') or (marketType == 'future'):
|
||
productType = None
|
||
productType, params = self.handle_product_type_and_params(None, params)
|
||
request['productType'] = productType
|
||
response = await self.privateMixGetV2MixAccountAccounts(self.extend(request, params))
|
||
elif marginMode == 'isolated':
|
||
response = await self.privateMarginGetV2MarginIsolatedAccountAssets(self.extend(request, params))
|
||
#
|
||
# {
|
||
# "code": "00000",
|
||
# "msg": "success",
|
||
# "requestTime": "1759829170717",
|
||
# "data": [
|
||
# {
|
||
# "symbol": "BTCUSDT",
|
||
# "coin": "USDT",
|
||
# "totalAmount": "0.000001",
|
||
# "available": "0.000001",
|
||
# "frozen": "0",
|
||
# "borrow": "0",
|
||
# "interest": "0",
|
||
# "net": "0.000001",
|
||
# "coupon": "0",
|
||
# "cTime": "1759826434145",
|
||
# "uTime": "1759826434146"
|
||
# },
|
||
# ]
|
||
# }
|
||
#
|
||
elif marginMode == 'cross':
|
||
response = await self.privateMarginGetV2MarginCrossedAccountAssets(self.extend(request, params))
|
||
#
|
||
# {
|
||
# "code": "00000",
|
||
# "msg": "success",
|
||
# "requestTime": "1759828519501",
|
||
# "data": [
|
||
# {
|
||
# "coin": "USDT",
|
||
# "totalAmount": "0.01",
|
||
# "available": "0.01",
|
||
# "frozen": "0",
|
||
# "borrow": "0",
|
||
# "interest": "0",
|
||
# "net": "0.01",
|
||
# "coupon": "0",
|
||
# "cTime": "1759828511592",
|
||
# "uTime": "1759828511592"
|
||
# }
|
||
# ]
|
||
# }
|
||
#
|
||
elif marketType == 'spot':
|
||
response = await self.privateSpotGetV2SpotAccountAssets(self.extend(request, params))
|
||
else:
|
||
raise NotSupported(self.id + ' fetchBalance() does not support ' + marketType + ' accounts')
|
||
# spot
|
||
#
|
||
# {
|
||
# "code": "00000",
|
||
# "msg": "success",
|
||
# "requestTime": 1700623852854,
|
||
# "data": [
|
||
# {
|
||
# "coin": "USDT",
|
||
# "available": "0.00000000",
|
||
# "limitAvailable": "0",
|
||
# "frozen": "0.00000000",
|
||
# "locked": "0.00000000",
|
||
# "uTime": "1699937566000"
|
||
# }
|
||
# ]
|
||
# }
|
||
#
|
||
# swap
|
||
#
|
||
# {
|
||
# "code": "00000",
|
||
# "msg": "success",
|
||
# "requestTime": 1700625127294,
|
||
# "data": [
|
||
# {
|
||
# "marginCoin": "USDT",
|
||
# "locked": "0",
|
||
# "available": "0",
|
||
# "crossedMaxAvailable": "0",
|
||
# "isolatedMaxAvailable": "0",
|
||
# "maxTransferOut": "0",
|
||
# "accountEquity": "0",
|
||
# "usdtEquity": "0.000000005166",
|
||
# "btcEquity": "0",
|
||
# "crossedRiskRate": "0",
|
||
# "unrealizedPL": "0",
|
||
# "coupon": "0",
|
||
# "crossedUnrealizedPL": null,
|
||
# "isolatedUnrealizedPL": null
|
||
# }
|
||
# ]
|
||
# }
|
||
#
|
||
# uta
|
||
#
|
||
# {
|
||
# "code": "00000",
|
||
# "msg": "success",
|
||
# "requestTime": 1749980065089,
|
||
# "data": {
|
||
# "accountEquity": "11.13919278",
|
||
# "usdtEquity": "11.13921165",
|
||
# "btcEquity": "0.00011256",
|
||
# "unrealisedPnl": "0",
|
||
# "usdtUnrealisedPnl": "0",
|
||
# "btcUnrealizedPnl": "0",
|
||
# "effEquity": "6.19299777",
|
||
# "mmr": "0",
|
||
# "imr": "0",
|
||
# "mgnRatio": "0",
|
||
# "positionMgnRatio": "0",
|
||
# "assets": [
|
||
# {
|
||
# "coin": "USDT",
|
||
# "equity": "6.19300826",
|
||
# "usdValue": "6.19299777",
|
||
# "balance": "6.19300826",
|
||
# "available": "6.19300826",
|
||
# "debt": "0",
|
||
# "locked": "0"
|
||
# }
|
||
# ]
|
||
# }
|
||
# }
|
||
#
|
||
data = self.safe_value(response, 'data', [])
|
||
return self.parse_balance(data)
|
||
|
||
def parse_uta_balance(self, balance) -> Balances:
|
||
result: dict = {'info': balance}
|
||
#
|
||
# {
|
||
# "coin": "USDT",
|
||
# "equity": "6.19300826",
|
||
# "usdValue": "6.19299777",
|
||
# "balance": "6.19300826",
|
||
# "available": "6.19300826",
|
||
# "debt": "0",
|
||
# "locked": "0"
|
||
# }
|
||
#
|
||
for i in range(0, len(balance)):
|
||
entry = balance[i]
|
||
account = self.account()
|
||
currencyId = self.safe_string(entry, 'coin')
|
||
code = self.safe_currency_code(currencyId)
|
||
account['debt'] = self.safe_string(entry, 'debt')
|
||
account['used'] = self.safe_string(entry, 'locked')
|
||
account['free'] = self.safe_string(entry, 'available')
|
||
account['total'] = self.safe_string(entry, 'balance')
|
||
result[code] = account
|
||
return self.safe_balance(result)
|
||
|
||
def parse_balance(self, balance) -> Balances:
|
||
result: dict = {'info': balance}
|
||
#
|
||
# spot
|
||
#
|
||
# {
|
||
# "coin": "USDT",
|
||
# "available": "0.00000000",
|
||
# "limitAvailable": "0",
|
||
# "frozen": "0.00000000",
|
||
# "locked": "0.00000000",
|
||
# "uTime": "1699937566000"
|
||
# }
|
||
#
|
||
# swap
|
||
#
|
||
# {
|
||
# "marginCoin": "USDT",
|
||
# "locked": "0",
|
||
# "available": "0",
|
||
# "crossedMaxAvailable": "0",
|
||
# "isolatedMaxAvailable": "0",
|
||
# "maxTransferOut": "0",
|
||
# "accountEquity": "0",
|
||
# "usdtEquity": "0.000000005166",
|
||
# "btcEquity": "0",
|
||
# "crossedRiskRate": "0",
|
||
# "unrealizedPL": "0",
|
||
# "coupon": "0",
|
||
# "crossedUnrealizedPL": null,
|
||
# "isolatedUnrealizedPL": null
|
||
# }
|
||
#
|
||
# cross & isolated margin
|
||
#
|
||
# {
|
||
# "coin": "USDT",
|
||
# "totalAmount": "0.01",
|
||
# "available": "0.01",
|
||
# "frozen": "0",
|
||
# "borrow": "0",
|
||
# "interest": "0",
|
||
# "net": "0.01",
|
||
# "coupon": "0",
|
||
# "cTime": "1759828511592",
|
||
# "uTime": "1759828511592"
|
||
# # "symbol": "BTCUSDT" # only for isolated margin
|
||
# }
|
||
#
|
||
for i in range(0, len(balance)):
|
||
entry = balance[i]
|
||
account = self.account()
|
||
currencyId = self.safe_string_2(entry, 'marginCoin', 'coin')
|
||
code = self.safe_currency_code(currencyId)
|
||
borrow = self.safe_string(entry, 'borrow')
|
||
if borrow is not None:
|
||
interest = self.safe_string(entry, 'interest')
|
||
account['free'] = self.safe_string(entry, 'transferable')
|
||
account['total'] = self.safe_string(entry, 'totalAmount')
|
||
account['debt'] = Precise.string_add(borrow, interest)
|
||
else:
|
||
# Use transferable instead of available for swap and margin https://github.com/ccxt/ccxt/pull/19127
|
||
spotAccountFree = self.safe_string(entry, 'available')
|
||
contractAccountFree = self.safe_string(entry, 'maxTransferOut')
|
||
if contractAccountFree is not None:
|
||
account['free'] = contractAccountFree
|
||
account['total'] = self.safe_string(entry, 'accountEquity')
|
||
else:
|
||
account['free'] = spotAccountFree
|
||
frozen = self.safe_string(entry, 'frozen')
|
||
locked = self.safe_string(entry, 'locked')
|
||
account['used'] = Precise.string_add(frozen, locked)
|
||
result[code] = account
|
||
return self.safe_balance(result)
|
||
|
||
def parse_order_status(self, status: Str):
|
||
statuses: dict = {
|
||
'new': 'open',
|
||
'init': 'open',
|
||
'not_trigger': 'open',
|
||
'partial_fill': 'open',
|
||
'partially_fill': 'open',
|
||
'partially_filled': 'open',
|
||
'triggered': 'closed',
|
||
'full_fill': 'closed',
|
||
'filled': 'closed',
|
||
'fail_trigger': 'rejected',
|
||
'cancel': 'canceled',
|
||
'cancelled': 'canceled',
|
||
'canceled': 'canceled',
|
||
'live': 'open',
|
||
'fail_execute': 'rejected',
|
||
'executed': 'closed',
|
||
}
|
||
return self.safe_string(statuses, status, status)
|
||
|
||
def parse_order(self, order: dict, market: Market = None) -> Order:
|
||
#
|
||
# createOrder, editOrder, closePosition
|
||
#
|
||
# {
|
||
# "clientOid": "abe95dbe-6081-4a6f-a2d3-ae49601cd479",
|
||
# "orderId": null
|
||
# }
|
||
#
|
||
# createOrders
|
||
#
|
||
# [
|
||
# {
|
||
# "orderId": "1111397214281175046",
|
||
# "clientOid": "766d3fc3-7321-4406-a689-15c9987a2e75"
|
||
# },
|
||
# {
|
||
# "orderId": "",
|
||
# "clientOid": "d1b75cb3-cc15-4ede-ad4c-3937396f75ab",
|
||
# "errorMsg": "less than the minimum amount 5 USDT",
|
||
# "errorCode": "45110"
|
||
# },
|
||
# ]
|
||
#
|
||
# spot, swap, future, spot margin and uta: cancelOrder, cancelOrders, cancelAllOrders
|
||
#
|
||
# {
|
||
# "orderId": "1098758604547850241",
|
||
# "clientOid": "1098758604585598977"
|
||
# }
|
||
#
|
||
# spot trigger: cancelOrder
|
||
#
|
||
# {
|
||
# "result": "success"
|
||
# }
|
||
#
|
||
# spot: fetchOrder, fetchOpenOrders, fetchCanceledAndClosedOrders
|
||
#
|
||
# {
|
||
# "userId": "7264631750",
|
||
# "symbol": "BTCUSDT",
|
||
# "orderId": "1111499608327360513",
|
||
# "clientOid": "d0d4dad5-18d0-4869-a074-ec40bb47cba6",
|
||
# "size": "0.0002000000000000", # COST for 'buy market' order! AMOUNT in all other cases
|
||
# "price": "0", # in fetchOrder: 0 for market order, otherwise limit price(field not present in fetchOpenOrders
|
||
# "orderType": "limit",
|
||
# "side": "buy",
|
||
# "status": "live",
|
||
# "basePrice": "0",
|
||
# "priceAvg": "25000.0000000000000000", # 0 if nothing filled
|
||
# "baseVolume": "0.0000000000000000", # 0 if nothing filled
|
||
# "quoteVolume": "0.0000000000000000", # 0 if nothing filled
|
||
# "enterPointSource": "WEB",
|
||
# "orderSource": "normal",
|
||
# "cTime": "1700728077966",
|
||
# "uTime": "1700728077966"
|
||
# "feeDetail": "{\\"newFees\\":{\\"c\\":0,\\"d\\":0,\\"deduction\\":false,\\"r\\":-0.0064699886,\\"t\\":-0.0064699886,\\"totalDeductionFee\\":0},\\"USDT\\":{\\"deduction\\":false,\\"feeCoinCode\\":\\"USDT\\",\\"totalDeductionFee\\":0,\\"totalFee\\":-0.0064699886000000}}", # might not be present in fetchOpenOrders
|
||
# "triggerPrice": null,
|
||
# "tpslType": "normal",
|
||
# "quoteCoin": "USDT", # not present in fetchOpenOrders
|
||
# "baseCoin": "DOT", # not present in fetchOpenOrders
|
||
# "cancelReason": "", # not present in fetchOpenOrders
|
||
# }
|
||
#
|
||
# spot trigger: fetchOpenOrders, fetchCanceledAndClosedOrders
|
||
#
|
||
# {
|
||
# "orderId": "1111503385931620352",
|
||
# "clientOid": "1111503385910648832",
|
||
# "symbol": "BTCUSDT",
|
||
# "size": "0.0002",
|
||
# "planType": "AMOUNT",
|
||
# "executePrice": "25000",
|
||
# "triggerPrice": "26000",
|
||
# "status": "live",
|
||
# "orderType": "limit",
|
||
# "side": "buy",
|
||
# "triggerType": "fill_price",
|
||
# "enterPointSource": "API",
|
||
# "cTime": "1700728978617",
|
||
# "uTime": "1700728978617"
|
||
# }
|
||
#
|
||
# spot margin: fetchOpenOrders, fetchCanceledAndClosedOrders
|
||
#
|
||
# {
|
||
# "symbol": "BTCUSDT",
|
||
# "orderType": "limit",
|
||
# "enterPointSource": "WEB",
|
||
# "orderId": "1111506377509580801",
|
||
# "clientOid": "2043a3b59a60445f9d9f7365bf3e960c",
|
||
# "loanType": "autoLoanAndRepay",
|
||
# "price": "25000",
|
||
# "side": "buy",
|
||
# "status": "live",
|
||
# "baseSize": "0.0002",
|
||
# "quoteSize": "5",
|
||
# "priceAvg": "0",
|
||
# "size": "0",
|
||
# "amount": "0",
|
||
# "force": "gtc",
|
||
# "cTime": "1700729691866",
|
||
# "uTime": "1700729691866"
|
||
# }
|
||
#
|
||
# swap and future: fetchOrder, fetchOpenOrders, fetchCanceledAndClosedOrders
|
||
#
|
||
# {
|
||
# "symbol": "BTCUSDT",
|
||
# "size": "0.001",
|
||
# "orderId": "1111465253393825792",
|
||
# "clientOid": "1111465253431574529",
|
||
# "baseVolume": "0",
|
||
# "fee": "0",
|
||
# "price": "27000",
|
||
# "priceAvg": "",
|
||
# "state": "live",
|
||
# # "status": "live", # key for fetchOpenOrders, fetchClosedOrders
|
||
# "side": "buy",
|
||
# "force": "gtc",
|
||
# "totalProfits": "0",
|
||
# "posSide": "long",
|
||
# "marginCoin": "USDT",
|
||
# "quoteVolume": "0",
|
||
# "leverage": "20",
|
||
# "marginMode": "crossed",
|
||
# "enterPointSource": "API",
|
||
# "tradeSide": "open",
|
||
# "posMode": "hedge_mode",
|
||
# "orderType": "limit",
|
||
# "orderSource": "normal",
|
||
# "presetStopSurplusPrice": "",
|
||
# "presetStopLossPrice": "",
|
||
# "reduceOnly": "NO",
|
||
# "cTime": "1700719887120",
|
||
# "uTime": "1700719887120"
|
||
#
|
||
# for swap trigger order, the additional below fields are present:
|
||
#
|
||
# "planType": "normal_plan",
|
||
# "callbackRatio": "",
|
||
# "triggerPrice": "24000",
|
||
# "triggerType": "mark_price",
|
||
# "planStatus": "live",
|
||
# "stopSurplusTriggerPrice": "",
|
||
# "stopSurplusExecutePrice": "",
|
||
# "stopSurplusTriggerType": "fill_price",
|
||
# "stopLossTriggerPrice": "",
|
||
# "stopLossExecutePrice": "",
|
||
# "stopLossTriggerType": "fill_price",
|
||
# }
|
||
#
|
||
# uta: fetchOrder, fetchOpenOrders, fetchCanceledAndClosedOrders
|
||
#
|
||
# {
|
||
# "orderId": "1320244799629316096",
|
||
# "clientOid": "1320244799633510400",
|
||
# "category": "USDT-FUTURES",
|
||
# "symbol": "BTCUSDT",
|
||
# "orderType": "limit",
|
||
# "side": "buy",
|
||
# "price": "50000",
|
||
# "qty": "0.001",
|
||
# "amount": "0",
|
||
# "cumExecQty": "0",
|
||
# "cumExecValue": "0",
|
||
# "avgPrice": "0",
|
||
# "timeInForce": "gtc",
|
||
# "orderStatus": "live",
|
||
# "posSide": "long",
|
||
# "holdMode": "hedge_mode",
|
||
# "reduceOnly": "NO",
|
||
# "feeDetail": [{
|
||
# "feeCoin": "",
|
||
# "fee": ""
|
||
# }],
|
||
# "createdTime": "1750496809871",
|
||
# "updatedTime": "1750496809886",
|
||
# "cancelReason": "",
|
||
# "execType": "normal",
|
||
# "stpMode": "none",
|
||
# "tpTriggerBy": null,
|
||
# "slTriggerBy": null,
|
||
# "takeProfit": null,
|
||
# "stopLoss": null,
|
||
# "tpOrderType": null,
|
||
# "slOrderType": null,
|
||
# "tpLimitPrice": null,
|
||
# "slLimitPrice": null
|
||
# }
|
||
#
|
||
# uta trigger: fetchClosedOrders, fetchCanceledOrders
|
||
#
|
||
# {
|
||
# "orderId": "1330984742276198400",
|
||
# "clientOid": "1330984742276198400",
|
||
# "symbol": "BTCUSDT",
|
||
# "category": "USDT-FUTURES",
|
||
# "qty": "0.001",
|
||
# "posSide": "long",
|
||
# "tpTriggerBy": "market",
|
||
# "slTriggerBy": "mark",
|
||
# "takeProfit": "",
|
||
# "stopLoss": "112000",
|
||
# "tpOrderType": "market",
|
||
# "slOrderType": "limit",
|
||
# "tpLimitPrice": "",
|
||
# "slLimitPrice": "111000",
|
||
# "createdTime": "1753057411736",
|
||
# "updatedTime": "1753058267412"
|
||
# }
|
||
#
|
||
errorMessage = self.safe_string(order, 'errorMsg')
|
||
if errorMessage is not None:
|
||
return self.safe_order({
|
||
'info': order,
|
||
'id': self.safe_string(order, 'orderId'),
|
||
'clientOrderId': self.safe_string_2(order, 'clientOrderId', 'clientOid'),
|
||
'status': 'rejected',
|
||
}, market)
|
||
posSide = self.safe_string(order, 'posSide')
|
||
isContractOrder = (posSide is not None)
|
||
marketType = 'contract' if isContractOrder else 'spot'
|
||
if market is not None:
|
||
marketType = market['type']
|
||
marketId = self.safe_string(order, 'symbol')
|
||
market = self.safe_market(marketId, market, None, marketType)
|
||
timestamp = self.safe_integer_n(order, ['cTime', 'ctime', 'createdTime'])
|
||
updateTimestamp = self.safe_integer_2(order, 'uTime', 'updatedTime')
|
||
rawStatus = self.safe_string_n(order, ['status', 'state', 'orderStatus', 'planStatus'])
|
||
fee = None
|
||
feeCostString = self.safe_string(order, 'fee')
|
||
if feeCostString is not None:
|
||
# swap
|
||
fee = {
|
||
'cost': self.parse_number(Precise.string_neg(feeCostString)),
|
||
'currency': market['settle'],
|
||
}
|
||
feeDetail = self.safe_value(order, 'feeDetail')
|
||
uta = self.safe_string(order, 'category') is not None
|
||
if uta:
|
||
feeResult = self.safe_dict(feeDetail, 0, {})
|
||
utaFee = self.safe_string(feeResult, 'fee')
|
||
fee = {
|
||
'cost': self.parse_number(Precise.string_neg(utaFee)),
|
||
'currency': market['settle'],
|
||
}
|
||
else:
|
||
if feeDetail is not None:
|
||
parsedFeeDetail = json.loads(feeDetail)
|
||
feeValues = list(parsedFeeDetail.values())
|
||
feeObject = None
|
||
for i in range(0, len(feeValues)):
|
||
feeValue = feeValues[i]
|
||
if self.safe_value(feeValue, 'feeCoinCode') is not None:
|
||
feeObject = feeValue
|
||
break
|
||
fee = {
|
||
'cost': self.parse_number(Precise.string_neg(self.safe_string(feeObject, 'totalFee'))),
|
||
'currency': self.safe_currency_code(self.safe_string(feeObject, 'feeCoinCode')),
|
||
}
|
||
postOnly = None
|
||
timeInForce = self.safe_string_upper_2(order, 'force', 'timeInForce')
|
||
if timeInForce == 'POST_ONLY':
|
||
postOnly = True
|
||
timeInForce = 'PO'
|
||
reduceOnly = None
|
||
reduceOnlyRaw = self.safe_string(order, 'reduceOnly')
|
||
if reduceOnlyRaw is not None:
|
||
reduceOnly = False if (reduceOnlyRaw == 'NO') else True
|
||
price = None
|
||
average = None
|
||
basePrice = self.safe_string(order, 'basePrice')
|
||
if basePrice is not None:
|
||
# for spot fetchOpenOrders, the price is priceAvg and the filled price is basePrice
|
||
price = self.safe_string(order, 'priceAvg')
|
||
average = self.safe_string(order, 'basePrice')
|
||
else:
|
||
price = self.safe_string_n(order, ['price', 'executePrice', 'slLimitPrice', 'tpLimitPrice'])
|
||
average = self.safe_string(order, 'priceAvg')
|
||
size = None
|
||
filled = None
|
||
baseSize = self.safe_string(order, 'baseSize')
|
||
if baseSize is not None:
|
||
# for spot margin fetchOpenOrders, the order size is baseSize and the filled amount is size
|
||
size = baseSize
|
||
filled = self.safe_string(order, 'size')
|
||
else:
|
||
size = self.safe_string_2(order, 'size', 'qty')
|
||
filled = self.safe_string_2(order, 'baseVolume', 'cumExecQty')
|
||
side = self.safe_string(order, 'side')
|
||
posMode = self.safe_string(order, 'posMode')
|
||
if posMode == 'hedge_mode' and reduceOnly:
|
||
side = 'sell' if (side == 'buy') else 'buy'
|
||
# on bitget hedge mode if the position is long the side is always buy, and if the position is short the side is always sell
|
||
# so the side of the reduceOnly order is inversed
|
||
orderType = self.safe_string(order, 'orderType')
|
||
isBuyMarket = (side == 'buy') and (orderType == 'market')
|
||
if market['spot'] and isBuyMarket:
|
||
# in top comment, for 'buy market' the 'size' field is COST, not AMOUNT
|
||
size = self.safe_string(order, 'baseVolume')
|
||
return self.safe_order({
|
||
'info': order,
|
||
'id': self.safe_string_2(order, 'orderId', 'data'),
|
||
'clientOrderId': self.safe_string_2(order, 'clientOrderId', 'clientOid'),
|
||
'timestamp': timestamp,
|
||
'datetime': self.iso8601(timestamp),
|
||
'lastTradeTimestamp': updateTimestamp,
|
||
'lastUpdateTimestamp': updateTimestamp,
|
||
'symbol': market['symbol'],
|
||
'type': orderType,
|
||
'side': side,
|
||
'price': price,
|
||
'amount': size,
|
||
'cost': self.safe_string_2(order, 'quoteVolume', 'quoteSize'),
|
||
'average': average,
|
||
'filled': filled,
|
||
'remaining': None,
|
||
'timeInForce': timeInForce,
|
||
'postOnly': postOnly,
|
||
'reduceOnly': reduceOnly,
|
||
'triggerPrice': self.safe_number(order, 'triggerPrice'),
|
||
'takeProfitPrice': self.safe_number_n(order, ['presetStopSurplusPrice', 'stopSurplusTriggerPrice', 'takeProfit']),
|
||
'stopLossPrice': self.safe_number_n(order, ['presetStopLossPrice', 'stopLossTriggerPrice', 'stopLoss']),
|
||
'status': self.parse_order_status(rawStatus),
|
||
'fee': fee,
|
||
'trades': None,
|
||
}, market)
|
||
|
||
async def create_market_buy_order_with_cost(self, symbol: str, cost: float, params={}):
|
||
"""
|
||
create a market buy order by providing the symbol and cost
|
||
|
||
https://www.bitget.com/api-doc/spot/trade/Place-Order
|
||
https://www.bitget.com/api-doc/margin/cross/trade/Cross-Place-Order
|
||
https://www.bitget.com/api-doc/margin/isolated/trade/Isolated-Place-Order
|
||
|
||
:param str symbol: unified symbol of the market to create an order in
|
||
:param float cost: how much you want to trade in units of the quote currency
|
||
:param dict [params]: extra parameters specific to the exchange API endpoint
|
||
:returns dict: an `order structure <https://docs.ccxt.com/#/?id=order-structure>`
|
||
"""
|
||
await self.load_markets()
|
||
market = self.market(symbol)
|
||
if not market['spot']:
|
||
raise NotSupported(self.id + ' createMarketBuyOrderWithCost() supports spot orders only')
|
||
req = {
|
||
'createMarketBuyOrderRequiresPrice': False,
|
||
}
|
||
return await self.create_order(symbol, 'market', 'buy', cost, None, self.extend(req, params))
|
||
|
||
async def create_order(self, symbol: str, type: OrderType, side: OrderSide, amount: float, price: Num = None, params={}):
|
||
"""
|
||
create a trade order
|
||
|
||
https://www.bitget.com/api-doc/spot/trade/Place-Order
|
||
https://www.bitget.com/api-doc/spot/plan/Place-Plan-Order
|
||
https://www.bitget.com/api-doc/contract/trade/Place-Order
|
||
https://www.bitget.com/api-doc/contract/plan/Place-Tpsl-Order
|
||
https://www.bitget.com/api-doc/contract/plan/Place-Plan-Order
|
||
https://www.bitget.com/api-doc/margin/cross/trade/Cross-Place-Order
|
||
https://www.bitget.com/api-doc/margin/isolated/trade/Isolated-Place-Order
|
||
https://www.bitget.com/api-doc/uta/trade/Place-Order
|
||
https://www.bitget.com/api-doc/uta/strategy/Place-Strategy-Order
|
||
|
||
:param str symbol: unified symbol of the market to create an order in
|
||
:param str type: 'market' or 'limit'
|
||
:param str side: 'buy' or 'sell'
|
||
:param float amount: how much you want to trade in units of the base currency
|
||
:param float [price]: the price at which the order is to be fulfilled, in units of the quote currency, ignored in market orders
|
||
:param dict [params]: extra parameters specific to the exchange API endpoint
|
||
:param float [params.cost]: *spot only* how much you want to trade in units of the quote currency, for market buy orders only
|
||
:param float [params.triggerPrice]: *swap only* The price at which a trigger order is triggered at
|
||
:param float [params.stopLossPrice]: *swap only* The price at which a stop loss order is triggered at
|
||
:param float [params.takeProfitPrice]: *swap only* The price at which a take profit order is triggered at
|
||
:param dict [params.takeProfit]: *takeProfit object in params* containing the triggerPrice at which the attached take profit order will be triggered(perpetual swap markets only)
|
||
:param float [params.takeProfit.triggerPrice]: *swap only* take profit trigger price
|
||
:param dict [params.stopLoss]: *stopLoss object in params* containing the triggerPrice at which the attached stop loss order will be triggered(perpetual swap markets only)
|
||
:param float [params.stopLoss.triggerPrice]: *swap only* stop loss trigger price
|
||
:param str [params.timeInForce]: "GTC", "IOC", "FOK", or "PO"
|
||
:param str [params.marginMode]: 'isolated' or 'cross' for spot margin trading
|
||
:param str [params.loanType]: *spot margin only* 'normal', 'autoLoan', 'autoRepay', or 'autoLoanAndRepay' default is 'normal'
|
||
:param str [params.holdSide]: *contract stopLossPrice, takeProfitPrice only* Two-way position: ('long' or 'short'), one-way position: ('buy' or 'sell')
|
||
:param float [params.stopLoss.price]: *swap only* the execution price for a stop loss attached to a trigger order
|
||
:param float [params.takeProfit.price]: *swap only* the execution price for a take profit attached to a trigger order
|
||
:param str [params.stopLoss.type]: *swap only* the type for a stop loss attached to a trigger order, 'fill_price', 'index_price' or 'mark_price', default is 'mark_price'
|
||
:param str [params.takeProfit.type]: *swap only* the type for a take profit attached to a trigger order, 'fill_price', 'index_price' or 'mark_price', default is 'mark_price'
|
||
:param str [params.trailingPercent]: *swap and future only* the percent to trail away from the current market price, rate can not be greater than 10
|
||
:param str [params.trailingTriggerPrice]: *swap and future only* the price to trigger a trailing stop order, default uses the price argument
|
||
:param str [params.triggerType]: *swap and future only* 'fill_price', 'mark_price' or 'index_price'
|
||
:param boolean [params.oneWayMode]: *swap and future only* required to set self to True in one_way_mode and you can leave self in hedge_mode, can adjust the mode using the setPositionMode() method
|
||
:param bool [params.hedged]: *swap and future only* True for hedged mode, False for one way mode, default is False
|
||
:param bool [params.reduceOnly]: True or False whether the order is reduce-only
|
||
:param boolean [params.uta]: set to True for the unified trading account(uta), defaults to False
|
||
:param str [params.posSide]: *uta only* hedged two-way position side, long or short
|
||
:returns dict: an `order structure <https://docs.ccxt.com/#/?id=order-structure>`
|
||
"""
|
||
await self.load_markets()
|
||
market = self.market(symbol)
|
||
marginParams = self.handle_margin_mode_and_params('createOrder', params)
|
||
marginMode = marginParams[0]
|
||
triggerPrice = self.safe_value_2(params, 'stopPrice', 'triggerPrice')
|
||
stopLossTriggerPrice = self.safe_value(params, 'stopLossPrice')
|
||
takeProfitTriggerPrice = self.safe_value(params, 'takeProfitPrice')
|
||
trailingPercent = self.safe_string_2(params, 'trailingPercent', 'callbackRatio')
|
||
isTrailingPercentOrder = trailingPercent is not None
|
||
isTriggerOrder = triggerPrice is not None
|
||
isStopLossTriggerOrder = stopLossTriggerPrice is not None
|
||
isTakeProfitTriggerOrder = takeProfitTriggerPrice is not None
|
||
isStopLossOrTakeProfitTrigger = isStopLossTriggerOrder or isTakeProfitTriggerOrder
|
||
response = None
|
||
uta = None
|
||
uta, params = self.handle_option_and_params(params, 'createOrder', 'uta', False)
|
||
if uta:
|
||
request = self.create_uta_order_request(symbol, type, side, amount, price, params)
|
||
if isStopLossOrTakeProfitTrigger:
|
||
response = await self.privateUtaPostV3TradePlaceStrategyOrder(request)
|
||
else:
|
||
response = await self.privateUtaPostV3TradePlaceOrder(request)
|
||
else:
|
||
request = self.create_order_request(symbol, type, side, amount, price, params)
|
||
if market['spot']:
|
||
if isTriggerOrder:
|
||
response = await self.privateSpotPostV2SpotTradePlacePlanOrder(request)
|
||
elif marginMode == 'isolated':
|
||
response = await self.privateMarginPostV2MarginIsolatedPlaceOrder(request)
|
||
elif marginMode == 'cross':
|
||
response = await self.privateMarginPostV2MarginCrossedPlaceOrder(request)
|
||
else:
|
||
response = await self.privateSpotPostV2SpotTradePlaceOrder(request)
|
||
else:
|
||
if isTriggerOrder or isTrailingPercentOrder:
|
||
response = await self.privateMixPostV2MixOrderPlacePlanOrder(request)
|
||
elif isStopLossOrTakeProfitTrigger:
|
||
response = await self.privateMixPostV2MixOrderPlaceTpslOrder(request)
|
||
else:
|
||
response = await self.privateMixPostV2MixOrderPlaceOrder(request)
|
||
#
|
||
# {
|
||
# "code": "00000",
|
||
# "msg": "success",
|
||
# "requestTime": 1645932209602,
|
||
# "data": {
|
||
# "orderId": "881669078313766912",
|
||
# "clientOid": "iauIBf#a45b595f96474d888d0ada"
|
||
# }
|
||
# }
|
||
#
|
||
data = self.safe_dict(response, 'data', {})
|
||
return self.parse_order(data, market)
|
||
|
||
def create_uta_order_request(self, symbol: str, type: OrderType, side: OrderSide, amount: float, price: Num = None, params={}):
|
||
market = self.market(symbol)
|
||
productType = None
|
||
productType, params = self.handle_product_type_and_params(market, params)
|
||
if productType == 'SPOT':
|
||
marginMode = None
|
||
marginMode, params = self.handle_margin_mode_and_params('createOrder', params)
|
||
if marginMode is not None:
|
||
productType = 'MARGIN'
|
||
request: dict = {
|
||
'category': productType,
|
||
'symbol': market['id'],
|
||
'qty': self.amount_to_precision(symbol, amount),
|
||
'side': side,
|
||
}
|
||
clientOrderId = self.safe_string_2(params, 'clientOid', 'clientOrderId')
|
||
if clientOrderId is not None:
|
||
request['clientOid'] = clientOrderId
|
||
params = self.omit(params, 'clientOrderId')
|
||
stopLossTriggerPrice = self.safe_number(params, 'stopLossPrice')
|
||
takeProfitTriggerPrice = self.safe_number(params, 'takeProfitPrice')
|
||
stopLoss = self.safe_value(params, 'stopLoss')
|
||
takeProfit = self.safe_value(params, 'takeProfit')
|
||
isStopLoss = stopLoss is not None
|
||
isTakeProfit = takeProfit is not None
|
||
isStopLossTrigger = stopLossTriggerPrice is not None
|
||
isTakeProfitTrigger = takeProfitTriggerPrice is not None
|
||
isStopLossOrTakeProfitTrigger = isStopLossTrigger or isTakeProfitTrigger
|
||
if isStopLossOrTakeProfitTrigger:
|
||
if isStopLossTrigger:
|
||
slType = self.safe_string(params, 'slTriggerBy', 'mark')
|
||
request['slTriggerBy'] = slType
|
||
request['stopLoss'] = self.price_to_precision(symbol, stopLossTriggerPrice)
|
||
if price is not None:
|
||
request['slLimitPrice'] = self.price_to_precision(symbol, price)
|
||
request['slOrderType'] = self.safe_string(params, 'slOrderType', 'limit')
|
||
else:
|
||
request['slOrderType'] = self.safe_string(params, 'slOrderType', 'market')
|
||
elif isTakeProfitTrigger:
|
||
tpType = self.safe_string(params, 'tpTriggerBy', 'mark')
|
||
request['tpTriggerBy'] = tpType
|
||
request['takeProfit'] = self.price_to_precision(symbol, takeProfitTriggerPrice)
|
||
if price is not None:
|
||
request['tpLimitPrice'] = self.price_to_precision(symbol, price)
|
||
request['tpOrderType'] = self.safe_string(params, 'tpOrderType', 'limit')
|
||
else:
|
||
request['tpOrderType'] = self.safe_string(params, 'tpOrderType', 'market')
|
||
params = self.omit(params, ['stopLossPrice', 'takeProfitPrice'])
|
||
else:
|
||
if isStopLoss:
|
||
slTriggerPrice = self.safe_number_2(stopLoss, 'triggerPrice', 'stopPrice')
|
||
slLimitPrice = self.safe_number(stopLoss, 'price')
|
||
request['stopLoss'] = self.price_to_precision(symbol, slTriggerPrice)
|
||
if slLimitPrice is not None:
|
||
request['slLimitPrice'] = self.price_to_precision(symbol, slLimitPrice)
|
||
request['slOrderType'] = self.safe_string(params, 'slOrderType', 'limit')
|
||
else:
|
||
request['slOrderType'] = self.safe_string(params, 'slOrderType', 'market')
|
||
if isTakeProfit:
|
||
tpTriggerPrice = self.safe_number_2(takeProfit, 'triggerPrice', 'stopPrice')
|
||
tpLimitPrice = self.safe_number(takeProfit, 'price')
|
||
request['takeProfit'] = self.price_to_precision(symbol, tpTriggerPrice)
|
||
if tpLimitPrice is not None:
|
||
request['tpLimitPrice'] = self.price_to_precision(symbol, tpLimitPrice)
|
||
request['tpOrderType'] = self.safe_string(params, 'tpOrderType', 'limit')
|
||
else:
|
||
request['tpOrderType'] = self.safe_string(params, 'tpOrderType', 'market')
|
||
isMarketOrder = type == 'market'
|
||
if not isMarketOrder:
|
||
request['price'] = self.price_to_precision(symbol, price)
|
||
request['orderType'] = type
|
||
exchangeSpecificTifParam = self.safe_string(params, 'timeInForce')
|
||
postOnly = None
|
||
postOnly, params = self.handle_post_only(isMarketOrder, exchangeSpecificTifParam == 'post_only', params)
|
||
defaultTimeInForce = self.safe_string_upper(self.options, 'defaultTimeInForce')
|
||
timeInForce = self.safe_string_upper(params, 'timeInForce', defaultTimeInForce)
|
||
if postOnly:
|
||
request['timeInForce'] = 'post_only'
|
||
elif timeInForce == 'GTC':
|
||
request['timeInForce'] = 'gtc'
|
||
elif timeInForce == 'FOK':
|
||
request['timeInForce'] = 'fok'
|
||
elif timeInForce == 'IOC':
|
||
request['timeInForce'] = 'ioc'
|
||
reduceOnly = self.safe_bool(params, 'reduceOnly', False)
|
||
hedged = None
|
||
hedged, params = self.handle_param_bool(params, 'hedged', False)
|
||
if reduceOnly:
|
||
if hedged or isStopLossOrTakeProfitTrigger:
|
||
reduceOnlyPosSide = 'long' if (side == 'sell') else 'short'
|
||
request['posSide'] = reduceOnlyPosSide
|
||
elif not isStopLossOrTakeProfitTrigger:
|
||
request['reduceOnly'] = 'yes'
|
||
else:
|
||
if hedged:
|
||
posSide = 'long' if (side == 'buy') else 'short'
|
||
request['posSide'] = posSide
|
||
params = self.omit(params, ['stopLoss', 'takeProfit', 'postOnly', 'reduceOnly', 'hedged'])
|
||
return self.extend(request, params)
|
||
|
||
def create_order_request(self, symbol: str, type: OrderType, side: OrderSide, amount: float, price: Num = None, params={}):
|
||
market = self.market(symbol)
|
||
marketType = None
|
||
marginMode = None
|
||
marketType, params = self.handle_market_type_and_params('createOrder', market, params)
|
||
marginMode, params = self.handle_margin_mode_and_params('createOrder', params)
|
||
request: dict = {
|
||
'symbol': market['id'],
|
||
'orderType': type,
|
||
}
|
||
hedged = None
|
||
hedged, params = self.handle_param_bool(params, 'hedged', False)
|
||
# backward compatibility for `oneWayMode`
|
||
oneWayMode = None
|
||
oneWayMode, params = self.handle_param_bool(params, 'oneWayMode')
|
||
if oneWayMode is not None:
|
||
hedged = not oneWayMode
|
||
isMarketOrder = type == 'market'
|
||
triggerPrice = self.safe_value_2(params, 'stopPrice', 'triggerPrice')
|
||
stopLossTriggerPrice = self.safe_value(params, 'stopLossPrice')
|
||
takeProfitTriggerPrice = self.safe_value(params, 'takeProfitPrice')
|
||
stopLoss = self.safe_value(params, 'stopLoss')
|
||
takeProfit = self.safe_value(params, 'takeProfit')
|
||
isTriggerOrder = triggerPrice is not None
|
||
isStopLossTriggerOrder = stopLossTriggerPrice is not None
|
||
isTakeProfitTriggerOrder = takeProfitTriggerPrice is not None
|
||
isStopLoss = stopLoss is not None
|
||
isTakeProfit = takeProfit is not None
|
||
isStopLossOrTakeProfitTrigger = isStopLossTriggerOrder or isTakeProfitTriggerOrder
|
||
isStopLossOrTakeProfit = isStopLoss or isTakeProfit
|
||
trailingTriggerPrice = self.safe_string(params, 'trailingTriggerPrice', self.number_to_string(price))
|
||
trailingPercent = self.safe_string_2(params, 'trailingPercent', 'callbackRatio')
|
||
isTrailingPercentOrder = trailingPercent is not None
|
||
if self.sum(isTriggerOrder, isStopLossTriggerOrder, isTakeProfitTriggerOrder, isTrailingPercentOrder) > 1:
|
||
raise ExchangeError(self.id + ' createOrder() params can only contain one of triggerPrice, stopLossPrice, takeProfitPrice, trailingPercent')
|
||
if type == 'limit':
|
||
request['price'] = self.price_to_precision(symbol, price)
|
||
triggerPriceType = self.safe_string_2(params, 'triggerPriceType', 'triggerType', 'mark_price')
|
||
reduceOnly = self.safe_bool(params, 'reduceOnly', False)
|
||
clientOrderId = self.safe_string_2(params, 'clientOid', 'clientOrderId')
|
||
exchangeSpecificTifParam = self.safe_string_2(params, 'force', 'timeInForce')
|
||
postOnly = None
|
||
postOnly, params = self.handle_post_only(isMarketOrder, exchangeSpecificTifParam == 'post_only', params)
|
||
defaultTimeInForce = self.safe_string_upper(self.options, 'defaultTimeInForce')
|
||
timeInForce = self.safe_string_upper(params, 'timeInForce', defaultTimeInForce)
|
||
if postOnly:
|
||
request['force'] = 'post_only'
|
||
elif timeInForce == 'GTC':
|
||
request['force'] = 'GTC'
|
||
elif timeInForce == 'FOK':
|
||
request['force'] = 'FOK'
|
||
elif timeInForce == 'IOC':
|
||
request['force'] = 'IOC'
|
||
params = self.omit(params, ['stopPrice', 'triggerType', 'stopLossPrice', 'takeProfitPrice', 'stopLoss', 'takeProfit', 'postOnly', 'reduceOnly', 'clientOrderId', 'trailingPercent', 'trailingTriggerPrice'])
|
||
if (marketType == 'swap') or (marketType == 'future'):
|
||
request['marginCoin'] = market['settleId']
|
||
request['size'] = self.amount_to_precision(symbol, amount)
|
||
productType = None
|
||
productType, params = self.handle_product_type_and_params(market, params)
|
||
request['productType'] = productType
|
||
if clientOrderId is not None:
|
||
request['clientOid'] = clientOrderId
|
||
if isTriggerOrder or isStopLossOrTakeProfitTrigger or isTrailingPercentOrder:
|
||
request['triggerType'] = triggerPriceType
|
||
if isTrailingPercentOrder:
|
||
if not isMarketOrder:
|
||
raise BadRequest(self.id + ' createOrder() bitget trailing orders must be market orders')
|
||
if trailingTriggerPrice is None:
|
||
raise ArgumentsRequired(self.id + ' createOrder() bitget trailing orders must have a trailingTriggerPrice param')
|
||
request['planType'] = 'track_plan'
|
||
request['triggerPrice'] = self.price_to_precision(symbol, trailingTriggerPrice)
|
||
request['callbackRatio'] = trailingPercent
|
||
elif isTriggerOrder:
|
||
request['planType'] = 'normal_plan'
|
||
request['triggerPrice'] = self.price_to_precision(symbol, triggerPrice)
|
||
if price is not None:
|
||
request['executePrice'] = self.price_to_precision(symbol, price)
|
||
if isStopLoss:
|
||
slTriggerPrice = self.safe_string_2(stopLoss, 'triggerPrice', 'stopPrice')
|
||
request['stopLossTriggerPrice'] = self.price_to_precision(symbol, slTriggerPrice)
|
||
slPrice = self.safe_string(stopLoss, 'price')
|
||
request['stopLossExecutePrice'] = self.price_to_precision(symbol, slPrice)
|
||
slType = self.safe_string(stopLoss, 'type', 'mark_price')
|
||
request['stopLossTriggerType'] = slType
|
||
if isTakeProfit:
|
||
tpTriggerPrice = self.safe_string_2(takeProfit, 'triggerPrice', 'stopPrice')
|
||
request['stopSurplusTriggerPrice'] = self.price_to_precision(symbol, tpTriggerPrice)
|
||
tpPrice = self.safe_string(takeProfit, 'price')
|
||
request['stopSurplusExecutePrice'] = self.price_to_precision(symbol, tpPrice)
|
||
tpType = self.safe_string(takeProfit, 'type', 'mark_price')
|
||
request['stopSurplusTriggerType'] = tpType
|
||
elif isStopLossOrTakeProfitTrigger:
|
||
if not isMarketOrder:
|
||
raise ExchangeError(self.id + ' createOrder() bitget stopLoss or takeProfit orders must be market orders')
|
||
if hedged:
|
||
request['holdSide'] = 'long' if (side == 'sell') else 'short'
|
||
else:
|
||
request['holdSide'] = 'buy' if (side == 'sell') else 'sell'
|
||
if isStopLossTriggerOrder:
|
||
request['triggerPrice'] = self.price_to_precision(symbol, stopLossTriggerPrice)
|
||
request['planType'] = 'pos_loss'
|
||
elif isTakeProfitTriggerOrder:
|
||
request['triggerPrice'] = self.price_to_precision(symbol, takeProfitTriggerPrice)
|
||
request['planType'] = 'pos_profit'
|
||
else:
|
||
if isStopLoss:
|
||
slTriggerPrice = self.safe_value_2(stopLoss, 'triggerPrice', 'stopPrice')
|
||
request['presetStopLossPrice'] = self.price_to_precision(symbol, slTriggerPrice)
|
||
if isTakeProfit:
|
||
tpTriggerPrice = self.safe_value_2(takeProfit, 'triggerPrice', 'stopPrice')
|
||
request['presetStopSurplusPrice'] = self.price_to_precision(symbol, tpTriggerPrice)
|
||
if not isStopLossOrTakeProfitTrigger:
|
||
if marginMode is None:
|
||
marginMode = 'cross'
|
||
marginModeRequest = 'crossed' if (marginMode == 'cross') else 'isolated'
|
||
request['marginMode'] = marginModeRequest
|
||
requestSide = side
|
||
if reduceOnly:
|
||
if not hedged:
|
||
request['reduceOnly'] = 'YES'
|
||
else:
|
||
# on bitget hedge mode if the position is long the side is always buy, and if the position is short the side is always sell
|
||
requestSide = 'sell' if (side == 'buy') else 'buy'
|
||
request['tradeSide'] = 'Close'
|
||
else:
|
||
if hedged:
|
||
request['tradeSide'] = 'Open'
|
||
request['side'] = requestSide
|
||
elif marketType == 'spot':
|
||
if isStopLossOrTakeProfitTrigger or isStopLossOrTakeProfit:
|
||
raise InvalidOrder(self.id + ' createOrder() does not support stop loss/take profit orders on spot markets, only swap markets')
|
||
request['side'] = side
|
||
quantity = None
|
||
planType = None
|
||
createMarketBuyOrderRequiresPrice = True
|
||
createMarketBuyOrderRequiresPrice, params = self.handle_option_and_params(params, 'createOrder', 'createMarketBuyOrderRequiresPrice', True)
|
||
if isMarketOrder and (side == 'buy'):
|
||
planType = 'total'
|
||
cost = self.safe_number(params, 'cost')
|
||
params = self.omit(params, 'cost')
|
||
if cost is not None:
|
||
quantity = self.cost_to_precision(symbol, cost)
|
||
elif createMarketBuyOrderRequiresPrice:
|
||
if price is None:
|
||
raise InvalidOrder(self.id + ' createOrder() requires the price argument for market buy orders to calculate the total cost to spend(amount * price), alternatively set the createMarketBuyOrderRequiresPrice option or param to False and pass the cost to spend in the amount argument')
|
||
else:
|
||
amountString = self.number_to_string(amount)
|
||
priceString = self.number_to_string(price)
|
||
quoteAmount = Precise.string_mul(amountString, priceString)
|
||
quantity = self.cost_to_precision(symbol, quoteAmount)
|
||
else:
|
||
quantity = self.cost_to_precision(symbol, amount)
|
||
else:
|
||
planType = 'amount'
|
||
quantity = self.amount_to_precision(symbol, amount)
|
||
if clientOrderId is not None:
|
||
request['clientOid'] = clientOrderId
|
||
if marginMode is not None:
|
||
request['loanType'] = 'normal'
|
||
if isMarketOrder and (side == 'buy'):
|
||
request['quoteSize'] = quantity
|
||
else:
|
||
request['baseSize'] = quantity
|
||
else:
|
||
if quantity is not None:
|
||
request['size'] = quantity
|
||
if triggerPrice is not None:
|
||
request['planType'] = planType
|
||
request['triggerType'] = triggerPriceType
|
||
request['triggerPrice'] = self.price_to_precision(symbol, triggerPrice)
|
||
if price is not None:
|
||
request['executePrice'] = self.price_to_precision(symbol, price)
|
||
else:
|
||
raise NotSupported(self.id + ' createOrder() does not support ' + marketType + ' orders')
|
||
return self.extend(request, params)
|
||
|
||
async def create_uta_orders(self, orders: List[OrderRequest], params={}):
|
||
await self.load_markets()
|
||
ordersRequests = []
|
||
symbol = None
|
||
marginMode = None
|
||
for i in range(0, len(orders)):
|
||
rawOrder = orders[i]
|
||
marketId = self.safe_string(rawOrder, 'symbol')
|
||
if symbol is None:
|
||
symbol = marketId
|
||
else:
|
||
if symbol != marketId:
|
||
raise BadRequest(self.id + ' createOrders() requires all orders to have the same symbol')
|
||
type = self.safe_string(rawOrder, 'type')
|
||
side = self.safe_string(rawOrder, 'side')
|
||
amount = self.safe_value(rawOrder, 'amount')
|
||
price = self.safe_value(rawOrder, 'price')
|
||
orderParams = self.safe_value(rawOrder, 'params', {})
|
||
marginResult = self.handle_margin_mode_and_params('createOrders', orderParams)
|
||
currentMarginMode = marginResult[0]
|
||
if currentMarginMode is not None:
|
||
if marginMode is None:
|
||
marginMode = currentMarginMode
|
||
else:
|
||
if marginMode != currentMarginMode:
|
||
raise BadRequest(self.id + ' createOrders() requires all orders to have the same margin mode(isolated or cross)')
|
||
orderRequest = self.create_uta_order_request(marketId, type, side, amount, price, orderParams)
|
||
ordersRequests.append(orderRequest)
|
||
market = self.market(symbol)
|
||
response = await self.privateUtaPostV3TradePlaceBatch(ordersRequests)
|
||
#
|
||
# {
|
||
# "code": "00000",
|
||
# "msg": "success",
|
||
# "requestTime": 1752810184560,
|
||
# "data": [
|
||
# {
|
||
# "orderId": "1329947796441513984",
|
||
# "clientOid": "1329947796483457024"
|
||
# },
|
||
# ]
|
||
# }
|
||
#
|
||
data = self.safe_list(response, 'data', [])
|
||
return self.parse_orders(data, market)
|
||
|
||
async def create_orders(self, orders: List[OrderRequest], params={}):
|
||
"""
|
||
create a list of trade orders(all orders should be of the same symbol)
|
||
|
||
https://www.bitget.com/api-doc/spot/trade/Batch-Place-Orders
|
||
https://www.bitget.com/api-doc/contract/trade/Batch-Order
|
||
https://www.bitget.com/api-doc/margin/isolated/trade/Isolated-Batch-Order
|
||
https://www.bitget.com/api-doc/margin/cross/trade/Cross-Batch-Order
|
||
https://www.bitget.com/api-doc/uta/trade/Place-Batch
|
||
|
||
:param Array orders: list of orders to create, each object should contain the parameters required by createOrder, namely symbol, type, side, amount, price and params
|
||
:param dict [params]: extra parameters specific to the api endpoint
|
||
:param boolean [params.uta]: set to True for the unified trading account(uta), defaults to False
|
||
:returns dict: an `order structure <https://docs.ccxt.com/#/?id=order-structure>`
|
||
"""
|
||
await self.load_markets()
|
||
uta = None
|
||
uta, params = self.handle_option_and_params(params, 'createOrders', 'uta', False)
|
||
if uta:
|
||
return await self.create_uta_orders(orders, params)
|
||
ordersRequests = []
|
||
symbol = None
|
||
marginMode = None
|
||
for i in range(0, len(orders)):
|
||
rawOrder = orders[i]
|
||
marketId = self.safe_string(rawOrder, 'symbol')
|
||
if symbol is None:
|
||
symbol = marketId
|
||
else:
|
||
if symbol != marketId:
|
||
raise BadRequest(self.id + ' createOrders() requires all orders to have the same symbol')
|
||
type = self.safe_string(rawOrder, 'type')
|
||
side = self.safe_string(rawOrder, 'side')
|
||
amount = self.safe_value(rawOrder, 'amount')
|
||
price = self.safe_value(rawOrder, 'price')
|
||
orderParams = self.safe_value(rawOrder, 'params', {})
|
||
marginResult = self.handle_margin_mode_and_params('createOrders', orderParams)
|
||
currentMarginMode = marginResult[0]
|
||
if currentMarginMode is not None:
|
||
if marginMode is None:
|
||
marginMode = currentMarginMode
|
||
else:
|
||
if marginMode != currentMarginMode:
|
||
raise BadRequest(self.id + ' createOrders() requires all orders to have the same margin mode(isolated or cross)')
|
||
orderRequest = self.create_order_request(marketId, type, side, amount, price, orderParams)
|
||
ordersRequests.append(orderRequest)
|
||
market = self.market(symbol)
|
||
request: dict = {
|
||
'symbol': market['id'],
|
||
'orderList': ordersRequests,
|
||
}
|
||
response = None
|
||
if (market['swap']) or (market['future']):
|
||
if marginMode is None:
|
||
marginMode = 'cross'
|
||
marginModeRequest = 'crossed' if (marginMode == 'cross') else 'isolated'
|
||
request['marginMode'] = marginModeRequest
|
||
request['marginCoin'] = market['settleId']
|
||
productType = None
|
||
productType, params = self.handle_product_type_and_params(market, params)
|
||
request['productType'] = productType
|
||
response = await self.privateMixPostV2MixOrderBatchPlaceOrder(request)
|
||
elif marginMode == 'isolated':
|
||
response = await self.privateMarginPostV2MarginIsolatedBatchPlaceOrder(request)
|
||
elif marginMode == 'cross':
|
||
response = await self.privateMarginPostV2MarginCrossedBatchPlaceOrder(request)
|
||
else:
|
||
response = await self.privateSpotPostV2SpotTradeBatchOrders(request)
|
||
#
|
||
# {
|
||
# "code": "00000",
|
||
# "msg": "success",
|
||
# "requestTime": 1700703539416,
|
||
# "data": {
|
||
# "successList": [
|
||
# {
|
||
# "orderId": "1111397214281175046",
|
||
# "clientOid": "766d3fc3-7321-4406-a689-15c9987a2e75"
|
||
# },
|
||
# ],
|
||
# "failureList": [
|
||
# {
|
||
# "orderId": "",
|
||
# "clientOid": "d1b75cb3-cc15-4ede-ad4c-3937396f75ab",
|
||
# "errorMsg": "less than the minimum amount 5 USDT",
|
||
# "errorCode": "45110"
|
||
# },
|
||
# ]
|
||
# }
|
||
# }
|
||
#
|
||
data = self.safe_value(response, 'data', {})
|
||
failure = self.safe_value(data, 'failureList', [])
|
||
orderInfo = self.safe_value(data, 'successList', [])
|
||
both = self.array_concat(orderInfo, failure)
|
||
return self.parse_orders(both, market)
|
||
|
||
async def edit_order(self, id: str, symbol: str, type: OrderType, side: OrderSide, amount: Num = None, price: Num = None, params={}):
|
||
"""
|
||
edit a trade order
|
||
|
||
https://www.bitget.com/api-doc/spot/plan/Modify-Plan-Order
|
||
https://www.bitget.com/api-doc/spot/trade/Cancel-Replace-Order
|
||
https://www.bitget.com/api-doc/contract/trade/Modify-Order
|
||
https://www.bitget.com/api-doc/contract/plan/Modify-Tpsl-Order
|
||
https://www.bitget.com/api-doc/contract/plan/Modify-Plan-Order
|
||
https://www.bitget.com/api-doc/uta/trade/Modify-Order
|
||
https://www.bitget.com/api-doc/uta/strategy/Modify-Strategy-Order
|
||
|
||
:param str id: cancel order id
|
||
:param str symbol: unified symbol of the market to create an order in
|
||
:param str type: 'market' or 'limit'
|
||
:param str side: 'buy' or 'sell'
|
||
:param float amount: how much you want to trade in units of the base currency
|
||
:param float [price]: the price at which the order is to be fulfilled, in units of the quote currency, ignored in market orders
|
||
:param dict [params]: extra parameters specific to the exchange API endpoint
|
||
:param float [params.triggerPrice]: the price that a trigger order is triggered at
|
||
:param float [params.stopLossPrice]: *swap only* The price at which a stop loss order is triggered at
|
||
:param float [params.takeProfitPrice]: *swap only* The price at which a take profit order is triggered at
|
||
:param dict [params.takeProfit]: *takeProfit object in params* containing the triggerPrice at which the attached take profit order will be triggered(perpetual swap markets only)
|
||
:param float [params.takeProfit.triggerPrice]: *swap only* take profit trigger price
|
||
:param dict [params.stopLoss]: *stopLoss object in params* containing the triggerPrice at which the attached stop loss order will be triggered(perpetual swap markets only)
|
||
:param float [params.stopLoss.triggerPrice]: *swap only* stop loss trigger price
|
||
:param float [params.stopLoss.price]: *swap only* the execution price for a stop loss attached to a trigger order
|
||
:param float [params.takeProfit.price]: *swap only* the execution price for a take profit attached to a trigger order
|
||
:param str [params.stopLoss.type]: *swap only* the type for a stop loss attached to a trigger order, 'fill_price', 'index_price' or 'mark_price', default is 'mark_price'
|
||
:param str [params.takeProfit.type]: *swap only* the type for a take profit attached to a trigger order, 'fill_price', 'index_price' or 'mark_price', default is 'mark_price'
|
||
:param str [params.trailingPercent]: *swap and future only* the percent to trail away from the current market price, rate can not be greater than 10
|
||
:param str [params.trailingTriggerPrice]: *swap and future only* the price to trigger a trailing stop order, default uses the price argument
|
||
:param str [params.newTriggerType]: *swap and future only* 'fill_price', 'mark_price' or 'index_price'
|
||
:param boolean [params.uta]: set to True for the unified trading account(uta), defaults to False
|
||
:returns dict: an `order structure <https://docs.ccxt.com/#/?id=order-structure>`
|
||
"""
|
||
await self.load_markets()
|
||
market = self.market(symbol)
|
||
request: dict = {
|
||
# 'orderId': id,
|
||
}
|
||
clientOrderId = self.safe_string_2(params, 'clientOrderId', 'clientOid')
|
||
if clientOrderId is not None:
|
||
params = self.omit(params, ['clientOrderId'])
|
||
request['clientOid'] = clientOrderId
|
||
else:
|
||
request['orderId'] = id
|
||
isMarketOrder = type == 'market'
|
||
triggerPrice = self.safe_value_2(params, 'stopPrice', 'triggerPrice')
|
||
isTriggerOrder = triggerPrice is not None
|
||
stopLossPrice = self.safe_value(params, 'stopLossPrice')
|
||
isStopLossOrder = stopLossPrice is not None
|
||
takeProfitPrice = self.safe_value(params, 'takeProfitPrice')
|
||
isTakeProfitOrder = takeProfitPrice is not None
|
||
stopLoss = self.safe_value(params, 'stopLoss')
|
||
takeProfit = self.safe_value(params, 'takeProfit')
|
||
isStopLoss = stopLoss is not None
|
||
isTakeProfit = takeProfit is not None
|
||
trailingTriggerPrice = self.safe_string(params, 'trailingTriggerPrice', self.number_to_string(price))
|
||
trailingPercent = self.safe_string_2(params, 'trailingPercent', 'newCallbackRatio')
|
||
isTrailingPercentOrder = trailingPercent is not None
|
||
if self.sum(isTriggerOrder, isStopLossOrder, isTakeProfitOrder, isTrailingPercentOrder) > 1:
|
||
raise ExchangeError(self.id + ' editOrder() params can only contain one of triggerPrice, stopLossPrice, takeProfitPrice, trailingPercent')
|
||
params = self.omit(params, ['stopPrice', 'triggerType', 'stopLossPrice', 'takeProfitPrice', 'stopLoss', 'takeProfit', 'clientOrderId', 'trailingTriggerPrice', 'trailingPercent'])
|
||
response = None
|
||
productType = None
|
||
uta = None
|
||
productType, params = self.handle_product_type_and_params(market, params)
|
||
uta, params = self.handle_option_and_params(params, 'editOrder', 'uta', False)
|
||
if uta:
|
||
if amount is not None:
|
||
request['qty'] = self.amount_to_precision(symbol, amount)
|
||
if isStopLossOrder or isTakeProfitOrder:
|
||
if isStopLossOrder:
|
||
slType = self.safe_string(params, 'slTriggerBy', 'mark')
|
||
request['slTriggerBy'] = slType
|
||
request['stopLoss'] = self.price_to_precision(symbol, stopLossPrice)
|
||
if price is not None:
|
||
request['slLimitPrice'] = self.price_to_precision(symbol, price)
|
||
request['slOrderType'] = self.safe_string(params, 'slOrderType', 'limit')
|
||
else:
|
||
request['slOrderType'] = self.safe_string(params, 'slOrderType', 'market')
|
||
elif isTakeProfitOrder:
|
||
tpType = self.safe_string(params, 'tpTriggerBy', 'mark')
|
||
request['tpTriggerBy'] = tpType
|
||
request['takeProfit'] = self.price_to_precision(symbol, takeProfitPrice)
|
||
if price is not None:
|
||
request['tpLimitPrice'] = self.price_to_precision(symbol, price)
|
||
request['tpOrderType'] = self.safe_string(params, 'tpOrderType', 'limit')
|
||
else:
|
||
request['tpOrderType'] = self.safe_string(params, 'tpOrderType', 'market')
|
||
params = self.omit(params, ['stopLossPrice', 'takeProfitPrice'])
|
||
response = await self.privateUtaPostV3TradeModifyStrategyOrder(self.extend(request, params))
|
||
else:
|
||
if price is not None:
|
||
request['price'] = self.price_to_precision(symbol, price)
|
||
response = await self.privateUtaPostV3TradeModifyOrder(self.extend(request, params))
|
||
elif market['spot']:
|
||
cost = self.safe_string(params, 'cost')
|
||
params = self.omit(params, 'cost')
|
||
editMarketBuyOrderRequiresPrice = self.safe_bool(self.options, 'editMarketBuyOrderRequiresPrice', True)
|
||
if (editMarketBuyOrderRequiresPrice or (cost is not None)) and isMarketOrder and (side == 'buy'):
|
||
if price is None and cost is None:
|
||
raise InvalidOrder(self.id + ' editOrder() requires price argument for market buy orders on spot markets to calculate the total amount to spend(amount * price), alternatively provide `cost` in the params')
|
||
else:
|
||
amountString = self.number_to_string(amount)
|
||
priceString = self.number_to_string(price)
|
||
finalCost = (Precise.string_mul(amountString, priceString)) if (cost is None) else cost
|
||
request['size'] = self.price_to_precision(symbol, finalCost)
|
||
else:
|
||
request['size'] = self.amount_to_precision(symbol, amount)
|
||
request['orderType'] = type
|
||
if triggerPrice is not None:
|
||
request['triggerPrice'] = self.price_to_precision(symbol, triggerPrice)
|
||
request['executePrice'] = self.price_to_precision(symbol, price)
|
||
else:
|
||
request['price'] = self.price_to_precision(symbol, price)
|
||
if triggerPrice is not None:
|
||
response = await self.privateSpotPostV2SpotTradeModifyPlanOrder(self.extend(request, params))
|
||
else:
|
||
request['symbol'] = market['id']
|
||
response = await self.privateSpotPostV2SpotTradeCancelReplaceOrder(self.extend(request, params))
|
||
else:
|
||
if (not market['swap']) and (not market['future']):
|
||
raise NotSupported(self.id + ' editOrder() does not support ' + market['type'] + ' orders')
|
||
request['symbol'] = market['id']
|
||
request['productType'] = productType
|
||
if not isTakeProfitOrder and not isStopLossOrder:
|
||
request['newSize'] = self.amount_to_precision(symbol, amount)
|
||
if (price is not None) and not isTrailingPercentOrder:
|
||
request['newPrice'] = self.price_to_precision(symbol, price)
|
||
if isTrailingPercentOrder:
|
||
if not isMarketOrder:
|
||
raise BadRequest(self.id + ' editOrder() bitget trailing orders must be market orders')
|
||
if trailingTriggerPrice is not None:
|
||
request['newTriggerPrice'] = self.price_to_precision(symbol, trailingTriggerPrice)
|
||
request['newCallbackRatio'] = trailingPercent
|
||
response = await self.privateMixPostV2MixOrderModifyPlanOrder(self.extend(request, params))
|
||
elif isTakeProfitOrder or isStopLossOrder:
|
||
request['marginCoin'] = market['settleId']
|
||
request['size'] = self.amount_to_precision(symbol, amount)
|
||
if price is not None:
|
||
request['executePrice'] = self.price_to_precision(symbol, price)
|
||
if isStopLossOrder:
|
||
request['triggerPrice'] = self.price_to_precision(symbol, stopLossPrice)
|
||
elif isTakeProfitOrder:
|
||
request['triggerPrice'] = self.price_to_precision(symbol, takeProfitPrice)
|
||
response = await self.privateMixPostV2MixOrderModifyTpslOrder(self.extend(request, params))
|
||
elif isTriggerOrder:
|
||
request['newTriggerPrice'] = self.price_to_precision(symbol, triggerPrice)
|
||
if isStopLoss:
|
||
slTriggerPrice = self.safe_number_2(stopLoss, 'triggerPrice', 'stopPrice')
|
||
request['newStopLossTriggerPrice'] = self.price_to_precision(symbol, slTriggerPrice)
|
||
slPrice = self.safe_number(stopLoss, 'price')
|
||
request['newStopLossExecutePrice'] = self.price_to_precision(symbol, slPrice)
|
||
slType = self.safe_string(stopLoss, 'type', 'mark_price')
|
||
request['newStopLossTriggerType'] = slType
|
||
if isTakeProfit:
|
||
tpTriggerPrice = self.safe_number_2(takeProfit, 'triggerPrice', 'stopPrice')
|
||
request['newSurplusTriggerPrice'] = self.price_to_precision(symbol, tpTriggerPrice)
|
||
tpPrice = self.safe_number(takeProfit, 'price')
|
||
request['newStopSurplusExecutePrice'] = self.price_to_precision(symbol, tpPrice)
|
||
tpType = self.safe_string(takeProfit, 'type', 'mark_price')
|
||
request['newStopSurplusTriggerType'] = tpType
|
||
response = await self.privateMixPostV2MixOrderModifyPlanOrder(self.extend(request, params))
|
||
else:
|
||
defaultNewClientOrderId = self.uuid()
|
||
newClientOrderId = self.safe_string_2(params, 'newClientOid', 'newClientOrderId', defaultNewClientOrderId)
|
||
params = self.omit(params, 'newClientOrderId')
|
||
request['newClientOid'] = newClientOrderId
|
||
if isStopLoss:
|
||
slTriggerPrice = self.safe_value_2(stopLoss, 'triggerPrice', 'stopPrice')
|
||
request['newPresetStopLossPrice'] = self.price_to_precision(symbol, slTriggerPrice)
|
||
if isTakeProfit:
|
||
tpTriggerPrice = self.safe_value_2(takeProfit, 'triggerPrice', 'stopPrice')
|
||
request['newPresetStopSurplusPrice'] = self.price_to_precision(symbol, tpTriggerPrice)
|
||
response = await self.privateMixPostV2MixOrderModifyOrder(self.extend(request, params))
|
||
#
|
||
# {
|
||
# "code": "00000",
|
||
# "msg": "success",
|
||
# "requestTime": 1700708275737,
|
||
# "data": {
|
||
# "clientOid": "abe95dbe-6081-4a6f-a2d3-ae49601cd459",
|
||
# "orderId": null
|
||
# }
|
||
# }
|
||
#
|
||
data = self.safe_dict(response, 'data', {})
|
||
return self.parse_order(data, market)
|
||
|
||
async def cancel_order(self, id: str, symbol: Str = None, params={}):
|
||
"""
|
||
cancels an open order
|
||
|
||
https://www.bitget.com/api-doc/spot/trade/Cancel-Order
|
||
https://www.bitget.com/api-doc/spot/plan/Cancel-Plan-Order
|
||
https://www.bitget.com/api-doc/contract/trade/Cancel-Order
|
||
https://www.bitget.com/api-doc/contract/plan/Cancel-Plan-Order
|
||
https://www.bitget.com/api-doc/margin/cross/trade/Cross-Cancel-Order
|
||
https://www.bitget.com/api-doc/margin/isolated/trade/Isolated-Cancel-Order
|
||
https://www.bitget.com/api-doc/uta/trade/Cancel-Order
|
||
https://www.bitget.com/api-doc/uta/strategy/Cancel-Strategy-Order
|
||
|
||
:param str id: order id
|
||
:param str symbol: unified symbol of the market the order was made in
|
||
:param dict [params]: extra parameters specific to the exchange API endpoint
|
||
:param str [params.marginMode]: 'isolated' or 'cross' for spot margin trading
|
||
:param boolean [params.trigger]: set to True for canceling trigger orders
|
||
:param str [params.planType]: *swap only* either profit_plan, loss_plan, normal_plan, pos_profit, pos_loss, moving_plan or track_plan
|
||
:param boolean [params.trailing]: set to True if you want to cancel a trailing order
|
||
:param boolean [params.uta]: set to True for the unified trading account(uta), defaults to False
|
||
:param str [params.clientOrderId]: the clientOrderId of the order, id does not need to be provided if clientOrderId is provided
|
||
:returns dict: An `order structure <https://docs.ccxt.com/#/?id=order-structure>`
|
||
"""
|
||
if symbol is None:
|
||
raise ArgumentsRequired(self.id + ' cancelOrder() requires a symbol argument')
|
||
await self.load_markets()
|
||
market = self.market(symbol)
|
||
marginMode = None
|
||
response = None
|
||
marginMode, params = self.handle_margin_mode_and_params('cancelOrder', params)
|
||
request: dict = {}
|
||
trailing = self.safe_value(params, 'trailing')
|
||
trigger = self.safe_value_2(params, 'stop', 'trigger')
|
||
params = self.omit(params, ['stop', 'trigger', 'trailing'])
|
||
if not (market['spot'] and trigger):
|
||
request['symbol'] = market['id']
|
||
uta = None
|
||
uta, params = self.handle_option_and_params(params, 'cancelOrder', 'uta', False)
|
||
isPlanOrder = trigger or trailing
|
||
isContract = market['swap'] or market['future']
|
||
isContractTriggerEndpoint = isContract and isPlanOrder and not uta
|
||
clientOrderId = self.safe_string_2(params, 'clientOrderId', 'clientOid')
|
||
if isContractTriggerEndpoint:
|
||
orderIdList = []
|
||
orderId: dict = {}
|
||
if clientOrderId is not None:
|
||
params = self.omit(params, 'clientOrderId')
|
||
orderId['clientOid'] = clientOrderId
|
||
else:
|
||
orderId['orderId'] = id
|
||
orderIdList.append(orderId)
|
||
request['orderIdList'] = orderIdList
|
||
else:
|
||
if clientOrderId is not None:
|
||
params = self.omit(params, 'clientOrderId')
|
||
request['clientOid'] = clientOrderId
|
||
else:
|
||
request['orderId'] = id
|
||
if uta:
|
||
if trigger:
|
||
response = await self.privateUtaPostV3TradeCancelStrategyOrder(self.extend(request, params))
|
||
else:
|
||
response = await self.privateUtaPostV3TradeCancelOrder(self.extend(request, params))
|
||
elif (market['swap']) or (market['future']):
|
||
productType = None
|
||
productType, params = self.handle_product_type_and_params(market, params)
|
||
request['productType'] = productType
|
||
if trailing:
|
||
planType = self.safe_string(params, 'planType', 'track_plan')
|
||
request['planType'] = planType
|
||
response = await self.privateMixPostV2MixOrderCancelPlanOrder(self.extend(request, params))
|
||
elif trigger:
|
||
response = await self.privateMixPostV2MixOrderCancelPlanOrder(self.extend(request, params))
|
||
else:
|
||
response = await self.privateMixPostV2MixOrderCancelOrder(self.extend(request, params))
|
||
elif market['spot']:
|
||
if marginMode is not None:
|
||
if marginMode == 'isolated':
|
||
response = await self.privateMarginPostV2MarginIsolatedCancelOrder(self.extend(request, params))
|
||
elif marginMode == 'cross':
|
||
response = await self.privateMarginPostV2MarginCrossedCancelOrder(self.extend(request, params))
|
||
else:
|
||
if trigger:
|
||
response = await self.privateSpotPostV2SpotTradeCancelPlanOrder(self.extend(request, params))
|
||
else:
|
||
response = await self.privateSpotPostV2SpotTradeCancelOrder(self.extend(request, params))
|
||
else:
|
||
raise NotSupported(self.id + ' cancelOrder() does not support ' + market['type'] + ' orders')
|
||
#
|
||
# spot, swap, future and spot margin
|
||
#
|
||
# {
|
||
# "code": "00000",
|
||
# "msg": "success",
|
||
# "requestTime": 1697690413177,
|
||
# "data": {
|
||
# "orderId": "1098758604547850241",
|
||
# "clientOid": "1098758604585598977"
|
||
# }
|
||
# }
|
||
#
|
||
# swap trigger
|
||
#
|
||
# {
|
||
# "code": "00000",
|
||
# "msg": "success",
|
||
# "requestTime": 1700711311791,
|
||
# "data": {
|
||
# "successList": [
|
||
# {
|
||
# "clientOid": "1111428059067125760",
|
||
# "orderId": "1111428059067125761"
|
||
# }
|
||
# ],
|
||
# "failureList": []
|
||
# }
|
||
# }
|
||
#
|
||
# spot trigger
|
||
#
|
||
# {
|
||
# "code": "00000",
|
||
# "msg": "success",
|
||
# "requestTime": 1700711728063,
|
||
# "data": {
|
||
# "result": "success"
|
||
# }
|
||
# }
|
||
#
|
||
# uta trigger
|
||
#
|
||
# {
|
||
# "code": "00000",
|
||
# "msg": "success",
|
||
# "requestTime": "1753058267399",
|
||
# "data": null
|
||
# }
|
||
#
|
||
data = self.safe_value(response, 'data', {})
|
||
order = None
|
||
if isContractTriggerEndpoint:
|
||
orderInfo = self.safe_value(data, 'successList', [])
|
||
order = orderInfo[0]
|
||
else:
|
||
if uta and trigger:
|
||
order = response
|
||
else:
|
||
order = data
|
||
return self.parse_order(order, market)
|
||
|
||
async def cancel_uta_orders(self, ids, symbol: Str = None, params={}):
|
||
if symbol is None:
|
||
raise ArgumentsRequired(self.id + ' cancelOrders() requires a symbol argument')
|
||
await self.load_markets()
|
||
market = self.market(symbol)
|
||
productType = None
|
||
productType, params = self.handle_product_type_and_params(market, params)
|
||
requestList = []
|
||
for i in range(0, len(ids)):
|
||
individualId = ids[i]
|
||
order: dict = {
|
||
'orderId': individualId,
|
||
'symbol': market['id'],
|
||
'category': productType,
|
||
}
|
||
requestList.append(order)
|
||
response = await self.privateUtaPostV3TradeCancelBatch(requestList)
|
||
#
|
||
# {
|
||
# "code": "00000",
|
||
# "msg": "success",
|
||
# "requestTime": 1752813731517,
|
||
# "data": [
|
||
# {
|
||
# "orderId": "1329948909442023424",
|
||
# "clientOid": "1329948909446217728"
|
||
# },
|
||
# ]
|
||
# }
|
||
#
|
||
data = self.safe_list(response, 'data', [])
|
||
return self.parse_orders(data, market)
|
||
|
||
async def cancel_orders(self, ids: List[str], symbol: Str = None, params={}):
|
||
"""
|
||
cancel multiple orders
|
||
|
||
https://www.bitget.com/api-doc/spot/trade/Batch-Cancel-Orders
|
||
https://www.bitget.com/api-doc/contract/trade/Batch-Cancel-Orders
|
||
https://www.bitget.com/api-doc/contract/plan/Cancel-Plan-Order
|
||
https://www.bitget.com/api-doc/margin/cross/trade/Cross-Batch-Cancel-Order
|
||
https://www.bitget.com/api-doc/margin/isolated/trade/Isolated-Batch-Cancel-Orders
|
||
https://www.bitget.com/api-doc/uta/trade/Cancel-Batch
|
||
|
||
:param str[] ids: order ids
|
||
:param str symbol: unified market symbol, default is None
|
||
:param dict [params]: extra parameters specific to the exchange API endpoint
|
||
:param str [params.marginMode]: 'isolated' or 'cross' for spot margin trading
|
||
:param boolean [params.trigger]: *contract only* set to True for canceling trigger orders
|
||
:param boolean [params.uta]: set to True for the unified trading account(uta), defaults to False
|
||
:returns dict: an array of `order structures <https://docs.ccxt.com/#/?id=order-structure>`
|
||
"""
|
||
if symbol is None:
|
||
raise ArgumentsRequired(self.id + ' cancelOrders() requires a symbol argument')
|
||
await self.load_markets()
|
||
market = self.market(symbol)
|
||
uta = None
|
||
uta, params = self.handle_option_and_params(params, 'cancelOrders', 'uta', False)
|
||
if uta:
|
||
return await self.cancel_uta_orders(ids, symbol, params)
|
||
marginMode = None
|
||
marginMode, params = self.handle_margin_mode_and_params('cancelOrders', params)
|
||
trigger = self.safe_value_2(params, 'stop', 'trigger')
|
||
params = self.omit(params, ['stop', 'trigger'])
|
||
orderIdList = []
|
||
for i in range(0, len(ids)):
|
||
individualId = ids[i]
|
||
orderId: dict = {
|
||
'orderId': individualId,
|
||
}
|
||
orderIdList.append(orderId)
|
||
request: dict = {
|
||
'symbol': market['id'],
|
||
}
|
||
if market['spot'] and (marginMode is None):
|
||
request['orderList'] = orderIdList
|
||
else:
|
||
request['orderIdList'] = orderIdList
|
||
response = None
|
||
if market['spot']:
|
||
if marginMode is not None:
|
||
if marginMode == 'cross':
|
||
response = await self.privateMarginPostV2MarginCrossedBatchCancelOrder(self.extend(request, params))
|
||
else:
|
||
response = await self.privateMarginPostV2MarginIsolatedBatchCancelOrder(self.extend(request, params))
|
||
else:
|
||
response = await self.privateSpotPostV2SpotTradeBatchCancelOrder(self.extend(request, params))
|
||
else:
|
||
productType = None
|
||
productType, params = self.handle_product_type_and_params(market, params)
|
||
request['productType'] = productType
|
||
if trigger:
|
||
response = await self.privateMixPostV2MixOrderCancelPlanOrder(self.extend(request, params))
|
||
else:
|
||
response = await self.privateMixPostV2MixOrderBatchCancelOrders(self.extend(request, params))
|
||
#
|
||
# {
|
||
# "code": "00000",
|
||
# "msg": "success",
|
||
# "requestTime": "1680008815965",
|
||
# "data": {
|
||
# "successList": [
|
||
# {
|
||
# "orderId": "1024598257429823488",
|
||
# "clientOid": "876493ce-c287-4bfc-9f4a-8b1905881313"
|
||
# },
|
||
# ],
|
||
# "failureList": []
|
||
# }
|
||
# }
|
||
#
|
||
data = self.safe_value(response, 'data', {})
|
||
orders = self.safe_list(data, 'successList', [])
|
||
return self.parse_orders(orders, market)
|
||
|
||
async def cancel_all_orders(self, symbol: Str = None, params={}):
|
||
"""
|
||
cancel all open orders
|
||
|
||
https://www.bitget.com/api-doc/spot/trade/Cancel-Symbol-Orders
|
||
https://www.bitget.com/api-doc/spot/plan/Batch-Cancel-Plan-Order
|
||
https://www.bitget.com/api-doc/contract/trade/Batch-Cancel-Orders
|
||
https://www.bitget.com/api-doc/margin/cross/trade/Cross-Batch-Cancel-Order
|
||
https://www.bitget.com/api-doc/margin/isolated/trade/Isolated-Batch-Cancel-Orders
|
||
|
||
:param str symbol: unified market symbol
|
||
:param dict [params]: extra parameters specific to the exchange API endpoint
|
||
:param str [params.marginMode]: 'isolated' or 'cross' for spot margin trading
|
||
:param boolean [params.trigger]: *contract only* set to True for canceling trigger orders
|
||
:param boolean [params.uta]: set to True for the unified trading account(uta), defaults to False
|
||
:returns dict[]: a list of `order structures <https://docs.ccxt.com/#/?id=order-structure>`
|
||
"""
|
||
if symbol is None:
|
||
raise ArgumentsRequired(self.id + ' cancelAllOrders() requires a symbol argument')
|
||
await self.load_markets()
|
||
market = self.market(symbol)
|
||
marginMode = None
|
||
marginMode, params = self.handle_margin_mode_and_params('cancelAllOrders', params)
|
||
productType = None
|
||
productType, params = self.handle_product_type_and_params(market, params)
|
||
request: dict = {
|
||
'symbol': market['id'],
|
||
}
|
||
trigger = self.safe_bool_2(params, 'stop', 'trigger')
|
||
params = self.omit(params, ['stop', 'trigger'])
|
||
response = None
|
||
uta = None
|
||
uta, params = self.handle_option_and_params(params, 'cancelAllOrders', 'uta', False)
|
||
if uta:
|
||
if productType == 'SPOT':
|
||
if marginMode is not None:
|
||
productType = 'MARGIN'
|
||
request['category'] = productType
|
||
response = await self.privateUtaPostV3TradeCancelSymbolOrder(self.extend(request, params))
|
||
#
|
||
# {
|
||
# "code": "00000",
|
||
# "msg": "success",
|
||
# "requestTime": 1750751578138,
|
||
# "data": {
|
||
# "list": [
|
||
# {
|
||
# "orderId": "1321313242969427968",
|
||
# "clientOid": "1321313242969427969"
|
||
# }
|
||
# ]
|
||
# }
|
||
# }
|
||
#
|
||
elif market['spot']:
|
||
if marginMode is not None:
|
||
raise NotSupported(self.id + ' cancelAllOrders() does not support margin markets, you can use cancelOrders() instead')
|
||
else:
|
||
if trigger:
|
||
stopRequest: dict = {
|
||
'symbolList': [market['id']],
|
||
}
|
||
response = await self.privateSpotPostV2SpotTradeBatchCancelPlanOrder(self.extend(stopRequest, params))
|
||
else:
|
||
response = await self.privateSpotPostV2SpotTradeCancelSymbolOrder(self.extend(request, params))
|
||
#
|
||
# {
|
||
# "code": "00000",
|
||
# "msg": "success",
|
||
# "requestTime": 1700716953996,
|
||
# "data": {
|
||
# "symbol": "BTCUSDT"
|
||
# }
|
||
# }
|
||
#
|
||
timestamp = self.safe_integer(response, 'requestTime')
|
||
responseData = self.safe_dict(response, 'data')
|
||
marketId = self.safe_string(responseData, 'symbol')
|
||
return [
|
||
self.safe_order({
|
||
'info': response,
|
||
'symbol': self.safe_symbol(marketId, None, None, 'spot'),
|
||
'timestamp': timestamp,
|
||
'datetime': self.iso8601(timestamp),
|
||
}),
|
||
]
|
||
else:
|
||
request['productType'] = productType
|
||
if trigger:
|
||
response = await self.privateMixPostV2MixOrderCancelPlanOrder(self.extend(request, params))
|
||
else:
|
||
response = await self.privateMixPostV2MixOrderBatchCancelOrders(self.extend(request, params))
|
||
# {
|
||
# "code": "00000",
|
||
# "msg": "success",
|
||
# "requestTime": "1680008815965",
|
||
# "data": {
|
||
# "successList": [
|
||
# {
|
||
# "orderId": "1024598257429823488",
|
||
# "clientOid": "876493ce-c287-4bfc-9f4a-8b1905881313"
|
||
# },
|
||
# ],
|
||
# "failureList": []
|
||
# }
|
||
# }
|
||
data = self.safe_dict(response, 'data')
|
||
resultList = self.safe_list_n(data, ['resultList', 'successList', 'list'])
|
||
failureList = self.safe_list_2(data, 'failure', 'failureList')
|
||
responseList = None
|
||
if (resultList is not None) and (failureList is not None):
|
||
responseList = self.array_concat(resultList, failureList)
|
||
else:
|
||
responseList = resultList
|
||
return self.parse_orders(responseList)
|
||
|
||
async def fetch_order(self, id: str, symbol: Str = None, params={}):
|
||
"""
|
||
fetches information on an order made by the user
|
||
|
||
https://www.bitget.com/api-doc/spot/trade/Get-Order-Info
|
||
https://www.bitget.com/api-doc/contract/trade/Get-Order-Details
|
||
https://www.bitget.com/api-doc/uta/trade/Get-Order-Details
|
||
|
||
:param str id: the order id
|
||
:param str symbol: unified symbol of the market the order was made in
|
||
:param dict [params]: extra parameters specific to the exchange API endpoint
|
||
:param boolean [params.uta]: set to True for the unified trading account(uta), defaults to False
|
||
:param str [params.clientOrderId]: the clientOrderId of the order, id does not need to be provided if clientOrderId is provided
|
||
:returns dict: An `order structure <https://docs.ccxt.com/#/?id=order-structure>`
|
||
"""
|
||
if symbol is None:
|
||
raise ArgumentsRequired(self.id + ' fetchOrder() requires a symbol argument')
|
||
await self.load_markets()
|
||
market = self.market(symbol)
|
||
request: dict = {
|
||
# 'orderId': id,
|
||
}
|
||
clientOrderId = self.safe_string_2(params, 'clientOrderId', 'clientOid')
|
||
if clientOrderId is not None:
|
||
params = self.omit(params, ['clientOrderId'])
|
||
request['clientOid'] = clientOrderId
|
||
else:
|
||
request['orderId'] = id
|
||
response = None
|
||
uta = None
|
||
uta, params = self.handle_option_and_params(params, 'fetchOrder', 'uta', False)
|
||
if uta:
|
||
response = await self.privateUtaGetV3TradeOrderInfo(self.extend(request, params))
|
||
elif market['spot']:
|
||
response = await self.privateSpotGetV2SpotTradeOrderInfo(self.extend(request, params))
|
||
elif market['swap'] or market['future']:
|
||
request['symbol'] = market['id']
|
||
productType = None
|
||
productType, params = self.handle_product_type_and_params(market, params)
|
||
request['productType'] = productType
|
||
response = await self.privateMixGetV2MixOrderDetail(self.extend(request, params))
|
||
else:
|
||
raise NotSupported(self.id + ' fetchOrder() does not support ' + market['type'] + ' orders')
|
||
#
|
||
# spot
|
||
#
|
||
# {
|
||
# "code": "00000",
|
||
# "msg": "success",
|
||
# "requestTime": 1700719076263,
|
||
# "data": [
|
||
# {
|
||
# "userId": "7264631750",
|
||
# "symbol": "BTCUSDT",
|
||
# "orderId": "1111461743123927040",
|
||
# "clientOid": "63f95110-93b5-4309-8f77-46339f1bcf3c",
|
||
# "price": "25000.0000000000000000",
|
||
# "size": "0.0002000000000000",
|
||
# "orderType": "limit",
|
||
# "side": "buy",
|
||
# "status": "live",
|
||
# "priceAvg": "0",
|
||
# "baseVolume": "0.0000000000000000",
|
||
# "quoteVolume": "0.0000000000000000",
|
||
# "enterPointSource": "API",
|
||
# "feeDetail": "",
|
||
# "orderSource": "normal",
|
||
# "cTime": "1700719050198",
|
||
# "uTime": "1700719050198"
|
||
# }
|
||
# ]
|
||
# }
|
||
#
|
||
# swap and future
|
||
#
|
||
# {
|
||
# "code": "00000",
|
||
# "msg": "success",
|
||
# "requestTime": 1700719918781,
|
||
# "data": {
|
||
# "symbol": "BTCUSDT",
|
||
# "size": "0.001",
|
||
# "orderId": "1111465253393825792",
|
||
# "clientOid": "1111465253431574529",
|
||
# "baseVolume": "0",
|
||
# "fee": "0",
|
||
# "price": "27000",
|
||
# "priceAvg": "",
|
||
# "state": "live",
|
||
# "side": "buy",
|
||
# "force": "gtc",
|
||
# "totalProfits": "0",
|
||
# "posSide": "long",
|
||
# "marginCoin": "USDT",
|
||
# "presetStopSurplusPrice": "",
|
||
# "presetStopLossPrice": "",
|
||
# "quoteVolume": "0",
|
||
# "orderType": "limit",
|
||
# "leverage": "20",
|
||
# "marginMode": "crossed",
|
||
# "reduceOnly": "NO",
|
||
# "enterPointSource": "API",
|
||
# "tradeSide": "open",
|
||
# "posMode": "hedge_mode",
|
||
# "orderSource": "normal",
|
||
# "cTime": "1700719887120",
|
||
# "uTime": "1700719887120"
|
||
# }
|
||
# }
|
||
#
|
||
# uta
|
||
#
|
||
# {
|
||
# "code": "00000",
|
||
# "msg": "success",
|
||
# "requestTime": 1750496858333,
|
||
# "data": {
|
||
# "orderId": "1320244799629316096",
|
||
# "clientOid": "1320244799633510400",
|
||
# "category": "USDT-FUTURES",
|
||
# "symbol": "BTCUSDT",
|
||
# "orderType": "limit",
|
||
# "side": "buy",
|
||
# "price": "50000",
|
||
# "qty": "0.001",
|
||
# "amount": "0",
|
||
# "cumExecQty": "0",
|
||
# "cumExecValue": "0",
|
||
# "avgPrice": "0",
|
||
# "timeInForce": "gtc",
|
||
# "orderStatus": "live",
|
||
# "posSide": "long",
|
||
# "holdMode": "hedge_mode",
|
||
# "reduceOnly": "NO",
|
||
# "feeDetail": [{
|
||
# "feeCoin": "",
|
||
# "fee": ""
|
||
# }],
|
||
# "createdTime": "1750496809871",
|
||
# "updatedTime": "1750496809886",
|
||
# "cancelReason": "",
|
||
# "execType": "normal",
|
||
# "stpMode": "none",
|
||
# "tpTriggerBy": null,
|
||
# "slTriggerBy": null,
|
||
# "takeProfit": null,
|
||
# "stopLoss": null,
|
||
# "tpOrderType": null,
|
||
# "slOrderType": null,
|
||
# "tpLimitPrice": null,
|
||
# "slLimitPrice": null
|
||
# }
|
||
# }
|
||
#
|
||
if not uta and (isinstance(response, str)):
|
||
response = json.loads(response)
|
||
data = self.safe_dict(response, 'data')
|
||
if (data is not None):
|
||
if not isinstance(data, list):
|
||
return self.parse_order(data, market)
|
||
dataList = self.safe_list(response, 'data', [])
|
||
dataListLength = len(dataList)
|
||
if dataListLength == 0:
|
||
raise OrderNotFound(self.id + ' fetchOrder() could not find order id ' + id + ' in ' + self.json(response))
|
||
first = self.safe_dict(dataList, 0, {})
|
||
return self.parse_order(first, market)
|
||
# first = self.safe_dict(data, 0, data)
|
||
# return self.parse_order(first, market)
|
||
|
||
async def fetch_open_orders(self, symbol: Str = None, since: Int = None, limit: Int = None, params={}) -> List[Order]:
|
||
"""
|
||
fetch all unfilled currently open orders
|
||
|
||
https://www.bitget.com/api-doc/spot/trade/Get-Unfilled-Orders
|
||
https://www.bitget.com/api-doc/spot/plan/Get-Current-Plan-Order
|
||
https://www.bitget.com/api-doc/contract/trade/Get-Orders-Pending
|
||
https://www.bitget.com/api-doc/contract/plan/get-orders-plan-pending
|
||
https://www.bitget.com/api-doc/margin/cross/trade/Get-Cross-Open-Orders
|
||
https://www.bitget.com/api-doc/margin/isolated/trade/Isolated-Open-Orders
|
||
https://www.bitget.com/api-doc/uta/strategy/Get-Unfilled-Strategy-Orders
|
||
|
||
:param str symbol: unified market symbol
|
||
:param int [since]: the earliest time in ms to fetch open orders for
|
||
:param int [limit]: the maximum number of open order structures to retrieve
|
||
:param dict [params]: extra parameters specific to the exchange API endpoint
|
||
:param int [params.until]: the latest time in ms to fetch orders for
|
||
:param str [params.planType]: *contract stop only* 'normal_plan': average trigger order, 'profit_loss': opened tp/sl orders, 'track_plan': trailing stop order, default is 'normal_plan'
|
||
:param boolean [params.trigger]: set to True for fetching trigger orders
|
||
:param boolean [params.paginate]: default False, when True will automatically paginate by calling self endpoint multiple times. See in the docs all the [available parameters](https://github.com/ccxt/ccxt/wiki/Manual#pagination-params)
|
||
:param str [params.isPlan]: *swap only* 'plan' for stop orders and 'profit_loss' for tp/sl orders, default is 'plan'
|
||
:param boolean [params.trailing]: set to True if you want to fetch trailing orders
|
||
:param boolean [params.uta]: set to True for the unified trading account(uta), defaults to False
|
||
:returns Order[]: a list of `order structures <https://docs.ccxt.com/#/?id=order-structure>`
|
||
"""
|
||
await self.load_markets()
|
||
market = None
|
||
type = None
|
||
request: dict = {}
|
||
marginMode = None
|
||
marginMode, params = self.handle_margin_mode_and_params('fetchOpenOrders', params)
|
||
uta = None
|
||
uta, params = self.handle_option_and_params(params, 'fetchOpenOrders', 'uta', False)
|
||
if symbol is not None:
|
||
market = self.market(symbol)
|
||
request['symbol'] = market['id']
|
||
defaultType = self.safe_string_2(self.options, 'fetchOpenOrders', 'defaultType', 'spot')
|
||
marketType = market['type'] if ('type' in market) else defaultType
|
||
type = self.safe_string(params, 'type', marketType)
|
||
else:
|
||
defaultType = self.safe_string_2(self.options, 'fetchOpenOrders', 'defaultType', 'spot')
|
||
type = self.safe_string(params, 'type', defaultType)
|
||
paginate = False
|
||
paginate, params = self.handle_option_and_params(params, 'fetchOpenOrders', 'paginate')
|
||
if paginate:
|
||
cursorReceived = None
|
||
cursorSent = None
|
||
if uta:
|
||
cursorReceived = 'cursor'
|
||
cursorSent = 'cursor'
|
||
elif type == 'spot':
|
||
if marginMode is not None:
|
||
cursorReceived = 'minId'
|
||
cursorSent = 'idLessThan'
|
||
else:
|
||
cursorReceived = 'endId'
|
||
cursorSent = 'idLessThan'
|
||
return await self.fetch_paginated_call_cursor('fetchOpenOrders', symbol, since, limit, params, cursorReceived, cursorSent)
|
||
response = None
|
||
trailing = self.safe_bool(params, 'trailing')
|
||
trigger = self.safe_bool_2(params, 'stop', 'trigger')
|
||
planTypeDefined = self.safe_string(params, 'planType') is not None
|
||
isTrigger = (trigger or planTypeDefined)
|
||
request, params = self.handle_until_option('endTime', request, params)
|
||
if since is not None:
|
||
request['startTime'] = since
|
||
if limit is not None:
|
||
request['limit'] = limit
|
||
if not uta and ((type == 'swap') or (type == 'future') or (marginMode is not None)):
|
||
clientOrderId = self.safe_string_2(params, 'clientOid', 'clientOrderId')
|
||
params = self.omit(params, 'clientOrderId')
|
||
if clientOrderId is not None:
|
||
request['clientOid'] = clientOrderId
|
||
productType = None
|
||
productType, params = self.handle_product_type_and_params(market, params)
|
||
params = self.omit(params, ['type', 'stop', 'trigger', 'trailing'])
|
||
if uta:
|
||
if type == 'spot':
|
||
if marginMode is not None:
|
||
productType = 'MARGIN'
|
||
request['category'] = productType
|
||
if trigger:
|
||
response = await self.privateUtaGetV3TradeUnfilledStrategyOrders(self.extend(request, params))
|
||
else:
|
||
response = await self.privateUtaGetV3TradeUnfilledOrders(self.extend(request, params))
|
||
elif type == 'spot':
|
||
if marginMode is not None:
|
||
if since is None:
|
||
since = self.milliseconds() - 7776000000
|
||
request['startTime'] = since
|
||
if marginMode == 'isolated':
|
||
response = await self.privateMarginGetV2MarginIsolatedOpenOrders(self.extend(request, params))
|
||
elif marginMode == 'cross':
|
||
response = await self.privateMarginGetV2MarginCrossedOpenOrders(self.extend(request, params))
|
||
else:
|
||
if trigger:
|
||
response = await self.privateSpotGetV2SpotTradeCurrentPlanOrder(self.extend(request, params))
|
||
else:
|
||
response = await self.privateSpotGetV2SpotTradeUnfilledOrders(self.extend(request, params))
|
||
else:
|
||
request['productType'] = productType
|
||
if trailing:
|
||
planType = self.safe_string(params, 'planType', 'track_plan')
|
||
request['planType'] = planType
|
||
response = await self.privateMixGetV2MixOrderOrdersPlanPending(self.extend(request, params))
|
||
elif isTrigger:
|
||
planType = self.safe_string(params, 'planType', 'normal_plan')
|
||
request['planType'] = planType
|
||
response = await self.privateMixGetV2MixOrderOrdersPlanPending(self.extend(request, params))
|
||
else:
|
||
response = await self.privateMixGetV2MixOrderOrdersPending(self.extend(request, params))
|
||
#
|
||
# spot
|
||
#
|
||
# {
|
||
# "code": "00000",
|
||
# "msg": "success",
|
||
# "requestTime": 1700728123994,
|
||
# "data": [
|
||
# {
|
||
# "userId": "7264631750",
|
||
# "symbol": "BTCUSDT",
|
||
# "orderId": "1111499608327360513",
|
||
# "clientOid": "d0d4dad5-18d0-4869-a074-ec40bb47cba6",
|
||
# "priceAvg": "25000.0000000000000000",
|
||
# "size": "0.0002000000000000",
|
||
# "orderType": "limit",
|
||
# "side": "buy",
|
||
# "status": "live",
|
||
# "basePrice": "0",
|
||
# "baseVolume": "0.0000000000000000",
|
||
# "quoteVolume": "0.0000000000000000",
|
||
# "enterPointSource": "WEB",
|
||
# "orderSource": "normal",
|
||
# "cTime": "1700728077966",
|
||
# "uTime": "1700728077966"
|
||
# }
|
||
# ]
|
||
# }
|
||
#
|
||
# spot stop
|
||
#
|
||
# {
|
||
# "code": "00000",
|
||
# "msg": "success",
|
||
# "requestTime": 1700729361609,
|
||
# "data": {
|
||
# "nextFlag": False,
|
||
# "idLessThan": "1111503385931620352",
|
||
# "orderList": [
|
||
# {
|
||
# "orderId": "1111503385931620352",
|
||
# "clientOid": "1111503385910648832",
|
||
# "symbol": "BTCUSDT",
|
||
# "size": "0.0002",
|
||
# "planType": "AMOUNT",
|
||
# "executePrice": "25000",
|
||
# "triggerPrice": "26000",
|
||
# "status": "live",
|
||
# "orderType": "limit",
|
||
# "side": "buy",
|
||
# "triggerType": "fill_price",
|
||
# "enterPointSource": "API",
|
||
# "cTime": "1700728978617",
|
||
# "uTime": "1700728978617"
|
||
# }
|
||
# ]
|
||
# }
|
||
# }
|
||
#
|
||
# spot margin
|
||
#
|
||
# {
|
||
# "code": "00000",
|
||
# "msg": "success",
|
||
# "requestTime": 1700729887686,
|
||
# "data": {
|
||
# "orderList": [
|
||
# {
|
||
# "symbol": "BTCUSDT",
|
||
# "orderType": "limit",
|
||
# "enterPointSource": "WEB",
|
||
# "orderId": "1111506377509580801",
|
||
# "clientOid": "2043a3b59a60445f9d9f7365bf3e960c",
|
||
# "loanType": "autoLoanAndRepay",
|
||
# "price": "25000",
|
||
# "side": "buy",
|
||
# "status": "live",
|
||
# "baseSize": "0.0002",
|
||
# "quoteSize": "5",
|
||
# "priceAvg": "0",
|
||
# "size": "0",
|
||
# "amount": "0",
|
||
# "force": "gtc",
|
||
# "cTime": "1700729691866",
|
||
# "uTime": "1700729691866"
|
||
# }
|
||
# ],
|
||
# "maxId": "1111506377509580801",
|
||
# "minId": "1111506377509580801"
|
||
# }
|
||
# }
|
||
#
|
||
# swap and future
|
||
#
|
||
# {
|
||
# "code": "00000",
|
||
# "msg": "success",
|
||
# "requestTime": 1700725609065,
|
||
# "data": {
|
||
# "entrustedList": [
|
||
# {
|
||
# "symbol": "BTCUSDT",
|
||
# "size": "0.002",
|
||
# "orderId": "1111488897767604224",
|
||
# "clientOid": "1111488897805352960",
|
||
# "baseVolume": "0",
|
||
# "fee": "0",
|
||
# "price": "25000",
|
||
# "priceAvg": "",
|
||
# "status": "live",
|
||
# "side": "buy",
|
||
# "force": "gtc",
|
||
# "totalProfits": "0",
|
||
# "posSide": "long",
|
||
# "marginCoin": "USDT",
|
||
# "quoteVolume": "0",
|
||
# "leverage": "20",
|
||
# "marginMode": "crossed",
|
||
# "enterPointSource": "web",
|
||
# "tradeSide": "open",
|
||
# "posMode": "hedge_mode",
|
||
# "orderType": "limit",
|
||
# "orderSource": "normal",
|
||
# "presetStopSurplusPrice": "",
|
||
# "presetStopLossPrice": "",
|
||
# "reduceOnly": "NO",
|
||
# "cTime": "1700725524378",
|
||
# "uTime": "1700725524378"
|
||
# }
|
||
# ],
|
||
# "endId": "1111488897767604224"
|
||
# }
|
||
# }
|
||
#
|
||
# swap and future stop
|
||
#
|
||
# {
|
||
# "code": "00000",\
|
||
# "msg": "success",
|
||
# "requestTime": 1700726417495,
|
||
# "data": {
|
||
# "entrustedList": [
|
||
# {
|
||
# "planType": "normal_plan",
|
||
# "symbol": "BTCUSDT",
|
||
# "size": "0.001",
|
||
# "orderId": "1111491399869075457",
|
||
# "clientOid": "1111491399869075456",
|
||
# "price": "27000",
|
||
# "callbackRatio": "",
|
||
# "triggerPrice": "24000",
|
||
# "triggerType": "mark_price",
|
||
# "planStatus": "live",
|
||
# "side": "buy",
|
||
# "posSide": "long",
|
||
# "marginCoin": "USDT",
|
||
# "marginMode": "crossed",
|
||
# "enterPointSource": "API",
|
||
# "tradeSide": "open",
|
||
# "posMode": "hedge_mode",
|
||
# "orderType": "limit",
|
||
# "stopSurplusTriggerPrice": "",
|
||
# "stopSurplusExecutePrice": "",
|
||
# "stopSurplusTriggerType": "fill_price",
|
||
# "stopLossTriggerPrice": "",
|
||
# "stopLossExecutePrice": "",
|
||
# "stopLossTriggerType": "fill_price",
|
||
# "cTime": "1700726120917",
|
||
# "uTime": "1700726120917"
|
||
# }
|
||
# ],
|
||
# "endId": "1111491399869075457"
|
||
# }
|
||
# }
|
||
#
|
||
# uta
|
||
#
|
||
# {
|
||
# "code": "00000",
|
||
# "msg": "success",
|
||
# "requestTime": 1750753395850,
|
||
# "data": {
|
||
# "list": [
|
||
# {
|
||
# "orderId": "1321320757371228160",
|
||
# "clientOid": "1321320757371228161",
|
||
# "category": "USDT-FUTURES",
|
||
# "symbol": "BTCUSDT",
|
||
# "orderType": "limit",
|
||
# "side": "buy",
|
||
# "price": "50000",
|
||
# "qty": "0.001",
|
||
# "amount": "0",
|
||
# "cumExecQty": "0",
|
||
# "cumExecValue": "0",
|
||
# "avgPrice": "0",
|
||
# "timeInForce": "gtc",
|
||
# "orderStatus": "live",
|
||
# "posSide": "long",
|
||
# "holdMode": "hedge_mode",
|
||
# "reduceOnly": "NO",
|
||
# "feeDetail": [
|
||
# {
|
||
# "feeCoin": "",
|
||
# "fee": ""
|
||
# }
|
||
# ],
|
||
# "createdTime": "1750753338186",
|
||
# "updatedTime": "1750753338203",
|
||
# "stpMode": "none",
|
||
# "tpTriggerBy": null,
|
||
# "slTriggerBy": null,
|
||
# "takeProfit": null,
|
||
# "stopLoss": null,
|
||
# "tpOrderType": null,
|
||
# "slOrderType": null,
|
||
# "tpLimitPrice": null,
|
||
# "slLimitPrice": null
|
||
# }
|
||
# ],
|
||
# "cursor": "1321320757371228160"
|
||
# }
|
||
# }
|
||
#
|
||
# uta trigger
|
||
#
|
||
# {
|
||
# "code": "00000",
|
||
# "msg": "success",
|
||
# "requestTime": 1753057527060,
|
||
# "data": [
|
||
# {
|
||
# "orderId": "1330984742276198400",
|
||
# "clientOid": "1330984742276198400",
|
||
# "symbol": "BTCUSDT",
|
||
# "category": "USDT-FUTURES",
|
||
# "qty": "0.001",
|
||
# "posSide": "long",
|
||
# "tpTriggerBy": "market",
|
||
# "slTriggerBy": "mark",
|
||
# "takeProfit": "",
|
||
# "stopLoss":"114000",
|
||
# "tpOrderType": "market",
|
||
# "slOrderType": "limit",
|
||
# "tpLimitPrice": "",
|
||
# "slLimitPrice": "113000",
|
||
# "createdTime": "1753057411736",
|
||
# "updatedTime": "1753057411747"
|
||
# }
|
||
# ]
|
||
# }
|
||
#
|
||
data = self.safe_value(response, 'data')
|
||
if uta:
|
||
result = None
|
||
if trigger:
|
||
result = self.safe_list(response, 'data', [])
|
||
else:
|
||
result = self.safe_list(data, 'list', [])
|
||
return self.parse_orders(result, market, since, limit)
|
||
elif type == 'spot':
|
||
if (marginMode is not None) or trigger:
|
||
resultList = self.safe_list(data, 'orderList', [])
|
||
return self.parse_orders(resultList, market, since, limit)
|
||
else:
|
||
result = self.safe_list(data, 'entrustedList', [])
|
||
return self.parse_orders(result, market, since, limit)
|
||
return self.parse_orders(data, market, since, limit)
|
||
|
||
async def fetch_closed_orders(self, symbol: Str = None, since: Int = None, limit: Int = None, params={}) -> List[Order]:
|
||
"""
|
||
fetches information on multiple closed orders made by the user
|
||
|
||
https://www.bitget.com/api-doc/spot/trade/Get-History-Orders
|
||
https://www.bitget.com/api-doc/spot/plan/Get-History-Plan-Order
|
||
https://www.bitget.com/api-doc/contract/trade/Get-Orders-History
|
||
https://www.bitget.com/api-doc/contract/plan/orders-plan-history
|
||
https://www.bitget.com/api-doc/margin/cross/trade/Get-Cross-Order-History
|
||
https://www.bitget.com/api-doc/margin/isolated/trade/Get-Isolated-Order-History
|
||
https://www.bitget.com/api-doc/uta/trade/Get-Order-History
|
||
|
||
:param str symbol: unified market symbol of the closed orders
|
||
:param int [since]: timestamp in ms of the earliest order
|
||
:param int [limit]: the max number of closed orders to return
|
||
:param dict [params]: extra parameters specific to the exchange API endpoint
|
||
:param int [params.until]: the latest time in ms to fetch orders for
|
||
:param str [params.planType]: *contract stop only* 'normal_plan': average trigger order, 'profit_loss': opened tp/sl orders, 'track_plan': trailing stop order, default is 'normal_plan'
|
||
:param boolean [params.trigger]: set to True for fetching trigger orders
|
||
:param boolean [params.paginate]: default False, when True will automatically paginate by calling self endpoint multiple times. See in the docs all the [available parameters](https://github.com/ccxt/ccxt/wiki/Manual#pagination-params)
|
||
:param str [params.isPlan]: *swap only* 'plan' for stop orders and 'profit_loss' for tp/sl orders, default is 'plan'
|
||
:param boolean [params.trailing]: set to True if you want to fetch trailing orders
|
||
:returns Order[]: a list of `order structures <https://docs.ccxt.com/#/?id=order-structure>`
|
||
"""
|
||
await self.load_markets()
|
||
orders = await self.fetch_canceled_and_closed_orders(symbol, since, limit, params)
|
||
return self.filter_by(orders, 'status', 'closed')
|
||
|
||
async def fetch_canceled_orders(self, symbol: Str = None, since: Int = None, limit: Int = None, params={}):
|
||
"""
|
||
fetches information on multiple canceled orders made by the user
|
||
|
||
https://www.bitget.com/api-doc/spot/trade/Get-History-Orders
|
||
https://www.bitget.com/api-doc/spot/plan/Get-History-Plan-Order
|
||
https://www.bitget.com/api-doc/contract/trade/Get-Orders-History
|
||
https://www.bitget.com/api-doc/contract/plan/orders-plan-history
|
||
https://www.bitget.com/api-doc/margin/cross/trade/Get-Cross-Order-History
|
||
https://www.bitget.com/api-doc/margin/isolated/trade/Get-Isolated-Order-History
|
||
https://www.bitget.com/api-doc/uta/trade/Get-Order-History
|
||
|
||
:param str symbol: unified market symbol of the canceled orders
|
||
:param int [since]: timestamp in ms of the earliest order
|
||
:param int [limit]: the max number of canceled orders to return
|
||
:param dict [params]: extra parameters specific to the exchange API endpoint
|
||
:param int [params.until]: the latest time in ms to fetch orders for
|
||
:param str [params.planType]: *contract stop only* 'normal_plan': average trigger order, 'profit_loss': opened tp/sl orders, 'track_plan': trailing stop order, default is 'normal_plan'
|
||
:param boolean [params.trigger]: set to True for fetching trigger orders
|
||
:param boolean [params.paginate]: default False, when True will automatically paginate by calling self endpoint multiple times. See in the docs all the [available parameters](https://github.com/ccxt/ccxt/wiki/Manual#pagination-params)
|
||
:param str [params.isPlan]: *swap only* 'plan' for stop orders and 'profit_loss' for tp/sl orders, default is 'plan'
|
||
:param boolean [params.trailing]: set to True if you want to fetch trailing orders
|
||
:returns dict: a list of `order structures <https://docs.ccxt.com/#/?id=order-structure>`
|
||
"""
|
||
await self.load_markets()
|
||
orders = await self.fetch_canceled_and_closed_orders(symbol, since, limit, params)
|
||
return self.filter_by(orders, 'status', 'canceled')
|
||
|
||
async def fetch_canceled_and_closed_orders(self, symbol: Str = None, since: Int = None, limit: Int = None, params={}):
|
||
"""
|
||
|
||
https://www.bitget.com/api-doc/spot/trade/Get-History-Orders
|
||
https://www.bitget.com/api-doc/spot/plan/Get-History-Plan-Order
|
||
https://www.bitget.com/api-doc/contract/trade/Get-Orders-History
|
||
https://www.bitget.com/api-doc/contract/plan/orders-plan-history
|
||
https://www.bitget.com/api-doc/margin/cross/trade/Get-Cross-Order-History
|
||
https://www.bitget.com/api-doc/margin/isolated/trade/Get-Isolated-Order-History
|
||
https://www.bitget.com/api-doc/uta/trade/Get-Order-History
|
||
https://www.bitget.com/api-doc/uta/strategy/Get-History-Strategy-Orders
|
||
|
||
fetches information on multiple canceled and closed orders made by the user
|
||
:param str symbol: unified market symbol of the market orders were made in
|
||
:param int [since]: the earliest time in ms to fetch orders for
|
||
:param int [limit]: the maximum number of order structures to retrieve
|
||
:param dict [params]: extra parameters specific to the exchange API endpoint
|
||
:param int [params.until]: the latest time in ms to fetch orders for
|
||
:param str [params.planType]: *contract stop only* 'normal_plan': average trigger order, 'profit_loss': opened tp/sl orders, 'track_plan': trailing stop order, default is 'normal_plan'
|
||
:param boolean [params.trigger]: set to True for fetching trigger orders
|
||
:param boolean [params.paginate]: default False, when True will automatically paginate by calling self endpoint multiple times. See in the docs all the [available parameters](https://github.com/ccxt/ccxt/wiki/Manual#pagination-params)
|
||
:param str [params.isPlan]: *swap only* 'plan' for stop orders and 'profit_loss' for tp/sl orders, default is 'plan'
|
||
:param boolean [params.trailing]: set to True if you want to fetch trailing orders
|
||
:param boolean [params.uta]: set to True for the unified trading account(uta), defaults to False
|
||
:returns Order[]: a list of `order structures <https://docs.ccxt.com/#/?id=order-structure>`
|
||
"""
|
||
uta = None
|
||
uta, params = self.handle_option_and_params(params, 'fetchCanceledAndClosedOrders', 'uta', False)
|
||
if uta:
|
||
return await self.fetch_uta_canceled_and_closed_orders(symbol, since, limit, params)
|
||
await self.load_markets()
|
||
market = None
|
||
request: dict = {}
|
||
if symbol is not None:
|
||
market = self.market(symbol)
|
||
request['symbol'] = market['id']
|
||
marketType = None
|
||
marketType, params = self.handle_market_type_and_params('fetchCanceledAndClosedOrders', market, params)
|
||
marginMode = None
|
||
marginMode, params = self.handle_margin_mode_and_params('fetchCanceledAndClosedOrders', params)
|
||
paginate = False
|
||
paginate, params = self.handle_option_and_params(params, 'fetchCanceledAndClosedOrders', 'paginate')
|
||
if paginate:
|
||
cursorReceived = None
|
||
if marketType == 'spot':
|
||
if marginMode is not None:
|
||
cursorReceived = 'minId'
|
||
else:
|
||
cursorReceived = 'endId'
|
||
return await self.fetch_paginated_call_cursor('fetchCanceledAndClosedOrders', symbol, since, limit, params, cursorReceived, 'idLessThan')
|
||
response = None
|
||
trailing = self.safe_bool(params, 'trailing')
|
||
trigger = self.safe_bool_2(params, 'stop', 'trigger')
|
||
params = self.omit(params, ['stop', 'trigger', 'trailing'])
|
||
request, params = self.handle_until_option('endTime', request, params)
|
||
if since is not None:
|
||
request['startTime'] = since
|
||
if limit is not None:
|
||
request['limit'] = limit
|
||
if (marketType == 'swap') or (marketType == 'future') or (marginMode is not None):
|
||
clientOrderId = self.safe_string_2(params, 'clientOid', 'clientOrderId')
|
||
params = self.omit(params, 'clientOrderId')
|
||
if clientOrderId is not None:
|
||
request['clientOid'] = clientOrderId
|
||
now = self.milliseconds()
|
||
if marketType == 'spot':
|
||
if marginMode is not None:
|
||
if since is None:
|
||
since = now - 7776000000
|
||
request['startTime'] = since
|
||
if marginMode == 'isolated':
|
||
response = await self.privateMarginGetV2MarginIsolatedHistoryOrders(self.extend(request, params))
|
||
elif marginMode == 'cross':
|
||
response = await self.privateMarginGetV2MarginCrossedHistoryOrders(self.extend(request, params))
|
||
elif trigger:
|
||
if symbol is None:
|
||
raise ArgumentsRequired(self.id + ' fetchCanceledAndClosedOrders() requires a symbol argument')
|
||
endTime = self.safe_integer_n(params, ['endTime', 'until'])
|
||
params = self.omit(params, ['until'])
|
||
if since is None:
|
||
since = now - 7776000000
|
||
request['startTime'] = since
|
||
if endTime is None:
|
||
request['endTime'] = now
|
||
response = await self.privateSpotGetV2SpotTradeHistoryPlanOrder(self.extend(request, params))
|
||
else:
|
||
response = await self.privateSpotGetV2SpotTradeHistoryOrders(self.extend(request, params))
|
||
else:
|
||
productType = None
|
||
productType, params = self.handle_product_type_and_params(market, params)
|
||
request['productType'] = productType
|
||
planTypeDefined = self.safe_string(params, 'planType') is not None
|
||
if trailing:
|
||
planType = self.safe_string(params, 'planType', 'track_plan')
|
||
request['planType'] = planType
|
||
response = await self.privateMixGetV2MixOrderOrdersPlanHistory(self.extend(request, params))
|
||
elif trigger or planTypeDefined:
|
||
planType = self.safe_string(params, 'planType', 'normal_plan')
|
||
request['planType'] = planType
|
||
response = await self.privateMixGetV2MixOrderOrdersPlanHistory(self.extend(request, params))
|
||
else:
|
||
response = await self.privateMixGetV2MixOrderOrdersHistory(self.extend(request, params))
|
||
#
|
||
# spot
|
||
#
|
||
# {
|
||
# "code": "00000",
|
||
# "msg": "success",
|
||
# "requestTime": 1700791085380,
|
||
# "data": [
|
||
# {
|
||
# "userId": "7264631750",
|
||
# "symbol": "BTCUSDT",
|
||
# "orderId": "1111499608327360513",
|
||
# "clientOid": "d0d4dad5-18d0-4869-a074-ec40bb47cba6",
|
||
# "price": "25000.0000000000000000",
|
||
# "size": "0.0002000000000000",
|
||
# "orderType": "limit",
|
||
# "side": "buy",
|
||
# "status": "cancelled",
|
||
# "priceAvg": "0",
|
||
# "baseVolume": "0.0000000000000000",
|
||
# "quoteVolume": "0.0000000000000000",
|
||
# "enterPointSource": "WEB",
|
||
# "feeDetail": "",
|
||
# "orderSource": "normal",
|
||
# "cTime": "1700728077966",
|
||
# "uTime": "1700728911471"
|
||
# },
|
||
# ]
|
||
# }
|
||
#
|
||
# spot stop
|
||
#
|
||
# {
|
||
# "code": "00000",
|
||
# "msg": "success",
|
||
# "requestTime": 1700792099146,
|
||
# "data": {
|
||
# "nextFlag": False,
|
||
# "idLessThan": "1098757597417775104",
|
||
# "orderList": [
|
||
# {
|
||
# "orderId": "1111503385931620352",
|
||
# "clientOid": "1111503385910648832",
|
||
# "symbol": "BTCUSDT",
|
||
# "size": "0.0002",
|
||
# "planType": "AMOUNT",
|
||
# "executePrice": "25000",
|
||
# "triggerPrice": "26000",
|
||
# "status": "cancelled",
|
||
# "orderType": "limit",
|
||
# "side": "buy",
|
||
# "triggerType": "fill_price",
|
||
# "enterPointSource": "API",
|
||
# "cTime": "1700728978617",
|
||
# "uTime": "1700729666868"
|
||
# },
|
||
# ]
|
||
# }
|
||
# }
|
||
#
|
||
# spot margin
|
||
#
|
||
# {
|
||
# "code": "00000",
|
||
# "msg": "success",
|
||
# "requestTime": 1700792381435,
|
||
# "data": {
|
||
# "orderList": [
|
||
# {
|
||
# "symbol": "BTCUSDT",
|
||
# "orderType": "limit",
|
||
# "enterPointSource": "WEB",
|
||
# "orderId": "1111456274707001345",
|
||
# "clientOid": "41e428dd305a4f668671b7f1ed00dc50",
|
||
# "loanType": "autoLoanAndRepay",
|
||
# "price": "27000",
|
||
# "side": "buy",
|
||
# "status": "cancelled",
|
||
# "baseSize": "0.0002",
|
||
# "quoteSize": "5.4",
|
||
# "priceAvg": "0",
|
||
# "size": "0",
|
||
# "amount": "0",
|
||
# "force": "gtc",
|
||
# "cTime": "1700717746427",
|
||
# "uTime": "1700717780636"
|
||
# },
|
||
# ],
|
||
# "maxId": "1111456274707001345",
|
||
# "minId": "1098396464990269440"
|
||
# }
|
||
# }
|
||
#
|
||
# swap and future
|
||
#
|
||
# {
|
||
# "code": "00000",
|
||
# "msg": "success",
|
||
# "requestTime": 1700792674673,
|
||
# "data": {
|
||
# "entrustedList": [
|
||
# {
|
||
# "symbol": "BTCUSDT",
|
||
# "size": "0.002",
|
||
# "orderId": "1111498800817143808",
|
||
# "clientOid": "1111498800850698240",
|
||
# "baseVolume": "0",
|
||
# "fee": "0",
|
||
# "price": "25000",
|
||
# "priceAvg": "",
|
||
# "status": "canceled",
|
||
# "side": "buy",
|
||
# "force": "gtc",
|
||
# "totalProfits": "0",
|
||
# "posSide": "long",
|
||
# "marginCoin": "USDT",
|
||
# "quoteVolume": "0",
|
||
# "leverage": "20",
|
||
# "marginMode": "crossed",
|
||
# "enterPointSource": "web",
|
||
# "tradeSide": "open",
|
||
# "posMode": "hedge_mode",
|
||
# "orderType": "limit",
|
||
# "orderSource": "normal",
|
||
# "presetStopSurplusPrice": "",
|
||
# "presetStopLossPrice": "",
|
||
# "reduceOnly": "NO",
|
||
# "cTime": "1700727885449",
|
||
# "uTime": "1700727944563"
|
||
# },
|
||
# ],
|
||
# "endId": "1098397008323575809"
|
||
# }
|
||
# }
|
||
#
|
||
# swap and future stop
|
||
#
|
||
# {
|
||
# "code": "00000",
|
||
# "msg": "success",
|
||
# "requestTime": 1700792938359,
|
||
# "data": {
|
||
# "entrustedList": [
|
||
# {
|
||
# "planType": "normal_plan",
|
||
# "symbol": "BTCUSDT",
|
||
# "size": "0.001",
|
||
# "orderId": "1111491399869075457",
|
||
# "clientOid": "1111491399869075456",
|
||
# "planStatus": "cancelled",
|
||
# "price": "27000",
|
||
# "feeDetail": null,
|
||
# "baseVolume": "0",
|
||
# "callbackRatio": "",
|
||
# "triggerPrice": "24000",
|
||
# "triggerType": "mark_price",
|
||
# "side": "buy",
|
||
# "posSide": "long",
|
||
# "marginCoin": "USDT",
|
||
# "marginMode": "crossed",
|
||
# "enterPointSource": "API",
|
||
# "tradeSide": "open",
|
||
# "posMode": "hedge_mode",
|
||
# "orderType": "limit",
|
||
# "stopSurplusTriggerPrice": "",
|
||
# "stopSurplusExecutePrice": "",
|
||
# "stopSurplusTriggerType": "fill_price",
|
||
# "stopLossTriggerPrice": "",
|
||
# "stopLossExecutePrice": "",
|
||
# "stopLossTriggerType": "fill_price",
|
||
# "cTime": "1700726120917",
|
||
# "uTime": "1700727879652"
|
||
# },
|
||
# ],
|
||
# "endId": "1098760007867502593"
|
||
# }
|
||
# }
|
||
#
|
||
data = self.safe_value(response, 'data', {})
|
||
if marketType == 'spot':
|
||
if (marginMode is not None) or trigger:
|
||
return self.parse_orders(self.safe_value(data, 'orderList', []), market, since, limit)
|
||
else:
|
||
return self.parse_orders(self.safe_value(data, 'entrustedList', []), market, since, limit)
|
||
if isinstance(response, str):
|
||
response = json.loads(response)
|
||
orders = self.safe_list(response, 'data', [])
|
||
return self.parse_orders(orders, market, since, limit)
|
||
|
||
async def fetch_uta_canceled_and_closed_orders(self, symbol: Str = None, since: Int = None, limit: Int = None, params={}):
|
||
await self.load_markets()
|
||
market = None
|
||
if symbol is not None:
|
||
market = self.market(symbol)
|
||
productType = None
|
||
productType, params = self.handle_product_type_and_params(market, params)
|
||
if productType == 'SPOT':
|
||
marginMode = None
|
||
marginMode, params = self.handle_margin_mode_and_params('fetchCanceledAndClosedOrders', params)
|
||
if marginMode is not None:
|
||
productType = 'MARGIN'
|
||
request: dict = {
|
||
'category': productType,
|
||
}
|
||
paginate = False
|
||
paginate, params = self.handle_option_and_params(params, 'fetchCanceledAndClosedOrders', 'paginate')
|
||
if paginate:
|
||
return await self.fetch_paginated_call_cursor('fetchCanceledAndClosedOrders', symbol, since, limit, params, 'cursor', 'cursor')
|
||
request, params = self.handle_until_option('endTime', request, params)
|
||
if since is not None:
|
||
request['startTime'] = since
|
||
if limit is not None:
|
||
request['limit'] = limit
|
||
response = None
|
||
trigger = self.safe_bool_2(params, 'stop', 'trigger')
|
||
params = self.omit(params, ['stop', 'trigger'])
|
||
if trigger:
|
||
response = await self.privateUtaGetV3TradeHistoryStrategyOrders(self.extend(request, params))
|
||
else:
|
||
response = await self.privateUtaGetV3TradeHistoryOrders(self.extend(request, params))
|
||
#
|
||
# uta
|
||
#
|
||
# {
|
||
# "code": "00000",
|
||
# "msg": "success",
|
||
# "requestTime": 1752531592855,
|
||
# "data": {
|
||
# "list": [
|
||
# {
|
||
# "orderId": "1322441400976261120",
|
||
# "clientOid": "1322441400976261121",
|
||
# "category": "USDT-FUTURES",
|
||
# "symbol": "BTCUSDT",
|
||
# "orderType": "market",
|
||
# "side": "sell",
|
||
# "price": "0",
|
||
# "qty": "0.0001",
|
||
# "amount": "0",
|
||
# "cumExecQty": "0.0001",
|
||
# "cumExecValue": "10.7005",
|
||
# "avgPrice": "107005.4",
|
||
# "timeInForce": "gtc",
|
||
# "orderStatus": "filled",
|
||
# "posSide": "long",
|
||
# "holdMode": "hedge_mode",
|
||
# "reduceOnly": "NO",
|
||
# "feeDetail": [
|
||
# {
|
||
# "feeCoin": "USDT",
|
||
# "fee": "0.00642032"
|
||
# }
|
||
# ],
|
||
# "createdTime": "1751020520442",
|
||
# "updatedTime": "1751020520457",
|
||
# "cancelReason": "",
|
||
# "execType": "normal",
|
||
# "stpMode": "none",
|
||
# "tpTriggerBy": null,
|
||
# "slTriggerBy": null,
|
||
# "takeProfit": null,
|
||
# "stopLoss": null,
|
||
# "tpOrderType": null,
|
||
# "slOrderType": null,
|
||
# "tpLimitPrice": null,
|
||
# "slLimitPrice": null
|
||
# },
|
||
# ],
|
||
# "cursor": "1322441328637100035"
|
||
# }
|
||
# }
|
||
#
|
||
# uta trigger
|
||
#
|
||
# {
|
||
# "code": "00000",
|
||
# "msg": "success",
|
||
# "requestTime": 1753058447920,
|
||
# "data": {
|
||
# "list": [
|
||
# {
|
||
# "orderId": "1330984742276198400",
|
||
# "clientOid": "1330984742276198400",
|
||
# "symbol": "BTCUSDT",
|
||
# "category": "USDT-FUTURES",
|
||
# "qty": "0.001",
|
||
# "posSide": "long",
|
||
# "tpTriggerBy": "market",
|
||
# "slTriggerBy": "mark",
|
||
# "takeProfit": "",
|
||
# "stopLoss": "112000",
|
||
# "tpOrderType": "market",
|
||
# "slOrderType": "limit",
|
||
# "tpLimitPrice": "",
|
||
# "slLimitPrice": "111000",
|
||
# "createdTime": "1753057411736",
|
||
# "updatedTime": "1753058267412"
|
||
# },
|
||
# ],
|
||
# "cursor": 1330960754317619202
|
||
# }
|
||
# }
|
||
#
|
||
data = self.safe_dict(response, 'data', {})
|
||
orders = self.safe_list(data, 'list', [])
|
||
return self.parse_orders(orders, market, since, limit)
|
||
|
||
async def fetch_ledger(self, code: Str = None, since: Int = None, limit: Int = None, params={}) -> List[LedgerEntry]:
|
||
"""
|
||
fetch the history of changes, actions done by the user or operations that altered the balance of the user
|
||
|
||
https://www.bitget.com/api-doc/spot/account/Get-Account-Bills
|
||
https://www.bitget.com/api-doc/contract/account/Get-Account-Bill
|
||
|
||
:param str [code]: unified currency code, default is None
|
||
:param int [since]: timestamp in ms of the earliest ledger entry, default is None
|
||
:param int [limit]: max number of ledger entries to return, default is None
|
||
:param dict [params]: extra parameters specific to the exchange API endpoint
|
||
:param int [params.until]: end time in ms
|
||
:param str [params.symbol]: *contract only* unified market symbol
|
||
:param str [params.productType]: *contract only* 'USDT-FUTURES', 'USDC-FUTURES', 'COIN-FUTURES', 'SUSDT-FUTURES', 'SUSDC-FUTURES' or 'SCOIN-FUTURES'
|
||
:param boolean [params.paginate]: default False, when True will automatically paginate by calling self endpoint multiple times. See in the docs all the [available parameters](https://github.com/ccxt/ccxt/wiki/Manual#pagination-params)
|
||
:returns dict: a `ledger structure <https://docs.ccxt.com/#/?id=ledger>`
|
||
"""
|
||
await self.load_markets()
|
||
symbol = self.safe_string(params, 'symbol')
|
||
params = self.omit(params, 'symbol')
|
||
market = None
|
||
if symbol is not None:
|
||
market = self.market(symbol)
|
||
marketType = None
|
||
marketType, params = self.handle_market_type_and_params('fetchLedger', market, params)
|
||
paginate = False
|
||
paginate, params = self.handle_option_and_params(params, 'fetchLedger', 'paginate')
|
||
if paginate:
|
||
cursorReceived = None
|
||
if marketType != 'spot':
|
||
cursorReceived = 'endId'
|
||
return await self.fetch_paginated_call_cursor('fetchLedger', symbol, since, limit, params, cursorReceived, 'idLessThan')
|
||
currency = None
|
||
request: dict = {}
|
||
if code is not None:
|
||
currency = self.currency(code)
|
||
request['coin'] = currency['id']
|
||
request, params = self.handle_until_option('endTime', request, params)
|
||
if since is not None:
|
||
request['startTime'] = since
|
||
if limit is not None:
|
||
request['limit'] = limit
|
||
response = None
|
||
if marketType == 'spot':
|
||
response = await self.privateSpotGetV2SpotAccountBills(self.extend(request, params))
|
||
else:
|
||
if symbol is not None:
|
||
request['symbol'] = market['id']
|
||
productType = None
|
||
productType, params = self.handle_product_type_and_params(market, params)
|
||
request['productType'] = productType
|
||
response = await self.privateMixGetV2MixAccountBill(self.extend(request, params))
|
||
#
|
||
# spot
|
||
#
|
||
# {
|
||
# "code": "00000",
|
||
# "msg": "success",
|
||
# "requestTime": 1700795836415,
|
||
# "data": [
|
||
# {
|
||
# "billId": "1111506298997215233",
|
||
# "coin": "USDT",
|
||
# "groupType": "transfer",
|
||
# "businessType": "transfer_out",
|
||
# "size": "-11.64958799",
|
||
# "balance": "0.00000000",
|
||
# "fees": "0.00000000",
|
||
# "cTime": "1700729673028"
|
||
# },
|
||
# ]
|
||
# }
|
||
#
|
||
# swap and future
|
||
#
|
||
# {
|
||
# "code": "00000",
|
||
# "msg": "success",
|
||
# "requestTime": 1700795977890,
|
||
# "data": {
|
||
# "bills": [
|
||
# {
|
||
# "billId": "1111499428100472833",
|
||
# "symbol": "",
|
||
# "amount": "-11.64958799",
|
||
# "fee": "0",
|
||
# "feeByCoupon": "",
|
||
# "businessType": "trans_to_exchange",
|
||
# "coin": "USDT",
|
||
# "cTime": "1700728034996"
|
||
# },
|
||
# ],
|
||
# "endId": "1098396773329305606"
|
||
# }
|
||
# }
|
||
#
|
||
data = self.safe_value(response, 'data')
|
||
if (marketType == 'swap') or (marketType == 'future'):
|
||
bills = self.safe_value(data, 'bills', [])
|
||
return self.parse_ledger(bills, currency, since, limit)
|
||
return self.parse_ledger(data, currency, since, limit)
|
||
|
||
def parse_ledger_entry(self, item: dict, currency: Currency = None) -> LedgerEntry:
|
||
#
|
||
# spot
|
||
#
|
||
# {
|
||
# "billId": "1111506298997215233",
|
||
# "coin": "USDT",
|
||
# "groupType": "transfer",
|
||
# "businessType": "transfer_out",
|
||
# "size": "-11.64958799",
|
||
# "balance": "0.00000000",
|
||
# "fees": "0.00000000",
|
||
# "cTime": "1700729673028"
|
||
# }
|
||
#
|
||
# swap and future
|
||
#
|
||
# {
|
||
# "billId": "1111499428100472833",
|
||
# "symbol": "",
|
||
# "amount": "-11.64958799",
|
||
# "fee": "0",
|
||
# "feeByCoupon": "",
|
||
# "businessType": "trans_to_exchange",
|
||
# "coin": "USDT",
|
||
# "cTime": "1700728034996"
|
||
# }
|
||
#
|
||
currencyId = self.safe_string(item, 'coin')
|
||
code = self.safe_currency_code(currencyId, currency)
|
||
currency = self.safe_currency(currencyId, currency)
|
||
timestamp = self.safe_integer(item, 'cTime')
|
||
after = self.safe_number(item, 'balance')
|
||
fee = self.safe_number_2(item, 'fees', 'fee')
|
||
amountRaw = self.safe_string_2(item, 'size', 'amount')
|
||
amount = self.parse_number(Precise.string_abs(amountRaw))
|
||
direction = 'in'
|
||
if amountRaw.find('-') >= 0:
|
||
direction = 'out'
|
||
return self.safe_ledger_entry({
|
||
'info': item,
|
||
'id': self.safe_string(item, 'billId'),
|
||
'timestamp': timestamp,
|
||
'datetime': self.iso8601(timestamp),
|
||
'direction': direction,
|
||
'account': None,
|
||
'referenceId': None,
|
||
'referenceAccount': None,
|
||
'type': self.parse_ledger_type(self.safe_string(item, 'businessType')),
|
||
'currency': code,
|
||
'amount': amount,
|
||
'before': None,
|
||
'after': after,
|
||
'status': None,
|
||
'fee': {
|
||
'currency': code,
|
||
'cost': fee,
|
||
},
|
||
}, currency)
|
||
|
||
def parse_ledger_type(self, type):
|
||
types: dict = {
|
||
'trans_to_cross': 'transfer',
|
||
'trans_from_cross': 'transfer',
|
||
'trans_to_exchange': 'transfer',
|
||
'trans_from_exchange': 'transfer',
|
||
'trans_to_isolated': 'transfer',
|
||
'trans_from_isolated': 'transfer',
|
||
'trans_to_contract': 'transfer',
|
||
'trans_from_contract': 'transfer',
|
||
'trans_to_otc': 'transfer',
|
||
'trans_from_otc': 'transfer',
|
||
'open_long': 'trade',
|
||
'close_long': 'trade',
|
||
'open_short': 'trade',
|
||
'close_short': 'trade',
|
||
'force_close_long': 'trade',
|
||
'force_close_short': 'trade',
|
||
'burst_long_loss_query': 'trade',
|
||
'burst_short_loss_query': 'trade',
|
||
'force_buy': 'trade',
|
||
'force_sell': 'trade',
|
||
'burst_buy': 'trade',
|
||
'burst_sell': 'trade',
|
||
'delivery_long': 'settlement',
|
||
'delivery_short': 'settlement',
|
||
'contract_settle_fee': 'fee',
|
||
'append_margin': 'transaction',
|
||
'adjust_down_lever_append_margin': 'transaction',
|
||
'reduce_margin': 'transaction',
|
||
'auto_append_margin': 'transaction',
|
||
'cash_gift_issue': 'cashback',
|
||
'cash_gift_recycle': 'cashback',
|
||
'bonus_issue': 'rebate',
|
||
'bonus_recycle': 'rebate',
|
||
'bonus_expired': 'rebate',
|
||
'transfer_in': 'transfer',
|
||
'transfer_out': 'transfer',
|
||
'deposit': 'deposit',
|
||
'withdraw': 'withdrawal',
|
||
'buy': 'trade',
|
||
'sell': 'trade',
|
||
}
|
||
return self.safe_string(types, type, type)
|
||
|
||
async def fetch_my_trades(self, symbol: Str = None, since: Int = None, limit: Int = None, params={}) -> List[Trade]:
|
||
"""
|
||
fetch all trades made by the user
|
||
|
||
https://www.bitget.com/api-doc/spot/trade/Get-Fills
|
||
https://www.bitget.com/api-doc/contract/trade/Get-Order-Fills
|
||
https://www.bitget.com/api-doc/margin/cross/trade/Get-Cross-Order-Fills
|
||
https://www.bitget.com/api-doc/margin/isolated/trade/Get-Isolated-Transaction-Details
|
||
https://www.bitget.com/api-doc/uta/trade/Get-Order-Fills
|
||
|
||
:param str symbol: unified market symbol
|
||
:param int [since]: the earliest time in ms to fetch trades for
|
||
:param int [limit]: the maximum number of trades structures to retrieve
|
||
:param dict [params]: extra parameters specific to the exchange API endpoint
|
||
:param int [params.until]: the latest time in ms to fetch trades for
|
||
:param boolean [params.uta]: set to True for the unified trading account(uta), defaults to False
|
||
:param boolean [params.paginate]: default False, when True will automatically paginate by calling self endpoint multiple times. See in the docs all the [available parameters](https://github.com/ccxt/ccxt/wiki/Manual#pagination-params)
|
||
:returns Trade[]: a list of `trade structures <https://docs.ccxt.com/#/?id=trade-structure>`
|
||
"""
|
||
uta = None
|
||
uta, params = self.handle_option_and_params(params, 'fetchMyTrades', 'uta', False)
|
||
if not uta and (symbol is None):
|
||
raise ArgumentsRequired(self.id + ' fetchMyTrades() requires a symbol argument')
|
||
await self.load_markets()
|
||
market = self.market(symbol)
|
||
request: dict = {}
|
||
request, params = self.handle_until_option('endTime', request, params)
|
||
if since is not None:
|
||
request['startTime'] = since
|
||
if limit is not None:
|
||
request['limit'] = limit
|
||
paginate = False
|
||
marginMode = None
|
||
paginate, params = self.handle_option_and_params(params, 'fetchMyTrades', 'paginate')
|
||
marginMode, params = self.handle_margin_mode_and_params('fetchMyTrades', params)
|
||
if paginate:
|
||
cursorReceived = None
|
||
cursorSent = None
|
||
if uta:
|
||
cursorReceived = 'cursor'
|
||
cursorSent = 'cursor'
|
||
elif market['spot']:
|
||
if marginMode is not None:
|
||
cursorReceived = 'minId'
|
||
cursorSent = 'idLessThan'
|
||
else:
|
||
cursorReceived = 'endId'
|
||
cursorSent = 'idLessThan'
|
||
return await self.fetch_paginated_call_cursor('fetchMyTrades', symbol, since, limit, params, cursorReceived, cursorSent)
|
||
response = None
|
||
if uta:
|
||
response = await self.privateUtaGetV3TradeFills(self.extend(request, params))
|
||
else:
|
||
request['symbol'] = market['id']
|
||
if market['spot']:
|
||
if marginMode is not None:
|
||
if since is None:
|
||
request['startTime'] = self.milliseconds() - 7776000000
|
||
if marginMode == 'isolated':
|
||
response = await self.privateMarginGetV2MarginIsolatedFills(self.extend(request, params))
|
||
elif marginMode == 'cross':
|
||
response = await self.privateMarginGetV2MarginCrossedFills(self.extend(request, params))
|
||
else:
|
||
response = await self.privateSpotGetV2SpotTradeFills(self.extend(request, params))
|
||
else:
|
||
productType = None
|
||
productType, params = self.handle_product_type_and_params(market, params)
|
||
request['productType'] = productType
|
||
response = await self.privateMixGetV2MixOrderFills(self.extend(request, params))
|
||
#
|
||
# spot
|
||
#
|
||
# {
|
||
# "code": "00000",
|
||
# "msg": "success",
|
||
# "requestTime": 1700802995406,
|
||
# "data": [
|
||
# {
|
||
# "userId": "7264631751",
|
||
# "symbol": "BTCUSDT",
|
||
# "orderId": "1098394344925597696",
|
||
# "tradeId": "1098394344974925824",
|
||
# "orderType": "market",
|
||
# "side": "sell",
|
||
# "priceAvg": "28467.68",
|
||
# "size": "0.0002",
|
||
# "amount": "5.693536",
|
||
# "feeDetail": {
|
||
# "deduction": "no",
|
||
# "feeCoin": "USDT",
|
||
# "totalDeductionFee": "",
|
||
# "totalFee": "-0.005693536"
|
||
# },
|
||
# "tradeScope": "taker",
|
||
# "cTime": "1697603539699",
|
||
# "uTime": "1697603539754"
|
||
# }
|
||
# ]
|
||
# }
|
||
#
|
||
# spot margin
|
||
#
|
||
# {
|
||
# "code": "00000",
|
||
# "msg": "success",
|
||
# "requestTime": 1700803176399,
|
||
# "data": {
|
||
# "fills": [
|
||
# {
|
||
# "orderId": "1099353730455318528",
|
||
# "tradeId": "1099353730627092481",
|
||
# "orderType": "market",
|
||
# "side": "sell",
|
||
# "priceAvg": "29543.7",
|
||
# "size": "0.0001",
|
||
# "amount": "2.95437",
|
||
# "tradeScope": "taker",
|
||
# "feeDetail": {
|
||
# "deduction": "no",
|
||
# "feeCoin": "USDT",
|
||
# "totalDeductionFee": "0",
|
||
# "totalFee": "-0.00295437"
|
||
# },
|
||
# "cTime": "1697832275063",
|
||
# "uTime": "1697832275150"
|
||
# },
|
||
# ],
|
||
# "minId": "1099353591699161118",
|
||
# "maxId": "1099353730627092481"
|
||
# }
|
||
# }
|
||
#
|
||
# swap and future
|
||
#
|
||
# {
|
||
# "code": "00000",
|
||
# "msg": "success",
|
||
# "requestTime": 1700803357487,
|
||
# "data": {
|
||
# "fillList": [
|
||
# {
|
||
# "tradeId": "1111468664328269825",
|
||
# "symbol": "BTCUSDT",
|
||
# "orderId": "1111468664264753162",
|
||
# "price": "37271.4",
|
||
# "baseVolume": "0.001",
|
||
# "feeDetail": [
|
||
# {
|
||
# "deduction": "no",
|
||
# "feeCoin": "USDT",
|
||
# "totalDeductionFee": null,
|
||
# "totalFee": "-0.02236284"
|
||
# }
|
||
# ],
|
||
# "side": "buy",
|
||
# "quoteVolume": "37.2714",
|
||
# "profit": "-0.0007",
|
||
# "enterPointSource": "web",
|
||
# "tradeSide": "close",
|
||
# "posMode": "hedge_mode",
|
||
# "tradeScope": "taker",
|
||
# "cTime": "1700720700342"
|
||
# },
|
||
# ],
|
||
# "endId": "1099351587643699201"
|
||
# }
|
||
# }
|
||
#
|
||
# uta
|
||
#
|
||
# {
|
||
# "code": "00000",
|
||
# "msg": "success",
|
||
# "requestTime": 1751099666579,
|
||
# "data": {
|
||
# "list": [
|
||
# {
|
||
# "execId": "1322441401010528257",
|
||
# "orderId": "1322441400976261120",
|
||
# "category": "USDT-FUTURES",
|
||
# "symbol": "BTCUSDT",
|
||
# "orderType": "market",
|
||
# "side": "sell",
|
||
# "execPrice": "107005.4",
|
||
# "execQty": "0.0001",
|
||
# "execValue": "10.7005",
|
||
# "tradeScope": "taker",
|
||
# "feeDetail": [{
|
||
# "feeCoin": "USDT",
|
||
# "fee":"0.00642032"
|
||
# }],
|
||
# "createdTime": "1751020520451",
|
||
# "updatedTime": "1751020520458",
|
||
# "execPnl": "0.00017"
|
||
# },
|
||
# ],
|
||
# "cursor": "1322061241878880257"
|
||
# }
|
||
# }
|
||
#
|
||
data = self.safe_value(response, 'data')
|
||
if uta:
|
||
fills = self.safe_list(data, 'list', [])
|
||
return self.parse_trades(fills, market, since, limit)
|
||
elif (market['swap'] or (market['future'])):
|
||
fills = self.safe_list(data, 'fillList', [])
|
||
return self.parse_trades(fills, market, since, limit)
|
||
elif marginMode is not None:
|
||
fills = self.safe_list(data, 'fills', [])
|
||
return self.parse_trades(fills, market, since, limit)
|
||
return self.parse_trades(data, market, since, limit)
|
||
|
||
async def fetch_position(self, symbol: str, params={}):
|
||
"""
|
||
fetch data on a single open contract trade position
|
||
|
||
https://www.bitget.com/api-doc/contract/position/get-single-position
|
||
https://www.bitget.com/api-doc/uta/trade/Get-Position
|
||
|
||
:param str symbol: unified market symbol of the market the position is held in
|
||
:param dict [params]: extra parameters specific to the exchange API endpoint
|
||
:param boolean [params.uta]: set to True for the unified trading account(uta), defaults to False
|
||
:returns dict: a `position structure <https://docs.ccxt.com/#/?id=position-structure>`
|
||
"""
|
||
await self.load_markets()
|
||
market = self.market(symbol)
|
||
productType = None
|
||
productType, params = self.handle_product_type_and_params(market, params)
|
||
request: dict = {
|
||
'symbol': market['id'],
|
||
}
|
||
response = None
|
||
uta = None
|
||
result = None
|
||
uta, params = self.handle_option_and_params(params, 'fetchPosition', 'uta', False)
|
||
if uta:
|
||
request['category'] = productType
|
||
response = await self.privateUtaGetV3PositionCurrentPosition(self.extend(request, params))
|
||
#
|
||
# {
|
||
# "code": "00000",
|
||
# "msg": "success",
|
||
# "requestTime": 1750929905423,
|
||
# "data": {
|
||
# "list": [
|
||
# {
|
||
# "category": "USDT-FUTURES",
|
||
# "symbol": "BTCUSDT",
|
||
# "marginCoin": "USDT",
|
||
# "holdMode": "hedge_mode",
|
||
# "posSide": "long",
|
||
# "marginMode": "crossed",
|
||
# "positionBalance": "5.435199",
|
||
# "available": "0.001",
|
||
# "frozen": "0",
|
||
# "total": "0.001",
|
||
# "leverage": "20",
|
||
# "curRealisedPnl": "0",
|
||
# "avgPrice": "107410.3",
|
||
# "positionStatus": "normal",
|
||
# "unrealisedPnl": "0.0047",
|
||
# "liquidationPrice": "0",
|
||
# "mmr": "0.004",
|
||
# "profitRate": "0.0008647337475591",
|
||
# "markPrice": "107415.3",
|
||
# "breakEvenPrice": "107539.2",
|
||
# "totalFunding": "0",
|
||
# "openFeeTotal": "-0.06444618",
|
||
# "closeFeeTotal": "0",
|
||
# "createdTime": "1750495670699",
|
||
# "updatedTime": "1750929883465"
|
||
# }
|
||
# ]
|
||
# }
|
||
# }
|
||
#
|
||
data = self.safe_dict(response, 'data', {})
|
||
result = self.safe_list(data, 'list', [])
|
||
else:
|
||
request['marginCoin'] = market['settleId']
|
||
request['productType'] = productType
|
||
response = await self.privateMixGetV2MixPositionSinglePosition(self.extend(request, params))
|
||
#
|
||
# {
|
||
# "code": "00000",
|
||
# "msg": "success",
|
||
# "requestTime": 1700807531673,
|
||
# "data": [
|
||
# {
|
||
# "marginCoin": "USDT",
|
||
# "symbol": "BTCUSDT",
|
||
# "holdSide": "long",
|
||
# "openDelegateSize": "0",
|
||
# "marginSize": "3.73555",
|
||
# "available": "0.002",
|
||
# "locked": "0",
|
||
# "total": "0.002",
|
||
# "leverage": "20",
|
||
# "achievedProfits": "0",
|
||
# "openPriceAvg": "37355.5",
|
||
# "marginMode": "crossed",
|
||
# "posMode": "hedge_mode",
|
||
# "unrealizedPL": "0.007",
|
||
# "liquidationPrice": "31724.970702417",
|
||
# "keepMarginRate": "0.004",
|
||
# "markPrice": "37359",
|
||
# "marginRatio": "0.029599540355",
|
||
# "cTime": "1700807507275"
|
||
# }
|
||
# ]
|
||
# }
|
||
#
|
||
result = self.safe_list(response, 'data', [])
|
||
first = self.safe_dict(result, 0, {})
|
||
return self.parse_position(first, market)
|
||
|
||
async def fetch_positions(self, symbols: Strings = None, params={}) -> List[Position]:
|
||
"""
|
||
fetch all open positions
|
||
|
||
https://www.bitget.com/api-doc/contract/position/get-all-position
|
||
https://www.bitget.com/api-doc/contract/position/Get-History-Position
|
||
https://www.bitget.com/api-doc/uta/trade/Get-Position
|
||
|
||
:param str[] [symbols]: list of unified market symbols
|
||
:param dict [params]: extra parameters specific to the exchange API endpoint
|
||
:param str [params.marginCoin]: the settle currency of the positions, needs to match the productType
|
||
:param str [params.productType]: 'USDT-FUTURES', 'USDC-FUTURES', 'COIN-FUTURES', 'SUSDT-FUTURES', 'SUSDC-FUTURES' or 'SCOIN-FUTURES'
|
||
:param boolean [params.paginate]: default False, when True will automatically paginate by calling self endpoint multiple times. See in the docs all the [available parameters](https://github.com/ccxt/ccxt/wiki/Manual#pagination-params)
|
||
:param boolean [params.useHistoryEndpoint]: default False, when True will use the historic endpoint to fetch positions
|
||
:param str [params.method]: either(default) 'privateMixGetV2MixPositionAllPosition', 'privateMixGetV2MixPositionHistoryPosition', or 'privateUtaGetV3PositionCurrentPosition'
|
||
:param boolean [params.uta]: set to True for the unified trading account(uta), defaults to False
|
||
:returns dict[]: a list of `position structure <https://docs.ccxt.com/#/?id=position-structure>`
|
||
"""
|
||
await self.load_markets()
|
||
paginate = False
|
||
paginate, params = self.handle_option_and_params(params, 'fetchPositions', 'paginate')
|
||
if paginate:
|
||
return await self.fetch_paginated_call_cursor('fetchPositions', None, None, None, params, 'endId', 'idLessThan')
|
||
method = None
|
||
useHistoryEndpoint = self.safe_bool(params, 'useHistoryEndpoint', False)
|
||
if useHistoryEndpoint:
|
||
method = 'privateMixGetV2MixPositionHistoryPosition'
|
||
else:
|
||
method, params = self.handle_option_and_params(params, 'fetchPositions', 'method', 'privateMixGetV2MixPositionAllPosition')
|
||
market = None
|
||
if symbols is not None:
|
||
first = self.safe_string(symbols, 0)
|
||
# symbols can be None or []
|
||
if first is not None:
|
||
market = self.market(first)
|
||
productType = None
|
||
productType, params = self.handle_product_type_and_params(market, params)
|
||
request: dict = {}
|
||
response = None
|
||
isHistory = False
|
||
uta = None
|
||
uta, params = self.handle_option_and_params(params, 'fetchPositions', 'uta', False)
|
||
if uta:
|
||
request['category'] = productType
|
||
response = await self.privateUtaGetV3PositionCurrentPosition(self.extend(request, params))
|
||
elif method == 'privateMixGetV2MixPositionAllPosition':
|
||
marginCoin = self.safe_string(params, 'marginCoin', 'USDT')
|
||
if market is not None:
|
||
marginCoin = market['settleId']
|
||
elif productType == 'USDT-FUTURES':
|
||
marginCoin = 'USDT'
|
||
elif productType == 'USDC-FUTURES':
|
||
marginCoin = 'USDC'
|
||
elif productType == 'SUSDT-FUTURES':
|
||
marginCoin = 'SUSDT'
|
||
elif productType == 'SUSDC-FUTURES':
|
||
marginCoin = 'SUSDC'
|
||
elif (productType == 'SCOIN-FUTURES') or (productType == 'COIN-FUTURES'):
|
||
if marginCoin is None:
|
||
raise ArgumentsRequired(self.id + ' fetchPositions() requires a marginCoin parameter that matches the productType')
|
||
request['marginCoin'] = marginCoin
|
||
request['productType'] = productType
|
||
response = await self.privateMixGetV2MixPositionAllPosition(self.extend(request, params))
|
||
else:
|
||
isHistory = True
|
||
if market is not None:
|
||
request['symbol'] = market['id']
|
||
request['productType'] = productType
|
||
response = await self.privateMixGetV2MixPositionHistoryPosition(self.extend(request, params))
|
||
#
|
||
# privateMixGetV2MixPositionAllPosition
|
||
#
|
||
# {
|
||
# "code": "00000",
|
||
# "msg": "success",
|
||
# "requestTime": 1700807810221,
|
||
# "data": [
|
||
# {
|
||
# "marginCoin": "USDT",
|
||
# "symbol": "BTCUSDT",
|
||
# "holdSide": "long",
|
||
# "openDelegateSize": "0",
|
||
# "marginSize": "3.73555",
|
||
# "available": "0.002",
|
||
# "locked": "0",
|
||
# "total": "0.002",
|
||
# "leverage": "20",
|
||
# "achievedProfits": "0",
|
||
# "openPriceAvg": "37355.5",
|
||
# "marginMode": "crossed",
|
||
# "posMode": "hedge_mode",
|
||
# "unrealizedPL": "0.03",
|
||
# "liquidationPrice": "31725.023602417",
|
||
# "keepMarginRate": "0.004",
|
||
# "markPrice": "37370.5",
|
||
# "marginRatio": "0.029550120396",
|
||
# "cTime": "1700807507275"
|
||
# }
|
||
# ]
|
||
# }
|
||
#
|
||
# privateMixGetV2MixPositionHistoryPosition
|
||
#
|
||
# {
|
||
# "code": "00000",
|
||
# "msg": "success",
|
||
# "requestTime": 1700808051002,
|
||
# "data": {
|
||
# "list": [
|
||
# {
|
||
# "symbol": "BTCUSDT",
|
||
# "marginCoin": "USDT",
|
||
# "holdSide": "long",
|
||
# "openAvgPrice": "37272.1",
|
||
# "closeAvgPrice": "37271.4",
|
||
# "marginMode": "crossed",
|
||
# "openTotalPos": "0.001",
|
||
# "closeTotalPos": "0.001",
|
||
# "pnl": "-0.0007",
|
||
# "netProfit": "-0.0454261",
|
||
# "totalFunding": "0",
|
||
# "openFee": "-0.02236326",
|
||
# "closeFee": "-0.02236284",
|
||
# "utime": "1700720700400",
|
||
# "ctime": "1700720651684"
|
||
# },
|
||
# ],
|
||
# "endId": "1099351653866962944"
|
||
# }
|
||
# }
|
||
#
|
||
# privateUtaGetV3PositionCurrentPosition
|
||
#
|
||
# {
|
||
# "code": "00000",
|
||
# "msg": "success",
|
||
# "requestTime": 1750929905423,
|
||
# "data": {
|
||
# "list": [
|
||
# {
|
||
# "category": "USDT-FUTURES",
|
||
# "symbol": "BTCUSDT",
|
||
# "marginCoin": "USDT",
|
||
# "holdMode": "hedge_mode",
|
||
# "posSide": "long",
|
||
# "marginMode": "crossed",
|
||
# "positionBalance": "5.435199",
|
||
# "available": "0.001",
|
||
# "frozen": "0",
|
||
# "total": "0.001",
|
||
# "leverage": "20",
|
||
# "curRealisedPnl": "0",
|
||
# "avgPrice": "107410.3",
|
||
# "positionStatus": "normal",
|
||
# "unrealisedPnl": "0.0047",
|
||
# "liquidationPrice": "0",
|
||
# "mmr": "0.004",
|
||
# "profitRate": "0.0008647337475591",
|
||
# "markPrice": "107415.3",
|
||
# "breakEvenPrice": "107539.2",
|
||
# "totalFunding": "0",
|
||
# "openFeeTotal": "-0.06444618",
|
||
# "closeFeeTotal": "0",
|
||
# "createdTime": "1750495670699",
|
||
# "updatedTime": "1750929883465"
|
||
# }
|
||
# ]
|
||
# }
|
||
# }
|
||
#
|
||
position = []
|
||
if uta or isHistory:
|
||
data = self.safe_dict(response, 'data', {})
|
||
position = self.safe_list(data, 'list', [])
|
||
else:
|
||
position = self.safe_list(response, 'data', [])
|
||
result = []
|
||
for i in range(0, len(position)):
|
||
result.append(self.parse_position(position[i], market))
|
||
symbols = self.market_symbols(symbols)
|
||
return self.filter_by_array_positions(result, 'symbol', symbols, False)
|
||
|
||
def parse_position(self, position: dict, market: Market = None):
|
||
#
|
||
# fetchPosition
|
||
#
|
||
# {
|
||
# "marginCoin": "USDT",
|
||
# "symbol": "BTCUSDT",
|
||
# "holdSide": "long",
|
||
# "openDelegateSize": "0",
|
||
# "marginSize": "3.73555",
|
||
# "available": "0.002",
|
||
# "locked": "0",
|
||
# "total": "0.002",
|
||
# "leverage": "20",
|
||
# "achievedProfits": "0",
|
||
# "openPriceAvg": "37355.5",
|
||
# "marginMode": "crossed",
|
||
# "posMode": "hedge_mode",
|
||
# "unrealizedPL": "0.007",
|
||
# "liquidationPrice": "31724.970702417",
|
||
# "keepMarginRate": "0.004",
|
||
# "markPrice": "37359",
|
||
# "marginRatio": "0.029599540355",
|
||
# "cTime": "1700807507275"
|
||
# }
|
||
#
|
||
# uta: fetchPosition
|
||
#
|
||
# {
|
||
# "category": "USDT-FUTURES",
|
||
# "symbol": "BTCUSDT",
|
||
# "marginCoin": "USDT",
|
||
# "holdMode": "hedge_mode",
|
||
# "posSide": "long",
|
||
# "marginMode": "crossed",
|
||
# "positionBalance": "5.435199",
|
||
# "available": "0.001",
|
||
# "frozen": "0",
|
||
# "total": "0.001",
|
||
# "leverage": "20",
|
||
# "curRealisedPnl": "0",
|
||
# "avgPrice": "107410.3",
|
||
# "positionStatus": "normal",
|
||
# "unrealisedPnl": "0.0047",
|
||
# "liquidationPrice": "0",
|
||
# "mmr": "0.004",
|
||
# "profitRate": "0.0008647337475591",
|
||
# "markPrice": "107415.3",
|
||
# "breakEvenPrice": "107539.2",
|
||
# "totalFunding": "0",
|
||
# "openFeeTotal": "-0.06444618",
|
||
# "closeFeeTotal": "0",
|
||
# "createdTime": "1750495670699",
|
||
# "updatedTime": "1750929883465"
|
||
# }
|
||
#
|
||
# fetchPositions: privateMixGetV2MixPositionAllPosition
|
||
#
|
||
# {
|
||
# "marginCoin": "USDT",
|
||
# "symbol": "BTCUSDT",
|
||
# "holdSide": "long",
|
||
# "openDelegateSize": "0",
|
||
# "marginSize": "3.73555",
|
||
# "available": "0.002",
|
||
# "locked": "0",
|
||
# "total": "0.002",
|
||
# "leverage": "20",
|
||
# "achievedProfits": "0",
|
||
# "openPriceAvg": "37355.5",
|
||
# "marginMode": "crossed",
|
||
# "posMode": "hedge_mode",
|
||
# "unrealizedPL": "0.03",
|
||
# "liquidationPrice": "31725.023602417",
|
||
# "keepMarginRate": "0.004",
|
||
# "markPrice": "37370.5",
|
||
# "marginRatio": "0.029550120396",
|
||
# "cTime": "1700807507275"
|
||
# }
|
||
#
|
||
# fetchPositionsHistory: privateMixGetV2MixPositionHistoryPosition
|
||
#
|
||
# {
|
||
# "symbol": "BTCUSDT",
|
||
# "marginCoin": "USDT",
|
||
# "holdSide": "long",
|
||
# "openAvgPrice": "37272.1",
|
||
# "closeAvgPrice": "37271.4",
|
||
# "marginMode": "crossed",
|
||
# "openTotalPos": "0.001",
|
||
# "closeTotalPos": "0.001",
|
||
# "pnl": "-0.0007",
|
||
# "netProfit": "-0.0454261",
|
||
# "totalFunding": "0",
|
||
# "openFee": "-0.02236326",
|
||
# "closeFee": "-0.02236284",
|
||
# "utime": "1700720700400",
|
||
# "ctime": "1700720651684"
|
||
# }
|
||
#
|
||
# closeAllPositions
|
||
#
|
||
# {
|
||
# "orderId": "1120923953904893955",
|
||
# "clientOid": "1120923953904893956"
|
||
# }
|
||
#
|
||
# uta: fetchPositionsHistory
|
||
#
|
||
# {
|
||
# "positionId": "1322441328637100049",
|
||
# "category": "USDT-FUTURES",
|
||
# "symbol": "BTCUSDT",
|
||
# "marginCoin": "USDT",
|
||
# "holdMode": "hedge_mode",
|
||
# "posSide": "long",
|
||
# "marginMode": "crossed",
|
||
# "openPriceAvg": "107003.7",
|
||
# "closePriceAvg": "107005.4",
|
||
# "openTotalPos": "0.0001",
|
||
# "closeTotalPos": "0.0001",
|
||
# "cumRealisedPnl": "0.00017",
|
||
# "netProfit": "-0.01267055",
|
||
# "totalFunding": "0",
|
||
# "openFeeTotal": "-0.00642022",
|
||
# "closeFeeTotal": "-0.00642032",
|
||
# "createdTime": "1751020503195",
|
||
# "updatedTime": "1751020520458"
|
||
# }
|
||
#
|
||
marketId = self.safe_string(position, 'symbol')
|
||
market = self.safe_market(marketId, market, None, 'contract')
|
||
symbol = market['symbol']
|
||
timestamp = self.safe_integer_n(position, ['cTime', 'ctime', 'createdTime'])
|
||
marginMode = self.safe_string(position, 'marginMode')
|
||
collateral = None
|
||
initialMargin = None
|
||
unrealizedPnl = self.safe_string_2(position, 'unrealizedPL', 'unrealisedPnl')
|
||
rawCollateral = self.safe_string_2(position, 'marginSize', 'positionBalance')
|
||
if marginMode == 'isolated':
|
||
collateral = Precise.string_add(rawCollateral, unrealizedPnl)
|
||
elif marginMode == 'crossed':
|
||
marginMode = 'cross'
|
||
initialMargin = rawCollateral
|
||
holdMode = self.safe_string_2(position, 'posMode', 'holdMode')
|
||
hedged = None
|
||
if holdMode == 'hedge_mode':
|
||
hedged = True
|
||
elif holdMode == 'one_way_mode':
|
||
hedged = False
|
||
side = self.safe_string_2(position, 'holdSide', 'posSide')
|
||
leverage = self.safe_string(position, 'leverage')
|
||
contractSizeNumber = self.safe_value(market, 'contractSize')
|
||
contractSize = self.number_to_string(contractSizeNumber)
|
||
baseAmount = self.safe_string_2(position, 'total', 'openTotalPos')
|
||
entryPrice = self.safe_string_n(position, ['openPriceAvg', 'openAvgPrice', 'avgPrice'])
|
||
maintenanceMarginPercentage = self.safe_string(position, 'keepMarginRate')
|
||
openNotional = Precise.string_mul(entryPrice, baseAmount)
|
||
if initialMargin is None:
|
||
initialMargin = Precise.string_div(openNotional, leverage)
|
||
contracts = self.parse_number(Precise.string_div(baseAmount, contractSize))
|
||
if contracts is None:
|
||
contracts = self.safe_number(position, 'closeTotalPos')
|
||
markPrice = self.safe_string(position, 'markPrice')
|
||
notional = Precise.string_mul(baseAmount, markPrice)
|
||
initialMarginPercentage = Precise.string_div(initialMargin, notional)
|
||
liquidationPrice = self.parse_number(self.omit_zero(self.safe_string(position, 'liquidationPrice')))
|
||
calcTakerFeeRate = '0.0006'
|
||
calcTakerFeeMult = '0.9994'
|
||
if (liquidationPrice is None) and (marginMode == 'isolated') and Precise.string_gt(baseAmount, '0'):
|
||
signedMargin = Precise.string_div(rawCollateral, baseAmount)
|
||
signedMmp = maintenanceMarginPercentage
|
||
if side == 'short':
|
||
signedMargin = Precise.string_neg(signedMargin)
|
||
signedMmp = Precise.string_neg(signedMmp)
|
||
mmrMinusOne = Precise.string_sub('1', signedMmp)
|
||
numerator = Precise.string_sub(entryPrice, signedMargin)
|
||
if side == 'long':
|
||
mmrMinusOne = Precise.string_mul(mmrMinusOne, calcTakerFeeMult)
|
||
else:
|
||
numerator = Precise.string_mul(numerator, calcTakerFeeMult)
|
||
liquidationPrice = self.parse_number(Precise.string_div(numerator, mmrMinusOne))
|
||
feeToClose = Precise.string_mul(notional, calcTakerFeeRate)
|
||
maintenanceMargin = Precise.string_add(Precise.string_mul(maintenanceMarginPercentage, notional), feeToClose)
|
||
percentage = Precise.string_mul(Precise.string_div(unrealizedPnl, initialMargin, 4), '100')
|
||
return self.safe_position({
|
||
'info': position,
|
||
'id': self.safe_string_2(position, 'orderId', 'positionId'),
|
||
'symbol': symbol,
|
||
'notional': self.parse_number(notional),
|
||
'marginMode': marginMode,
|
||
'liquidationPrice': liquidationPrice,
|
||
'entryPrice': self.parse_number(entryPrice),
|
||
'unrealizedPnl': self.parse_number(unrealizedPnl),
|
||
'realizedPnl': self.safe_number_n(position, ['pnl', 'curRealisedPnl', 'cumRealisedPnl']),
|
||
'percentage': self.parse_number(percentage),
|
||
'contracts': contracts,
|
||
'contractSize': contractSizeNumber,
|
||
'markPrice': self.parse_number(markPrice),
|
||
'lastPrice': self.safe_number_2(position, 'closeAvgPrice', 'closePriceAvg'),
|
||
'side': side,
|
||
'hedged': hedged,
|
||
'timestamp': timestamp,
|
||
'datetime': self.iso8601(timestamp),
|
||
'lastUpdateTimestamp': self.safe_integer_2(position, 'utime', 'updatedTime'),
|
||
'maintenanceMargin': self.parse_number(maintenanceMargin),
|
||
'maintenanceMarginPercentage': self.parse_number(maintenanceMarginPercentage),
|
||
'collateral': self.parse_number(collateral),
|
||
'initialMargin': self.parse_number(initialMargin),
|
||
'initialMarginPercentage': self.parse_number(initialMarginPercentage),
|
||
'leverage': self.parse_number(leverage),
|
||
'marginRatio': self.safe_number_2(position, 'marginRatio', 'mmr'),
|
||
'stopLossPrice': None,
|
||
'takeProfitPrice': None,
|
||
})
|
||
|
||
async def fetch_funding_rate_history(self, symbol: Str = None, since: Int = None, limit: Int = None, params={}):
|
||
"""
|
||
fetches historical funding rate prices
|
||
|
||
https://www.bitget.com/api-doc/contract/market/Get-History-Funding-Rate
|
||
https://www.bitget.com/api-doc/uta/public/Get-History-Funding-Rate
|
||
|
||
:param str symbol: unified symbol of the market to fetch the funding rate history for
|
||
:param int [since]: timestamp in ms of the earliest funding rate to fetch
|
||
:param int [limit]: the maximum amount of funding rate structures to fetch
|
||
:param dict [params]: extra parameters specific to the exchange API endpoint
|
||
:param boolean [params.uta]: set to True for the unified trading account(uta), defaults to False
|
||
:param boolean [params.paginate]: default False, when True will automatically paginate by calling self endpoint multiple times. See in the docs all the [availble parameters](https://github.com/ccxt/ccxt/wiki/Manual#pagination-params)
|
||
:returns dict[]: a list of `funding rate structures <https://docs.ccxt.com/#/?id=funding-rate-history-structure>`
|
||
"""
|
||
if symbol is None:
|
||
raise ArgumentsRequired(self.id + ' fetchFundingRateHistory() requires a symbol argument')
|
||
await self.load_markets()
|
||
market = self.market(symbol)
|
||
request: dict = {
|
||
'symbol': market['id'],
|
||
}
|
||
productType = None
|
||
uta = None
|
||
response = None
|
||
result = None
|
||
productType, params = self.handle_product_type_and_params(market, params)
|
||
uta, params = self.handle_option_and_params(params, 'fetchFundingRateHistory', 'uta', False)
|
||
if uta:
|
||
if limit is not None:
|
||
request['limit'] = limit
|
||
request['category'] = productType
|
||
response = await self.publicUtaGetV3MarketHistoryFundRate(self.extend(request, params))
|
||
#
|
||
# {
|
||
# "code": "00000",
|
||
# "msg": "success",
|
||
# "requestTime": 1750435113658,
|
||
# "data": {
|
||
# "resultList": [
|
||
# {
|
||
# "symbol": "BTCUSDT",
|
||
# "fundingRate": "-0.000017",
|
||
# "fundingRateTimestamp": "1750431600000"
|
||
# },
|
||
# ]
|
||
# }
|
||
# }
|
||
#
|
||
data = self.safe_dict(response, 'data', {})
|
||
result = self.safe_list(data, 'resultList', [])
|
||
else:
|
||
paginate = False
|
||
paginate, params = self.handle_option_and_params(params, 'fetchFundingRateHistory', 'paginate')
|
||
if paginate:
|
||
return await self.fetch_paginated_call_incremental('fetchFundingRateHistory', symbol, since, limit, params, 'pageNo', 100)
|
||
if limit is not None:
|
||
request['pageSize'] = limit
|
||
request['productType'] = productType
|
||
response = await self.publicMixGetV2MixMarketHistoryFundRate(self.extend(request, params))
|
||
#
|
||
# {
|
||
# "code": "00000",
|
||
# "msg": "success",
|
||
# "requestTime": 1652406728393,
|
||
# "data": [
|
||
# {
|
||
# "symbol": "BTCUSDT",
|
||
# "fundingRate": "-0.0003",
|
||
# "fundingTime": "1652396400000"
|
||
# },
|
||
# ]
|
||
# }
|
||
#
|
||
result = self.safe_list(response, 'data', [])
|
||
rates = []
|
||
for i in range(0, len(result)):
|
||
entry = result[i]
|
||
marketId = self.safe_string(entry, 'symbol')
|
||
symbolInner = self.safe_symbol(marketId, market)
|
||
timestamp = self.safe_integer_2(entry, 'fundingTime', 'fundingRateTimestamp')
|
||
rates.append({
|
||
'info': entry,
|
||
'symbol': symbolInner,
|
||
'fundingRate': self.safe_number(entry, 'fundingRate'),
|
||
'timestamp': timestamp,
|
||
'datetime': self.iso8601(timestamp),
|
||
})
|
||
sorted = self.sort_by(rates, 'timestamp')
|
||
return self.filter_by_symbol_since_limit(sorted, market['symbol'], since, limit)
|
||
|
||
async def fetch_funding_rate(self, symbol: str, params={}) -> FundingRate:
|
||
"""
|
||
fetch the current funding rate
|
||
|
||
https://www.bitget.com/api-doc/contract/market/Get-Current-Funding-Rate
|
||
https://www.bitget.com/api-doc/contract/market/Get-Symbol-Next-Funding-Time
|
||
https://www.bitget.com/api-doc/uta/public/Get-Current-Funding-Rate
|
||
|
||
:param str symbol: unified market symbol
|
||
:param dict [params]: extra parameters specific to the exchange API endpoint
|
||
:param boolean [params.uta]: set to True for the unified trading account(uta), defaults to False
|
||
:param str [params.method]: either(default) 'publicMixGetV2MixMarketCurrentFundRate' or 'publicMixGetV2MixMarketFundingTime'
|
||
:returns dict: a `funding rate structure <https://docs.ccxt.com/#/?id=funding-rate-structure>`
|
||
"""
|
||
await self.load_markets()
|
||
market = self.market(symbol)
|
||
if not market['swap']:
|
||
raise BadSymbol(self.id + ' fetchFundingRate() supports swap contracts only')
|
||
productType = None
|
||
productType, params = self.handle_product_type_and_params(market, params)
|
||
request: dict = {
|
||
'symbol': market['id'],
|
||
}
|
||
uta = None
|
||
response = None
|
||
uta, params = self.handle_option_and_params(params, 'fetchFundingRate', 'uta', False)
|
||
if uta:
|
||
response = await self.publicUtaGetV3MarketCurrentFundRate(self.extend(request, params))
|
||
#
|
||
# {
|
||
# "code": "00000",
|
||
# "msg": "success",
|
||
# "requestTime": 1750897372153,
|
||
# "data": [
|
||
# {
|
||
# "symbol": "BTCUSDT",
|
||
# "fundingRate": "0.00001",
|
||
# "fundingRateInterval": "8",
|
||
# "nextUpdate": "1750924800000",
|
||
# "minFundingRate": "-0.003",
|
||
# "maxFundingRate": "0.003"
|
||
# }
|
||
# ]
|
||
# }
|
||
#
|
||
else:
|
||
request['productType'] = productType
|
||
method = None
|
||
method, params = self.handle_option_and_params(params, 'fetchFundingRate', 'method', 'publicMixGetV2MixMarketCurrentFundRate')
|
||
if method == 'publicMixGetV2MixMarketCurrentFundRate':
|
||
response = await self.publicMixGetV2MixMarketCurrentFundRate(self.extend(request, params))
|
||
#
|
||
# {
|
||
# "code": "00000",
|
||
# "msg": "success",
|
||
# "requestTime": 1745500709429,
|
||
# "data": [
|
||
# {
|
||
# "symbol": "BTCUSDT",
|
||
# "fundingRate": "-0.000013",
|
||
# "fundingRateInterval": "8",
|
||
# "nextUpdate": "1745510400000",
|
||
# "minFundingRate": "-0.003",
|
||
# "maxFundingRate": "0.003"
|
||
# }
|
||
# ]
|
||
# }
|
||
#
|
||
elif method == 'publicMixGetV2MixMarketFundingTime':
|
||
response = await self.publicMixGetV2MixMarketFundingTime(self.extend(request, params))
|
||
#
|
||
# {
|
||
# "code": "00000",
|
||
# "msg": "success",
|
||
# "requestTime": 1745402092428,
|
||
# "data": [
|
||
# {
|
||
# "symbol": "BTCUSDT",
|
||
# "nextFundingTime": "1745424000000",
|
||
# "ratePeriod": "8"
|
||
# }
|
||
# ]
|
||
# }
|
||
#
|
||
data = self.safe_list(response, 'data', [])
|
||
return self.parse_funding_rate(data[0], market)
|
||
|
||
async def fetch_funding_rates(self, symbols: Strings = None, params={}) -> FundingRates:
|
||
"""
|
||
fetch the current funding rates for all markets
|
||
|
||
https://www.bitget.com/api-doc/contract/market/Get-All-Symbol-Ticker
|
||
|
||
:param str[] [symbols]: list of unified market symbols
|
||
:param dict [params]: extra parameters specific to the exchange API endpoint
|
||
:param str [params.subType]: *contract only* 'linear', 'inverse'
|
||
:param str [params.productType]: *contract only* 'USDT-FUTURES', 'USDC-FUTURES', 'COIN-FUTURES', 'SUSDT-FUTURES', 'SUSDC-FUTURES' or 'SCOIN-FUTURES'
|
||
:param str [params.method]: either(default) 'publicMixGetV2MixMarketTickers' or 'publicMixGetV2MixMarketCurrentFundRate'
|
||
:returns dict: a dictionary of `funding rate structures <https://docs.ccxt.com/#/?id=funding-rates-structure>`, indexed by market symbols
|
||
"""
|
||
await self.load_markets()
|
||
market = None
|
||
if symbols is not None:
|
||
symbol = self.safe_value(symbols, 0)
|
||
market = self.market(symbol)
|
||
request: dict = {}
|
||
productType = None
|
||
productType, params = self.handle_product_type_and_params(market, params)
|
||
method = 'publicMixGetV2MixMarketTickers'
|
||
method, params = self.handle_option_and_params(params, 'fetchFundingRates', 'method', method)
|
||
response = None
|
||
request['productType'] = productType
|
||
if method == 'publicMixGetV2MixMarketTickers':
|
||
# {
|
||
# "code": "00000",
|
||
# "msg": "success",
|
||
# "requestTime": 1700533773477,
|
||
# "data": [
|
||
# {
|
||
# "symbol": "BTCUSD",
|
||
# "lastPr": "29904.5",
|
||
# "askPr": "29904.5",
|
||
# "bidPr": "29903.5",
|
||
# "bidSz": "0.5091",
|
||
# "askSz": "2.2694",
|
||
# "high24h": "0",
|
||
# "low24h": "0",
|
||
# "ts": "1695794271400",
|
||
# "change24h": "0",
|
||
# "baseVolume": "0",
|
||
# "quoteVolume": "0",
|
||
# "usdtVolume": "0",
|
||
# "openUtc": "0",
|
||
# "changeUtc24h": "0",
|
||
# "indexPrice": "29132.353333",
|
||
# "fundingRate": "-0.0007",
|
||
# "holdingAmount": "125.6844",
|
||
# "deliveryStartTime": null,
|
||
# "deliveryTime": null,
|
||
# "deliveryStatus": "delivery_normal",
|
||
# "open24h": "0",
|
||
# "markPrice": "12345"
|
||
# },
|
||
# ]
|
||
# }
|
||
response = await self.publicMixGetV2MixMarketTickers(self.extend(request, params))
|
||
elif method == 'publicMixGetV2MixMarketCurrentFundRate':
|
||
#
|
||
# {
|
||
# "code": "00000",
|
||
# "msg": "success",
|
||
# "requestTime":1761659449917,
|
||
# "data":[
|
||
# {
|
||
# "symbol": "BTCUSDT",
|
||
# "fundingRate": "-0.000024",
|
||
# "fundingRateInterval": "8",
|
||
# "nextUpdate": "1761667200000",
|
||
# "minFundingRate": "-0.003",
|
||
# "maxFundingRate": "0.003"
|
||
# }
|
||
# ]
|
||
# }
|
||
#
|
||
response = await self.publicMixGetV2MixMarketCurrentFundRate(self.extend(request, params))
|
||
symbols = self.market_symbols(symbols)
|
||
data = self.safe_list(response, 'data', [])
|
||
return self.parse_funding_rates(data, symbols)
|
||
|
||
async def fetch_funding_intervals(self, symbols: Strings = None, params={}) -> FundingRates:
|
||
"""
|
||
fetch the funding rate interval for multiple markets
|
||
|
||
https://www.bitget.com/api-doc/contract/market/Get-All-Symbol-Ticker
|
||
|
||
:param str[] [symbols]: list of unified market symbols
|
||
:param dict [params]: extra parameters specific to the exchange API endpoint
|
||
:param str [params.productType]: 'USDT-FUTURES'(default), 'USDC-FUTURES', 'COIN-FUTURES', 'SUSDT-FUTURES', 'SUSDC-FUTURES' or 'SCOIN-FUTURES'
|
||
:returns dict[]: a list of `funding rate structures <https://docs.ccxt.com/#/?id=funding-rate-structure>`
|
||
"""
|
||
await self.load_markets()
|
||
params = self.extend({'method': 'publicMixGetV2MixMarketCurrentFundRate'}, params)
|
||
return await self.fetch_funding_rates(symbols, params)
|
||
|
||
def parse_funding_rate(self, contract, market: Market = None) -> FundingRate:
|
||
#
|
||
# fetchFundingRate: publicMixGetV2MixMarketCurrentFundRate, publicUtaGetV3MarketCurrentFundRate
|
||
#
|
||
# {
|
||
# "symbol": "BTCUSDT",
|
||
# "fundingRate": "-0.000013",
|
||
# "fundingRateInterval": "8",
|
||
# "nextUpdate": "1745510400000",
|
||
# "minFundingRate": "-0.003",
|
||
# "maxFundingRate": "0.003"
|
||
# }
|
||
#
|
||
# fetchFundingRate: publicMixGetV2MixMarketFundingTime
|
||
#
|
||
# {
|
||
# "symbol": "BTCUSDT",
|
||
# "nextFundingTime": "1745424000000",
|
||
# "ratePeriod": "8"
|
||
# }
|
||
#
|
||
# fetchFundingInterval
|
||
#
|
||
# {
|
||
# "symbol": "BTCUSDT",
|
||
# "nextFundingTime": "1727942400000",
|
||
# "ratePeriod": "8"
|
||
# }
|
||
#
|
||
# fetchFundingRates
|
||
#
|
||
# {
|
||
# "symbol": "BTCUSD",
|
||
# "lastPr": "29904.5",
|
||
# "askPr": "29904.5",
|
||
# "bidPr": "29903.5",
|
||
# "bidSz": "0.5091",
|
||
# "askSz": "2.2694",
|
||
# "high24h": "0",
|
||
# "low24h": "0",
|
||
# "ts": "1695794271400",
|
||
# "change24h": "0",
|
||
# "baseVolume": "0",
|
||
# "quoteVolume": "0",
|
||
# "usdtVolume": "0",
|
||
# "openUtc": "0",
|
||
# "changeUtc24h": "0",
|
||
# "indexPrice": "29132.353333",
|
||
# "fundingRate": "-0.0007",
|
||
# "holdingAmount": "125.6844",
|
||
# "deliveryStartTime": null,
|
||
# "deliveryTime": null,
|
||
# "deliveryStatus": "delivery_normal",
|
||
# "open24h": "0",
|
||
# "markPrice": "12345"
|
||
# }
|
||
#
|
||
marketId = self.safe_string(contract, 'symbol')
|
||
symbol = self.safe_symbol(marketId, market, None, 'swap')
|
||
fundingTimestamp = self.safe_integer_2(contract, 'nextFundingTime', 'nextUpdate')
|
||
interval = self.safe_string_2(contract, 'ratePeriod', 'fundingRateInterval')
|
||
timestamp = self.safe_integer(contract, 'ts')
|
||
markPrice = self.safe_number(contract, 'markPrice')
|
||
indexPrice = self.safe_number(contract, 'indexPrice')
|
||
intervalString = None
|
||
if interval is not None:
|
||
intervalString = interval + 'h'
|
||
return {
|
||
'info': contract,
|
||
'symbol': symbol,
|
||
'markPrice': markPrice,
|
||
'indexPrice': indexPrice,
|
||
'interestRate': None,
|
||
'estimatedSettlePrice': None,
|
||
'timestamp': timestamp,
|
||
'datetime': self.iso8601(timestamp),
|
||
'fundingRate': self.safe_number(contract, 'fundingRate'),
|
||
'fundingTimestamp': fundingTimestamp,
|
||
'fundingDatetime': self.iso8601(fundingTimestamp),
|
||
'nextFundingRate': None,
|
||
'nextFundingTimestamp': None,
|
||
'nextFundingDatetime': None,
|
||
'previousFundingRate': None,
|
||
'previousFundingTimestamp': None,
|
||
'previousFundingDatetime': None,
|
||
'interval': intervalString,
|
||
}
|
||
|
||
async def fetch_funding_history(self, symbol: Str = None, since: Int = None, limit: Int = None, params={}) -> List[FundingHistory]:
|
||
"""
|
||
fetch the funding history
|
||
|
||
https://www.bitget.com/api-doc/contract/account/Get-Account-Bill
|
||
|
||
:param str symbol: unified market symbol
|
||
:param int [since]: the starting timestamp in milliseconds
|
||
:param int [limit]: the number of entries to return
|
||
:param dict [params]: extra parameters specific to the exchange API endpoint
|
||
:param int [params.until]: the latest time in ms to fetch funding history for
|
||
:param boolean [params.paginate]: default False, when True will automatically paginate by calling self endpoint multiple times. See in the docs all the [available parameters](https://github.com/ccxt/ccxt/wiki/Manual#pagination-params)
|
||
:returns dict[]: a list of `funding history structures <https://docs.ccxt.com/#/?id=funding-history-structure>`
|
||
"""
|
||
await self.load_markets()
|
||
if symbol is None:
|
||
raise ArgumentsRequired(self.id + ' fetchFundingHistory() requires a symbol argument')
|
||
paginate = False
|
||
paginate, params = self.handle_option_and_params(params, 'fetchFundingHistory', 'paginate')
|
||
if paginate:
|
||
return await self.fetch_paginated_call_cursor('fetchFundingHistory', symbol, since, limit, params, 'endId', 'idLessThan')
|
||
market = self.market(symbol)
|
||
if not market['swap']:
|
||
raise BadSymbol(self.id + ' fetchFundingHistory() supports swap contracts only')
|
||
productType = None
|
||
productType, params = self.handle_product_type_and_params(market, params)
|
||
request: dict = {
|
||
'symbol': market['id'],
|
||
'marginCoin': market['settleId'],
|
||
'businessType': 'contract_settle_fee',
|
||
'productType': productType,
|
||
}
|
||
request, params = self.handle_until_option('endTime', request, params)
|
||
if since is not None:
|
||
request['startTime'] = since
|
||
if limit is not None:
|
||
request['limit'] = limit
|
||
response = await self.privateMixGetV2MixAccountBill(self.extend(request, params))
|
||
#
|
||
# {
|
||
# "code": "00000",
|
||
# "msg": "success",
|
||
# "requestTime": 1700795977890,
|
||
# "data": {
|
||
# "bills": [
|
||
# {
|
||
# "billId": "1111499428100472833",
|
||
# "symbol": "BTCUSDT",
|
||
# "amount": "-0.004992",
|
||
# "fee": "0",
|
||
# "feeByCoupon": "",
|
||
# "businessType": "contract_settle_fee",
|
||
# "coin": "USDT",
|
||
# "cTime": "1700728034996"
|
||
# },
|
||
# ],
|
||
# "endId": "1098396773329305606"
|
||
# }
|
||
# }
|
||
#
|
||
data = self.safe_value(response, 'data', {})
|
||
result = self.safe_value(data, 'bills', [])
|
||
return self.parse_funding_histories(result, market, since, limit)
|
||
|
||
def parse_funding_history(self, contract, market: Market = None):
|
||
#
|
||
# {
|
||
# "billId": "1111499428100472833",
|
||
# "symbol": "BTCUSDT",
|
||
# "amount": "-0.004992",
|
||
# "fee": "0",
|
||
# "feeByCoupon": "",
|
||
# "businessType": "contract_settle_fee",
|
||
# "coin": "USDT",
|
||
# "cTime": "1700728034996"
|
||
# }
|
||
#
|
||
marketId = self.safe_string(contract, 'symbol')
|
||
currencyId = self.safe_string(contract, 'coin')
|
||
timestamp = self.safe_integer(contract, 'cTime')
|
||
return {
|
||
'info': contract,
|
||
'symbol': self.safe_symbol(marketId, market, None, 'swap'),
|
||
'timestamp': timestamp,
|
||
'datetime': self.iso8601(timestamp),
|
||
'code': self.safe_currency_code(currencyId),
|
||
'amount': self.safe_number(contract, 'amount'),
|
||
'id': self.safe_string(contract, 'billId'),
|
||
}
|
||
|
||
def parse_funding_histories(self, contracts, market=None, since: Int = None, limit: Int = None) -> List[FundingHistory]:
|
||
result = []
|
||
for i in range(0, len(contracts)):
|
||
contract = contracts[i]
|
||
business = self.safe_string(contract, 'businessType')
|
||
if business != 'contract_settle_fee':
|
||
continue
|
||
result.append(self.parse_funding_history(contract, market))
|
||
sorted = self.sort_by(result, 'timestamp')
|
||
symbol = None
|
||
if market is not None:
|
||
symbol = market['symbol']
|
||
return self.filter_by_symbol_since_limit(sorted, symbol, since, limit)
|
||
|
||
async def modify_margin_helper(self, symbol: str, amount, type, params={}) -> MarginModification:
|
||
await self.load_markets()
|
||
holdSide = self.safe_string(params, 'holdSide')
|
||
market = self.market(symbol)
|
||
productType = None
|
||
productType, params = self.handle_product_type_and_params(market, params)
|
||
request: dict = {
|
||
'symbol': market['id'],
|
||
'marginCoin': market['settleId'],
|
||
'amount': self.amount_to_precision(symbol, amount), # positive value for adding margin, negative for reducing
|
||
'holdSide': holdSide, # long or short
|
||
'productType': productType,
|
||
}
|
||
params = self.omit(params, 'holdSide')
|
||
response = await self.privateMixPostV2MixAccountSetMargin(self.extend(request, params))
|
||
#
|
||
# {
|
||
# "code": "00000",
|
||
# "msg": "success",
|
||
# "requestTime": 1700813444618,
|
||
# "data": ""
|
||
# }
|
||
#
|
||
return self.extend(self.parse_margin_modification(response, market), {
|
||
'amount': self.parse_number(amount),
|
||
'type': type,
|
||
})
|
||
|
||
def parse_margin_modification(self, data: dict, market: Market = None) -> MarginModification:
|
||
#
|
||
# addMargin/reduceMargin
|
||
#
|
||
# {
|
||
# "code": "00000",
|
||
# "msg": "success",
|
||
# "requestTime": 1700813444618,
|
||
# "data": ""
|
||
# }
|
||
#
|
||
errorCode = self.safe_string(data, 'code')
|
||
status = 'ok' if (errorCode == '00000') else 'failed'
|
||
return {
|
||
'info': data,
|
||
'symbol': market['symbol'],
|
||
'type': None,
|
||
'marginMode': 'isolated',
|
||
'amount': None,
|
||
'total': None,
|
||
'code': market['settle'],
|
||
'status': status,
|
||
'timestamp': None,
|
||
'datetime': None,
|
||
}
|
||
|
||
async def reduce_margin(self, symbol: str, amount: float, params={}) -> MarginModification:
|
||
"""
|
||
remove margin from a position
|
||
|
||
https://www.bitget.com/api-doc/contract/account/Change-Margin
|
||
|
||
:param str symbol: unified market symbol
|
||
:param float amount: the amount of margin to remove
|
||
:param dict [params]: extra parameters specific to the exchange API endpoint
|
||
:returns dict: a `margin structure <https://docs.ccxt.com/#/?id=reduce-margin-structure>`
|
||
"""
|
||
if amount > 0:
|
||
raise BadRequest(self.id + ' reduceMargin() amount parameter must be a negative value')
|
||
holdSide = self.safe_string(params, 'holdSide')
|
||
if holdSide is None:
|
||
raise ArgumentsRequired(self.id + ' reduceMargin() requires a holdSide parameter, either long or short')
|
||
return await self.modify_margin_helper(symbol, amount, 'reduce', params)
|
||
|
||
async def add_margin(self, symbol: str, amount: float, params={}) -> MarginModification:
|
||
"""
|
||
add margin
|
||
|
||
https://www.bitget.com/api-doc/contract/account/Change-Margin
|
||
|
||
:param str symbol: unified market symbol
|
||
:param float amount: the amount of margin to add
|
||
:param dict [params]: extra parameters specific to the exchange API endpoint
|
||
:returns dict: a `margin structure <https://docs.ccxt.com/#/?id=add-margin-structure>`
|
||
"""
|
||
holdSide = self.safe_string(params, 'holdSide')
|
||
if holdSide is None:
|
||
raise ArgumentsRequired(self.id + ' addMargin() requires a holdSide parameter, either long or short')
|
||
return await self.modify_margin_helper(symbol, amount, 'add', params)
|
||
|
||
async def fetch_leverage(self, symbol: str, params={}) -> Leverage:
|
||
"""
|
||
fetch the set leverage for a market
|
||
|
||
https://www.bitget.com/api-doc/contract/account/Get-Single-Account
|
||
|
||
:param str symbol: unified market symbol
|
||
:param dict [params]: extra parameters specific to the exchange API endpoint
|
||
:returns dict: a `leverage structure <https://docs.ccxt.com/#/?id=leverage-structure>`
|
||
"""
|
||
await self.load_markets()
|
||
market = self.market(symbol)
|
||
productType = None
|
||
productType, params = self.handle_product_type_and_params(market, params)
|
||
request: dict = {
|
||
'symbol': market['id'],
|
||
'marginCoin': market['settleId'],
|
||
'productType': productType,
|
||
}
|
||
response = await self.privateMixGetV2MixAccountAccount(self.extend(request, params))
|
||
#
|
||
# {
|
||
# "code": "00000",
|
||
# "msg": "success",
|
||
# "requestTime": 1709366911964,
|
||
# "data": {
|
||
# "marginCoin": "USDT",
|
||
# "locked": "0",
|
||
# "available": "0",
|
||
# "crossedMaxAvailable": "0",
|
||
# "isolatedMaxAvailable": "0",
|
||
# "maxTransferOut": "0",
|
||
# "accountEquity": "0",
|
||
# "usdtEquity": "0.000000009166",
|
||
# "btcEquity": "0",
|
||
# "crossedRiskRate": "0",
|
||
# "crossedMarginLeverage": 20,
|
||
# "isolatedLongLever": 20,
|
||
# "isolatedShortLever": 20,
|
||
# "marginMode": "crossed",
|
||
# "posMode": "hedge_mode",
|
||
# "unrealizedPL": "0",
|
||
# "coupon": "0",
|
||
# "crossedUnrealizedPL": "0",
|
||
# "isolatedUnrealizedPL": ""
|
||
# }
|
||
# }
|
||
#
|
||
data = self.safe_dict(response, 'data', {})
|
||
return self.parse_leverage(data, market)
|
||
|
||
def parse_leverage(self, leverage: dict, market: Market = None) -> Leverage:
|
||
isCrossMarginMode = self.safe_string(leverage, 'marginMode') == 'crossed'
|
||
longLevKey = 'crossedMarginLeverage' if isCrossMarginMode else 'isolatedLongLever'
|
||
shortLevKey = 'crossedMarginLeverage' if isCrossMarginMode else 'isolatedShortLever'
|
||
return {
|
||
'info': leverage,
|
||
'symbol': market['symbol'],
|
||
'marginMode': 'cross' if isCrossMarginMode else 'isolated',
|
||
'longLeverage': self.safe_integer(leverage, longLevKey),
|
||
'shortLeverage': self.safe_integer(leverage, shortLevKey),
|
||
}
|
||
|
||
async def set_leverage(self, leverage: int, symbol: Str = None, params={}):
|
||
"""
|
||
set the level of leverage for a market
|
||
|
||
https://www.bitget.com/api-doc/contract/account/Change-Leverage
|
||
https://www.bitget.com/api-doc/uta/account/Change-Leverage
|
||
|
||
:param int leverage: the rate of leverage
|
||
:param str symbol: unified market symbol
|
||
:param dict [params]: extra parameters specific to the exchange API endpoint
|
||
:param str [params.holdSide]: *isolated only* position direction, 'long' or 'short'
|
||
:param boolean [params.uta]: set to True for the unified trading account(uta), defaults to False
|
||
:param boolean [params.posSide]: required for uta isolated margin, long or short
|
||
:returns dict: response from the exchange
|
||
"""
|
||
if symbol is None:
|
||
raise ArgumentsRequired(self.id + ' setLeverage() requires a symbol argument')
|
||
await self.load_markets()
|
||
market = self.market(symbol)
|
||
productType = None
|
||
productType, params = self.handle_product_type_and_params(market, params)
|
||
request: dict = {
|
||
'symbol': market['id'],
|
||
'leverage': self.number_to_string(leverage),
|
||
}
|
||
uta = None
|
||
response = None
|
||
uta, params = self.handle_option_and_params(params, 'setLeverage', 'uta', False)
|
||
if uta:
|
||
if productType == 'SPOT':
|
||
marginMode = None
|
||
marginMode, params = self.handle_margin_mode_and_params('fetchTrades', params)
|
||
if marginMode is not None:
|
||
productType = 'MARGIN'
|
||
request['coin'] = market['settleId']
|
||
request['category'] = productType
|
||
response = await self.privateUtaPostV3AccountSetLeverage(self.extend(request, params))
|
||
#
|
||
# {
|
||
# "code": "00000",
|
||
# "msg": "success",
|
||
# "requestTime": 1752815940833,
|
||
# "data": "success"
|
||
# }
|
||
#
|
||
else:
|
||
request['marginCoin'] = market['settleId']
|
||
request['productType'] = productType
|
||
response = await self.privateMixPostV2MixAccountSetLeverage(self.extend(request, params))
|
||
#
|
||
# {
|
||
# "code": "00000",
|
||
# "msg": "success",
|
||
# "requestTime": 1700864711517,
|
||
# "data": {
|
||
# "symbol": "BTCUSDT",
|
||
# "marginCoin": "USDT",
|
||
# "longLeverage": "25",
|
||
# "shortLeverage": "25",
|
||
# "crossMarginLeverage": "25",
|
||
# "marginMode": "crossed"
|
||
# }
|
||
# }
|
||
#
|
||
return response
|
||
|
||
async def set_margin_mode(self, marginMode: str, symbol: Str = None, params={}):
|
||
"""
|
||
set margin mode to 'cross' or 'isolated'
|
||
|
||
https://www.bitget.com/api-doc/contract/account/Change-Margin-Mode
|
||
|
||
:param str marginMode: 'cross' or 'isolated'
|
||
:param str symbol: unified market symbol
|
||
:param dict [params]: extra parameters specific to the exchange API endpoint
|
||
:returns dict: response from the exchange
|
||
"""
|
||
if symbol is None:
|
||
raise ArgumentsRequired(self.id + ' setMarginMode() requires a symbol argument')
|
||
marginMode = marginMode.lower()
|
||
if marginMode == 'cross':
|
||
marginMode = 'crossed'
|
||
if (marginMode != 'isolated') and (marginMode != 'crossed'):
|
||
raise ArgumentsRequired(self.id + ' setMarginMode() marginMode must be either isolated or crossed(cross)')
|
||
await self.load_markets()
|
||
market = self.market(symbol)
|
||
productType = None
|
||
productType, params = self.handle_product_type_and_params(market, params)
|
||
request: dict = {
|
||
'symbol': market['id'],
|
||
'marginCoin': market['settleId'],
|
||
'marginMode': marginMode,
|
||
'productType': productType,
|
||
}
|
||
response = await self.privateMixPostV2MixAccountSetMarginMode(self.extend(request, params))
|
||
#
|
||
# {
|
||
# "code": "00000",
|
||
# "msg": "success",
|
||
# "requestTime": 1700865205552,
|
||
# "data": {
|
||
# "symbol": "BTCUSDT",
|
||
# "marginCoin": "USDT",
|
||
# "longLeverage": "20",
|
||
# "shortLeverage": "3",
|
||
# "marginMode": "isolated"
|
||
# }
|
||
# }
|
||
#
|
||
return response
|
||
|
||
async def set_position_mode(self, hedged: bool, symbol: Str = None, params={}):
|
||
"""
|
||
set hedged to True or False for a market
|
||
|
||
https://www.bitget.com/api-doc/contract/account/Change-Hold-Mode
|
||
https://www.bitget.com/api-doc/uta/account/Change-Position-Mode
|
||
|
||
:param bool hedged: set to True to use dualSidePosition
|
||
:param str symbol: not used by bitget setPositionMode()
|
||
:param dict [params]: extra parameters specific to the exchange API endpoint
|
||
:param str [params.productType]: required if not uta and symbol is None: 'USDT-FUTURES', 'USDC-FUTURES', 'COIN-FUTURES', 'SUSDT-FUTURES', 'SUSDC-FUTURES' or 'SCOIN-FUTURES'
|
||
:param boolean [params.uta]: set to True for the unified trading account(uta), defaults to False
|
||
:returns dict: response from the exchange
|
||
"""
|
||
await self.load_markets()
|
||
posMode = 'hedge_mode' if hedged else 'one_way_mode'
|
||
request: dict = {}
|
||
market = None
|
||
if symbol is not None:
|
||
market = self.market(symbol)
|
||
productType = None
|
||
uta = None
|
||
response = None
|
||
productType, params = self.handle_product_type_and_params(market, params)
|
||
uta, params = self.handle_option_and_params(params, 'setPositionMode', 'uta', False)
|
||
if uta:
|
||
request['holdMode'] = posMode
|
||
response = await self.privateUtaPostV3AccountSetHoldMode(self.extend(request, params))
|
||
#
|
||
# {
|
||
# "code": "00000",
|
||
# "msg": "success",
|
||
# "requestTime": 1752816734592,
|
||
# "data": "success"
|
||
# }
|
||
#
|
||
else:
|
||
request['posMode'] = posMode
|
||
request['productType'] = productType
|
||
response = await self.privateMixPostV2MixAccountSetPositionMode(self.extend(request, params))
|
||
#
|
||
# {
|
||
# "code": "00000",
|
||
# "msg": "success",
|
||
# "requestTime": 1700865608009,
|
||
# "data": {
|
||
# "posMode": "hedge_mode"
|
||
# }
|
||
# }
|
||
#
|
||
return response
|
||
|
||
async def fetch_open_interest(self, symbol: str, params={}):
|
||
"""
|
||
retrieves the open interest of a contract trading pair
|
||
|
||
https://www.bitget.com/api-doc/contract/market/Get-Open-Interest
|
||
https://www.bitget.com/api-doc/uta/public/Get-Open-Interest
|
||
|
||
:param str symbol: unified CCXT market symbol
|
||
:param dict [params]: exchange specific parameters
|
||
:param boolean [params.uta]: set to True for the unified trading account(uta), defaults to False
|
||
:returns dict} an open interest structure{@link https://docs.ccxt.com/#/?id=open-interest-structure:
|
||
"""
|
||
await self.load_markets()
|
||
market = self.market(symbol)
|
||
if not market['contract']:
|
||
raise BadRequest(self.id + ' fetchOpenInterest() supports contract markets only')
|
||
productType = None
|
||
productType, params = self.handle_product_type_and_params(market, params)
|
||
request: dict = {
|
||
'symbol': market['id'],
|
||
}
|
||
uta = None
|
||
response = None
|
||
uta, params = self.handle_option_and_params(params, 'fetchOpenInterest', 'uta', False)
|
||
if uta:
|
||
request['category'] = productType
|
||
response = await self.publicUtaGetV3MarketOpenInterest(self.extend(request, params))
|
||
#
|
||
# {
|
||
# "code": "00000",
|
||
# "msg": "success",
|
||
# "requestTime": 1751101221545,
|
||
# "data": {
|
||
# "list": [
|
||
# {
|
||
# "symbol": "BTCUSDT",
|
||
# "openInterest": "18166.3583"
|
||
# }
|
||
# ],
|
||
# "ts": "1751101220993"
|
||
# }
|
||
# }
|
||
#
|
||
else:
|
||
request['productType'] = productType
|
||
response = await self.publicMixGetV2MixMarketOpenInterest(self.extend(request, params))
|
||
#
|
||
# {
|
||
# "code": "00000",
|
||
# "msg": "success",
|
||
# "requestTime": 1700866041022,
|
||
# "data": {
|
||
# "openInterestList": [
|
||
# {
|
||
# "symbol": "BTCUSDT",
|
||
# "size": "52234.134"
|
||
# }
|
||
# ],
|
||
# "ts": "1700866041023"
|
||
# }
|
||
# }
|
||
#
|
||
data = self.safe_dict(response, 'data', {})
|
||
return self.parse_open_interest(data, market)
|
||
|
||
def parse_open_interest(self, interest, market: Market = None):
|
||
#
|
||
# default
|
||
#
|
||
# {
|
||
# "openInterestList": [
|
||
# {
|
||
# "symbol": "BTCUSDT",
|
||
# "size": "52234.134"
|
||
# }
|
||
# ],
|
||
# "ts": "1700866041023"
|
||
# }
|
||
#
|
||
# uta
|
||
#
|
||
# {
|
||
# "list": [
|
||
# {
|
||
# "symbol": "BTCUSDT",
|
||
# "openInterest": "18166.3583"
|
||
# }
|
||
# ],
|
||
# "ts": "1751101220993"
|
||
# }
|
||
#
|
||
data = self.safe_list_2(interest, 'openInterestList', 'list', [])
|
||
timestamp = self.safe_integer(interest, 'ts')
|
||
marketId = self.safe_string(data[0], 'symbol')
|
||
return self.safe_open_interest({
|
||
'symbol': self.safe_symbol(marketId, market, None, 'contract'),
|
||
'openInterestAmount': self.safe_number_2(data[0], 'size', 'openInterest'),
|
||
'openInterestValue': None,
|
||
'timestamp': timestamp,
|
||
'datetime': self.iso8601(timestamp),
|
||
'info': interest,
|
||
}, market)
|
||
|
||
async def fetch_transfers(self, code: Str = None, since: Int = None, limit: Int = None, params={}) -> List[TransferEntry]:
|
||
"""
|
||
fetch a history of internal transfers made on an account
|
||
|
||
https://www.bitget.com/api-doc/spot/account/Get-Account-TransferRecords
|
||
|
||
:param str code: unified currency code of the currency transferred
|
||
:param int [since]: the earliest time in ms to fetch transfers for
|
||
:param int [limit]: the maximum number of transfers structures to retrieve
|
||
:param dict [params]: extra parameters specific to the exchange API endpoint
|
||
:param int [params.until]: the latest time in ms to fetch entries for
|
||
:returns dict[]: a list of `transfer structures <https://docs.ccxt.com/#/?id=transfer-structure>`
|
||
"""
|
||
if code is None:
|
||
raise ArgumentsRequired(self.id + ' fetchTransfers() requires a code argument')
|
||
await self.load_markets()
|
||
type = None
|
||
type, params = self.handle_market_type_and_params('fetchTransfers', None, params)
|
||
fromAccount = self.safe_string(params, 'fromAccount', type)
|
||
params = self.omit(params, 'fromAccount')
|
||
accountsByType = self.safe_value(self.options, 'accountsByType', {})
|
||
type = self.safe_string(accountsByType, fromAccount)
|
||
currency = self.currency(code)
|
||
request: dict = {
|
||
'coin': currency['id'],
|
||
'fromType': type,
|
||
}
|
||
if since is not None:
|
||
request['startTime'] = since
|
||
if limit is not None:
|
||
request['limit'] = limit
|
||
request, params = self.handle_until_option('endTime', request, params)
|
||
response = await self.privateSpotGetV2SpotAccountTransferRecords(self.extend(request, params))
|
||
#
|
||
# {
|
||
# "code": "00000",
|
||
# "msg": "success",
|
||
# "requestTime": 1700873854651,
|
||
# "data": [
|
||
# {
|
||
# "coin": "USDT",
|
||
# "status": "Successful",
|
||
# "toType": "crossed_margin",
|
||
# "toSymbol": "",
|
||
# "fromType": "spot",
|
||
# "fromSymbol": "",
|
||
# "size": "11.64958799",
|
||
# "ts": "1700729673028",
|
||
# "clientOid": "1111506298504744960",
|
||
# "transferId": "24930940"
|
||
# },
|
||
# ]
|
||
# }
|
||
#
|
||
data = self.safe_list(response, 'data', [])
|
||
return self.parse_transfers(data, currency, since, limit)
|
||
|
||
async def transfer(self, code: str, amount: float, fromAccount: str, toAccount: str, params={}) -> TransferEntry:
|
||
"""
|
||
transfer currency internally between wallets on the same account
|
||
|
||
https://www.bitget.com/api-doc/spot/account/Wallet-Transfer
|
||
|
||
:param str code: unified currency code
|
||
:param float amount: amount to transfer
|
||
:param str fromAccount: account to transfer from
|
||
:param str toAccount: account to transfer to
|
||
:param dict [params]: extra parameters specific to the exchange API endpoint
|
||
:param str [params.symbol]: unified CCXT market symbol, required when transferring to or from an account type that is a leveraged position-by-position account
|
||
:param str [params.clientOid]: custom id
|
||
:returns dict: a `transfer structure <https://docs.ccxt.com/#/?id=transfer-structure>`
|
||
"""
|
||
await self.load_markets()
|
||
currency = self.currency(code)
|
||
accountsByType = self.safe_value(self.options, 'accountsByType', {})
|
||
fromType = self.safe_string(accountsByType, fromAccount)
|
||
toType = self.safe_string(accountsByType, toAccount)
|
||
request: dict = {
|
||
'fromType': fromType,
|
||
'toType': toType,
|
||
'amount': amount,
|
||
'coin': currency['id'],
|
||
}
|
||
symbol = self.safe_string(params, 'symbol')
|
||
params = self.omit(params, 'symbol')
|
||
market = None
|
||
if symbol is not None:
|
||
market = self.market(symbol)
|
||
request['symbol'] = market['id']
|
||
response = await self.privateSpotPostV2SpotWalletTransfer(self.extend(request, params))
|
||
#
|
||
# {
|
||
# "code": "00000",
|
||
# "msg": "success",
|
||
# "requestTime": 1700874302021,
|
||
# "data": {
|
||
# "transferId": "1112112916581847040",
|
||
# "clientOrderId": null
|
||
# }
|
||
# }
|
||
#
|
||
data = self.safe_value(response, 'data', {})
|
||
data['ts'] = self.safe_integer(response, 'requestTime')
|
||
return self.parse_transfer(data, currency)
|
||
|
||
def parse_transfer(self, transfer: dict, currency: Currency = None) -> TransferEntry:
|
||
#
|
||
# transfer
|
||
#
|
||
# {
|
||
# "transferId": "1112112916581847040",
|
||
# "clientOrderId": null,
|
||
# "ts": 1700874302021
|
||
# }
|
||
#
|
||
# fetchTransfers
|
||
#
|
||
# {
|
||
# "coin": "USDT",
|
||
# "status": "Successful",
|
||
# "toType": "crossed_margin",
|
||
# "toSymbol": "",
|
||
# "fromType": "spot",
|
||
# "fromSymbol": "",
|
||
# "size": "11.64958799",
|
||
# "ts": "1700729673028",
|
||
# "clientOid": "1111506298504744960",
|
||
# "transferId": "24930940"
|
||
# }
|
||
#
|
||
timestamp = self.safe_integer(transfer, 'ts')
|
||
status = self.safe_string_lower(transfer, 'status')
|
||
currencyId = self.safe_string(transfer, 'coin')
|
||
fromAccountRaw = self.safe_string(transfer, 'fromType')
|
||
accountsById = self.safe_value(self.options, 'accountsById', {})
|
||
fromAccount = self.safe_string(accountsById, fromAccountRaw, fromAccountRaw)
|
||
toAccountRaw = self.safe_string(transfer, 'toType')
|
||
toAccount = self.safe_string(accountsById, toAccountRaw, toAccountRaw)
|
||
return {
|
||
'info': transfer,
|
||
'id': self.safe_string(transfer, 'transferId'),
|
||
'timestamp': timestamp,
|
||
'datetime': self.iso8601(timestamp),
|
||
'currency': self.safe_currency_code(currencyId, currency),
|
||
'amount': self.safe_number(transfer, 'size'),
|
||
'fromAccount': fromAccount,
|
||
'toAccount': toAccount,
|
||
'status': self.parse_transfer_status(status),
|
||
}
|
||
|
||
def parse_transfer_status(self, status: Str) -> Str:
|
||
statuses: dict = {
|
||
'successful': 'ok',
|
||
}
|
||
return self.safe_string(statuses, status, status)
|
||
|
||
def parse_deposit_withdraw_fee(self, fee, currency: Currency = None):
|
||
#
|
||
# {
|
||
# "chains": [
|
||
# {
|
||
# "browserUrl": "https://blockchair.com/bitcoin/transaction/",
|
||
# "chain": "BTC",
|
||
# "depositConfirm": "1",
|
||
# "extraWithdrawFee": "0",
|
||
# "minDepositAmount": "0.0001",
|
||
# "minWithdrawAmount": "0.005",
|
||
# "needTag": "false",
|
||
# "rechargeable": "true",
|
||
# "withdrawConfirm": "1",
|
||
# "withdrawFee": "0.0004",
|
||
# "withdrawable": "true"
|
||
# },
|
||
# ],
|
||
# "coin": "BTC",
|
||
# "coinId": "1",
|
||
# "transfer": "true""
|
||
# }
|
||
#
|
||
chains = self.safe_value(fee, 'chains', [])
|
||
chainsLength = len(chains)
|
||
result: dict = {
|
||
'info': fee,
|
||
'withdraw': {
|
||
'fee': None,
|
||
'percentage': None,
|
||
},
|
||
'deposit': {
|
||
'fee': None,
|
||
'percentage': None,
|
||
},
|
||
'networks': {},
|
||
}
|
||
for i in range(0, chainsLength):
|
||
chain = chains[i]
|
||
networkId = self.safe_string(chain, 'chain')
|
||
currencyCode = self.safe_string(currency, 'code')
|
||
networkCode = self.network_id_to_code(networkId, currencyCode)
|
||
result['networks'][networkCode] = {
|
||
'deposit': {'fee': None, 'percentage': None},
|
||
'withdraw': {'fee': self.safe_number(chain, 'withdrawFee'), 'percentage': False},
|
||
}
|
||
if chainsLength == 1:
|
||
result['withdraw']['fee'] = self.safe_number(chain, 'withdrawFee')
|
||
result['withdraw']['percentage'] = False
|
||
return result
|
||
|
||
async def fetch_deposit_withdraw_fees(self, codes: Strings = None, params={}):
|
||
"""
|
||
fetch deposit and withdraw fees
|
||
|
||
https://www.bitget.com/api-doc/spot/market/Get-Coin-List
|
||
|
||
:param str[]|None codes: list of unified currency codes
|
||
:param dict [params]: extra parameters specific to the exchange API endpoint
|
||
:returns dict: a list of `fee structures <https://docs.ccxt.com/#/?id=fee-structure>`
|
||
"""
|
||
await self.load_markets()
|
||
response = await self.publicSpotGetV2SpotPublicCoins(params)
|
||
#
|
||
# {
|
||
# "code": "00000",
|
||
# "data": [
|
||
# {
|
||
# "chains": [
|
||
# {
|
||
# "browserUrl": "https://blockchair.com/bitcoin/transaction/",
|
||
# "chain": "BTC",
|
||
# "depositConfirm": "1",
|
||
# "extraWithdrawFee": "0",
|
||
# "minDepositAmount": "0.0001",
|
||
# "minWithdrawAmount": "0.005",
|
||
# "needTag": "false",
|
||
# "rechargeable": "true",
|
||
# "withdrawConfirm": "1",
|
||
# "withdrawFee": "0.0004",
|
||
# "withdrawable": "true"
|
||
# },
|
||
# ],
|
||
# "coin": "BTC",
|
||
# "coinId": "1",
|
||
# "transfer": "true""
|
||
# }
|
||
# ],
|
||
# "msg": "success",
|
||
# "requestTime": "1700120731773"
|
||
# }
|
||
#
|
||
data = self.safe_list(response, 'data', [])
|
||
return self.parse_deposit_withdraw_fees(data, codes, 'coin')
|
||
|
||
async def borrow_cross_margin(self, code: str, amount: float, params={}):
|
||
"""
|
||
create a loan to borrow margin
|
||
|
||
https://www.bitget.com/api-doc/margin/cross/account/Cross-Borrow
|
||
|
||
:param str code: unified currency code of the currency to borrow
|
||
:param str amount: the amount to borrow
|
||
:param dict [params]: extra parameters specific to the exchange API endpoint
|
||
:returns dict: a `margin loan structure <https://docs.ccxt.com/#/?id=margin-loan-structure>`
|
||
"""
|
||
await self.load_markets()
|
||
currency = self.currency(code)
|
||
request: dict = {
|
||
'coin': currency['id'],
|
||
'borrowAmount': self.currency_to_precision(code, amount),
|
||
}
|
||
response = await self.privateMarginPostV2MarginCrossedAccountBorrow(self.extend(request, params))
|
||
#
|
||
# {
|
||
# "code": "00000",
|
||
# "msg": "success",
|
||
# "requestTime": 1700876470931,
|
||
# "data": {
|
||
# "loanId": "1112122013642272769",
|
||
# "coin": "USDT",
|
||
# "borrowAmount": "4"
|
||
# }
|
||
# }
|
||
#
|
||
data = self.safe_value(response, 'data', {})
|
||
return self.parse_margin_loan(data, currency)
|
||
|
||
async def borrow_isolated_margin(self, symbol: str, code: str, amount: float, params={}):
|
||
"""
|
||
create a loan to borrow margin
|
||
|
||
https://www.bitget.com/api-doc/margin/isolated/account/Isolated-Borrow
|
||
|
||
:param str symbol: unified market symbol
|
||
:param str code: unified currency code of the currency to borrow
|
||
:param str amount: the amount to borrow
|
||
:param dict [params]: extra parameters specific to the exchange API endpoint
|
||
:returns dict: a `margin loan structure <https://docs.ccxt.com/#/?id=margin-loan-structure>`
|
||
"""
|
||
await self.load_markets()
|
||
currency = self.currency(code)
|
||
market = self.market(symbol)
|
||
request: dict = {
|
||
'coin': currency['id'],
|
||
'borrowAmount': self.currency_to_precision(code, amount),
|
||
'symbol': market['id'],
|
||
}
|
||
response = await self.privateMarginPostV2MarginIsolatedAccountBorrow(self.extend(request, params))
|
||
#
|
||
# {
|
||
# "code": "00000",
|
||
# "msg": "success",
|
||
# "requestTime": 1700877255605,
|
||
# "data": {
|
||
# "loanId": "1112125304879067137",
|
||
# "symbol": "BTCUSDT",
|
||
# "coin": "USDT",
|
||
# "borrowAmount": "4"
|
||
# }
|
||
# }
|
||
#
|
||
data = self.safe_value(response, 'data', {})
|
||
return self.parse_margin_loan(data, currency, market)
|
||
|
||
async def repay_isolated_margin(self, symbol: str, code: str, amount, params={}):
|
||
"""
|
||
repay borrowed margin and interest
|
||
|
||
https://www.bitget.com/api-doc/margin/isolated/account/Isolated-Repay
|
||
|
||
:param str symbol: unified market symbol
|
||
:param str code: unified currency code of the currency to repay
|
||
:param str amount: the amount to repay
|
||
:param dict [params]: extra parameters specific to the exchange API endpoint
|
||
:returns dict: a `margin loan structure <https://docs.ccxt.com/#/?id=margin-loan-structure>`
|
||
"""
|
||
await self.load_markets()
|
||
currency = self.currency(code)
|
||
market = self.market(symbol)
|
||
request: dict = {
|
||
'coin': currency['id'],
|
||
'repayAmount': self.currency_to_precision(code, amount),
|
||
'symbol': market['id'],
|
||
}
|
||
response = await self.privateMarginPostV2MarginIsolatedAccountRepay(self.extend(request, params))
|
||
#
|
||
# {
|
||
# "code": "00000",
|
||
# "msg": "success",
|
||
# "requestTime": 1700877518012,
|
||
# "data": {
|
||
# "remainDebtAmount": "0",
|
||
# "repayId": "1112126405439270912",
|
||
# "symbol": "BTCUSDT",
|
||
# "coin": "USDT",
|
||
# "repayAmount": "8.000137"
|
||
# }
|
||
# }
|
||
#
|
||
data = self.safe_value(response, 'data', {})
|
||
return self.parse_margin_loan(data, currency, market)
|
||
|
||
async def repay_cross_margin(self, code: str, amount, params={}):
|
||
"""
|
||
repay borrowed margin and interest
|
||
|
||
https://www.bitget.com/api-doc/margin/cross/account/Cross-Repay
|
||
|
||
:param str code: unified currency code of the currency to repay
|
||
:param str amount: the amount to repay
|
||
:param dict [params]: extra parameters specific to the exchange API endpoint
|
||
:returns dict: a `margin loan structure <https://docs.ccxt.com/#/?id=margin-loan-structure>`
|
||
"""
|
||
await self.load_markets()
|
||
currency = self.currency(code)
|
||
request: dict = {
|
||
'coin': currency['id'],
|
||
'repayAmount': self.currency_to_precision(code, amount),
|
||
}
|
||
response = await self.privateMarginPostV2MarginCrossedAccountRepay(self.extend(request, params))
|
||
#
|
||
# {
|
||
# "code": "00000",
|
||
# "msg": "success",
|
||
# "requestTime": 1700876704885,
|
||
# "data": {
|
||
# "remainDebtAmount": "0",
|
||
# "repayId": "1112122994945830912",
|
||
# "coin": "USDT",
|
||
# "repayAmount": "4.00006834"
|
||
# }
|
||
# }
|
||
#
|
||
data = self.safe_value(response, 'data', {})
|
||
return self.parse_margin_loan(data, currency)
|
||
|
||
def parse_margin_loan(self, info, currency: Currency = None, market: Market = None):
|
||
#
|
||
# isolated: borrowMargin
|
||
#
|
||
# {
|
||
# "loanId": "1112125304879067137",
|
||
# "symbol": "BTCUSDT",
|
||
# "coin": "USDT",
|
||
# "borrowAmount": "4"
|
||
# }
|
||
#
|
||
# cross: borrowMargin
|
||
#
|
||
# {
|
||
# "loanId": "1112122013642272769",
|
||
# "coin": "USDT",
|
||
# "borrowAmount": "4"
|
||
# }
|
||
#
|
||
# isolated: repayMargin
|
||
#
|
||
# {
|
||
# "remainDebtAmount": "0",
|
||
# "repayId": "1112126405439270912",
|
||
# "symbol": "BTCUSDT",
|
||
# "coin": "USDT",
|
||
# "repayAmount": "8.000137"
|
||
# }
|
||
#
|
||
# cross: repayMargin
|
||
#
|
||
# {
|
||
# "remainDebtAmount": "0",
|
||
# "repayId": "1112122994945830912",
|
||
# "coin": "USDT",
|
||
# "repayAmount": "4.00006834"
|
||
# }
|
||
#
|
||
currencyId = self.safe_string(info, 'coin')
|
||
marketId = self.safe_string(info, 'symbol')
|
||
symbol = None
|
||
if marketId is not None:
|
||
symbol = self.safe_symbol(marketId, market, None, 'spot')
|
||
return {
|
||
'id': self.safe_string_2(info, 'loanId', 'repayId'),
|
||
'currency': self.safe_currency_code(currencyId, currency),
|
||
'amount': self.safe_number_2(info, 'borrowAmount', 'repayAmount'),
|
||
'symbol': symbol,
|
||
'timestamp': None,
|
||
'datetime': None,
|
||
'info': info,
|
||
}
|
||
|
||
async def fetch_my_liquidations(self, symbol: Str = None, since: Int = None, limit: Int = None, params={}) -> List[Liquidation]:
|
||
"""
|
||
retrieves the users liquidated positions
|
||
|
||
https://www.bitget.com/api-doc/margin/cross/record/Get-Cross-Liquidation-Records
|
||
https://www.bitget.com/api-doc/margin/isolated/record/Get-Isolated-Liquidation-Records
|
||
|
||
:param str [symbol]: unified CCXT market symbol
|
||
:param int [since]: the earliest time in ms to fetch liquidations for
|
||
:param int [limit]: the maximum number of liquidation structures to retrieve
|
||
:param dict [params]: exchange specific parameters for the bitget api endpoint
|
||
:param int [params.until]: timestamp in ms of the latest liquidation
|
||
:param str [params.marginMode]: 'cross' or 'isolated' default value is 'cross'
|
||
:param boolean [params.paginate]: default False, when True will automatically paginate by calling self endpoint multiple times. See in the docs all the [available parameters](https://github.com/ccxt/ccxt/wiki/Manual#pagination-params)
|
||
:returns dict: an array of `liquidation structures <https://docs.ccxt.com/#/?id=liquidation-structure>`
|
||
"""
|
||
await self.load_markets()
|
||
paginate = False
|
||
paginate, params = self.handle_option_and_params(params, 'fetchMyLiquidations', 'paginate')
|
||
if paginate:
|
||
return await self.fetch_paginated_call_cursor('fetchMyLiquidations', symbol, since, limit, params, 'minId', 'idLessThan')
|
||
market = None
|
||
if symbol is not None:
|
||
market = self.market(symbol)
|
||
type = None
|
||
type, params = self.handle_market_type_and_params('fetchMyLiquidations', market, params)
|
||
if type != 'spot':
|
||
raise NotSupported(self.id + ' fetchMyLiquidations() supports spot margin markets only')
|
||
request: dict = {}
|
||
request, params = self.handle_until_option('endTime', request, params)
|
||
if since is not None:
|
||
request['startTime'] = since
|
||
else:
|
||
request['startTime'] = self.milliseconds() - 7776000000
|
||
if limit is not None:
|
||
request['limit'] = limit
|
||
response = None
|
||
marginMode = None
|
||
marginMode, params = self.handle_margin_mode_and_params('fetchMyLiquidations', params, 'cross')
|
||
if marginMode == 'isolated':
|
||
if symbol is None:
|
||
raise ArgumentsRequired(self.id + ' fetchMyLiquidations() requires a symbol argument')
|
||
request['symbol'] = market['id']
|
||
response = await self.privateMarginGetV2MarginIsolatedLiquidationHistory(self.extend(request, params))
|
||
elif marginMode == 'cross':
|
||
response = await self.privateMarginGetV2MarginCrossedLiquidationHistory(self.extend(request, params))
|
||
#
|
||
# isolated
|
||
#
|
||
# {
|
||
# "code": "00000",
|
||
# "msg": "success",
|
||
# "requestTime": 1698114119193,
|
||
# "data": {
|
||
# "resultList": [
|
||
# {
|
||
# "liqId": "123",
|
||
# "symbol": "BTCUSDT",
|
||
# "liqStartTime": "1653453245342",
|
||
# "liqEndTime": "16312423423432",
|
||
# "liqRiskRatio": "1.01",
|
||
# "totalAssets": "1242.34",
|
||
# "totalDebt": "1100",
|
||
# "liqFee": "1.2",
|
||
# "uTime": "1668134458717",
|
||
# "cTime": "1653453245342"
|
||
# }
|
||
# ],
|
||
# "maxId": "0",
|
||
# "minId": "0"
|
||
# }
|
||
# }
|
||
#
|
||
# cross
|
||
#
|
||
# {
|
||
# "code": "00000",
|
||
# "msg": "success",
|
||
# "requestTime": 1698114119193,
|
||
# "data": {
|
||
# "resultList": [
|
||
# {
|
||
# "liqId": "123",
|
||
# "liqStartTime": "1653453245342",
|
||
# "liqEndTime": "16312423423432",
|
||
# "liqRiskRatio": "1.01",
|
||
# "totalAssets": "1242.34",
|
||
# "totalDebt": "1100",
|
||
# "LiqFee": "1.2",
|
||
# "uTime": "1668134458717",
|
||
# "cTime": "1653453245342"
|
||
# }
|
||
# ],
|
||
# "maxId": "0",
|
||
# "minId": "0"
|
||
# }
|
||
# }
|
||
#
|
||
data = self.safe_value(response, 'data', {})
|
||
liquidations = self.safe_list(data, 'resultList', [])
|
||
return self.parse_liquidations(liquidations, market, since, limit)
|
||
|
||
def parse_liquidation(self, liquidation, market: Market = None):
|
||
#
|
||
# isolated
|
||
#
|
||
# {
|
||
# "liqId": "123",
|
||
# "symbol": "BTCUSDT",
|
||
# "liqStartTime": "1653453245342",
|
||
# "liqEndTime": "16312423423432",
|
||
# "liqRiskRatio": "1.01",
|
||
# "totalAssets": "1242.34",
|
||
# "totalDebt": "1100",
|
||
# "liqFee": "1.2",
|
||
# "uTime": "1692690126000"
|
||
# "cTime": "1653453245342"
|
||
# }
|
||
#
|
||
# cross
|
||
#
|
||
# {
|
||
# "liqId": "123",
|
||
# "liqStartTime": "1653453245342",
|
||
# "liqEndTime": "16312423423432",
|
||
# "liqRiskRatio": "1.01",
|
||
# "totalAssets": "1242.34",
|
||
# "totalDebt": "1100",
|
||
# "LiqFee": "1.2",
|
||
# "uTime": "1692690126000"
|
||
# "cTime": "1653453245342"
|
||
# }
|
||
#
|
||
marketId = self.safe_string(liquidation, 'symbol')
|
||
timestamp = self.safe_integer(liquidation, 'liqEndTime')
|
||
liquidationFee = self.safe_string_2(liquidation, 'LiqFee', 'liqFee')
|
||
totalDebt = self.safe_string(liquidation, 'totalDebt')
|
||
quoteValueString = Precise.string_add(liquidationFee, totalDebt)
|
||
return self.safe_liquidation({
|
||
'info': liquidation,
|
||
'symbol': self.safe_symbol(marketId, market),
|
||
'contracts': None,
|
||
'contractSize': None,
|
||
'price': None,
|
||
'baseValue': None,
|
||
'quoteValue': self.parse_number(quoteValueString),
|
||
'timestamp': timestamp,
|
||
'datetime': self.iso8601(timestamp),
|
||
})
|
||
|
||
async def fetch_isolated_borrow_rate(self, symbol: str, params={}) -> IsolatedBorrowRate:
|
||
"""
|
||
fetch the rate of interest to borrow a currency for margin trading
|
||
|
||
https://www.bitget.com/api-doc/margin/isolated/account/Isolated-Margin-Interest-Rate-And-Max-Borrowable-Amount
|
||
|
||
:param str symbol: unified market symbol
|
||
:param dict [params]: extra parameters specific to the exchange API endpoint
|
||
:returns dict: an `isolated borrow rate structure <https://docs.ccxt.com/#/?id=isolated-borrow-rate-structure>`
|
||
"""
|
||
await self.load_markets()
|
||
market = self.market(symbol)
|
||
request: dict = {
|
||
'symbol': market['id'],
|
||
}
|
||
response = await self.privateMarginGetV2MarginIsolatedInterestRateAndLimit(self.extend(request, params))
|
||
#
|
||
# {
|
||
# "code": "00000",
|
||
# "msg": "success",
|
||
# "requestTime": 1700878692567,
|
||
# "data": [
|
||
# {
|
||
# "symbol": "BTCUSDT",
|
||
# "leverage": "10",
|
||
# "baseCoin": "BTC",
|
||
# "baseTransferable": True,
|
||
# "baseBorrowable": True,
|
||
# "baseDailyInterestRate": "0.00007",
|
||
# "baseAnnuallyInterestRate": "0.02555",
|
||
# "baseMaxBorrowableAmount": "27",
|
||
# "baseVipList": [
|
||
# {"level":"0","dailyInterestRate":"0.00007","limit":"27","annuallyInterestRate":"0.02555","discountRate":"1"},
|
||
# {"level":"1","dailyInterestRate":"0.0000679","limit":"27.81","annuallyInterestRate":"0.0247835","discountRate":"0.97"},
|
||
# {"level":"2","dailyInterestRate":"0.0000644","limit":"29.16","annuallyInterestRate":"0.023506","discountRate":"0.92"},
|
||
# {"level":"3","dailyInterestRate":"0.0000602","limit":"31.32","annuallyInterestRate":"0.021973","discountRate":"0.86"},
|
||
# {"level":"4","dailyInterestRate":"0.0000525","limit":"35.91","annuallyInterestRate":"0.0191625","discountRate":"0.75"},
|
||
# {"level":"5","dailyInterestRate":"0.000042","limit":"44.82","annuallyInterestRate":"0.01533","discountRate":"0.6"}
|
||
# ],
|
||
# "quoteCoin": "USDT",
|
||
# "quoteTransferable": True,
|
||
# "quoteBorrowable": True,
|
||
# "quoteDailyInterestRate": "0.00041095",
|
||
# "quoteAnnuallyInterestRate": "0.14999675",
|
||
# "quoteMaxBorrowableAmount": "300000",
|
||
# "quoteList": [
|
||
# {"level":"0","dailyInterestRate":"0.00041095","limit":"300000","annuallyInterestRate":"0.14999675","discountRate":"1"},
|
||
# {"level":"1","dailyInterestRate":"0.00039863","limit":"309000","annuallyInterestRate":"0.14549995","discountRate":"0.97"},
|
||
# {"level":"2","dailyInterestRate":"0.00037808","limit":"324000","annuallyInterestRate":"0.1379992","discountRate":"0.92"},
|
||
# {"level":"3","dailyInterestRate":"0.00035342","limit":"348000","annuallyInterestRate":"0.1289983","discountRate":"0.86"},
|
||
# {"level":"4","dailyInterestRate":"0.00030822","limit":"399000","annuallyInterestRate":"0.1125003","discountRate":"0.75"},
|
||
# {"level":"5","dailyInterestRate":"0.00024657","limit":"498000","annuallyInterestRate":"0.08999805","discountRate":"0.6"}
|
||
# ]
|
||
# }
|
||
# ]
|
||
# }
|
||
#
|
||
timestamp = self.safe_integer(response, 'requestTime')
|
||
data = self.safe_value(response, 'data', [])
|
||
first = self.safe_value(data, 0, {})
|
||
first['timestamp'] = timestamp
|
||
return self.parse_isolated_borrow_rate(first, market)
|
||
|
||
def parse_isolated_borrow_rate(self, info: dict, market: Market = None) -> IsolatedBorrowRate:
|
||
#
|
||
# {
|
||
# "symbol": "BTCUSDT",
|
||
# "leverage": "10",
|
||
# "baseCoin": "BTC",
|
||
# "baseTransferable": True,
|
||
# "baseBorrowable": True,
|
||
# "baseDailyInterestRate": "0.00007",
|
||
# "baseAnnuallyInterestRate": "0.02555",
|
||
# "baseMaxBorrowableAmount": "27",
|
||
# "baseVipList": [
|
||
# {"level":"0","dailyInterestRate":"0.00007","limit":"27","annuallyInterestRate":"0.02555","discountRate":"1"},
|
||
# {"level":"1","dailyInterestRate":"0.0000679","limit":"27.81","annuallyInterestRate":"0.0247835","discountRate":"0.97"},
|
||
# {"level":"2","dailyInterestRate":"0.0000644","limit":"29.16","annuallyInterestRate":"0.023506","discountRate":"0.92"},
|
||
# {"level":"3","dailyInterestRate":"0.0000602","limit":"31.32","annuallyInterestRate":"0.021973","discountRate":"0.86"},
|
||
# {"level":"4","dailyInterestRate":"0.0000525","limit":"35.91","annuallyInterestRate":"0.0191625","discountRate":"0.75"},
|
||
# {"level":"5","dailyInterestRate":"0.000042","limit":"44.82","annuallyInterestRate":"0.01533","discountRate":"0.6"}
|
||
# ],
|
||
# "quoteCoin": "USDT",
|
||
# "quoteTransferable": True,
|
||
# "quoteBorrowable": True,
|
||
# "quoteDailyInterestRate": "0.00041095",
|
||
# "quoteAnnuallyInterestRate": "0.14999675",
|
||
# "quoteMaxBorrowableAmount": "300000",
|
||
# "quoteList": [
|
||
# {"level":"0","dailyInterestRate":"0.00041095","limit":"300000","annuallyInterestRate":"0.14999675","discountRate":"1"},
|
||
# {"level":"1","dailyInterestRate":"0.00039863","limit":"309000","annuallyInterestRate":"0.14549995","discountRate":"0.97"},
|
||
# {"level":"2","dailyInterestRate":"0.00037808","limit":"324000","annuallyInterestRate":"0.1379992","discountRate":"0.92"},
|
||
# {"level":"3","dailyInterestRate":"0.00035342","limit":"348000","annuallyInterestRate":"0.1289983","discountRate":"0.86"},
|
||
# {"level":"4","dailyInterestRate":"0.00030822","limit":"399000","annuallyInterestRate":"0.1125003","discountRate":"0.75"},
|
||
# {"level":"5","dailyInterestRate":"0.00024657","limit":"498000","annuallyInterestRate":"0.08999805","discountRate":"0.6"}
|
||
# ]
|
||
# }
|
||
#
|
||
marketId = self.safe_string(info, 'symbol')
|
||
symbol = self.safe_symbol(marketId, market, None, 'spot')
|
||
baseId = self.safe_string(info, 'baseCoin')
|
||
quoteId = self.safe_string(info, 'quoteCoin')
|
||
timestamp = self.safe_integer(info, 'timestamp')
|
||
return {
|
||
'symbol': symbol,
|
||
'base': self.safe_currency_code(baseId),
|
||
'baseRate': self.safe_number(info, 'baseDailyInterestRate'),
|
||
'quote': self.safe_currency_code(quoteId),
|
||
'quoteRate': self.safe_number(info, 'quoteDailyInterestRate'),
|
||
'period': 86400000, # 1-Day
|
||
'timestamp': timestamp,
|
||
'datetime': self.iso8601(timestamp),
|
||
'info': info,
|
||
}
|
||
|
||
async def fetch_cross_borrow_rate(self, code: str, params={}) -> CrossBorrowRate:
|
||
"""
|
||
fetch the rate of interest to borrow a currency for margin trading
|
||
|
||
https://www.bitget.com/api-doc/margin/cross/account/Get-Cross-Margin-Interest-Rate-And-Borrowable
|
||
https://www.bitget.com/api-doc/uta/public/Get-Margin-Loans
|
||
|
||
:param str code: unified currency code
|
||
:param dict [params]: extra parameters specific to the exchange API endpoint
|
||
:param boolean [params.uta]: set to True for the unified trading account(uta), defaults to False
|
||
:returns dict: a `borrow rate structure <https://github.com/ccxt/ccxt/wiki/Manual#borrow-rate-structure>`
|
||
"""
|
||
await self.load_markets()
|
||
currency = self.currency(code)
|
||
request: dict = {
|
||
'coin': currency['id'],
|
||
}
|
||
uta = None
|
||
response = None
|
||
result = None
|
||
uta, params = self.handle_option_and_params(params, 'fetchCrossBorrowRate', 'uta', False)
|
||
if uta:
|
||
response = await self.publicUtaGetV3MarketMarginLoans(self.extend(request, params))
|
||
#
|
||
# {
|
||
# "code": "00000",
|
||
# "msg": "success",
|
||
# "requestTime": 1752817798893,
|
||
# "data": {
|
||
# "dailyInterest": "0.00100008",
|
||
# "annualInterest": "0.3650292",
|
||
# "limit": "100"
|
||
# }
|
||
# }
|
||
#
|
||
result = self.safe_dict(response, 'data', {})
|
||
else:
|
||
response = await self.privateMarginGetV2MarginCrossedInterestRateAndLimit(self.extend(request, params))
|
||
#
|
||
# {
|
||
# "code": "00000",
|
||
# "msg": "success",
|
||
# "requestTime": 1700879047861,
|
||
# "data": [
|
||
# {
|
||
# "coin": "BTC",
|
||
# "leverage": "3",
|
||
# "transferable": True,
|
||
# "borrowable": True,
|
||
# "dailyInterestRate": "0.00007",
|
||
# "annualInterestRate": "0.02555",
|
||
# "maxBorrowableAmount": "26",
|
||
# "vipList": [
|
||
# {"level":"0","limit":"26","dailyInterestRate":"0.00007","annualInterestRate":"0.02555","discountRate":"1"},
|
||
# {"level":"1","limit":"26.78","dailyInterestRate":"0.0000679","annualInterestRate":"0.0247835","discountRate":"0.97"},
|
||
# {"level":"2","limit":"28.08","dailyInterestRate":"0.0000644","annualInterestRate":"0.023506","discountRate":"0.92"},
|
||
# {"level":"3","limit":"30.16","dailyInterestRate":"0.0000602","annualInterestRate":"0.021973","discountRate":"0.86"},
|
||
# {"level":"4","limit":"34.58","dailyInterestRate":"0.0000525","annualInterestRate":"0.0191625","discountRate":"0.75"},
|
||
# {"level":"5","limit":"43.16","dailyInterestRate":"0.000042","annualInterestRate":"0.01533","discountRate":"0.6"}
|
||
# ]
|
||
# }
|
||
# ]
|
||
# }
|
||
#
|
||
data = self.safe_value(response, 'data', [])
|
||
result = self.safe_value(data, 0, {})
|
||
timestamp = self.safe_integer(response, 'requestTime')
|
||
result['timestamp'] = timestamp
|
||
return self.parse_borrow_rate(result, currency)
|
||
|
||
def parse_borrow_rate(self, info, currency: Currency = None):
|
||
#
|
||
# default
|
||
#
|
||
# {
|
||
# "coin": "BTC",
|
||
# "leverage": "3",
|
||
# "transferable": True,
|
||
# "borrowable": True,
|
||
# "dailyInterestRate": "0.00007",
|
||
# "annualInterestRate": "0.02555",
|
||
# "maxBorrowableAmount": "26",
|
||
# "vipList": [
|
||
# {"level":"0","limit":"26","dailyInterestRate":"0.00007","annualInterestRate":"0.02555","discountRate":"1"},
|
||
# {"level":"1","limit":"26.78","dailyInterestRate":"0.0000679","annualInterestRate":"0.0247835","discountRate":"0.97"},
|
||
# {"level":"2","limit":"28.08","dailyInterestRate":"0.0000644","annualInterestRate":"0.023506","discountRate":"0.92"},
|
||
# {"level":"3","limit":"30.16","dailyInterestRate":"0.0000602","annualInterestRate":"0.021973","discountRate":"0.86"},
|
||
# {"level":"4","limit":"34.58","dailyInterestRate":"0.0000525","annualInterestRate":"0.0191625","discountRate":"0.75"},
|
||
# {"level":"5","limit":"43.16","dailyInterestRate":"0.000042","annualInterestRate":"0.01533","discountRate":"0.6"}
|
||
# ]
|
||
# }
|
||
#
|
||
# uta
|
||
#
|
||
# {
|
||
# "dailyInterest": "0.00100008",
|
||
# "annualInterest": "0.3650292",
|
||
# "limit": "100"
|
||
# }
|
||
#
|
||
currencyId = self.safe_string(info, 'coin')
|
||
timestamp = self.safe_integer(info, 'timestamp')
|
||
return {
|
||
'currency': self.safe_currency_code(currencyId, currency),
|
||
'rate': self.safe_number_2(info, 'dailyInterestRate', 'dailyInterest'),
|
||
'period': 86400000, # 1-Day
|
||
'timestamp': timestamp,
|
||
'datetime': self.iso8601(timestamp),
|
||
'info': info,
|
||
}
|
||
|
||
async def fetch_borrow_interest(self, code: Str = None, symbol: Str = None, since: Int = None, limit: Int = None, params={}) -> List[BorrowInterest]:
|
||
"""
|
||
fetch the interest owed by the user for borrowing currency for margin trading
|
||
|
||
https://www.bitget.com/api-doc/margin/cross/record/Get-Cross-Interest-Records
|
||
https://www.bitget.com/api-doc/margin/isolated/record/Get-Isolated-Interest-Records
|
||
|
||
:param str [code]: unified currency code
|
||
:param str [symbol]: unified market symbol when fetching interest in isolated markets
|
||
:param int [since]: the earliest time in ms to fetch borrow interest for
|
||
:param int [limit]: the maximum number of structures to retrieve
|
||
:param dict [params]: extra parameters specific to the exchange API endpoint
|
||
:param boolean [params.paginate]: default False, when True will automatically paginate by calling self endpoint multiple times. See in the docs all the [available parameters](https://github.com/ccxt/ccxt/wiki/Manual#pagination-params)
|
||
:returns dict[]: a list of `borrow interest structures <https://docs.ccxt.com/#/?id=borrow-interest-structure>`
|
||
"""
|
||
await self.load_markets()
|
||
paginate = False
|
||
paginate, params = self.handle_option_and_params(params, 'fetchBorrowInterest', 'paginate')
|
||
if paginate:
|
||
return await self.fetch_paginated_call_cursor('fetchBorrowInterest', symbol, since, limit, params, 'minId', 'idLessThan')
|
||
market = None
|
||
if symbol is not None:
|
||
market = self.market(symbol)
|
||
request: dict = {}
|
||
currency = None
|
||
if code is not None:
|
||
currency = self.currency(code)
|
||
request['coin'] = currency['id']
|
||
if since is not None:
|
||
request['startTime'] = since
|
||
else:
|
||
request['startTime'] = self.milliseconds() - 7776000000
|
||
if limit is not None:
|
||
request['limit'] = limit
|
||
response = None
|
||
marginMode = None
|
||
marginMode, params = self.handle_margin_mode_and_params('fetchBorrowInterest', params, 'cross')
|
||
if marginMode == 'isolated':
|
||
if symbol is None:
|
||
raise ArgumentsRequired(self.id + ' fetchBorrowInterest() requires a symbol argument')
|
||
request['symbol'] = market['id']
|
||
response = await self.privateMarginGetV2MarginIsolatedInterestHistory(self.extend(request, params))
|
||
elif marginMode == 'cross':
|
||
response = await self.privateMarginGetV2MarginCrossedInterestHistory(self.extend(request, params))
|
||
#
|
||
# isolated
|
||
#
|
||
# {
|
||
# "code": "00000",
|
||
# "msg": "success",
|
||
# "requestTime": 1700879935189,
|
||
# "data": {
|
||
# "resultList": [
|
||
# {
|
||
# "interestId": "1112125304879067137",
|
||
# "interestCoin": "USDT",
|
||
# "dailyInterestRate": "0.00041095",
|
||
# "loanCoin": "USDT",
|
||
# "interestAmount": "0.0000685",
|
||
# "interstType": "first",
|
||
# "symbol": "BTCUSDT",
|
||
# "cTime": "1700877255648",
|
||
# "uTime": "1700877255648"
|
||
# },
|
||
# ],
|
||
# "maxId": "1112125304879067137",
|
||
# "minId": "1100138015672119298"
|
||
# }
|
||
# }
|
||
#
|
||
# cross
|
||
#
|
||
# {
|
||
# "code": "00000",
|
||
# "msg": "success",
|
||
# "requestTime": 1700879597044,
|
||
# "data": {
|
||
# "resultList": [
|
||
# {
|
||
# "interestId": "1112122013642272769",
|
||
# "interestCoin": "USDT",
|
||
# "dailyInterestRate": "0.00041",
|
||
# "loanCoin": "USDT",
|
||
# "interestAmount": "0.00006834",
|
||
# "interstType": "first",
|
||
# "cTime": "1700876470957",
|
||
# "uTime": "1700876470957"
|
||
# },
|
||
# ],
|
||
# "maxId": "1112122013642272769",
|
||
# "minId": "1096917004629716993"
|
||
# }
|
||
# }
|
||
#
|
||
data = self.safe_value(response, 'data', {})
|
||
rows = self.safe_value(data, 'resultList', [])
|
||
interest = self.parse_borrow_interests(rows, market)
|
||
return self.filter_by_currency_since_limit(interest, code, since, limit)
|
||
|
||
def parse_borrow_interest(self, info: dict, market: Market = None) -> BorrowInterest:
|
||
#
|
||
# isolated
|
||
#
|
||
# {
|
||
# "interestId": "1112125304879067137",
|
||
# "interestCoin": "USDT",
|
||
# "dailyInterestRate": "0.00041095",
|
||
# "loanCoin": "USDT",
|
||
# "interestAmount": "0.0000685",
|
||
# "interstType": "first",
|
||
# "symbol": "BTCUSDT",
|
||
# "cTime": "1700877255648",
|
||
# "uTime": "1700877255648"
|
||
# }
|
||
#
|
||
# cross
|
||
#
|
||
# {
|
||
# "interestId": "1112122013642272769",
|
||
# "interestCoin": "USDT",
|
||
# "dailyInterestRate": "0.00041",
|
||
# "loanCoin": "USDT",
|
||
# "interestAmount": "0.00006834",
|
||
# "interstType": "first",
|
||
# "cTime": "1700876470957",
|
||
# "uTime": "1700876470957"
|
||
# }
|
||
#
|
||
marketId = self.safe_string(info, 'symbol')
|
||
market = self.safe_market(marketId, market)
|
||
marginMode = 'isolated' if (marketId is not None) else 'cross'
|
||
timestamp = self.safe_integer(info, 'cTime')
|
||
return {
|
||
'info': info,
|
||
'symbol': self.safe_string(market, 'symbol'),
|
||
'currency': self.safe_currency_code(self.safe_string(info, 'interestCoin')),
|
||
'interest': self.safe_number(info, 'interestAmount'),
|
||
'interestRate': self.safe_number(info, 'dailyInterestRate'),
|
||
'amountBorrowed': None,
|
||
'marginMode': marginMode,
|
||
'timestamp': timestamp,
|
||
'datetime': self.iso8601(timestamp),
|
||
}
|
||
|
||
async def close_position(self, symbol: str, side: OrderSide = None, params={}) -> Order:
|
||
"""
|
||
closes an open position for a market
|
||
|
||
https://www.bitget.com/api-doc/contract/trade/Flash-Close-Position
|
||
https://www.bitget.com/api-doc/uta/trade/Close-All-Positions
|
||
|
||
:param str symbol: unified CCXT market symbol
|
||
:param str [side]: one-way mode: 'buy' or 'sell', hedge-mode: 'long' or 'short'
|
||
:param dict [params]: extra parameters specific to the exchange API endpoint
|
||
:param boolean [params.uta]: set to True for the unified trading account(uta), defaults to False
|
||
:returns dict: An `order structure <https://docs.ccxt.com/#/?id=order-structure>`
|
||
"""
|
||
await self.load_markets()
|
||
market = self.market(symbol)
|
||
request: dict = {
|
||
'symbol': market['id'],
|
||
}
|
||
productType = None
|
||
uta = None
|
||
response = None
|
||
productType, params = self.handle_product_type_and_params(market, params)
|
||
uta, params = self.handle_option_and_params(params, 'closePosition', 'uta', False)
|
||
if uta:
|
||
if side is not None:
|
||
request['posSide'] = side
|
||
request['category'] = productType
|
||
response = await self.privateUtaPostV3TradeClosePositions(self.extend(request, params))
|
||
#
|
||
# {
|
||
# "code": "00000",
|
||
# "msg": "success",
|
||
# "requestTime": 1751020218384,
|
||
# "data": {
|
||
# "list": [
|
||
# {
|
||
# "orderId": "1322440134099320832",
|
||
# "clientOid": "1322440134099320833"
|
||
# }
|
||
# ]
|
||
# }
|
||
# }
|
||
#
|
||
else:
|
||
if side is not None:
|
||
request['holdSide'] = side
|
||
request['productType'] = productType
|
||
response = await self.privateMixPostV2MixOrderClosePositions(self.extend(request, params))
|
||
#
|
||
# {
|
||
# "code": "00000",
|
||
# "msg": "success",
|
||
# "requestTime": 1702975017017,
|
||
# "data": {
|
||
# "successList": [
|
||
# {
|
||
# "orderId": "1120923953904893955",
|
||
# "clientOid": "1120923953904893956"
|
||
# }
|
||
# ],
|
||
# "failureList": [],
|
||
# "result": False
|
||
# }
|
||
# }
|
||
#
|
||
data = self.safe_value(response, 'data', {})
|
||
order = self.safe_list_2(data, 'successList', 'list', [])
|
||
return self.parse_order(order[0], market)
|
||
|
||
async def close_all_positions(self, params={}) -> List[Position]:
|
||
"""
|
||
closes all open positions for a market type
|
||
|
||
https://www.bitget.com/api-doc/contract/trade/Flash-Close-Position
|
||
https://www.bitget.com/api-doc/uta/trade/Close-All-Positions
|
||
|
||
:param dict [params]: extra parameters specific to the exchange API endpoint
|
||
:param str [params.productType]: 'USDT-FUTURES', 'USDC-FUTURES', 'COIN-FUTURES', 'SUSDT-FUTURES', 'SUSDC-FUTURES' or 'SCOIN-FUTURES'
|
||
:param boolean [params.uta]: set to True for the unified trading account(uta), defaults to False
|
||
:returns dict[]: A list of `position structures <https://docs.ccxt.com/#/?id=position-structure>`
|
||
"""
|
||
await self.load_markets()
|
||
request: dict = {}
|
||
productType = None
|
||
uta = None
|
||
response = None
|
||
productType, params = self.handle_product_type_and_params(None, params)
|
||
uta, params = self.handle_option_and_params(params, 'closeAllPositions', 'uta', False)
|
||
if uta:
|
||
request['category'] = productType
|
||
response = await self.privateUtaPostV3TradeClosePositions(self.extend(request, params))
|
||
#
|
||
# {
|
||
# "code": "00000",
|
||
# "msg": "success",
|
||
# "requestTime": 1751020218384,
|
||
# "data": {
|
||
# "list": [
|
||
# {
|
||
# "orderId": "1322440134099320832",
|
||
# "clientOid": "1322440134099320833"
|
||
# }
|
||
# ]
|
||
# }
|
||
# }
|
||
#
|
||
else:
|
||
request['productType'] = productType
|
||
response = await self.privateMixPostV2MixOrderClosePositions(self.extend(request, params))
|
||
#
|
||
# {
|
||
# "code": "00000",
|
||
# "msg": "success",
|
||
# "requestTime": 1702975017017,
|
||
# "data": {
|
||
# "successList": [
|
||
# {
|
||
# "orderId": "1120923953904893955",
|
||
# "clientOid": "1120923953904893956"
|
||
# }
|
||
# ],
|
||
# "failureList": [],
|
||
# "result": False
|
||
# }
|
||
# }
|
||
#
|
||
data = self.safe_value(response, 'data', {})
|
||
orderInfo = self.safe_list_2(data, 'successList', 'list', [])
|
||
return self.parse_positions(orderInfo, None, params)
|
||
|
||
async def fetch_margin_mode(self, symbol: str, params={}) -> MarginMode:
|
||
"""
|
||
fetches the margin mode of a trading pair
|
||
|
||
https://www.bitget.com/api-doc/contract/account/Get-Single-Account
|
||
|
||
:param str symbol: unified symbol of the market to fetch the margin mode for
|
||
:param dict [params]: extra parameters specific to the exchange API endpoint
|
||
:returns dict: a `margin mode structure <https://docs.ccxt.com/#/?id=margin-mode-structure>`
|
||
"""
|
||
await self.load_markets()
|
||
market = self.market(symbol)
|
||
productType = None
|
||
productType, params = self.handle_product_type_and_params(market, params)
|
||
request: dict = {
|
||
'symbol': market['id'],
|
||
'marginCoin': market['settleId'],
|
||
'productType': productType,
|
||
}
|
||
response = await self.privateMixGetV2MixAccountAccount(self.extend(request, params))
|
||
#
|
||
# {
|
||
# "code": "00000",
|
||
# "msg": "success",
|
||
# "requestTime": 1709791216652,
|
||
# "data": {
|
||
# "marginCoin": "USDT",
|
||
# "locked": "0",
|
||
# "available": "19.88811074",
|
||
# "crossedMaxAvailable": "19.88811074",
|
||
# "isolatedMaxAvailable": "19.88811074",
|
||
# "maxTransferOut": "19.88811074",
|
||
# "accountEquity": "19.88811074",
|
||
# "usdtEquity": "19.888110749166",
|
||
# "btcEquity": "0.000302183391",
|
||
# "crossedRiskRate": "0",
|
||
# "crossedMarginLeverage": 20,
|
||
# "isolatedLongLever": 20,
|
||
# "isolatedShortLever": 20,
|
||
# "marginMode": "crossed",
|
||
# "posMode": "hedge_mode",
|
||
# "unrealizedPL": "0",
|
||
# "coupon": "0",
|
||
# "crossedUnrealizedPL": "0",
|
||
# "isolatedUnrealizedPL": ""
|
||
# }
|
||
# }
|
||
#
|
||
data = self.safe_dict(response, 'data', {})
|
||
return self.parse_margin_mode(data, market)
|
||
|
||
def parse_margin_mode(self, marginMode: dict, market=None) -> MarginMode:
|
||
marginType = self.safe_string(marginMode, 'marginMode')
|
||
marginType = 'cross' if (marginType == 'crossed') else marginType
|
||
return {
|
||
'info': marginMode,
|
||
'symbol': market['symbol'],
|
||
'marginMode': marginType,
|
||
}
|
||
|
||
async def fetch_positions_history(self, symbols: Strings = None, since: Int = None, limit: Int = None, params={}) -> List[Position]:
|
||
"""
|
||
fetches historical positions
|
||
|
||
https://www.bitget.com/api-doc/contract/position/Get-History-Position
|
||
https://www.bitget.com/api-doc/uta/trade/Get-Position-History
|
||
|
||
:param str[] [symbols]: unified contract symbols
|
||
:param int [since]: timestamp in ms of the earliest position to fetch, default=3 months ago, max range for params["until"] - since is 3 months
|
||
:param int [limit]: the maximum amount of records to fetch, default=20, max=100
|
||
:param dict params: extra parameters specific to the exchange api endpoint
|
||
:param int [params.until]: timestamp in ms of the latest position to fetch, max range for params["until"] - since is 3 months
|
||
:param str [params.productType]: USDT-FUTURES(default), COIN-FUTURES, USDC-FUTURES, SUSDT-FUTURES, SCOIN-FUTURES, or SUSDC-FUTURES
|
||
:param boolean [params.uta]: set to True for the unified trading account(uta), defaults to False
|
||
:returns dict[]: a list of `position structures <https://docs.ccxt.com/#/?id=position-structure>`
|
||
"""
|
||
await self.load_markets()
|
||
request: dict = {}
|
||
market = None
|
||
productType = None
|
||
uta = None
|
||
response = None
|
||
if symbols is not None:
|
||
symbolsLength = len(symbols)
|
||
if symbolsLength > 0:
|
||
market = self.market(symbols[0])
|
||
request['symbol'] = market['id']
|
||
if since is not None:
|
||
request['startTime'] = since
|
||
if limit is not None:
|
||
request['limit'] = limit
|
||
request, params = self.handle_until_option('endTime', request, params)
|
||
productType, params = self.handle_product_type_and_params(market, params)
|
||
uta, params = self.handle_option_and_params(params, 'fetchPositionsHistory', 'uta', False)
|
||
if uta:
|
||
request['category'] = productType
|
||
response = await self.privateUtaGetV3PositionHistoryPosition(self.extend(request, params))
|
||
#
|
||
# {
|
||
# "code": "00000",
|
||
# "msg": "success",
|
||
# "requestTime": 1751020950427,
|
||
# "data": {
|
||
# "list": [
|
||
# {
|
||
# "positionId": "1322441328637100049",
|
||
# "category": "USDT-FUTURES",
|
||
# "symbol": "BTCUSDT",
|
||
# "marginCoin": "USDT",
|
||
# "holdMode": "hedge_mode",
|
||
# "posSide": "long",
|
||
# "marginMode": "crossed",
|
||
# "openPriceAvg": "107003.7",
|
||
# "closePriceAvg": "107005.4",
|
||
# "openTotalPos": "0.0001",
|
||
# "closeTotalPos": "0.0001",
|
||
# "cumRealisedPnl": "0.00017",
|
||
# "netProfit": "-0.01267055",
|
||
# "totalFunding": "0",
|
||
# "openFeeTotal": "-0.00642022",
|
||
# "closeFeeTotal": "-0.00642032",
|
||
# "createdTime": "1751020503195",
|
||
# "updatedTime": "1751020520458"
|
||
# },
|
||
# ],
|
||
# "cursor": "1322440134158041089"
|
||
# }
|
||
# }
|
||
#
|
||
else:
|
||
response = await self.privateMixGetV2MixPositionHistoryPosition(self.extend(request, params))
|
||
#
|
||
# {
|
||
# code: '00000',
|
||
# msg: 'success',
|
||
# requestTime: '1712794148791',
|
||
# data: {
|
||
# list: [
|
||
# {
|
||
# symbol: 'XRPUSDT',
|
||
# marginCoin: 'USDT',
|
||
# holdSide: 'long',
|
||
# openAvgPrice: '0.64967',
|
||
# closeAvgPrice: '0.58799',
|
||
# marginMode: 'isolated',
|
||
# openTotalPos: '10',
|
||
# closeTotalPos: '10',
|
||
# pnl: '-0.62976205',
|
||
# netProfit: '-0.65356802',
|
||
# totalFunding: '-0.01638',
|
||
# openFee: '-0.00389802',
|
||
# closeFee: '-0.00352794',
|
||
# ctime: '1709590322199',
|
||
# utime: '1709667583395'
|
||
# },
|
||
# ...
|
||
# ]
|
||
# }
|
||
# }
|
||
#
|
||
data = self.safe_dict(response, 'data', {})
|
||
responseList = self.safe_list(data, 'list', [])
|
||
positions = self.parse_positions(responseList, symbols, params)
|
||
return self.filter_by_since_limit(positions, since, limit)
|
||
|
||
async def fetch_convert_quote(self, fromCode: str, toCode: str, amount: Num = None, params={}) -> Conversion:
|
||
"""
|
||
fetch a quote for converting from one currency to another
|
||
|
||
https://www.bitget.com/api-doc/common/convert/Get-Quoted-Price
|
||
|
||
:param str fromCode: the currency that you want to sell and convert from
|
||
:param str toCode: the currency that you want to buy and convert into
|
||
:param float [amount]: how much you want to trade in units of the from currency
|
||
:param dict [params]: extra parameters specific to the exchange API endpoint
|
||
:returns dict: a `conversion structure <https://docs.ccxt.com/#/?id=conversion-structure>`
|
||
"""
|
||
await self.load_markets()
|
||
request: dict = {
|
||
'fromCoin': fromCode,
|
||
'toCoin': toCode,
|
||
'fromCoinSize': self.number_to_string(amount),
|
||
}
|
||
response = await self.privateConvertGetV2ConvertQuotedPrice(self.extend(request, params))
|
||
#
|
||
# {
|
||
# "code": "00000",
|
||
# "msg": "success",
|
||
# "requestTime": 1712121940158,
|
||
# "data": {
|
||
# "fromCoin": "USDT",
|
||
# "fromCoinSize": "5",
|
||
# "cnvtPrice": "0.9993007892377704",
|
||
# "toCoin": "USDC",
|
||
# "toCoinSize": "4.99650394",
|
||
# "traceId": "1159288930228187140",
|
||
# "fee": "0"
|
||
# }
|
||
# }
|
||
#
|
||
data = self.safe_dict(response, 'data', {})
|
||
fromCurrencyId = self.safe_string(data, 'fromCoin', fromCode)
|
||
fromCurrency = self.currency(fromCurrencyId)
|
||
toCurrencyId = self.safe_string(data, 'toCoin', toCode)
|
||
toCurrency = self.currency(toCurrencyId)
|
||
return self.parse_conversion(data, fromCurrency, toCurrency)
|
||
|
||
async def create_convert_trade(self, id: str, fromCode: str, toCode: str, amount: Num = None, params={}) -> Conversion:
|
||
"""
|
||
convert from one currency to another
|
||
|
||
https://www.bitget.com/api-doc/common/convert/Trade
|
||
|
||
:param str id: the id of the trade that you want to make
|
||
:param str fromCode: the currency that you want to sell and convert from
|
||
:param str toCode: the currency that you want to buy and convert into
|
||
:param float amount: how much you want to trade in units of the from currency
|
||
:param dict [params]: extra parameters specific to the exchange API endpoint
|
||
:param str params['price']: the price of the conversion, obtained from fetchConvertQuote()
|
||
:param str params['toAmount']: the amount you want to trade in units of the toCurrency, obtained from fetchConvertQuote()
|
||
:returns dict: a `conversion structure <https://docs.ccxt.com/#/?id=conversion-structure>`
|
||
"""
|
||
await self.load_markets()
|
||
price = self.safe_string_2(params, 'price', 'cnvtPrice')
|
||
if price is None:
|
||
raise ArgumentsRequired(self.id + ' createConvertTrade() requires a price parameter')
|
||
toAmount = self.safe_string_2(params, 'toAmount', 'toCoinSize')
|
||
if toAmount is None:
|
||
raise ArgumentsRequired(self.id + ' createConvertTrade() requires a toAmount parameter')
|
||
params = self.omit(params, ['price', 'toAmount'])
|
||
request: dict = {
|
||
'traceId': id,
|
||
'fromCoin': fromCode,
|
||
'toCoin': toCode,
|
||
'fromCoinSize': self.number_to_string(amount),
|
||
'toCoinSize': toAmount,
|
||
'cnvtPrice': price,
|
||
}
|
||
response = await self.privateConvertPostV2ConvertTrade(self.extend(request, params))
|
||
#
|
||
# {
|
||
# "code": "00000",
|
||
# "msg": "success",
|
||
# "requestTime": 1712123746203,
|
||
# "data": {
|
||
# "cnvtPrice": "0.99940076",
|
||
# "toCoin": "USDC",
|
||
# "toCoinSize": "4.99700379",
|
||
# "ts": "1712123746217"
|
||
# }
|
||
# }
|
||
#
|
||
data = self.safe_dict(response, 'data', {})
|
||
toCurrencyId = self.safe_string(data, 'toCoin', toCode)
|
||
toCurrency = self.currency(toCurrencyId)
|
||
return self.parse_conversion(data, None, toCurrency)
|
||
|
||
async def fetch_convert_trade_history(self, code: Str = None, since: Int = None, limit: Int = None, params={}) -> List[Conversion]:
|
||
"""
|
||
fetch the users history of conversion trades
|
||
|
||
https://www.bitget.com/api-doc/common/convert/Get-Convert-Record
|
||
|
||
:param str [code]: the unified currency code
|
||
:param int [since]: the earliest time in ms to fetch conversions for
|
||
:param int [limit]: the maximum number of conversion structures to retrieve
|
||
:param dict [params]: extra parameters specific to the exchange API endpoint
|
||
:returns dict[]: a list of `conversion structures <https://docs.ccxt.com/#/?id=conversion-structure>`
|
||
"""
|
||
await self.load_markets()
|
||
request: dict = {}
|
||
msInDay = 86400000
|
||
now = self.milliseconds()
|
||
if since is not None:
|
||
request['startTime'] = since
|
||
else:
|
||
request['startTime'] = now - msInDay
|
||
endTime = self.safe_string_2(params, 'endTime', 'until')
|
||
if endTime is not None:
|
||
request['endTime'] = endTime
|
||
else:
|
||
request['endTime'] = now
|
||
if limit is not None:
|
||
request['limit'] = limit
|
||
params = self.omit(params, 'until')
|
||
response = await self.privateConvertGetV2ConvertConvertRecord(self.extend(request, params))
|
||
#
|
||
# {
|
||
# "code": "00000",
|
||
# "msg": "success",
|
||
# "requestTime": 1712124371799,
|
||
# "data": {
|
||
# "dataList": [
|
||
# {
|
||
# "id": "1159296505255219205",
|
||
# "fromCoin": "USDT",
|
||
# "fromCoinSize": "5",
|
||
# "cnvtPrice": "0.99940076",
|
||
# "toCoin": "USDC",
|
||
# "toCoinSize": "4.99700379",
|
||
# "ts": "1712123746217",
|
||
# "fee": "0"
|
||
# }
|
||
# ],
|
||
# "endId": "1159296505255219205"
|
||
# }
|
||
# }
|
||
#
|
||
data = self.safe_dict(response, 'data', {})
|
||
dataList = self.safe_list(data, 'dataList', [])
|
||
return self.parse_conversions(dataList, code, 'fromCoin', 'toCoin', since, limit)
|
||
|
||
def parse_conversion(self, conversion: dict, fromCurrency: Currency = None, toCurrency: Currency = None) -> Conversion:
|
||
#
|
||
# fetchConvertQuote
|
||
#
|
||
# {
|
||
# "fromCoin": "USDT",
|
||
# "fromCoinSize": "5",
|
||
# "cnvtPrice": "0.9993007892377704",
|
||
# "toCoin": "USDC",
|
||
# "toCoinSize": "4.99650394",
|
||
# "traceId": "1159288930228187140",
|
||
# "fee": "0"
|
||
# }
|
||
#
|
||
# createConvertTrade
|
||
#
|
||
# {
|
||
# "cnvtPrice": "0.99940076",
|
||
# "toCoin": "USDC",
|
||
# "toCoinSize": "4.99700379",
|
||
# "ts": "1712123746217"
|
||
# }
|
||
#
|
||
# fetchConvertTradeHistory
|
||
#
|
||
# {
|
||
# "id": "1159296505255219205",
|
||
# "fromCoin": "USDT",
|
||
# "fromCoinSize": "5",
|
||
# "cnvtPrice": "0.99940076",
|
||
# "toCoin": "USDC",
|
||
# "toCoinSize": "4.99700379",
|
||
# "ts": "1712123746217",
|
||
# "fee": "0"
|
||
# }
|
||
#
|
||
timestamp = self.safe_integer(conversion, 'ts')
|
||
fromCoin = self.safe_string(conversion, 'fromCoin')
|
||
fromCode = self.safe_currency_code(fromCoin, fromCurrency)
|
||
to = self.safe_string(conversion, 'toCoin')
|
||
toCode = self.safe_currency_code(to, toCurrency)
|
||
return {
|
||
'info': conversion,
|
||
'timestamp': timestamp,
|
||
'datetime': self.iso8601(timestamp),
|
||
'id': self.safe_string_2(conversion, 'id', 'traceId'),
|
||
'fromCurrency': fromCode,
|
||
'fromAmount': self.safe_number(conversion, 'fromCoinSize'),
|
||
'toCurrency': toCode,
|
||
'toAmount': self.safe_number(conversion, 'toCoinSize'),
|
||
'price': self.safe_number(conversion, 'cnvtPrice'),
|
||
'fee': self.safe_number(conversion, 'fee'),
|
||
}
|
||
|
||
async def fetch_convert_currencies(self, params={}) -> Currencies:
|
||
"""
|
||
fetches all available currencies that can be converted
|
||
|
||
https://www.bitget.com/api-doc/common/convert/Get-Convert-Currencies
|
||
|
||
:param dict [params]: extra parameters specific to the exchange API endpoint
|
||
:returns dict: an associative dictionary of currencies
|
||
"""
|
||
await self.load_markets()
|
||
response = await self.privateConvertGetV2ConvertCurrencies(params)
|
||
#
|
||
# {
|
||
# "code": "00000",
|
||
# "msg": "success",
|
||
# "requestTime": 1712121755897,
|
||
# "data": [
|
||
# {
|
||
# "coin": "BTC",
|
||
# "available": "0.00009850",
|
||
# "maxAmount": "0.756266",
|
||
# "minAmount": "0.00001"
|
||
# },
|
||
# ]
|
||
# }
|
||
#
|
||
result: dict = {}
|
||
data = self.safe_list(response, 'data', [])
|
||
for i in range(0, len(data)):
|
||
entry = data[i]
|
||
id = self.safe_string(entry, 'coin')
|
||
code = self.safe_currency_code(id)
|
||
result[code] = {
|
||
'info': entry,
|
||
'id': id,
|
||
'code': code,
|
||
'networks': None,
|
||
'type': None,
|
||
'name': None,
|
||
'active': None,
|
||
'deposit': None,
|
||
'withdraw': self.safe_number(entry, 'available'),
|
||
'fee': None,
|
||
'precision': None,
|
||
'limits': {
|
||
'amount': {
|
||
'min': self.safe_number(entry, 'minAmount'),
|
||
'max': self.safe_number(entry, 'maxAmount'),
|
||
},
|
||
'withdraw': {
|
||
'min': None,
|
||
'max': None,
|
||
},
|
||
'deposit': {
|
||
'min': None,
|
||
'max': None,
|
||
},
|
||
},
|
||
'created': None,
|
||
}
|
||
return result
|
||
|
||
async def fetch_funding_interval(self, symbol: str, params={}) -> FundingRate:
|
||
"""
|
||
fetch the current funding rate interval
|
||
|
||
https://www.bitget.com/api-doc/contract/market/Get-Symbol-Next-Funding-Time
|
||
https://www.bitget.com/api-doc/uta/public/Get-Current-Funding-Rate
|
||
|
||
:param str symbol: unified market symbol
|
||
:param dict [params]: extra parameters specific to the exchange API endpoint
|
||
:param boolean [params.uta]: set to True for the unified trading account(uta), defaults to False
|
||
:returns dict: a `funding rate structure <https://docs.ccxt.com/#/?id=funding-rate-structure>`
|
||
"""
|
||
await self.load_markets()
|
||
market = self.market(symbol)
|
||
productType = None
|
||
productType, params = self.handle_product_type_and_params(market, params)
|
||
request: dict = {
|
||
'symbol': market['id'],
|
||
}
|
||
response = None
|
||
uta = None
|
||
uta, params = self.handle_option_and_params(params, 'fetchFundingInterval', 'uta', False)
|
||
if uta:
|
||
response = await self.publicUtaGetV3MarketCurrentFundRate(self.extend(request, params))
|
||
#
|
||
# {
|
||
# "code": "00000",
|
||
# "msg": "success",
|
||
# "requestTime": 1752880157959,
|
||
# "data": [
|
||
# {
|
||
# "symbol": "BTCUSDT",
|
||
# "fundingRate": "0.0001",
|
||
# "fundingRateInterval": "8",
|
||
# "nextUpdate": "1752883200000",
|
||
# "minFundingRate": "-0.003",
|
||
# "maxFundingRate": "0.003"
|
||
# }
|
||
# ]
|
||
# }
|
||
#
|
||
else:
|
||
request['productType'] = productType
|
||
response = await self.publicMixGetV2MixMarketFundingTime(self.extend(request, params))
|
||
#
|
||
# {
|
||
# "code": "00000",
|
||
# "msg": "success",
|
||
# "requestTime": 1727930153888,
|
||
# "data": [
|
||
# {
|
||
# "symbol": "BTCUSDT",
|
||
# "nextFundingTime": "1727942400000",
|
||
# "ratePeriod": "8"
|
||
# }
|
||
# ]
|
||
# }
|
||
#
|
||
data = self.safe_list(response, 'data', [])
|
||
first = self.safe_dict(data, 0, {})
|
||
return self.parse_funding_rate(first, market)
|
||
|
||
async def fetch_long_short_ratio_history(self, symbol: Str = None, timeframe: Str = None, since: Int = None, limit: Int = None, params={}) -> List[LongShortRatio]:
|
||
"""
|
||
fetches the long short ratio history for a unified market symbol
|
||
|
||
https://www.bitget.com/api-doc/common/apidata/Margin-Ls-Ratio
|
||
https://www.bitget.com/api-doc/common/apidata/Account-Long-Short
|
||
|
||
:param str symbol: unified symbol of the market to fetch the long short ratio for
|
||
:param str [timeframe]: the period for the ratio
|
||
:param int [since]: the earliest time in ms to fetch ratios for
|
||
:param int [limit]: the maximum number of long short ratio structures to retrieve
|
||
:param dict [params]: extra parameters specific to the exchange API endpoint
|
||
:returns dict[]: an array of `long short ratio structures <https://docs.ccxt.com/#/?id=long-short-ratio-structure>`
|
||
"""
|
||
await self.load_markets()
|
||
market = self.market(symbol)
|
||
request: dict = {
|
||
'symbol': market['id'],
|
||
}
|
||
if timeframe is not None:
|
||
request['period'] = timeframe
|
||
response = None
|
||
if market['swap'] or market['future']:
|
||
response = await self.publicMixGetV2MixMarketAccountLongShort(self.extend(request, params))
|
||
#
|
||
# {
|
||
# "code": "00000",
|
||
# "msg": "success",
|
||
# "requestTime": 1729321233281,
|
||
# "data": [
|
||
# {
|
||
# "longAccountRatio": "0.58",
|
||
# "shortAccountRatio": "0.42",
|
||
# "longShortAccountRatio": "0.0138",
|
||
# "ts": "1729312200000"
|
||
# },
|
||
# ]
|
||
# }
|
||
#
|
||
else:
|
||
response = await self.publicMarginGetV2MarginMarketLongShortRatio(self.extend(request, params))
|
||
#
|
||
# {
|
||
# "code": "00000",
|
||
# "msg": "success",
|
||
# "requestTime": 1729306974712,
|
||
# "data": [
|
||
# {
|
||
# "longShortRatio": "40.66",
|
||
# "ts": "1729306800000"
|
||
# },
|
||
# ]
|
||
# }
|
||
#
|
||
data = self.safe_list(response, 'data', [])
|
||
return self.parse_long_short_ratio_history(data, market)
|
||
|
||
def parse_long_short_ratio(self, info: dict, market: Market = None) -> LongShortRatio:
|
||
marketId = self.safe_string(info, 'symbol')
|
||
timestamp = self.safe_integer_omit_zero(info, 'ts')
|
||
return {
|
||
'info': info,
|
||
'symbol': self.safe_symbol(marketId, market, None, 'contract'),
|
||
'timestamp': timestamp,
|
||
'datetime': self.iso8601(timestamp),
|
||
'timeframe': None,
|
||
'longShortRatio': self.safe_number_2(info, 'longShortRatio', 'longShortAccountRatio'),
|
||
}
|
||
|
||
def handle_errors(self, code: int, reason: str, url: str, method: str, headers: dict, body: str, response, requestHeaders, requestBody):
|
||
if not response:
|
||
return None # fallback to default error handler
|
||
#
|
||
# spot
|
||
#
|
||
# {"code":"00000","msg":"success","requestTime":1713294492511,"data":[...]}"
|
||
#
|
||
# {"status":"fail","err_code":"01001","err_msg":"系统异常,请稍后重试"}
|
||
# {"status":"error","ts":1595594160149,"err_code":"invalid-parameter","err_msg":"invalid size, valid range: [1,2000]"}
|
||
# {"status":"error","ts":1595684716042,"err_code":"invalid-parameter","err_msg":"illegal sign invalid"}
|
||
# {"status":"error","ts":1595700216275,"err_code":"bad-request","err_msg":"your balance is low!"}
|
||
# {"status":"error","ts":1595700344504,"err_code":"invalid-parameter","err_msg":"invalid type"}
|
||
# {"status":"error","ts":1595703343035,"err_code":"bad-request","err_msg":"order cancel fail"}
|
||
# {"status":"error","ts":1595704360508,"err_code":"invalid-parameter","err_msg":"accesskey not null"}
|
||
# {"status":"error","ts":1595704490084,"err_code":"invalid-parameter","err_msg":"permissions not right"}
|
||
# {"status":"error","ts":1595711862763,"err_code":"system exception","err_msg":"system exception"}
|
||
# {"status":"error","ts":1595730308979,"err_code":"bad-request","err_msg":"20003"}
|
||
#
|
||
# swap
|
||
#
|
||
# {"code":"40015","msg":"","requestTime":1595698564931,"data":null}
|
||
# {"code":"40017","msg":"Order id must not be blank","requestTime":1595702477835,"data":null}
|
||
# {"code":"40017","msg":"Order Type must not be blank","requestTime":1595698516162,"data":null}
|
||
# {"code":"40301","msg":"","requestTime":1595667662503,"data":null}
|
||
# {"code":"40017","msg":"Contract code must not be blank","requestTime":1595703151651,"data":null}
|
||
# {"code":"40108","msg":"","requestTime":1595885064600,"data":null}
|
||
# {"order_id":"513468410013679613","client_oid":null,"symbol":"ethusd","result":false,"err_code":"order_no_exist_error","err_msg":"订单不存在!"}
|
||
#
|
||
message = self.safe_string_2(response, 'err_msg', 'msg')
|
||
feedback = self.id + ' ' + body
|
||
nonEmptyMessage = ((message is not None) and (message != '') and (message != 'success'))
|
||
if nonEmptyMessage:
|
||
self.throw_exactly_matched_exception(self.exceptions['exact'], message, feedback)
|
||
self.throw_broadly_matched_exception(self.exceptions['broad'], message, feedback)
|
||
errorCode = self.safe_string_2(response, 'code', 'err_code')
|
||
nonZeroErrorCode = (errorCode is not None) and (errorCode != '00000')
|
||
if nonZeroErrorCode:
|
||
self.throw_exactly_matched_exception(self.exceptions['exact'], errorCode, feedback)
|
||
if nonZeroErrorCode or nonEmptyMessage:
|
||
raise ExchangeError(feedback) # unknown message
|
||
return None
|
||
|
||
def nonce(self):
|
||
return self.milliseconds() - self.options['timeDifference']
|
||
|
||
def sign(self, path, api=[], method='GET', params={}, headers=None, body=None):
|
||
signed = api[0] == 'private'
|
||
endpoint = api[1]
|
||
pathPart = '/api'
|
||
request = '/' + self.implode_params(path, params)
|
||
payload = pathPart + request
|
||
url = self.implode_hostname(self.urls['api'][endpoint]) + payload
|
||
query = self.omit(params, self.extract_params(path))
|
||
if not signed and (method == 'GET'):
|
||
keys = list(query.keys())
|
||
keysLength = len(keys)
|
||
if keysLength > 0:
|
||
url = url + '?' + self.urlencode(query)
|
||
if signed:
|
||
self.check_required_credentials()
|
||
timestamp = str(self.nonce())
|
||
auth = timestamp + method + payload
|
||
if method == 'POST':
|
||
body = self.json(params)
|
||
auth += body
|
||
else:
|
||
if params:
|
||
queryInner = '?' + self.urlencode(self.keysort(params))
|
||
# check #21169 pr
|
||
if queryInner.find('%24') > -1:
|
||
queryInner = queryInner.replace('%24', '$')
|
||
url += queryInner
|
||
auth += queryInner
|
||
signature = self.hmac(self.encode(auth), self.encode(self.secret), hashlib.sha256, 'base64')
|
||
broker = self.safe_string(self.options, 'broker')
|
||
headers = {
|
||
'ACCESS-KEY': self.apiKey,
|
||
'ACCESS-SIGN': signature,
|
||
'ACCESS-TIMESTAMP': timestamp,
|
||
'ACCESS-PASSPHRASE': self.password,
|
||
'X-CHANNEL-API-CODE': broker,
|
||
}
|
||
if method == 'POST':
|
||
headers['Content-Type'] = 'application/json'
|
||
sandboxMode = self.safe_bool_2(self.options, 'sandboxMode', 'sandbox', False)
|
||
if sandboxMode and (path != 'v2/public/time') and (path != 'v3/market/current-fund-rate'):
|
||
# https://github.com/ccxt/ccxt/issues/25252#issuecomment-2662742336
|
||
if headers is None:
|
||
headers = {}
|
||
productType = self.safe_string(params, 'productType')
|
||
if (productType != 'SCOIN-FUTURES') and (productType != 'SUSDT-FUTURES') and (productType != 'SUSDC-FUTURES'):
|
||
headers['PAPTRADING'] = '1'
|
||
return {'url': url, 'method': method, 'body': body, 'headers': headers}
|