Files
ccxt_with_mt5/ccxt/binance.py
lz_db 0fab423a18 add
2025-11-16 12:31:03 +08:00

13487 lines
673 KiB
Python
Raw Blame History

This file contains ambiguous Unicode characters
This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.
# -*- 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.base.exchange import Exchange
from ccxt.abstract.binance import ImplicitAPI
import hashlib
import json
from ccxt.base.types import Any, Balances, BorrowInterest, Conversion, CrossBorrowRate, Currencies, Currency, DepositAddress, Greeks, Int, IsolatedBorrowRate, IsolatedBorrowRates, LedgerEntry, Leverage, Leverages, LeverageTier, LeverageTiers, LongShortRatio, MarginMode, MarginModes, MarginModification, Market, Num, Option, Order, OrderBook, OrderRequest, OrderSide, OrderType, Position, Str, Strings, Ticker, Tickers, FundingRate, FundingRates, Trade, TradingFeeInterface, TradingFees, Transaction, MarketInterface, 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 OperationRejected
from ccxt.base.errors import MarginModeAlreadySet
from ccxt.base.errors import MarketClosed
from ccxt.base.errors import InsufficientFunds
from ccxt.base.errors import InvalidOrder
from ccxt.base.errors import OrderNotFound
from ccxt.base.errors import OrderImmediatelyFillable
from ccxt.base.errors import OrderNotFillable
from ccxt.base.errors import NotSupported
from ccxt.base.errors import OperationFailed
from ccxt.base.errors import DDoSProtection
from ccxt.base.errors import RateLimitExceeded
from ccxt.base.errors import OnMaintenance
from ccxt.base.errors import InvalidNonce
from ccxt.base.errors import RequestTimeout
from ccxt.base.errors import BadResponse
from ccxt.base.decimal_to_precision import TRUNCATE
from ccxt.base.decimal_to_precision import TICK_SIZE
from ccxt.base.precise import Precise
class binance(Exchange, ImplicitAPI):
def describe(self) -> Any:
return self.deep_extend(super(binance, self).describe(), {
'id': 'binance',
'name': 'Binance',
'countries': [], # Japan
'rateLimit': 50,
'certified': True,
'pro': True,
# new metainfo2 interface
'has': {
'CORS': None,
'spot': True,
'margin': True,
'swap': True,
'future': True,
'option': True,
'addMargin': True,
'borrowCrossMargin': True,
'borrowIsolatedMargin': True,
'cancelAllOrders': True,
'cancelOrder': True,
'cancelOrders': True, # contract only
'closeAllPositions': False,
'closePosition': False, # exchange specific closePosition parameter for binance createOrder is not synonymous with how CCXT uses closePositions
'createConvertTrade': True,
'createDepositAddress': False,
'createLimitBuyOrder': True,
'createLimitSellOrder': True,
'createMarketBuyOrder': True,
'createMarketBuyOrderWithCost': True,
'createMarketOrderWithCost': True,
'createMarketSellOrder': True,
'createMarketSellOrderWithCost': True,
'createOrder': True,
'createOrders': True,
'createOrderWithTakeProfitAndStopLoss': False,
'createPostOnlyOrder': True,
'createReduceOnlyOrder': True,
'createStopLimitOrder': True,
'createStopLossOrder': True,
'createStopMarketOrder': False,
'createStopOrder': True,
'createTakeProfitOrder': True,
'createTrailingPercentOrder': True,
'createTriggerOrder': True,
'editOrder': True,
'editOrders': True,
'fetchAccounts': None,
'fetchAllGreeks': True,
'fetchBalance': True,
'fetchBidsAsks': True,
'fetchBorrowInterest': True,
'fetchBorrowRateHistories': False,
'fetchBorrowRateHistory': True,
'fetchCanceledAndClosedOrders': 'emulated',
'fetchCanceledOrders': 'emulated',
'fetchClosedOrder': False,
'fetchClosedOrders': 'emulated',
'fetchConvertCurrencies': True,
'fetchConvertQuote': True,
'fetchConvertTrade': True,
'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': 'emulated',
'fetchFundingIntervals': True,
'fetchFundingRate': True,
'fetchFundingRateHistory': True,
'fetchFundingRates': True,
'fetchGreeks': True,
'fetchIndexOHLCV': True,
'fetchIsolatedBorrowRate': 'emulated',
'fetchIsolatedBorrowRates': True,
'fetchL3OrderBook': False,
'fetchLastPrices': True,
'fetchLedger': True,
'fetchLedgerEntry': True,
'fetchLeverage': 'emulated',
'fetchLeverages': True,
'fetchLeverageTiers': True,
'fetchLiquidations': False,
'fetchLongShortRatio': False,
'fetchLongShortRatioHistory': True,
'fetchMarginAdjustmentHistory': True,
'fetchMarginMode': True,
'fetchMarginModes': True,
'fetchMarketLeverageTiers': 'emulated',
'fetchMarkets': True,
'fetchMarkOHLCV': True,
'fetchMarkPrice': True,
'fetchMarkPrices': True,
'fetchMyLiquidations': True,
'fetchMySettlementHistory': True,
'fetchMyTrades': True,
'fetchOHLCV': True,
'fetchOpenInterest': True,
'fetchOpenInterestHistory': True,
'fetchOpenOrder': True,
'fetchOpenOrders': True,
'fetchOption': True,
'fetchOptionChain': False,
'fetchOrder': True,
'fetchOrderBook': True,
'fetchOrderBooks': False,
'fetchOrders': True,
'fetchOrderTrades': True,
'fetchPosition': True,
'fetchPositionHistory': False,
'fetchPositionMode': True,
'fetchPositions': True,
'fetchPositionsHistory': False,
'fetchPositionsRisk': True,
'fetchPremiumIndexOHLCV': True,
'fetchSettlementHistory': True,
'fetchStatus': True,
'fetchTicker': True,
'fetchTickers': True,
'fetchTime': True,
'fetchTrades': True,
'fetchTradingFee': True,
'fetchTradingFees': True,
'fetchTradingLimits': 'emulated',
'fetchTransactionFee': 'emulated',
'fetchTransactionFees': True,
'fetchTransactions': False,
'fetchTransfer': False,
'fetchTransfers': True,
'fetchUnderlyingAssets': False,
'fetchVolatilityHistory': False,
'fetchWithdrawAddresses': False,
'fetchWithdrawal': False,
'fetchWithdrawals': True,
'fetchWithdrawalWhitelist': False,
'reduceMargin': True,
'repayCrossMargin': True,
'repayIsolatedMargin': True,
'sandbox': True,
'setLeverage': True,
'setMargin': False,
'setMarginMode': True,
'setPositionMode': True,
'signIn': False,
'transfer': True,
'withdraw': True,
},
'timeframes': {
'1s': '1s', # spot only for now
'1m': '1m',
'3m': '3m',
'5m': '5m',
'15m': '15m',
'30m': '30m',
'1h': '1h',
'2h': '2h',
'4h': '4h',
'6h': '6h',
'8h': '8h',
'12h': '12h',
'1d': '1d',
'3d': '3d',
'1w': '1w',
'1M': '1M',
},
'urls': {
'logo': 'https://github.com/user-attachments/assets/e9419b93-ccb0-46aa-9bff-c883f096274b',
'test': {
'dapiPublic': 'https://testnet.binancefuture.com/dapi/v1',
'dapiPrivate': 'https://testnet.binancefuture.com/dapi/v1',
'dapiPrivateV2': 'https://testnet.binancefuture.com/dapi/v2',
'fapiPublic': 'https://testnet.binancefuture.com/fapi/v1',
'fapiPublicV2': 'https://testnet.binancefuture.com/fapi/v2',
'fapiPublicV3': 'https://testnet.binancefuture.com/fapi/v3',
'fapiPrivate': 'https://testnet.binancefuture.com/fapi/v1',
'fapiPrivateV2': 'https://testnet.binancefuture.com/fapi/v2',
'fapiPrivateV3': 'https://testnet.binancefuture.com/fapi/v3',
'public': 'https://testnet.binance.vision/api/v3',
'private': 'https://testnet.binance.vision/api/v3',
'v1': 'https://testnet.binance.vision/api/v1',
},
'demo': {
'dapiPublic': 'https://demo-dapi.binance.com/dapi/v1',
'dapiPrivate': 'https://demo-dapi.binance.com/dapi/v1',
'dapiPrivateV2': 'https://demo-dapi.binance.com/dapi/v2',
'fapiPublic': 'https://demo-fapi.binance.com/fapi/v1',
'fapiPublicV2': 'https://demo-fapi.binance.com/fapi/v2',
'fapiPublicV3': 'https://demo-fapi.binance.com/fapi/v3',
'fapiPrivate': 'https://demo-fapi.binance.com/fapi/v1',
'fapiPrivateV2': 'https://demo-fapi.binance.com/fapi/v2',
'fapiPrivateV3': 'https://demo-fapi.binance.com/fapi/v3',
'public': 'https://demo-api.binance.com/api/v3',
'private': 'https://demo-api.binance.com/api/v3',
'v1': 'https://demo-api.binance.com/api/v1',
},
'api': {
'sapi': 'https://api.binance.com/sapi/v1',
'sapiV2': 'https://api.binance.com/sapi/v2',
'sapiV3': 'https://api.binance.com/sapi/v3',
'sapiV4': 'https://api.binance.com/sapi/v4',
'dapiPublic': 'https://dapi.binance.com/dapi/v1',
'dapiPrivate': 'https://dapi.binance.com/dapi/v1',
'eapiPublic': 'https://eapi.binance.com/eapi/v1',
'eapiPrivate': 'https://eapi.binance.com/eapi/v1',
'dapiPrivateV2': 'https://dapi.binance.com/dapi/v2',
'dapiData': 'https://dapi.binance.com/futures/data',
'fapiPublic': 'https://fapi.binance.com/fapi/v1',
'fapiPublicV2': 'https://fapi.binance.com/fapi/v2',
'fapiPublicV3': 'https://fapi.binance.com/fapi/v3',
'fapiPrivate': 'https://fapi.binance.com/fapi/v1',
'fapiPrivateV2': 'https://fapi.binance.com/fapi/v2',
'fapiPrivateV3': 'https://fapi.binance.com/fapi/v3',
'fapiData': 'https://fapi.binance.com/futures/data',
'public': 'https://api.binance.com/api/v3',
'private': 'https://api.binance.com/api/v3',
'v1': 'https://api.binance.com/api/v1',
'papi': 'https://papi.binance.com/papi/v1',
'papiV2': 'https://papi.binance.com/papi/v2',
},
'www': 'https://www.binance.com',
'referral': {
'url': 'https://accounts.binance.com/register?ref=CCXTCOM',
'discount': 0.1,
},
'doc': [
'https://developers.binance.com/en',
],
'api_management': 'https://www.binance.com/en/usercenter/settings/api-management',
'fees': 'https://www.binance.com/en/fee/schedule',
},
'api': {
# the API structure below will need 3-layer apidefs
'sapi': {
# IP(sapi) request rate limit of 12 000 per minute
# 1 IP(sapi) => cost = 0.1 =>(1000 / (50 * 0.1)) * 60 = 12000
# 10 IP(sapi) => cost = 1
# UID(sapi) request rate limit of 180 000 per minute
# 1 UID(sapi) => cost = 0.006667 =>(1000 / (50 * 0.006667)) * 60 = 180000
'get': {
# copy trading
'copyTrading/futures/userStatus': 2,
'copyTrading/futures/leadSymbol': 2,
'system/status': 0.1,
# these endpoints require self.apiKey
'accountSnapshot': 240, # Weight(IP): 2400 => cost = 0.1 * 2400 = 240
'account/info': 0.1,
'margin/asset': 1, # Weight(IP): 10 => cost = 0.1 * 10 = 1
'margin/pair': 1,
'margin/allAssets': 0.1,
'margin/allPairs': 0.1,
'margin/priceIndex': 1,
# these endpoints require self.apiKey + self.secret
'spot/delist-schedule': 10,
'asset/assetDividend': 1,
'asset/dribblet': 0.1,
'asset/transfer': 0.1,
'asset/assetDetail': 0.1,
'asset/tradeFee': 0.1,
'asset/ledger-transfer/cloud-mining/queryByPage': 4.0002, # Weight(UID): 600 => cost = 0.006667 * 600 = 4.0002
'asset/convert-transfer/queryByPage': 0.033335,
'asset/wallet/balance': 6, # Weight(IP): 60 => cost = 0.1 * 60 = 6
'asset/custody/transfer-history': 6, # Weight(IP): 60 => cost = 0.1 * 60 = 6
'margin/borrow-repay': 1,
'margin/loan': 1,
'margin/repay': 1,
'margin/account': 1,
'margin/transfer': 0.1,
'margin/interestHistory': 0.1,
'margin/forceLiquidationRec': 0.1,
'margin/order': 1,
'margin/openOrders': 1,
'margin/allOrders': 20, # Weight(IP): 200 => cost = 0.1 * 200 = 20
'margin/myTrades': 1,
'margin/maxBorrowable': 5, # Weight(IP): 50 => cost = 0.1 * 50 = 5
'margin/maxTransferable': 5,
'margin/tradeCoeff': 1,
'margin/isolated/transfer': 0.1,
'margin/isolated/account': 1,
'margin/isolated/pair': 1,
'margin/isolated/allPairs': 1,
'margin/isolated/accountLimit': 0.1,
'margin/interestRateHistory': 0.1,
'margin/orderList': 1,
'margin/allOrderList': 20, # Weight(IP): 200 => cost = 0.1 * 200 = 20
'margin/openOrderList': 1,
'margin/crossMarginData': {'cost': 0.1, 'noCoin': 0.5},
'margin/isolatedMarginData': {'cost': 0.1, 'noCoin': 1},
'margin/isolatedMarginTier': 0.1,
'margin/rateLimit/order': 2,
'margin/dribblet': 0.1,
'margin/dust': 20.001, # Weight(UID): 3000 => cost = 0.006667 * 3000 = 20
'margin/crossMarginCollateralRatio': 10,
'margin/exchange-small-liability': 0.6667,
'margin/exchange-small-liability-history': 0.6667,
'margin/next-hourly-interest-rate': 0.6667,
'margin/capital-flow': 10, # Weight(IP): 100 => cost = 0.1 * 100 = 10
'margin/delist-schedule': 10, # Weight(IP): 100 => cost = 0.1 * 100 = 10
'margin/available-inventory': 0.3334, # Weight(UID): 50 => cost = 0.006667 * 50 = 0.3334
'margin/leverageBracket': 0.1, # Weight(IP): 1 => cost = 0.1 * 1 = 0.1
'loan/vip/loanable/data': 40, # Weight(IP): 400 => cost = 0.1 * 400 = 40
'loan/vip/collateral/data': 40, # Weight(IP): 400 => cost = 0.1 * 400 = 40
'loan/vip/request/data': 2.6668, # Weight(UID): 400 => cost = 0.006667 * 400 = 2.6668
'loan/vip/request/interestRate': 2.6668, # Weight(UID): 400 => cost = 0.006667 * 400 = 2.6668
'loan/income': 40.002, # Weight(UID): 6000 => cost = 0.006667 * 6000 = 40.002
'loan/ongoing/orders': 40, # Weight(IP): 400 => cost = 0.1 * 400 = 40
'loan/ltv/adjustment/history': 40, # Weight(IP): 400 => cost = 0.1 * 400 = 40
'loan/borrow/history': 40, # Weight(IP): 400 => cost = 0.1 * 400 = 40
'loan/repay/history': 40, # Weight(IP): 400 => cost = 0.1 * 400 = 40
'loan/loanable/data': 40, # Weight(IP): 400 => cost = 0.1 * 400 = 40
'loan/collateral/data': 40, # Weight(IP): 400 => cost = 0.1 * 400 = 40
'loan/repay/collateral/rate': 600, # Weight(IP): 6000 => cost = 0.1 * 6000 = 600
'loan/flexible/ongoing/orders': 30, # TODO: Deprecating at 2024-04-24 03:00(UTC)
'loan/flexible/borrow/history': 40, # Weight(IP): 400 => cost = 0.1 * 400 = 40, check flexible rate loans order history before 2024-02-27 08:00(UTC)
'loan/flexible/repay/history': 40, # Weight(IP): 400 => cost = 0.1 * 400 = 40, check flexible rate loans order history before 2024-02-27 08:00(UTC)
'loan/flexible/ltv/adjustment/history': 40, # Weight(IP): 400 => cost = 0.1 * 400 = 40, check flexible rate loans order history before 2024-02-27 08:00(UTC)
'loan/vip/ongoing/orders': 40, # Weight(IP): 400 => cost = 0.1 * 400 = 40
'loan/vip/repay/history': 40, # Weight(IP): 400 => cost = 0.1 * 400 = 40
'loan/vip/collateral/account': 600, # Weight(IP): 6000 => cost = 0.1 * 6000 = 600
'fiat/orders': 600.03, # Weight(UID): 90000 => cost = 0.006667 * 90000 = 600.03
'fiat/payments': 0.1,
'futures/transfer': 1,
'futures/histDataLink': 0.1, # Weight(IP): 1 => cost = 0.1 * 1 = 0.1
'rebate/taxQuery': 80.004, # Weight(UID): 12000 => cost = 0.006667 * 12000 = 80.004
'capital/config/getall': 1, # get networks for withdrawing USDT ERC20 vs USDT Omni
'capital/deposit/address': 1,
'capital/deposit/address/list': 1,
'capital/deposit/hisrec': 0.1,
'capital/deposit/subAddress': 0.1,
'capital/deposit/subHisrec': 0.1,
'capital/withdraw/history': 2, # Weight(UID): 18000 + (Additional: 10 requests per second => cost = ( 1000 / rateLimit ) / 10 = 2
'capital/withdraw/address/list': 10,
'capital/contract/convertible-coins': 4.0002, # Weight(UID): 600 => cost = 0.006667 * 600 = 4.0002
'convert/tradeFlow': 20.001, # Weight(UID): 3000 => cost = 0.006667 * 3000 = 20.001
'convert/exchangeInfo': 50,
'convert/assetInfo': 10,
'convert/orderStatus': 0.6667,
'convert/limit/queryOpenOrders': 20.001, # Weight(UID): 3000 => cost = 0.006667 * 3000 = 20.001
'account/status': 0.1,
'account/apiTradingStatus': 0.1,
'account/apiRestrictions/ipRestriction': 0.1,
'bnbBurn': 0.1,
'sub-account/futures/account': 1,
'sub-account/futures/accountSummary': 0.1,
'sub-account/futures/positionRisk': 1,
'sub-account/futures/internalTransfer': 0.1,
'sub-account/list': 0.1,
'sub-account/margin/account': 1,
'sub-account/margin/accountSummary': 1,
'sub-account/spotSummary': 0.1,
'sub-account/status': 1,
'sub-account/sub/transfer/history': 0.1,
'sub-account/transfer/subUserHistory': 0.1,
'sub-account/universalTransfer': 0.1,
'sub-account/apiRestrictions/ipRestriction/thirdPartyList': 1,
'sub-account/transaction-statistics': 0.40002, # Weight(UID): 60 => cost = 0.006667 * 60 = 0.40002
'sub-account/subAccountApi/ipRestriction': 20.001, # Weight(UID): 3000 => cost = 0.006667 * 3000 = 20.001
'managed-subaccount/asset': 0.1,
'managed-subaccount/accountSnapshot': 240,
'managed-subaccount/queryTransLogForInvestor': 0.1,
'managed-subaccount/queryTransLogForTradeParent': 0.40002, # Weight(UID): 60 => cost = 0.006667 * 60 = 0.40002
'managed-subaccount/fetch-future-asset': 0.40002, # Weight(UID): 60 => cost = 0.006667 * 60 = 0.40002
'managed-subaccount/marginAsset': 0.1,
'managed-subaccount/info': 0.40002, # Weight(UID): 60 => cost = 0.006667 * 60 = 0.40002
'managed-subaccount/deposit/address': 0.006667, # Weight(UID): 1 => cost = 0.006667 * 1 = 0.006667
'managed-subaccount/query-trans-log': 0.40002,
# lending endpoints
'lending/daily/product/list': 0.1,
'lending/daily/userLeftQuota': 0.1,
'lending/daily/userRedemptionQuota': 0.1,
'lending/daily/token/position': 0.1,
'lending/union/account': 0.1,
'lending/union/purchaseRecord': 0.1,
'lending/union/redemptionRecord': 0.1,
'lending/union/interestHistory': 0.1,
'lending/project/list': 0.1,
'lending/project/position/list': 0.1,
# eth-staking
'eth-staking/eth/history/stakingHistory': 15, # Weight(IP): 150 => cost = 0.1 * 150 = 15
'eth-staking/eth/history/redemptionHistory': 15, # Weight(IP): 150 => cost = 0.1 * 150 = 15
'eth-staking/eth/history/rewardsHistory': 15, # Weight(IP): 150 => cost = 0.1 * 150 = 15
'eth-staking/eth/quota': 15, # Weight(IP): 150 => cost = 0.1 * 150 = 15
'eth-staking/eth/history/rateHistory': 15, # Weight(IP): 150 => cost = 0.1 * 150 = 15
'eth-staking/account': 15, # Weight(IP): 150 => cost = 0.1 * 150 = 15
'eth-staking/wbeth/history/wrapHistory': 15, # Weight(IP): 150 => cost = 0.1 * 150 = 15
'eth-staking/wbeth/history/unwrapHistory': 15, # Weight(IP): 150 => cost = 0.1 * 150 = 15
'eth-staking/eth/history/wbethRewardsHistory': 15, # Weight(IP): 150 => cost = 0.1 * 150 = 15
'sol-staking/sol/history/stakingHistory': 15,
'sol-staking/sol/history/redemptionHistory': 15,
'sol-staking/sol/history/bnsolRewardsHistory': 15,
'sol-staking/sol/history/rateHistory': 15,
'sol-staking/account': 15,
'sol-staking/sol/quota': 15,
# mining endpoints
'mining/pub/algoList': 0.1,
'mining/pub/coinList': 0.1,
'mining/worker/detail': 0.5, # Weight(IP): 5 => cost = 0.1 * 5 = 0.5
'mining/worker/list': 0.5,
'mining/payment/list': 0.5,
'mining/statistics/user/status': 0.5,
'mining/statistics/user/list': 0.5,
'mining/payment/uid': 0.5,
# liquid swap endpoints
'bswap/pools': 0.1,
'bswap/liquidity': {'cost': 0.1, 'noPoolId': 1},
'bswap/liquidityOps': 20.001, # Weight(UID): 3000 => cost = 0.006667 * 3000 = 20.001
'bswap/quote': 1.00005, # Weight(UID): 150 => cost = 0.006667 * 150 = 1.00005
'bswap/swap': 20.001, # Weight(UID): 3000 => cost = 0.006667 * 3000 = 20.001
'bswap/poolConfigure': 1.00005, # Weight(UID): 150 => cost = 0.006667 * 150 = 1.00005
'bswap/addLiquidityPreview': 1.00005, # Weight(UID): 150 => cost = 0.006667 * 150 = 1.00005
'bswap/removeLiquidityPreview': 1.00005, # Weight(UID): 150 => cost = 0.006667 * 150 = 1.00005
'bswap/unclaimedRewards': 6.667, # Weight(UID): 1000 => cost = 0.006667 * 1000 = 6.667
'bswap/claimedHistory': 6.667, # Weight(UID): 1000 => cost = 0.006667 * 1000 = 6.667
# leveraged token endpoints
'blvt/tokenInfo': 0.1,
'blvt/subscribe/record': 0.1,
'blvt/redeem/record': 0.1,
'blvt/userLimit': 0.1,
# broker api TODO(NOT IN DOCS)
'apiReferral/ifNewUser': 1,
'apiReferral/customization': 1,
'apiReferral/userCustomization': 1,
'apiReferral/rebate/recentRecord': 1,
'apiReferral/rebate/historicalRecord': 1,
'apiReferral/kickback/recentRecord': 1,
'apiReferral/kickback/historicalRecord': 1,
# brokerage API TODO https://binance-docs.github.io/Brokerage-API/General/ does not state ratelimits
'broker/subAccountApi': 1,
'broker/subAccount': 1,
'broker/subAccountApi/commission/futures': 1,
'broker/subAccountApi/commission/coinFutures': 1,
'broker/info': 1,
'broker/transfer': 1,
'broker/transfer/futures': 1,
'broker/rebate/recentRecord': 1,
'broker/rebate/historicalRecord': 1,
'broker/subAccount/bnbBurn/status': 1,
'broker/subAccount/depositHist': 1,
'broker/subAccount/spotSummary': 1,
'broker/subAccount/marginSummary': 1,
'broker/subAccount/futuresSummary': 1,
'broker/rebate/futures/recentRecord': 1,
'broker/subAccountApi/ipRestriction': 1,
'broker/universalTransfer': 1,
# v2 not supported yet
# GET /sapi/v2/broker/subAccount/futuresSummary
'account/apiRestrictions': 0.1,
# c2c / p2p
'c2c/orderMatch/listUserOrderHistory': 0.1,
# nft endpoints
'nft/history/transactions': 20.001, # Weight(UID): 3000 => cost = 0.006667 * 3000 = 20.001
'nft/history/deposit': 20.001,
'nft/history/withdraw': 20.001,
'nft/user/getAsset': 20.001,
'pay/transactions': 20.001,
'giftcard/verify': 0.1,
'giftcard/cryptography/rsa-public-key': 0.1,
'giftcard/buyCode/token-limit': 0.1,
'algo/spot/openOrders': 0.1,
'algo/spot/historicalOrders': 0.1,
'algo/spot/subOrders': 0.1,
'algo/futures/openOrders': 0.1,
'algo/futures/historicalOrders': 0.1,
'algo/futures/subOrders': 0.1,
'portfolio/account': 0.1,
'portfolio/collateralRate': 5,
'portfolio/pmLoan': 3.3335,
'portfolio/interest-history': 0.6667,
'portfolio/asset-index-price': 0.1,
'portfolio/repay-futures-switch': 3, # Weight(IP): 30 => cost = 0.1 * 30 = 3
'portfolio/margin-asset-leverage': 5, # Weight(IP): 50 => cost = 0.1 * 50 = 5
'portfolio/balance': 2,
'portfolio/negative-balance-exchange-record': 2,
'portfolio/pmloan-history': 5,
'portfolio/earn-asset-balance': 150, # Weight(IP): 1500 => cost = 0.1 * 1500 = 150
# staking
'staking/productList': 0.1,
'staking/position': 0.1,
'staking/stakingRecord': 0.1,
'staking/personalLeftQuota': 0.1,
'lending/auto-invest/target-asset/list': 0.1, # Weight(IP): 1 => cost = 0.1 * 1 = 0.1
'lending/auto-invest/target-asset/roi/list': 0.1,
'lending/auto-invest/all/asset': 0.1,
'lending/auto-invest/source-asset/list': 0.1,
'lending/auto-invest/plan/list': 0.1,
'lending/auto-invest/plan/id': 0.1,
'lending/auto-invest/history/list': 0.1,
'lending/auto-invest/index/info': 0.1, # Weight(IP): 1 => cost = 0.1 * 1 = 0.1
'lending/auto-invest/index/user-summary': 0.1, # Weight(IP): 1 => cost = 0.1 * 1 = 0.1
'lending/auto-invest/one-off/status': 0.1, # Weight(IP): 1 => cost = 0.1 * 1 = 0.1
'lending/auto-invest/redeem/history': 0.1, # Weight(IP): 1 => cost = 0.1 * 1 = 0.1
'lending/auto-invest/rebalance/history': 0.1, # Weight(IP): 1 => cost = 0.1 * 1 = 0.1
# simple earn
'simple-earn/flexible/list': 15,
'simple-earn/locked/list': 15,
'simple-earn/flexible/personalLeftQuota': 15,
'simple-earn/locked/personalLeftQuota': 15,
'simple-earn/flexible/subscriptionPreview': 15,
'simple-earn/locked/subscriptionPreview': 15,
'simple-earn/flexible/history/rateHistory': 15,
'simple-earn/flexible/position': 15,
'simple-earn/locked/position': 15,
'simple-earn/account': 15,
'simple-earn/flexible/history/subscriptionRecord': 15,
'simple-earn/locked/history/subscriptionRecord': 15,
'simple-earn/flexible/history/redemptionRecord': 15,
'simple-earn/locked/history/redemptionRecord': 15,
'simple-earn/flexible/history/rewardsRecord': 15,
'simple-earn/locked/history/rewardsRecord': 15,
'simple-earn/flexible/history/collateralRecord': 0.1,
# Convert
'dci/product/list': 0.1,
'dci/product/positions': 0.1,
'dci/product/accounts': 0.1,
},
'post': {
'asset/dust': 0.06667, # Weight(UID): 10 => cost = 0.006667 * 10 = 0.06667
'asset/dust-btc': 0.1,
'asset/transfer': 6.0003, # Weight(UID): 900 => cost = 0.006667 * 900 = 6.0003
'asset/get-funding-asset': 0.1,
'asset/convert-transfer': 0.033335,
'account/disableFastWithdrawSwitch': 0.1,
'account/enableFastWithdrawSwitch': 0.1,
# 'account/apiRestrictions/ipRestriction': 1, discontinued
# 'account/apiRestrictions/ipRestriction/ipList': 1, discontinued
'capital/withdraw/apply': 4.0002, # Weight(UID): 600 => cost = 0.006667 * 600 = 4.0002
'capital/contract/convertible-coins': 4.0002,
'capital/deposit/credit-apply': 0.1, # Weight(IP): 1 => cost = 0.1 * 1 = 0.1
'margin/borrow-repay': 20.001,
'margin/transfer': 4.0002,
'margin/loan': 20.001, # Weight(UID): 3000 => cost = 0.006667 * 3000 = 20.001
'margin/repay': 20.001,
'margin/order': 0.040002, # Weight(UID): 6 => cost = 0.006667 * 6 = 0.040002
'margin/order/oco': 0.040002,
'margin/dust': 20.001, # Weight(UID): 3000 => cost = 0.006667 * 3000 = 20.001
'margin/exchange-small-liability': 20.001,
# 'margin/isolated/create': 1, discontinued
'margin/isolated/transfer': 4.0002, # Weight(UID): 600 => cost = 0.006667 * 600 = 4.0002
'margin/isolated/account': 2.0001, # Weight(UID): 300 => cost = 0.006667 * 300 = 2.0001
'margin/max-leverage': 300, # Weight(IP): 3000 => cost = 0.1 * 3000 = 300
'bnbBurn': 0.1,
'sub-account/virtualSubAccount': 0.1,
'sub-account/margin/transfer': 4.0002, # Weight(UID): 600 => cost = 0.006667 * 600 = 4.0002
'sub-account/margin/enable': 0.1,
'sub-account/futures/enable': 0.1,
'sub-account/futures/transfer': 0.1,
'sub-account/futures/internalTransfer': 0.1,
'sub-account/transfer/subToSub': 0.1,
'sub-account/transfer/subToMaster': 0.1,
'sub-account/universalTransfer': 0.1,
'sub-account/options/enable': 0.1,
'managed-subaccount/deposit': 0.1,
'managed-subaccount/withdraw': 0.1,
'userDataStream': 0.1,
'userDataStream/isolated': 0.1,
'futures/transfer': 0.1,
# lending
'lending/customizedFixed/purchase': 0.1,
'lending/daily/purchase': 0.1,
'lending/daily/redeem': 0.1,
# liquid swap endpoints
'bswap/liquidityAdd': 60, # Weight(UID): 1000 + (Additional: 1 request every 3 seconds = 0.333 requests per second) => cost = ( 1000 / rateLimit ) / 0.333 = 60.0000006
'bswap/liquidityRemove': 60, # Weight(UID): 1000 + (Additional: 1 request every three seconds)
'bswap/swap': 60, # Weight(UID): 1000 + (Additional: 1 request every three seconds)
'bswap/claimRewards': 6.667, # Weight(UID): 1000 => cost = 0.006667 * 1000 = 6.667
# leveraged token endpoints
'blvt/subscribe': 0.1,
'blvt/redeem': 0.1,
# brokerage API TODO: NO MENTION OF RATELIMITS IN BROKERAGE DOCS
'apiReferral/customization': 1,
'apiReferral/userCustomization': 1,
'apiReferral/rebate/historicalRecord': 1,
'apiReferral/kickback/historicalRecord': 1,
'broker/subAccount': 1,
'broker/subAccount/margin': 1,
'broker/subAccount/futures': 1,
'broker/subAccountApi': 1,
'broker/subAccountApi/permission': 1,
'broker/subAccountApi/commission': 1,
'broker/subAccountApi/commission/futures': 1,
'broker/subAccountApi/commission/coinFutures': 1,
'broker/transfer': 1,
'broker/transfer/futures': 1,
'broker/rebate/historicalRecord': 1,
'broker/subAccount/bnbBurn/spot': 1,
'broker/subAccount/bnbBurn/marginInterest': 1,
'broker/subAccount/blvt': 1,
'broker/subAccountApi/ipRestriction': 1,
'broker/subAccountApi/ipRestriction/ipList': 1,
'broker/universalTransfer': 1,
'broker/subAccountApi/permission/universalTransfer': 1,
'broker/subAccountApi/permission/vanillaOptions': 1,
#
'giftcard/createCode': 0.1,
'giftcard/redeemCode': 0.1,
'giftcard/buyCode': 0.1,
'algo/spot/newOrderTwap': 20.001,
'algo/futures/newOrderVp': 20.001,
'algo/futures/newOrderTwap': 20.001,
# staking
'staking/purchase': 0.1,
'staking/redeem': 0.1,
'staking/setAutoStaking': 0.1,
# eth-staking
'eth-staking/eth/stake': 15, # Weight(IP): 150 => cost = 0.1 * 150 = 15
'eth-staking/eth/redeem': 15, # Weight(IP): 150 => cost = 0.1 * 150 = 15
'eth-staking/wbeth/wrap': 15, # Weight(IP): 150 => cost = 0.1 * 150 = 15
'sol-staking/sol/stake': 15,
'sol-staking/sol/redeem': 15,
# mining endpoints
'mining/hash-transfer/config': 0.5, # Weight(IP): 5 => cost = 0.1 * 5 = 0.5
'mining/hash-transfer/config/cancel': 0.5, # Weight(IP): 5 => cost = 0.1 * 5 = 0.5
'portfolio/repay': 20.001,
'loan/vip/renew': 40.002, # Weight(UID): 6000 => cost = 0.006667 * 6000 = 40.002
'loan/vip/borrow': 40.002,
'loan/borrow': 40.002,
'loan/repay': 40.002,
'loan/adjust/ltv': 40.002,
'loan/customize/margin_call': 40.002,
'loan/flexible/repay': 40.002, # TODO: Deprecating at 2024-04-24 03:00(UTC)
'loan/flexible/adjust/ltv': 40.002, # TODO: Deprecating at 2024-04-24 03:00(UTC)
'loan/vip/repay': 40.002,
'convert/getQuote': 1.3334, # Weight(UID): 200 => cost = 0.006667 * 200 = 1.3334
'convert/acceptQuote': 3.3335, # Weight(UID): 500 => cost = 0.006667 * 500 = 3.3335
'convert/limit/placeOrder': 3.3335, # Weight(UID): 500 => cost = 0.006667 * 500 = 3.3335
'convert/limit/cancelOrder': 1.3334, # Weight(UID): 200 => cost = 0.006667 * 200 = 1.3334
'portfolio/auto-collection': 150, # Weight(IP): 1500 => cost = 0.1 * 1500 = 150
'portfolio/asset-collection': 6, # Weight(IP): 60 => cost = 0.1 * 60 = 6
'portfolio/bnb-transfer': 150, # Weight(IP): 1500 => cost = 0.1 * 1500 = 150
'portfolio/repay-futures-switch': 150, # Weight(IP): 1500 => cost = 0.1 * 1500 = 150
'portfolio/repay-futures-negative-balance': 150, # Weight(IP): 1500 => cost = 0.1 * 1500 = 150
'portfolio/mint': 20,
'portfolio/redeem': 20,
'portfolio/earn-asset-transfer': 150, # Weight(IP): 1500 => cost = 0.1 * 1500 = 150
'lending/auto-invest/plan/add': 0.1, # Weight(IP): 1 => cost = 0.1 * 1 = 0.1
'lending/auto-invest/plan/edit': 0.1, # Weight(IP): 1 => cost = 0.1 * 1 = 0.1
'lending/auto-invest/plan/edit-status': 0.1, # Weight(IP): 1 => cost = 0.1 * 1 = 0.1
'lending/auto-invest/one-off': 0.1, # Weight(IP): 1 => cost = 0.1 * 1 = 0.1
'lending/auto-invest/redeem': 0.1, # Weight(IP): 1 => cost = 0.1 * 1 = 0.1
# simple earn
'simple-earn/flexible/subscribe': 0.1,
'simple-earn/locked/subscribe': 0.1,
'simple-earn/flexible/redeem': 0.1,
'simple-earn/locked/redeem': 0.1,
'simple-earn/flexible/setAutoSubscribe': 15,
'simple-earn/locked/setAutoSubscribe': 15,
'simple-earn/locked/setRedeemOption': 5,
# convert
'dci/product/subscribe': 0.1,
'dci/product/auto_compound/edit': 0.1,
},
'put': {
'userDataStream': 0.1,
'userDataStream/isolated': 0.1,
},
'delete': {
# 'account/apiRestrictions/ipRestriction/ipList': 1, discontinued
'margin/openOrders': 0.1,
'margin/order': 0.006667, # Weight(UID): 1 => cost = 0.006667
'margin/orderList': 0.006667,
'margin/isolated/account': 2.0001, # Weight(UID): 300 => cost = 0.006667 * 300 = 2.0001
'userDataStream': 0.1,
'userDataStream/isolated': 0.1,
# brokerage API TODO NO MENTION OF RATELIMIT IN BROKERAGE DOCS
'broker/subAccountApi': 1,
'broker/subAccountApi/ipRestriction/ipList': 1,
'algo/spot/order': 0.1,
'algo/futures/order': 0.1,
'sub-account/subAccountApi/ipRestriction/ipList': 20.001, # Weight(UID): 3000 => cost = 0.006667 * 3000 = 20.001
},
},
'sapiV2': {
'get': {
'eth-staking/account': 15, # Weight(IP): 150 => cost = 0.1 * 150 = 15
'sub-account/futures/account': 0.1,
'sub-account/futures/accountSummary': 1,
'sub-account/futures/positionRisk': 0.1,
'loan/flexible/ongoing/orders': 30, # Weight(IP): 300 => cost = 0.1 * 300 = 30
'loan/flexible/borrow/history': 40, # Weight(IP): 400 => cost = 0.1 * 400 = 40
'loan/flexible/repay/history': 40, # Weight(IP): 400 => cost = 0.1 * 400 = 40
'loan/flexible/ltv/adjustment/history': 40, # Weight(IP): 400 => cost = 0.1 * 400 = 40
'loan/flexible/loanable/data': 40, # Weight(IP): 400 => cost = 0.1 * 400 = 40
'loan/flexible/collateral/data': 40, # Weight(IP): 400 => cost = 0.1 * 400 = 40
'portfolio/account': 2,
},
'post': {
'eth-staking/eth/stake': 15, # Weight(IP): 150 => cost = 0.1 * 150 = 15
'sub-account/subAccountApi/ipRestriction': 20.001, # Weight(UID): 3000 => cost = 0.006667 * 3000 = 20.001
'loan/flexible/borrow': 40.002, # Weight(UID): 6000 => cost = 0.006667 * 6000 = 40.002
'loan/flexible/repay': 40.002, # Weight(UID): 6000 => cost = 0.006667 * 6000 = 40.002
'loan/flexible/adjust/ltv': 40.002, # Weight(UID): 6000 => cost = 0.006667 * 6000 = 40.002
},
},
'sapiV3': {
'get': {
'sub-account/assets': 0.40002, # Weight(UID): 60 => cost = 0.006667 * 60 = 0.40002
},
'post': {
'asset/getUserAsset': 0.5,
},
},
'sapiV4': {
'get': {
'sub-account/assets': 0.40002, # Weight(UID): 60 => cost = 0.006667 * 60 = 0.40002
},
},
'dapiPublic': {
'get': {
'ping': 1,
'time': 1,
'exchangeInfo': 1,
'depth': {'cost': 2, 'byLimit': [[50, 2], [100, 5], [500, 10], [1000, 20]]},
'trades': 5,
'historicalTrades': 20,
'aggTrades': 20,
'premiumIndex': 10,
'fundingRate': 1,
'klines': {'cost': 1, 'byLimit': [[99, 1], [499, 2], [1000, 5], [10000, 10]]},
'continuousKlines': {'cost': 1, 'byLimit': [[99, 1], [499, 2], [1000, 5], [10000, 10]]},
'indexPriceKlines': {'cost': 1, 'byLimit': [[99, 1], [499, 2], [1000, 5], [10000, 10]]},
'markPriceKlines': {'cost': 1, 'byLimit': [[99, 1], [499, 2], [1000, 5], [10000, 10]]},
'premiumIndexKlines': {'cost': 1, 'byLimit': [[99, 1], [499, 2], [1000, 5], [10000, 10]]},
'ticker/24hr': {'cost': 1, 'noSymbol': 40},
'ticker/price': {'cost': 1, 'noSymbol': 2},
'ticker/bookTicker': {'cost': 2, 'noSymbol': 5},
'constituents': 2,
'openInterest': 1,
'fundingInfo': 1,
},
},
'dapiData': {
'get': {
'delivery-price': 1,
'openInterestHist': 1,
'topLongShortAccountRatio': 1,
'topLongShortPositionRatio': 1,
'globalLongShortAccountRatio': 1,
'takerBuySellVol': 1,
'basis': 1,
},
},
'dapiPrivate': {
'get': {
'positionSide/dual': 30,
'orderAmendment': 1,
'order': 1,
'openOrder': 1,
'openOrders': {'cost': 1, 'noSymbol': 5},
'allOrders': {'cost': 20, 'noSymbol': 40},
'balance': 1,
'account': 5,
'positionMargin/history': 1,
'positionRisk': 1,
'userTrades': {'cost': 20, 'noSymbol': 40},
'income': 20,
'leverageBracket': 1,
'forceOrders': {'cost': 20, 'noSymbol': 50},
'adlQuantile': 5,
'commissionRate': 20,
'income/asyn': 5,
'income/asyn/id': 5,
'trade/asyn': 0.5,
'trade/asyn/id': 0.5,
'order/asyn': 0.5,
'order/asyn/id': 0.5,
'pmExchangeInfo': 0.5, # Weight(IP): 5 => cost = 0.1 * 5 = 0.5
'pmAccountInfo': 0.5, # Weight(IP): 5 => cost = 0.1 * 5 = 0.5
},
'post': {
'positionSide/dual': 1,
'order': 4,
'batchOrders': 5,
'countdownCancelAll': 10,
'leverage': 1,
'marginType': 1,
'positionMargin': 1,
'listenKey': 1,
},
'put': {
'listenKey': 1,
'order': 1,
'batchOrders': 5,
},
'delete': {
'order': 1,
'allOpenOrders': 1,
'batchOrders': 5,
'listenKey': 1,
},
},
'dapiPrivateV2': {
'get': {
'leverageBracket': 1,
},
},
'fapiPublic': {
'get': {
'ping': 1,
'time': 1,
'exchangeInfo': 1,
'depth': {'cost': 2, 'byLimit': [[50, 2], [100, 5], [500, 10], [1000, 20]]},
'trades': 5,
'historicalTrades': 20,
'aggTrades': 20,
'klines': {'cost': 1, 'byLimit': [[99, 1], [499, 2], [1000, 5], [10000, 10]]},
'continuousKlines': {'cost': 1, 'byLimit': [[99, 1], [499, 2], [1000, 5], [10000, 10]]},
'markPriceKlines': {'cost': 1, 'byLimit': [[99, 1], [499, 2], [1000, 5], [10000, 10]]},
'indexPriceKlines': {'cost': 1, 'byLimit': [[99, 1], [499, 2], [1000, 5], [10000, 10]]},
'premiumIndexKlines': {'cost': 1, 'byLimit': [[99, 1], [499, 2], [1000, 5], [10000, 10]]},
'fundingRate': 1,
'fundingInfo': 1,
'premiumIndex': 1,
'ticker/24hr': {'cost': 1, 'noSymbol': 40},
'ticker/price': {'cost': 1, 'noSymbol': 2},
'ticker/bookTicker': {'cost': 1, 'noSymbol': 2},
'openInterest': 1,
'indexInfo': 1,
'assetIndex': {'cost': 1, 'noSymbol': 10},
'constituents': 2,
'apiTradingStatus': {'cost': 1, 'noSymbol': 10},
'lvtKlines': 1,
'convert/exchangeInfo': 4,
'insuranceBalance': 1,
},
},
'fapiData': {
'get': {
'delivery-price': 1,
'openInterestHist': 1,
'topLongShortAccountRatio': 1,
'topLongShortPositionRatio': 1,
'globalLongShortAccountRatio': 1,
'takerlongshortRatio': 1,
'basis': 1,
},
},
'fapiPrivate': {
'get': {
'forceOrders': {'cost': 20, 'noSymbol': 50},
'allOrders': 5,
'openOrder': 1,
'openOrders': {'cost': 1, 'noSymbol': 40},
'order': 1,
'account': 5,
'balance': 5,
'leverageBracket': 1,
'positionMargin/history': 1,
'positionRisk': 5,
'positionSide/dual': 30,
'userTrades': 5,
'income': 30,
'commissionRate': 20,
'rateLimit/order': 1,
'apiTradingStatus': 1,
'multiAssetsMargin': 30,
# broker endpoints
'apiReferral/ifNewUser': 1,
'apiReferral/customization': 1,
'apiReferral/userCustomization': 1,
'apiReferral/traderNum': 1,
'apiReferral/overview': 1,
'apiReferral/tradeVol': 1,
'apiReferral/rebateVol': 1,
'apiReferral/traderSummary': 1,
'adlQuantile': 5,
'pmAccountInfo': 5,
'orderAmendment': 1,
'income/asyn': 1000,
'income/asyn/id': 10,
'order/asyn': 1000,
'order/asyn/id': 10,
'trade/asyn': 1000,
'trade/asyn/id': 10,
'feeBurn': 1,
'symbolConfig': 5,
'accountConfig': 5,
'convert/orderStatus': 5,
},
'post': {
'batchOrders': 5,
'positionSide/dual': 1,
'positionMargin': 1,
'marginType': 1,
'order': 4,
'leverage': 1,
'listenKey': 1,
'countdownCancelAll': 10,
'multiAssetsMargin': 1,
# broker endpoints
'apiReferral/customization': 1,
'apiReferral/userCustomization': 1,
'feeBurn': 1,
'convert/getQuote': 200, # 360 requests per hour
'convert/acceptQuote': 20,
},
'put': {
'listenKey': 1,
'order': 1,
'batchOrders': 5,
},
'delete': {
'batchOrders': 1,
'order': 1,
'allOpenOrders': 1,
'listenKey': 1,
},
},
'fapiPublicV2': {
'get': {
'ticker/price': 0,
},
},
'fapiPrivateV2': {
'get': {
'account': 1,
'balance': 1,
'positionRisk': 1,
},
},
'fapiPublicV3': {
'get': {},
},
'fapiPrivateV3': {
'get': {
'account': 1,
'balance': 1,
'positionRisk': 1,
},
},
'eapiPublic': {
'get': {
'ping': 1,
'time': 1,
'exchangeInfo': 1,
'index': 1,
'ticker': 5,
'mark': 5,
'depth': 1,
'klines': 1,
'trades': 5,
'historicalTrades': 20,
'exerciseHistory': 3,
'openInterest': 3,
},
},
'eapiPrivate': {
'get': {
'account': 3,
'position': 5,
'openOrders': {'cost': 1, 'noSymbol': 40},
'historyOrders': 3,
'userTrades': 5,
'exerciseRecord': 5,
'bill': 1,
'income/asyn': 5,
'income/asyn/id': 5,
'marginAccount': 3,
'mmp': 1,
'countdownCancelAll': 1,
'order': 1,
'block/order/orders': 5,
'block/order/execute': 5,
'block/user-trades': 5,
'blockTrades': 5,
},
'post': {
'order': 1,
'batchOrders': 5,
'listenKey': 1,
'mmpSet': 1,
'mmpReset': 1,
'countdownCancelAll': 1,
'countdownCancelAllHeartBeat': 10,
'block/order/create': 5,
'block/order/execute': 5,
},
'put': {
'listenKey': 1,
'block/order/create': 5,
},
'delete': {
'order': 1,
'batchOrders': 1,
'allOpenOrders': 1,
'allOpenOrdersByUnderlying': 1,
'listenKey': 1,
'block/order/create': 5,
},
},
'public': {
# IP(api) request rate limit of 6000 per minute
# 1 IP(api) => cost = 0.2 =>(1000 / (50 * 0.2)) * 60 = 6000
'get': {
'ping': 0.2, # Weight(IP): 1 => cost = 0.2 * 1 = 0.2
'time': 0.2,
'depth': {'cost': 1, 'byLimit': [[100, 1], [500, 5], [1000, 10], [5000, 50]]},
'trades': 2, # Weight(IP): 10 => cost = 0.2 * 10 = 2
'aggTrades': 0.4,
'historicalTrades': 2, # Weight(IP): 10 => cost = 0.2 * 10 = 2
'klines': 0.4,
'uiKlines': 0.4,
'ticker/24hr': {'cost': 0.4, 'noSymbol': 16},
'ticker': {'cost': 0.4, 'noSymbol': 16},
'ticker/tradingDay': 0.8,
'ticker/price': {'cost': 0.4, 'noSymbol': 0.8},
'ticker/bookTicker': {'cost': 0.4, 'noSymbol': 0.8},
'exchangeInfo': 4, # Weight(IP): 20 => cost = 0.2 * 20 = 4
'avgPrice': 0.4,
},
'put': {
'userDataStream': 0.4,
},
'post': {
'userDataStream': 0.4,
},
'delete': {
'userDataStream': 0.4,
},
},
'private': {
'get': {
'allOrderList': 4, # oco Weight(IP): 20 => cost = 0.2 * 20 = 4
'openOrderList': 1.2, # oco Weight(IP): 6 => cost = 0.2 * 6 = 1.2
'orderList': 0.8, # oco
'order': 0.8,
'openOrders': {'cost': 1.2, 'noSymbol': 16},
'allOrders': 4,
'account': 4,
'myTrades': 4,
'rateLimit/order': 8, # Weight(IP): 40 => cost = 0.2 * 40 = 8
'myPreventedMatches': 4, # Weight(IP): 20 => cost = 0.2 * 20 = 4
'myAllocations': 4,
'account/commission': 4,
},
'post': {
'order/oco': 0.2,
'orderList/oco': 0.2,
'orderList/oto': 0.2,
'orderList/otoco': 0.2,
'sor/order': 0.2,
'sor/order/test': 0.2,
'order': 0.2,
'order/cancelReplace': 0.2,
'order/test': 0.2,
},
'delete': {
'openOrders': 0.2,
'orderList': 0.2, # oco
'order': 0.2,
},
},
'papi': {
# IP(papi) request rate limit of 6000 per minute
# 1 IP(papi) => cost = 0.2 =>(1000 / (50 * 0.2)) * 60 = 6000
# Order(papi) request rate limit of 1200 per minute
# 1 Order(papi) => cost = 1 =>(1000 / (50 * 1)) * 60 = 1200
'get': {
'ping': 0.2,
'um/order': 1,
'um/openOrder': 1,
'um/openOrders': {'cost': 1, 'noSymbol': 40},
'um/allOrders': 5,
'cm/order': 1,
'cm/openOrder': 1,
'cm/openOrders': {'cost': 1, 'noSymbol': 40},
'cm/allOrders': 20,
'um/conditional/openOrder': 1,
'um/conditional/openOrders': {'cost': 1, 'noSymbol': 40},
'um/conditional/orderHistory': 1,
'um/conditional/allOrders': {'cost': 1, 'noSymbol': 40},
'cm/conditional/openOrder': 1,
'cm/conditional/openOrders': {'cost': 1, 'noSymbol': 40},
'cm/conditional/orderHistory': 1,
'cm/conditional/allOrders': 40,
'margin/order': 10,
'margin/openOrders': 5,
'margin/allOrders': 100,
'margin/orderList': 5,
'margin/allOrderList': 100,
'margin/openOrderList': 5,
'margin/myTrades': 5,
'balance': 4,
'account': 4,
'margin/maxBorrowable': 1,
'margin/maxWithdraw': 1,
'um/positionRisk': 1,
'cm/positionRisk': 0.2,
'um/positionSide/dual': 6,
'cm/positionSide/dual': 6,
'um/userTrades': 5,
'cm/userTrades': 20,
'um/leverageBracket': 0.2,
'cm/leverageBracket': 0.2,
'margin/forceOrders': 1,
'um/forceOrders': {'cost': 20, 'noSymbol': 50},
'cm/forceOrders': {'cost': 20, 'noSymbol': 50},
'um/apiTradingStatus': {'cost': 0.2, 'noSymbol': 2},
'um/commissionRate': 4,
'cm/commissionRate': 4,
'margin/marginLoan': 2,
'margin/repayLoan': 2,
'margin/marginInterestHistory': 0.2,
'portfolio/interest-history': 10,
'um/income': 6,
'cm/income': 6,
'um/account': 1,
'cm/account': 1,
'repay-futures-switch': 6,
'um/adlQuantile': 5,
'cm/adlQuantile': 5,
'um/trade/asyn': 300,
'um/trade/asyn/id': 2,
'um/order/asyn': 300,
'um/order/asyn/id': 2,
'um/income/asyn': 300,
'um/income/asyn/id': 2,
'um/orderAmendment': 1,
'cm/orderAmendment': 1,
'um/feeBurn': 30,
'um/accountConfig': 1,
'um/symbolConfig': 1,
'cm/accountConfig': 1,
'cm/symbolConfig': 1,
'rateLimit/order': 1,
},
'post': {
'um/order': 1,
'um/conditional/order': 1,
'cm/order': 1,
'cm/conditional/order': 1,
'margin/order': 1,
'marginLoan': 100,
'repayLoan': 100,
'margin/order/oco': 1,
'um/leverage': 0.2,
'cm/leverage': 0.2,
'um/positionSide/dual': 0.2,
'cm/positionSide/dual': 0.2,
'auto-collection': 150,
'bnb-transfer': 150,
'repay-futures-switch': 150,
'repay-futures-negative-balance': 150,
'listenKey': 0.2,
'asset-collection': 6,
'margin/repay-debt': 3000,
'um/feeBurn': 1,
},
'put': {
'listenKey': 0.2,
'um/order': 1,
'cm/order': 1,
},
'delete': {
'um/order': 1,
'um/conditional/order': 1,
'um/allOpenOrders': 1,
'um/conditional/allOpenOrders': 1,
'cm/order': 1,
'cm/conditional/order': 1,
'cm/allOpenOrders': 1,
'cm/conditional/allOpenOrders': 1,
'margin/order': 2,
'margin/allOpenOrders': 5,
'margin/orderList': 2,
'listenKey': 0.2,
},
},
'papiV2': {
'get': {
'um/account': 1,
},
},
},
'fees': {
'trading': {
'feeSide': 'get',
'tierBased': False,
'percentage': True,
'taker': self.parse_number('0.001'),
'maker': self.parse_number('0.001'),
},
'linear': {
'trading': {
'feeSide': 'quote',
'tierBased': True,
'percentage': True,
'taker': self.parse_number('0.000500'),
'maker': self.parse_number('0.000200'),
'tiers': {
'taker': [
[self.parse_number('0'), self.parse_number('0.000400')],
[self.parse_number('250'), self.parse_number('0.000400')],
[self.parse_number('2500'), self.parse_number('0.000350')],
[self.parse_number('7500'), self.parse_number('0.000320')],
[self.parse_number('22500'), self.parse_number('0.000300')],
[self.parse_number('50000'), self.parse_number('0.000270')],
[self.parse_number('100000'), self.parse_number('0.000250')],
[self.parse_number('200000'), self.parse_number('0.000220')],
[self.parse_number('400000'), self.parse_number('0.000200')],
[self.parse_number('750000'), self.parse_number('0.000170')],
],
'maker': [
[self.parse_number('0'), self.parse_number('0.000200')],
[self.parse_number('250'), self.parse_number('0.000160')],
[self.parse_number('2500'), self.parse_number('0.000140')],
[self.parse_number('7500'), self.parse_number('0.000120')],
[self.parse_number('22500'), self.parse_number('0.000100')],
[self.parse_number('50000'), self.parse_number('0.000080')],
[self.parse_number('100000'), self.parse_number('0.000060')],
[self.parse_number('200000'), self.parse_number('0.000040')],
[self.parse_number('400000'), self.parse_number('0.000020')],
[self.parse_number('750000'), self.parse_number('0')],
],
},
},
},
'inverse': {
'trading': {
'feeSide': 'base',
'tierBased': True,
'percentage': True,
'taker': self.parse_number('0.000500'),
'maker': self.parse_number('0.000100'),
'tiers': {
'taker': [
[self.parse_number('0'), self.parse_number('0.000500')],
[self.parse_number('250'), self.parse_number('0.000450')],
[self.parse_number('2500'), self.parse_number('0.000400')],
[self.parse_number('7500'), self.parse_number('0.000300')],
[self.parse_number('22500'), self.parse_number('0.000250')],
[self.parse_number('50000'), self.parse_number('0.000240')],
[self.parse_number('100000'), self.parse_number('0.000240')],
[self.parse_number('200000'), self.parse_number('0.000240')],
[self.parse_number('400000'), self.parse_number('0.000240')],
[self.parse_number('750000'), self.parse_number('0.000240')],
],
'maker': [
[self.parse_number('0'), self.parse_number('0.000100')],
[self.parse_number('250'), self.parse_number('0.000080')],
[self.parse_number('2500'), self.parse_number('0.000050')],
[self.parse_number('7500'), self.parse_number('0.0000030')],
[self.parse_number('22500'), self.parse_number('0')],
[self.parse_number('50000'), self.parse_number('-0.000050')],
[self.parse_number('100000'), self.parse_number('-0.000060')],
[self.parse_number('200000'), self.parse_number('-0.000070')],
[self.parse_number('400000'), self.parse_number('-0.000080')],
[self.parse_number('750000'), self.parse_number('-0.000090')],
],
},
},
},
'option': {},
},
'currencies': {
'BNFCR': self.safe_currency_structure({'id': 'BNFCR', 'code': 'BNFCR', 'precision': self.parse_number('0.001')}),
},
'commonCurrencies': {
'BCC': 'BCC', # kept for backward-compatibility https://github.com/ccxt/ccxt/issues/4848
'YOYO': 'YOYOW',
},
'precisionMode': TICK_SIZE,
# exchange-specific options
'options': {
'sandboxMode': False,
'fetchMargins': True,
'fetchMarkets': {
'types': [
'spot', # allows CORS in browsers
'linear', # allows CORS in browsers
'inverse', # allows CORS in browsers
# 'option', # does not allow CORS, enable outside of the browser only
],
},
'loadAllOptions': False,
'fetchCurrencies': True, # self is a private call and it requires API keys
# 'fetchTradesMethod': 'publicGetAggTrades', # publicGetTrades, publicGetHistoricalTrades, eapiPublicGetTrades
# 'repayCrossMarginMethod': 'papiPostRepayLoan', # papiPostMarginRepayDebt
'defaultTimeInForce': 'GTC', # 'GTC' = Good To Cancel(default), 'IOC' = Immediate Or Cancel
'defaultType': 'spot', # 'spot', 'future', 'margin', 'delivery', 'option'
'defaultSubType': None, # 'linear', 'inverse'
'hasAlreadyAuthenticatedSuccessfully': False,
'warnOnFetchOpenOrdersWithoutSymbol': True,
'currencyToPrecisionRoundingMode': TRUNCATE,
# not an error
# https://github.com/ccxt/ccxt/issues/11268
# https://github.com/ccxt/ccxt/pull/11624
# POST https://fapi.binance.com/fapi/v1/marginType 400 Bad Request
# binanceusdm
'throwMarginModeAlreadySet': False,
'fetchPositions': 'positionRisk', # or 'account' or 'option'
'recvWindow': 10 * 1000, # 10 sec
'timeDifference': 0, # the difference between system clock and Binance clock
'adjustForTimeDifference': False, # controls the adjustment logic upon instantiation
'newOrderRespType': {
'market': 'FULL', # 'ACK' for order id, 'RESULT' for full order or 'FULL' for order with fills
'limit': 'FULL', # we change it from 'ACK' by default to 'FULL'(returns immediately if limit is not hit)
},
'quoteOrderQty': True, # whether market orders support amounts in quote currency
'broker': {
'spot': 'x-TKT5PX2F',
'margin': 'x-TKT5PX2F',
'future': 'x-cvBPrNm9',
'delivery': 'x-xcKtGhcu',
'swap': 'x-cvBPrNm9',
'option': 'x-xcKtGhcu',
'inverse': 'x-xcKtGhcu',
},
'accountsByType': {
'main': 'MAIN',
'spot': 'MAIN',
'funding': 'FUNDING',
'margin': 'MARGIN',
'cross': 'MARGIN',
'future': 'UMFUTURE', # backwards compatibility
'delivery': 'CMFUTURE', # backwards compatbility
'linear': 'UMFUTURE',
'swap': 'UMFUTURE',
'inverse': 'CMFUTURE',
'option': 'OPTION',
},
'accountsById': {
'MAIN': 'spot',
'FUNDING': 'funding',
'MARGIN': 'margin',
'UMFUTURE': 'linear',
'CMFUTURE': 'inverse',
'OPTION': 'option',
},
'networks': {
'ERC20': 'ETH',
'TRC20': 'TRX',
'BEP2': 'BNB',
'BEP20': 'BSC',
'OMNI': 'OMNI',
'EOS': 'EOS',
'SPL': 'SOL', # temporarily keep support for SPL(old name)
'SOL': 'SOL', # we shouldn't rename SOL
},
'networksById': {
'SOL': 'SOL', # temporary fix for SPL definition
},
'impliedNetworks': {
'ETH': {'ERC20': 'ETH'},
'TRX': {'TRC20': 'TRX'},
},
'legalMoney': {
'MXN': True,
'UGX': True,
'SEK': True,
'CHF': True,
'VND': True,
'AED': True,
'DKK': True,
'KZT': True,
'HUF': True,
'PEN': True,
'PHP': True,
'USD': True,
'TRY': True,
'EUR': True,
'NGN': True,
'PLN': True,
'BRL': True,
'ZAR': True,
'KES': True,
'ARS': True,
'RUB': True,
'AUD': True,
'NOK': True,
'CZK': True,
'GBP': True,
'UAH': True,
'GHS': True,
'HKD': True,
'CAD': True,
'INR': True,
'JPY': True,
'NZD': True,
},
'legalMoneyCurrenciesById': {
'BUSD': 'USD',
},
'defaultWithdrawPrecision': 0.00000001,
},
'features': {
'spot': {
'sandbox': True,
'fetchCurrencies': {
'private': True,
},
'createOrder': {
'marginMode': True,
'triggerPrice': True,
'triggerPriceType': None,
'triggerDirection': False,
'stopLossPrice': True,
'takeProfitPrice': True,
'attachedStopLossTakeProfit': None,
'timeInForce': {
'IOC': True,
'FOK': True,
'PO': True,
'GTD': False,
},
'hedged': True,
'leverage': False,
'marketBuyByCost': True,
'marketBuyRequiresPrice': False,
'selfTradePrevention': {
'EXPIRE_MAKER': True,
'EXPIRE_TAKER': True,
'EXPIRE_BOTH': True,
'NONE': True,
},
'trailing': False, # todo: self is different from standard trailing https://github.com/binance/binance-spot-api-docs/blob/master/faqs/trailing-stop-faq.md
'icebergAmount': True,
},
'createOrders': None,
'fetchMyTrades': {
'marginMode': False,
'limit': 1000,
'daysBack': None,
'untilDays': 1, # days between start-end
'symbolRequired': True,
},
'fetchOrder': {
'marginMode': True,
'trigger': False,
'trailing': False,
'symbolRequired': True,
},
'fetchOpenOrders': {
'marginMode': True,
'limit': None,
'trigger': False,
'trailing': False,
'symbolRequired': False,
},
'fetchOrders': {
'marginMode': True,
'limit': 1000,
'daysBack': None,
'untilDays': 10000,
'trigger': False,
'trailing': False,
'symbolRequired': True,
},
'fetchClosedOrders': {
'marginMode': True,
'limit': 1000,
'daysBack': None,
'daysBackCanceled': None,
'untilDays': 10000,
'trigger': False,
'trailing': False,
'symbolRequired': True,
},
'fetchOHLCV': {
'limit': 1000,
},
},
'forDerivatives': {
'sandbox': True,
'createOrder': {
'marginMode': False,
'triggerPrice': True,
'triggerPriceType': {
'mark': True,
'last': True,
'index': False,
},
'stopLossPrice': True,
'takeProfitPrice': True,
'attachedStopLossTakeProfit': None, # not supported
'timeInForce': {
'IOC': True,
'FOK': True,
'PO': True,
'GTD': True,
# 'GTX': True,
},
'hedged': True,
# exchange-supported features
'selfTradePrevention': True,
'trailing': True,
'iceberg': False,
'leverage': False,
'marketBuyRequiresPrice': False,
'marketBuyByCost': True,
},
'createOrders': {
'max': 5,
},
'fetchMyTrades': {
'marginMode': False,
'daysBack': None,
'limit': 1000,
'untilDays': 7,
'symbolRequired': True,
},
'fetchOrder': {
'marginMode': False,
'trigger': False,
'trailing': False,
'symbolRequired': True,
},
'fetchOpenOrders': {
'marginMode': True,
'limit': 500,
'trigger': False,
'trailing': False,
'symbolRequired': False,
},
'fetchOrders': {
'marginMode': True,
'limit': 1000,
'daysBack': 90,
'untilDays': 7,
'trigger': False,
'trailing': False,
'symbolRequired': True,
},
'fetchClosedOrders': {
'marginMode': True,
'limit': 1000,
'daysBack': 90,
'daysBackCanceled': 3,
'untilDays': 7,
'trigger': False,
'trailing': False,
'symbolRequired': True,
},
'fetchOHLCV': {
'limit': 1500,
},
},
'swap': {
'linear': {
'extends': 'forDerivatives',
},
'inverse': {
'extends': 'forDerivatives',
},
},
'future': {
'linear': {
'extends': 'forDerivatives',
},
'inverse': {
'extends': 'forDerivatives',
},
},
},
'exceptions': {
'spot': {
'exact': {
#
# 1xxx
#
'-1004': OperationFailed, # {"code":-1004,"msg":"Server is busy, please wait and try again"}
'-1008': OperationFailed, # undocumented, but mentioned: This is sent whenever the servers are overloaded with requests.
'-1099': AuthenticationError, # {"code":-1099,"msg":"Not found, authenticated, or authorized"}
'-1108': BadRequest, # undocumented, but mentioned: This error will occur if a value to a parameter being sent was too large, potentially causing overflow
'-1131': BadRequest, # {"code":-1131,"msg":"recvWindow must be less than 60000"}
'-1134': BadRequest, # strategyType was less than 1000000.
'-1135': BadRequest, # undocumented, but mentioned: This error code will occur if a parameter requiring a JSON object is invalid.
'-1145': BadRequest, # cancelRestrictions has to be either ONLY_NEW or ONLY_PARTIALLY_FILLED.
'-1151': BadSymbol, # Symbol is present multiple times in the list.
#
# 2xxx
#
'-2008': AuthenticationError, # undocumented, Invalid Api-Key ID
'-2016': OperationRejected, # {"code":-2016,"msg":"No trading window could be found for the symbol. Try ticker/24hrs instead."}
'-2021': BadResponse, # This code is sent when either the cancellation of the order failed or the new order placement failed but not both.
'-2022': BadResponse, # This code is sent when both the cancellation of the order failed and the new order placement failed.
'-2026': InvalidOrder, # Order was canceled or expired with no executed qty over 90 days ago and has been archived.
#
# 3xxx(these errors are available only for spot atm)
#
'-3000': OperationFailed, # {"code":-3000,"msg":"Internal server error."}
'-3001': AuthenticationError, # {"code":-3001,"msg":"Please enable 2FA first."}
'-3002': BadSymbol, # {"code":-3002,"msg":"We don't have self asset."}
'-3003': BadRequest, # {"code":-3003,"msg":"Margin account does not exist."}
'-3004': OperationRejected, # {"code":-3004,"msg":"Trade not allowed."}
'-3005': BadRequest, # {"code":-3005,"msg":"Transferring out not allowed. Transfer out amount exceeds max amount."}
'-3006': BadRequest, # {"code":-3006,"msg":"Your borrow amount has exceed maximum borrow amount."}
'-3007': OperationFailed, # {"code":-3007,"msg":"You have pending transaction, please try again later.."}
'-3008': BadRequest, # {"code":-3008,"msg":"Borrow not allowed. Your borrow amount has exceed maximum borrow amount."}
'-3009': OperationRejected, # {"code":-3009,"msg":"This asset are not allowed to transfer into margin account currently."}
'-3010': BadRequest, # {"code":-3010,"msg":"Repay not allowed. Repay amount exceeds borrow amount."}
'-3011': BadRequest, # {"code":-3011,"msg":"Your input date is invalid."}
'-3012': OperationRejected, # {"code":-3012,"msg":"Borrow is banned for self asset."}
'-3013': BadRequest, # {"code":-3013,"msg":"Borrow amount less than minimum borrow amount."}
'-3014': AccountSuspended, # {"code":-3014,"msg":"Borrow is banned for self account."}
'-3015': BadRequest, # {"code":-3015,"msg":"Repay amount exceeds borrow amount."}
'-3016': BadRequest, # {"code":-3016,"msg":"Repay amount less than minimum repay amount."}
'-3017': OperationRejected, # {"code":-3017,"msg":"This asset are not allowed to transfer into margin account currently."}
'-3018': AccountSuspended, # {"code":-3018,"msg":"Transferring in has been banned for self account."}
'-3019': AccountSuspended, # {"code":-3019,"msg":"Transferring out has been banned for self account."}
'-3020': BadRequest, # {"code":-3020,"msg":"Transfer out amount exceeds max amount."}
'-3021': BadRequest, # {"code":-3021,"msg":"Margin account are not allowed to trade self trading pair."}
'-3022': AccountSuspended, # {"code":-3022,"msg":"You account's trading is banned."}
'-3023': OperationRejected, # {"code":-3023,"msg":"You can't transfer out/place order under current margin level."}
'-3024': OperationRejected, # {"code":-3024,"msg":"The unpaid debt is too small after self repayment."}
'-3025': BadRequest, # {"code":-3025,"msg":"Your input date is invalid."}
'-3026': BadRequest, # {"code":-3026,"msg":"Your input param is invalid."}
'-3027': BadSymbol, # {"code":-3027,"msg":"Not a valid margin asset."}
'-3028': BadSymbol, # {"code":-3028,"msg":"Not a valid margin pair."}
'-3029': OperationFailed, # {"code":-3029,"msg":"Transfer failed."}
'-3036': AccountSuspended, # {"code":-3036,"msg":"This account is not allowed to repay."}
'-3037': OperationFailed, # {"code":-3037,"msg":"PNL is clearing. Wait a second."}
'-3038': BadRequest, # {"code":-3038,"msg":"Listen key not found."}
'-3041': InsufficientFunds, # {"code":-3041,"msg":"Balance is not enough"}
'-3042': BadRequest, # {"code":-3042,"msg":"PriceIndex not available for self margin pair."}
'-3043': PermissionDenied, # {"code":-3043,"msg":"Transferring in not allowed."}
'-3044': OperationFailed, # {"code":-3044,"msg":"System busy."}
'-3045': OperationRejected, # {"code":-3045,"msg":"The system doesn't have enough asset now."}
'-3999': PermissionDenied, # {"code":-3999,"msg":"This function is only available for invited users."}
#
# 4xxx(different from contract markets)
#
'-4000': ExchangeError, # override commons
'-4001': BadRequest, # {"code":-4001 ,"msg":"Invalid operation."}
'-4002': BadRequest, # {"code":-4002 ,"msg":"Invalid get."}
'-4003': BadRequest, # {"code":-4003 ,"msg":"Your input email is invalid."}
'-4004': AuthenticationError, # {"code":-4004,"msg":"You don't login or auth."}
'-4005': RateLimitExceeded, # {"code":-4005 ,"msg":"Too many new requests."}
'-4006': BadRequest, # {"code":-4006 ,"msg":"Support main account only."}
'-4007': PermissionDenied, # {"code":-4007 ,"msg":"Address validation is not passed."}
'-4008': PermissionDenied, # {"code":-4008 ,"msg":"Address tag validation is not passed."}
'-4009': ExchangeError, # undocumented
'-4010': PermissionDenied, # {"code":-4010 ,"msg":"White list mail has been confirmed."} # [TODO] possible bug: it should probably be "has not been confirmed"
'-4011': BadRequest, # {"code":-4011 ,"msg":"White list mail is invalid."}
'-4012': PermissionDenied, # {"code":-4012 ,"msg":"White list is not opened."}
'-4013': AuthenticationError, # {"code":-4013 ,"msg":"2FA is not opened."}
'-4014': OperationRejected, # {"code":-4014 ,"msg":"Withdraw is not allowed within 2 min login."}
'-4015': PermissionDenied, # {"code":-4015 ,"msg":"Withdraw is limited."}
'-4016': PermissionDenied, # {"code":-4016 ,"msg":"Within 24 hours after password modification, withdrawal is prohibited."}
'-4017': PermissionDenied, # {"code":-4017 ,"msg":"Within 24 hours after the release of 2FA, withdrawal is prohibited."}
'-4018': BadSymbol, # {"code":-4018,"msg":"We don't have self asset."}
'-4019': BadRequest, # {"code":-4019,"msg":"Current asset is not open for withdrawal."}
'-4020': ExchangeError, # override commons
'-4021': BadRequest, # {"code":-4021,"msg":"Asset withdrawal must be an %s multiple of %s."}
'-4022': BadRequest, # {"code":-4022,"msg":"Not less than the minimum pick-up quantity %s."}
'-4023': OperationRejected, # {"code":-4023,"msg":"Within 24 hours, the withdrawal exceeds the maximum amount."}
'-4024': InsufficientFunds, # {"code":-4024,"msg":"You don't have self asset."}
'-4025': InsufficientFunds, # {"code":-4025,"msg":"The number of hold asset is less than zero."}
'-4026': InsufficientFunds, # {"code":-4026,"msg":"You have insufficient balance."}
'-4027': OperationFailed, # {"code":-4027,"msg":"Failed to obtain tranId."}
'-4028': BadRequest, # {"code":-4028,"msg":"The amount of withdrawal must be greater than the Commission."}
'-4029': BadRequest, # {"code":-4029,"msg":"The withdrawal record does not exist."}
'-4030': BadResponse, # {"code":-4030,"msg":"Confirmation of successful asset withdrawal. [TODO] possible bug in docs"}
'-4031': OperationFailed, # {"code":-4031,"msg":"Cancellation failed."}
'-4032': OperationRejected, # {"code":-4032,"msg":"Withdraw verification exception."}
'-4033': BadRequest, # {"code":-4033,"msg":"Illegal address."}
'-4034': OperationRejected, # {"code":-4034,"msg":"The address is suspected of fake."}
'-4035': PermissionDenied, # {"code":-4035,"msg":"This address is not on the whitelist. Please join and try again."}
'-4036': PermissionDenied, # {"code":-4036,"msg":"The new address needs to be withdrawn in {0} hours."}
'-4037': OperationFailed, # {"code":-4037,"msg":"Re-sending Mail failed."}
'-4038': OperationFailed, # {"code":-4038,"msg":"Please try again in 5 minutes."}
'-4039': PermissionDenied, # {"code":-4039,"msg":"The user does not exist."}
'-4040': OperationRejected, # {"code":-4040,"msg":"This address not charged."}
'-4041': OperationFailed, # {"code":-4041,"msg":"Please try again in one minute."}
'-4042': OperationRejected, # {"code":-4042,"msg":"This asset cannot get deposit address again."}
'-4043': OperationRejected, # {"code":-4043,"msg":"More than 100 recharge addresses were used in 24 hours."}
'-4044': PermissionDenied, # {"code":-4044,"msg":"This is a blacklist country."}
'-4045': OperationFailed, # {"code":-4045,"msg":"Failure to acquire assets."}
'-4046': AuthenticationError, # {"code":-4046,"msg":"Agreement not confirmed."}
'-4047': BadRequest, # {"code":-4047,"msg":"Time interval must be within 0-90 days"}
'-4048': ExchangeError, # override commons
'-4049': ExchangeError, # override commons
'-4050': ExchangeError, # override commons
'-4051': ExchangeError, # override commons
'-4052': ExchangeError, # override commons
'-4053': ExchangeError, # override commons
'-4054': ExchangeError, # override commons
'-4055': ExchangeError, # override commons
'-4056': ExchangeError, # override commons
'-4057': ExchangeError, # override commons
'-4058': ExchangeError, # override commons
'-4059': ExchangeError, # override commons
'-4060': OperationFailed, # As your deposit has not reached the required block confirmations, we have temporarily locked {0} asset
'-4061': ExchangeError, # override commons
'-4062': ExchangeError, # override commons
'-4063': ExchangeError, # override commons
'-4064': ExchangeError, # override commons
'-4065': ExchangeError, # override commons
'-4066': ExchangeError, # override commons
'-4067': ExchangeError, # override commons
'-4068': ExchangeError, # override commons
'-4069': ExchangeError, # override commons
'-4070': ExchangeError, # override commons
'-4071': ExchangeError, # override commons
'-4072': ExchangeError, # override commons
'-4073': ExchangeError, # override commons
'-4074': ExchangeError, # override commons
'-4075': ExchangeError, # override commons
'-4076': ExchangeError, # override commons
'-4077': ExchangeError, # override commons
'-4078': ExchangeError, # override commons
'-4079': ExchangeError, # override commons
'-4080': ExchangeError, # override commons
'-4081': ExchangeError, # override commons
'-4082': ExchangeError, # override commons
'-4083': ExchangeError, # override commons
'-4084': ExchangeError, # override commons
'-4085': ExchangeError, # override commons
'-4086': ExchangeError, # override commons
'-4087': ExchangeError, # override commons
'-4088': ExchangeError, # override commons
'-4089': ExchangeError, # override commons
'-4091': ExchangeError, # override commons
'-4092': ExchangeError, # override commons
'-4093': ExchangeError, # override commons
'-4094': ExchangeError, # override commons
'-4095': ExchangeError, # override commons
'-4096': ExchangeError, # override commons
'-4097': ExchangeError, # override commons
'-4098': ExchangeError, # override commons
'-4099': ExchangeError, # override commons
'-4101': ExchangeError, # override commons
'-4102': ExchangeError, # override commons
'-4103': ExchangeError, # override commons
'-4104': ExchangeError, # override commons
'-4105': ExchangeError, # override commons
'-4106': ExchangeError, # override commons
'-4107': ExchangeError, # override commons
'-4108': ExchangeError, # override commons
'-4109': ExchangeError, # override commons
'-4110': ExchangeError, # override commons
'-4112': ExchangeError, # override commons
'-4113': ExchangeError, # override commons
'-4114': ExchangeError, # override commons
'-4115': ExchangeError, # override commons
'-4116': ExchangeError, # override commons
'-4117': ExchangeError, # override commons
'-4118': ExchangeError, # override commons
'-4119': ExchangeError, # override commons
'-4120': ExchangeError, # override commons
'-4121': ExchangeError, # override commons
'-4122': ExchangeError, # override commons
'-4123': ExchangeError, # override commons
'-4124': ExchangeError, # override commons
'-4125': ExchangeError, # override commons
'-4126': ExchangeError, # override commons
'-4127': ExchangeError, # override commons
'-4128': ExchangeError, # override commons
'-4129': ExchangeError, # override commons
'-4130': ExchangeError, # override commons
'-4131': ExchangeError, # override commons
'-4132': ExchangeError, # override commons
'-4133': ExchangeError, # override commons
'-4134': ExchangeError, # override commons
'-4135': ExchangeError, # override commons
'-4136': ExchangeError, # override commons
'-4137': ExchangeError, # override commons
'-4138': ExchangeError, # override commons
'-4139': ExchangeError, # override commons
'-4141': ExchangeError, # override commons
'-4142': ExchangeError, # override commons
'-4143': ExchangeError, # override commons
'-4144': ExchangeError, # override commons
'-4145': ExchangeError, # override commons
'-4146': ExchangeError, # override commons
'-4147': ExchangeError, # override commons
'-4148': ExchangeError, # override commons
'-4149': ExchangeError, # override commons
'-4150': ExchangeError, # override commons
#
# 5xxx
#
'-5001': BadRequest, # Don't allow transfer to micro assets.
'-5002': InsufficientFunds, # You have insufficient balance.
'-5003': InsufficientFunds, # You don't have self asset.
'-5004': OperationRejected, # The residual balances of %s have exceeded 0.001BTC, Please re-choose.
'-5005': OperationRejected, # The residual balances of %s is too low, Please re-choose.
'-5006': OperationRejected, # Only transfer once in 24 hours.
'-5007': BadRequest, # Quantity must be greater than zero.
'-5008': OperationRejected, # Insufficient amount of returnable assets.
'-5009': BadSymbol, # Product does not exist.
'-5010': OperationFailed, # Asset transfer fail.
'-5011': BadRequest, # future account not exists.
'-5012': OperationFailed, # Asset transfer is in pending.
'-5013': InsufficientFunds, # {"code":-5013,"msg":"Asset transfer failed: insufficient balance""} # undocumented
'-5021': BadRequest, # This parent sub have no relation
'-5022': BadRequest, # future account or sub relation not exists.
#
# 6xxx
#
'-6001': BadSymbol, # Daily product not exists.
'-6003': PermissionDenied, # Product not exist or you don't have permission
'-6004': BadRequest, # Product not in purchase status
'-6005': BadRequest, # Smaller than min purchase limit
'-6006': BadRequest, # Redeem amount error
'-6007': OperationRejected, # Not in redeem time
'-6008': OperationRejected, # Product not in redeem status
'-6009': RateLimitExceeded, # Request frequency too high
'-6011': OperationRejected, # Exceeding the maximum num allowed to purchase per user
'-6012': InsufficientFunds, # Balance not enough
'-6013': BadResponse, # Purchasing failed
'-6014': OperationRejected, # Exceed up-limit allowed to purchased
'-6015': BadRequest, # Empty request body
'-6016': BadRequest, # Parameter err
'-6017': PermissionDenied, # Not in whitelist
'-6018': InsufficientFunds, # Asset not enough
'-6019': OperationRejected, # Need confirm
'-6020': BadRequest, # Project not exists
#
# 7xxx
#
'-7001': BadRequest, # Date range is not supported.
'-7002': BadRequest, # Data request type is not supported.
#
# 1xxxx
#
'-10001': OperationFailed, # The system is under maintenance, please try again later.
'-10002': BadRequest, # Invalid input parameters.
'-10005': BadResponse, # No records found.
'-10007': BadRequest, # This coin is not loanable
'-10008': BadRequest, # This coin is not loanable
'-10009': BadRequest, # This coin can not be used.
'-10010': BadRequest, # This coin can not be used.
'-10011': InsufficientFunds, # Insufficient spot assets.
'-10012': BadRequest, # Invalid repayment amount.
'-10013': InsufficientFunds, # Insufficient collateral amount.
'-10015': OperationFailed, # Collateral deduction failed.
'-10016': OperationFailed, # Failed to provide loan.
'-10017': OperationRejected, # {"code":-10017,"msg":"Repay amount should not be larger than liability."}
'-10018': BadRequest, # Invalid repayment amount.
'-10019': BadRequest, # Configuration does not exists.
'-10020': BadRequest, # User ID does not exist.
'-10021': InvalidOrder, # Order does not exist.
'-10022': BadRequest, # Invalid adjustment amount.
'-10023': OperationFailed, # Failed to adjust LTV.
'-10024': BadRequest, # LTV adjustment not supported.
'-10025': OperationFailed, # Repayment failed.
'-10026': BadRequest, # Invalid parameter.
'-10028': BadRequest, # Invalid parameter.
'-10029': OperationRejected, # Loan amount is too small.
'-10030': OperationRejected, # Loan amount is too much.
'-10031': OperationRejected, # Individual loan quota reached.
'-10032': OperationFailed, # Repayment is temporarily unavailable.
'-10034': OperationRejected, # Repay with collateral is not available currently, please try to repay with borrowed coin.
'-10039': OperationRejected, # Repayment amount is too small.
'-10040': OperationRejected, # Repayment amount is too large.
'-10041': OperationFailed, # Due to high demand, there are currently insufficient loanable assets for {0}. Please adjust your borrow amount or try again tomorrow.
'-10042': BadSymbol, # asset %s is not supported
'-10043': OperationRejected, # {0} borrowing is currently not supported.
'-10044': OperationRejected, # Collateral amount has reached the limit. Please reduce your collateral amount or try with other collaterals.
'-10045': OperationRejected, # The loan coin does not support collateral repayment. Please try again later.
'-10046': OperationRejected, # Collateral Adjustment exceeds the maximum limit. Please try again.
'-10047': PermissionDenied, # This coin is currently not supported in your location due to local regulations.
'-11008': OperationRejected, # undocumented: Exceeding the accounts maximum borrowable limit
'-12014': RateLimitExceeded, # More than 1 request in 2 seconds
# BLVT
'-13000': OperationRejected, # Redeption of the token is forbiden now
'-13001': OperationRejected, # Exceeds individual 24h redemption limit of the token
'-13002': OperationRejected, # Exceeds total 24h redemption limit of the token
'-13003': PermissionDenied, # Subscription of the token is forbiden now
'-13004': OperationRejected, # Exceeds individual 24h subscription limit of the token
'-13005': OperationRejected, # Exceeds total 24h subscription limit of the token
'-13006': OperationRejected, # Subscription amount is too small
'-13007': PermissionDenied, # The Agreement is not signed
# 18xxx - BINANCE CODE
'-18002': OperationRejected, # The total amount of codes you created has exceeded the 24-hour limit, please try again after UTC 0
'-18003': OperationRejected, # Too many codes created in 24 hours, please try again after UTC 0
'-18004': OperationRejected, # Too many invalid redeem attempts in 24 hours, please try again after UTC 0
'-18005': PermissionDenied, # Too many invalid verify attempts, please try later
'-18006': OperationRejected, # The amount is too small, please re-enter
'-18007': OperationRejected, # This token is not currently supported, please re-enter
#
# 2xxxx
#
# 21xxx - PORTFOLIO MARGIN(documented in spot docs)
'-21001': BadRequest, # Request ID is not a Portfolio Margin Account.
'-21002': BadRequest, # Portfolio Margin Account doesn't support transfer from margin to futures.
'-21003': BadResponse, # Fail to retrieve margin assets.
'-21004': OperationRejected, # User doesnt have portfolio margin bankruptcy loan
'-21005': InsufficientFunds, # Users spot wallet doesnt have enough BUSD to repay portfolio margin bankruptcy loan
'-21006': OperationFailed, # User had portfolio margin bankruptcy loan repayment in process
'-21007': OperationFailed, # User failed to repay portfolio margin bankruptcy loan since liquidation was in process
#
# misc
#
'-32603': BadRequest, # undocumented, Filter failure: LOT_SIZE & precision
'400002': BadRequest, # undocumented, {“status”: “FAIL”, “code”: “400002”, “errorMessage”: “Signature for self request is not valid.”}
'100001003': AuthenticationError, # undocumented, {"code":100001003,"msg":"Verification failed"}
'200003903': AuthenticationError, # undocumented, {"code":200003903,"msg":"Your identity verification has been rejected. Please complete identity verification again."}
},
},
'linear': {
'exact': {
#
# 1xxx
#
'-1005': PermissionDenied, # {"code":-1005,"msg":"No such IP has been white listed"}
'-1008': OperationFailed, # -1008 SERVER_BUSY: Server is currently overloaded with other requests. Please try again in a few minutes.
'-1011': PermissionDenied, # {"code":-1011,"msg":"This IP cannot access self route."}
'-1023': BadRequest, # {"code":-1023,"msg":"Start time is greater than end time."}
'-1099': AuthenticationError, # {"code":-1099,"msg":"Not found, authenticated, or authorized"}
'-1109': PermissionDenied, # {"code":-1109,"msg":"Invalid account."}
'-1110': BadRequest, # {"code":-1110,"msg":"Invalid symbolType."}
'-1113': BadRequest, # {"code":-1113,"msg":"Withdrawal amount must be negative."}
'-1122': BadRequest, # INVALID_SYMBOL_STATUS
'-1126': BadSymbol, # ASSET_NOT_SUPPORTED
'-1136': BadRequest, # {"code":-1136,"msg":"Invalid newOrderRespType"}
#
# 2xxx
#
'-2012': OperationFailed, # CANCEL_ALL_FAIL
'-2016': OperationRejected, # {"code":-2016,"msg":"No trading window could be found for the symbol. Try ticker/24hrs instead."}
'-2017': PermissionDenied, # API Keys are locked on self account.
'-2018': InsufficientFunds, # {"code":-2018,"msg":"Balance is insufficient"}
'-2019': InsufficientFunds, # {"code":-2019,"msg":"Margin is insufficient."}
'-2020': OperationFailed, # {"code":-2020,"msg":"Unable to fill."}
'-2021': OrderImmediatelyFillable, # {"code":-2021,"msg":"Order would immediately trigger."}
'-2022': InvalidOrder, # {"code":-2022,"msg":"ReduceOnly Order is rejected."}
'-2023': OperationFailed, # {"code":-2023,"msg":"User in liquidation mode now."}
'-2024': InsufficientFunds, # {"code":-2024,"msg":"Position is not sufficient."}
'-2025': OperationRejected, # {"code":-2025,"msg":"Reach max open order limit."}
'-2026': InvalidOrder, # {"code":-2026,"msg":"This OrderType is not supported when reduceOnly."}
'-2027': OperationRejected, # {"code":-2027,"msg":"Exceeded the maximum allowable position at current leverage."}
'-2028': OperationRejected, # {"code":-2028,"msg":"Leverage is smaller than permitted: insufficient margin balance"}
#
# 4xxx
#
'-4063': BadRequest, # INVALID_OPTIONS_REQUEST_TYPE
'-4064': BadRequest, # INVALID_OPTIONS_TIME_FRAME
'-4065': BadRequest, # INVALID_OPTIONS_AMOUNT
'-4066': BadRequest, # INVALID_OPTIONS_EVENT_TYPE
'-4069': BadRequest, # Position INVALID_OPTIONS_PREMIUM_FEE
'-4070': BadRequest, # Client options id is not valid.
'-4071': BadRequest, # Invalid options direction
'-4072': OperationRejected, # premium fee is not updated, reject order
'-4073': BadRequest, # OPTIONS_PREMIUM_INPUT_LESS_THAN_ZERO
'-4074': OperationRejected, # Order amount is bigger than upper boundary or less than 0, reject order
'-4075': BadRequest, # output premium fee is less than 0, reject order
'-4076': OperationRejected, # original fee is too much higher than last fee
'-4077': OperationRejected, # place order amount has reached to limit, reject order
'-4078': OperationFailed, # options internal error
'-4079': BadRequest, # invalid options id
'-4080': PermissionDenied, # user not found with id: %s
'-4081': BadRequest, # OPTIONS_NOT_FOUND
'-4085': BadRequest, # Invalid notional limit coefficient
'-4087': PermissionDenied, # User can only place reduce only order
'-4088': PermissionDenied, # User can not place order currently
'-4114': BadRequest, # INVALID_CLIENT_TRAN_ID_LEN
'-4115': BadRequest, # DUPLICATED_CLIENT_TRAN_ID
'-4116': InvalidOrder, # DUPLICATED_CLIENT_ORDER_ID
'-4117': OperationRejected, # STOP_ORDER_TRIGGERING
'-4118': OperationRejected, # REDUCE_ONLY_MARGIN_CHECK_FAILED
'-4131': OperationRejected, # The counterparty's best price does not meet the PERCENT_PRICE filter limit
'-4140': BadRequest, # Invalid symbol status for opening position
'-4141': OperationRejected, # Symbol is closed
'-4144': BadSymbol, # Invalid pair
'-4164': InvalidOrder, # {"code":-4164,"msg":"Order's notional must be no smaller than 20(unless you choose reduce only)."},
'-4136': InvalidOrder, # {"code":-4136,"msg":"Target strategy invalid for orderType TRAILING_STOP_MARKET,closePosition True"}
'-4165': BadRequest, # Invalid time interval
'-4167': BadRequest, # Unable to adjust to Multi-Assets mode with symbols of USDⓈ-M Futures under isolated-margin mode.
'-4168': BadRequest, # Unable to adjust to isolated-margin mode under the Multi-Assets mode.
'-4169': OperationRejected, # Unable to adjust Multi-Assets Mode with insufficient margin balance in USDⓈ-M Futures
'-4170': OperationRejected, # Unable to adjust Multi-Assets Mode with open orders in USDⓈ-M Futures
'-4171': OperationRejected, # Adjusted asset mode is currently set and does not need to be adjusted repeatedly
'-4172': OperationRejected, # Unable to adjust Multi-Assets Mode with a negative wallet balance of margin available asset in USDⓈ-M Futures account.
'-4183': BadRequest, # Price is higher than stop price multiplier cap.
'-4184': BadRequest, # Price is lower than stop price multiplier floor.
'-4192': PermissionDenied, # Trade forbidden due to Cooling-off Period.
'-4202': PermissionDenied, # Intermediate Personal Verification is required for adjusting leverage over 20x
'-4203': PermissionDenied, # More than 20x leverage is available one month after account registration.
'-4205': PermissionDenied, # More than 20x leverage is available %s days after Futures account registration.
'-4206': PermissionDenied, # hasattr(self, Users) country has limited adjust leverage.
'-4208': OperationRejected, # Current symbol leverage cannot exceed 20 when using position limit adjustment service.
'-4209': OperationRejected, # Leverage adjustment failed. Current symbol max leverage limit is %sx
'-4210': BadRequest, # Stop price is higher than price multiplier cap
'-4211': BadRequest, # Stop price is lower than price multiplier floor
'-4400': PermissionDenied, # Futures Trading Quantitative Rules violated, only reduceOnly order is allowed, please try again later.
'-4401': PermissionDenied, # Compliance restricted account permission: can only place reduceOnly order.
'-4402': PermissionDenied, # Dear user, our Terms of Use and compliance with local regulations, self feature is currently not available in your region.
'-4403': PermissionDenied, # Dear user, our Terms of Use and compliance with local regulations, the leverage can only up to %sx in your region
#
# 5xxx
#
'-5021': OrderNotFillable, # Due to the order could not be filled immediately, the FOK order has been rejected.
'-5022': OrderNotFillable, # Due to the order could not be executed, the Post Only order will be rejected.
'-5024': OperationRejected, # Symbol is not in trading status. Order amendment is not permitted.
'-5025': OperationRejected, # Only limit order is supported.
'-5026': OperationRejected, # Exceed maximum modify order limit.
'-5027': OperationRejected, # No need to modify the order.
'-5028': BadRequest, # Timestamp for self request is outside of the ME recvWindow.
'-5037': BadRequest, # Invalid price match
'-5038': BadRequest, # Price match only supports order type: LIMIT, STOP AND TAKE_PROFIT
'-5039': BadRequest, # Invalid self trade prevention mode
'-5040': BadRequest, # The goodTillDate timestamp must be greater than the current time plus 600 seconds and smaller than 253402300799000
'-5041': OperationFailed, # No depth matches self BBO order
},
},
'inverse': {
'exact': {
#
# 1xxx
#
'-1005': PermissionDenied, # {"code":-1005,"msg":"No such IP has been white listed"}
'-1011': PermissionDenied, # {"code":-1011,"msg":"This IP cannot access self route."}
'-1023': BadRequest, # {"code":-1023,"msg":"Start time is greater than end time."}
'-1109': AuthenticationError, # {"code":-1109,"msg":"Invalid account."}
'-1110': BadSymbol, # {"code":-1110,"msg":"Invalid symbolType."}
'-1113': BadRequest, # {"code":-1113,"msg":"Withdrawal amount must be negative."}
'-1128': BadRequest, # {"code":-1128,"msg":"Combination of optional parameters invalid."}
'-1136': BadRequest, # {"code":-1136,"msg":"Invalid newOrderRespType"}
#
# 2xxx
#
'-2016': OperationRejected, # {"code":-2016,"msg":"No trading window could be found for the symbol. Try ticker/24hrs instead."}
'-2018': InsufficientFunds, # {"code":-2018,"msg":"Balance is insufficient"}
'-2019': InsufficientFunds, # {"code":-2019,"msg":"Margin is insufficient."}
'-2020': OperationFailed, # {"code":-2020,"msg":"Unable to fill."}
'-2021': OrderImmediatelyFillable, # {"code":-2021,"msg":"Order would immediately trigger."}
'-2022': InvalidOrder, # {"code":-2022,"msg":"ReduceOnly Order is rejected."}
'-2023': OperationFailed, # {"code":-2023,"msg":"User in liquidation mode now."}
'-2024': BadRequest, # {"code":-2024,"msg":"Position is not sufficient."}
'-2025': OperationRejected, # {"code":-2025,"msg":"Reach max open order limit."}
'-2026': InvalidOrder, # {"code":-2026,"msg":"This OrderType is not supported when reduceOnly."}
'-2027': OperationRejected, # {"code":-2027,"msg":"Exceeded the maximum allowable position at current leverage."}
'-2028': OperationRejected, # {"code":-2028,"msg":"Leverage is smaller than permitted: insufficient margin balance"}
#
# 4xxx
#
'-4086': BadRequest, # Invalid price spread threshold.
'-4087': BadSymbol, # Invalid pair
'-4088': BadRequest, # Invalid time interval
'-4089': PermissionDenied, # User can only place reduce only order.
'-4090': PermissionDenied, # User can not place order currently.
'-4110': BadRequest, # clientTranId is not valid
'-4111': BadRequest, # clientTranId is duplicated.
'-4112': OperationRejected, # ReduceOnly Order Failed. Please check your existing position and open orders.
'-4113': OperationRejected, # The counterparty's best price does not meet the PERCENT_PRICE filter limit.
'-4150': OperationRejected, # Leverage reduction is not supported in Isolated Margin Mode with open positions.
'-4151': BadRequest, # Price is higher than stop price multiplier cap.
'-4152': BadRequest, # Price is lower than stop price multiplier floor.
'-4154': BadRequest, # Stop price is higher than price multiplier cap.
'-4155': BadRequest, # Stop price is lower than price multiplier floor
'-4178': BadRequest, # Order's notional must be no smaller than one(unless you choose reduce only)
'-4188': BadRequest, # Timestamp for self request is outside of the ME recvWindow.
'-4192': PermissionDenied, # Trade forbidden due to Cooling-off Period.
'-4194': PermissionDenied, # Intermediate Personal Verification is required for adjusting leverage over 20x.
'-4195': PermissionDenied, # More than 20x leverage is available one month after account registration.
'-4196': BadRequest, # Only limit order is supported.
'-4197': OperationRejected, # No need to modify the order.
'-4198': OperationRejected, # Exceed maximum modify order limit.
'-4199': BadRequest, # Symbol is not in trading status. Order amendment is not permitted.
'-4200': PermissionDenied, # More than 20x leverage is available %s days after Futures account registration.
'-4201': PermissionDenied, # Users in your location/country can only access a maximum leverage of %s
'-4202': OperationRejected, # Current symbol leverage cannot exceed 20 when using position limit adjustment service.
},
},
'option': {
'exact': {
#
# 1xxx
#
'-1003': ExchangeError, # override common
'-1004': ExchangeError, # override common
'-1006': ExchangeError, # override common
'-1007': ExchangeError, # override common
'-1008': RateLimitExceeded, # TOO_MANY_REQUEST
'-1010': ExchangeError, # override common
'-1013': ExchangeError, # override common
'-1108': ExchangeError, # override common
'-1112': ExchangeError, # override common
'-1114': ExchangeError, # override common
'-1128': BadSymbol, # BAD_CONTRACT
'-1129': BadSymbol, # BAD_CURRENCY
'-1131': BadRequest, # {"code":-1131,"msg":"recvWindow must be less than 60000"}
#
# 2xxx
#
'-2011': ExchangeError, # override common
'-2018': InsufficientFunds, # BALANCE_NOT_SUFFICIENT
'-2027': InsufficientFunds, # OPTION_MARGIN_NOT_SUFFICIENT
#
# 3xxx
#
'-3029': OperationFailed, # {"code":-3029,"msg":"Transfer failed."}
#
# 4xxx
#
# -4001 inherited
# -4002 inherited
# -4003 inherited
# -4004 inherited
# -4005 inherited
'-4006': ExchangeError, # override commons
'-4007': ExchangeError, # override commons
'-4008': ExchangeError, # override commons
'-4009': ExchangeError, # override commons
'-4010': ExchangeError, # override commons
'-4011': ExchangeError, # override commons
'-4012': ExchangeError, # override commons
# -4013 inherited
'-4014': ExchangeError, # override commons
'-4015': ExchangeError, # override commons
'-4016': ExchangeError, # override commons
'-4017': ExchangeError, # override commons
'-4018': ExchangeError, # override commons
'-4019': ExchangeError, # override commons
'-4020': ExchangeError, # override commons
'-4021': ExchangeError, # override commons
'-4022': ExchangeError, # override commons
'-4023': ExchangeError, # override commons
'-4024': ExchangeError, # override commons
'-4025': ExchangeError, # override commons
'-4026': ExchangeError, # override commons
'-4027': ExchangeError, # override commons
'-4028': ExchangeError, # override commons
# -4029 inherited
# -4030 inherited
'-4031': ExchangeError, # override commons
'-4032': ExchangeError, # override commons
'-4033': ExchangeError, # override commons
'-4034': ExchangeError, # override commons
'-4035': ExchangeError, # override commons
'-4036': ExchangeError, # override commons
'-4037': ExchangeError, # override commons
'-4038': ExchangeError, # override commons
'-4039': ExchangeError, # override commons
'-4040': ExchangeError, # override commons
'-4041': ExchangeError, # override commons
'-4042': ExchangeError, # override commons
'-4043': ExchangeError, # override commons
'-4044': ExchangeError, # override commons
'-4045': ExchangeError, # override commons
'-4046': ExchangeError, # override commons
'-4047': ExchangeError, # override commons
'-4048': ExchangeError, # override commons
'-4049': ExchangeError, # override commons
'-4050': ExchangeError, # override commons
'-4051': ExchangeError, # override commons
'-4052': ExchangeError, # override commons
'-4053': ExchangeError, # override commons
'-4054': ExchangeError, # override commons
# -4055 inherited
'-4056': ExchangeError, # override commons
'-4057': ExchangeError, # override commons
'-4058': ExchangeError, # override commons
'-4059': ExchangeError, # override commons
'-4060': ExchangeError, # override commons
'-4061': ExchangeError, # override commons
'-4062': ExchangeError, # override commons
'-4063': ExchangeError, # override commons
'-4064': ExchangeError, # override commons
'-4065': ExchangeError, # override commons
'-4066': ExchangeError, # override commons
'-4067': ExchangeError, # override commons
'-4068': ExchangeError, # override commons
'-4069': ExchangeError, # override commons
'-4070': ExchangeError, # override commons
'-4071': ExchangeError, # override commons
'-4072': ExchangeError, # override commons
'-4073': ExchangeError, # override commons
'-4074': ExchangeError, # override commons
'-4075': ExchangeError, # override commons
'-4076': ExchangeError, # override commons
'-4077': ExchangeError, # override commons
'-4078': ExchangeError, # override commons
'-4079': ExchangeError, # override commons
'-4080': ExchangeError, # override commons
'-4081': ExchangeError, # override commons
'-4082': ExchangeError, # override commons
'-4083': ExchangeError, # override commons
'-4084': ExchangeError, # override commons
'-4085': ExchangeError, # override commons
'-4086': ExchangeError, # override commons
'-4087': ExchangeError, # override commons
'-4088': ExchangeError, # override commons
'-4089': ExchangeError, # override commons
'-4091': ExchangeError, # override commons
'-4092': ExchangeError, # override commons
'-4093': ExchangeError, # override commons
'-4094': ExchangeError, # override commons
'-4095': ExchangeError, # override commons
'-4096': ExchangeError, # override commons
'-4097': ExchangeError, # override commons
'-4098': ExchangeError, # override commons
'-4099': ExchangeError, # override commons
'-4101': ExchangeError, # override commons
'-4102': ExchangeError, # override commons
'-4103': ExchangeError, # override commons
'-4104': ExchangeError, # override commons
'-4105': ExchangeError, # override commons
'-4106': ExchangeError, # override commons
'-4107': ExchangeError, # override commons
'-4108': ExchangeError, # override commons
'-4109': ExchangeError, # override commons
'-4110': ExchangeError, # override commons
'-4112': ExchangeError, # override commons
'-4113': ExchangeError, # override commons
'-4114': ExchangeError, # override commons
'-4115': ExchangeError, # override commons
'-4116': ExchangeError, # override commons
'-4117': ExchangeError, # override commons
'-4118': ExchangeError, # override commons
'-4119': ExchangeError, # override commons
'-4120': ExchangeError, # override commons
'-4121': ExchangeError, # override commons
'-4122': ExchangeError, # override commons
'-4123': ExchangeError, # override commons
'-4124': ExchangeError, # override commons
'-4125': ExchangeError, # override commons
'-4126': ExchangeError, # override commons
'-4127': ExchangeError, # override commons
'-4128': ExchangeError, # override commons
'-4129': ExchangeError, # override commons
'-4130': ExchangeError, # override commons
'-4131': ExchangeError, # override commons
'-4132': ExchangeError, # override commons
'-4133': ExchangeError, # override commons
'-4134': ExchangeError, # override commons
'-4135': ExchangeError, # override commons
'-4136': ExchangeError, # override commons
'-4137': ExchangeError, # override commons
'-4138': ExchangeError, # override commons
'-4139': ExchangeError, # override commons
'-4141': ExchangeError, # override commons
'-4142': ExchangeError, # override commons
'-4143': ExchangeError, # override commons
'-4144': ExchangeError, # override commons
'-4145': ExchangeError, # override commons
'-4146': ExchangeError, # override commons
'-4147': ExchangeError, # override commons
'-4148': ExchangeError, # override commons
'-4149': ExchangeError, # override commons
'-4150': ExchangeError, # override commons
#
# 2xxxx
#
'-20121': ExchangeError, # override commons
'-20124': ExchangeError, # override commons
'-20130': ExchangeError, # override commons
'-20132': ExchangeError, # override commons
'-20194': ExchangeError, # override commons
'-20195': ExchangeError, # override commons
'-20196': ExchangeError, # override commons
'-20198': ExchangeError, # override commons
'-20204': ExchangeError, # override commons
},
},
'portfolioMargin': {
'exact': {
#
# 10xx General Server or Network Issues
#
'-1000': OperationFailed, # An unknown error occured while processing the request.
'-1001': ExchangeError, # Internal error; unable to process your request. Please try again.
'-1002': PermissionDenied, # You are not authorized to execute self request.
'-1003': RateLimitExceeded, # Too many requests use the websocket for live updates to avoid polling the API.
'-1004': BadRequest, # This IP is already on the white list.
'-1005': PermissionDenied, # No such IP has been white listed.
'-1006': BadResponse, # An unexpected response was received from the message bus. Execution status unknown.
'-1007': BadResponse, # Timeout waiting for response from backend server. Send status unknown, execution status unknown.
'-1008': OperationFailed, # WS Spot server is currently overloaded with other requests. Please try again in a few minutes.
'-1010': ExchangeError, # ERROR_MSG_RECEIVED
'-1011': PermissionDenied, # This IP cannot access self route.
'-1013': ExchangeError, # INVALID_MESSAGE.
'-1014': InvalidOrder, # Unsupported order combination.
'-1015': InvalidOrder, # Too many new orders.
'-1016': NotSupported, # This service is no longer available.
'-1020': NotSupported, # This operation is not supported.
'-1021': BadRequest, # Timestamp for self request is outside of the recvWindow 1000ms ahead of the servers time.
'-1022': BadRequest, # Signature for self request is not valid.
'-1023': BadRequest, # Start time is greater than end time
'-1099': OperationFailed, # WS not found authenticated or authorized
#
# 11xx Request Issues
#
'-1100': BadRequest, # Illegal characters found in a parameter.
'-1101': BadRequest, # Too many parameters sent for self endpoint.
'-1102': BadRequest, # A mandatory parameter was not sent, was empty/null, or malformed.
'-1103': BadRequest, # An unknown parameter was sent.
'-1104': BadRequest, # Not all sent parameters were read.
'-1105': BadRequest, # A parameter was empty.
'-1106': BadRequest, # A parameter was sent when not required.
'-1108': BadRequest, # Invalid asset.
'-1109': BadRequest, # Invalid account.
'-1110': BadSymbol, # Invalid symbolType.
'-1111': BadRequest, # Precision is over the maximum defined for self asset.
'-1112': BadRequest, # No orders on book for symbol.
'-1113': BadRequest, # Withdrawal amount must be negative.
'-1114': BadRequest, # TimeInForce parameter sent when not required.
'-1115': BadRequest, # Invalid timeInForce.
'-1116': BadRequest, # Invalid orderType.
'-1117': BadRequest, # Invalid side.
'-1118': BadRequest, # New client order ID was empty.
'-1119': BadRequest, # Original client order ID was empty.
'-1120': BadRequest, # Invalid interval.
'-1121': BadSymbol, # Invalid symbol.
'-1125': BadRequest, # This listenKey does not exist.
'-1127': BadRequest, # Lookup interval is too big.
'-1128': BadRequest, # Combination of optional parameters invalid.
'-1130': BadRequest, # Invalid data sent for a parameter.
'-1131': BadRequest, # WS recvWindow must be less than 60000
'-1134': BadRequest, # WS strategyType was less than 1000000.
'-1136': BadRequest, # Invalid newOrderRespType.
'-1145': BadRequest, # WS cancelRestrictions has to be either ONLY_NEW or ONLY_PARTIALLY_FILLED.
'-1151': BadRequest, # WS Symbol is present multiple times in the list.
#
# 20xx Processing Issues
#
'-2010': InvalidOrder, # NEW_ORDER_REJECTED
'-2011': OperationRejected, # CANCEL_REJECTED
'-2013': OrderNotFound, # Order does not exist.
'-2014': OperationRejected, # API-key format invalid.
'-2015': OperationRejected, # Invalid API-key, IP, or permissions for action.
'-2016': OperationFailed, # No trading window could be found for the symbol. Try ticker/24hrs instead.
'-2018': OperationFailed, # Balance is insufficient.
'-2019': OperationFailed, # Margin is insufficient.
'-2020': OrderNotFillable, # Unable to fill.
'-2021': OrderImmediatelyFillable, # Order would immediately trigger.
'-2022': InvalidOrder, # ReduceOnly Order is rejected.
'-2023': OperationFailed, # User in liquidation mode now.
'-2024': OperationRejected, # Position is not sufficient.
'-2025': OperationRejected, # Reach max open order limit.
'-2026': InvalidOrder, # This OrderType is not supported when reduceOnly.
'-2027': OperationRejected, # Exceeded the maximum allowable position at current leverage.
'-2028': OperationRejected, # Leverage is smaller than permitted: insufficient margin balance.
#
# 4xxx Filters and other issues
#
'-4000': BadRequest, # Invalid order status.
'-4001': BadRequest, # Price less than 0.
'-4002': BadRequest, # Price greater than max price.
'-4003': BadRequest, # Quantity less than zero.
'-4004': BadRequest, # Quantity less than min quantity.
'-4005': BadRequest, # Quantity greater than max quantity.
'-4006': BadRequest, # Stop price less than zero.
'-4007': BadRequest, # Stop price greater than max price.
'-4008': BadRequest, # Tick size less than zero.
'-4009': BadRequest, # Max price less than min price.
'-4010': BadRequest, # Max qty less than min qty.
'-4011': BadRequest, # Step size less than zero.
'-4012': BadRequest, # Max mum orders less than zero.
'-4013': BadRequest, # Price less than min price.
'-4014': BadRequest, # Price not increased by tick size.
'-4015': BadRequest, # Client order id is not valid.
'-4016': BadRequest, # Price is higher than mark price multiplier cap.
'-4017': BadRequest, # Multiplier up less than zero.
'-4018': BadRequest, # Multiplier down less than zero.
'-4019': BadRequest, # Composite scale too large.
'-4020': BadRequest, # Target strategy invalid for orderType '%s',reduceOnly '%b'.
'-4021': BadRequest, # Invalid depth limit.
'-4022': BadRequest, # market status sent is not valid.
'-4023': BadRequest, # Qty not increased by step size.
'-4024': BadRequest, # Price is lower than mark price multiplier floor.
'-4025': BadRequest, # Multiplier decimal less than zero.
'-4026': BadRequest, # Commission invalid.
'-4027': BadRequest, # Invalid account type.
'-4028': BadRequest, # Invalid leverage
'-4029': BadRequest, # Tick size precision is invalid.
'-4030': BadRequest, # Step size precision is invalid.
'-4031': BadRequest, # Invalid parameter working type
'-4032': BadRequest, # Exceed maximum cancel order size.
'-4033': BadRequest, # Insurance account not found.
'-4044': BadRequest, # Balance Type is invalid.
'-4045': BadRequest, # Reach max stop order limit.
'-4046': BadRequest, # No need to change margin type.
'-4047': BadRequest, # Margin type cannot be changed if there exists open orders.
'-4048': BadRequest, # Margin type cannot be changed if there exists position.
'-4049': BadRequest, # Add margin only support for isolated position.
'-4050': BadRequest, # Cross balance insufficient.
'-4051': BadRequest, # Isolated balance insufficient.
'-4052': BadRequest, # No need to change auto add margin.
'-4053': BadRequest, # Auto add margin only support for isolated position.
'-4054': BadRequest, # Cannot add position margin: position is 0.
'-4055': BadRequest, # Amount must be positive.
'-4056': PermissionDenied, # Invalid api key type.
'-4057': PermissionDenied, # Invalid api public key
'-4058': BadRequest, # maxPrice and priceDecimal too large,please check.
'-4059': BadRequest, # No need to change position side.
'-4060': BadRequest, # Invalid position side.
'-4061': InvalidOrder, # Order's position side does not match user's setting.
'-4062': BadRequest, # Invalid or improper reduceOnly value.
'-4063': BadRequest, # Invalid options request type
'-4064': BadRequest, # Invalid options time frame
'-4065': BadRequest, # Invalid options amount
'-4066': BadRequest, # Invalid options event type
'-4067': BadRequest, # Position side cannot be changed if there exists open orders.
'-4068': BadRequest, # Position side cannot be changed if there exists position.
'-4069': BadRequest, # Invalid options premium fee
'-4070': BadRequest, # Client options id is not valid.
'-4071': BadRequest, # Invalid options direction
'-4072': OperationRejected, # premium fee is not updated, reject order
'-4073': BadRequest, # input premium fee is less than 0, reject order
'-4074': BadRequest, # Order amount is bigger than upper boundary or less than 0, reject order
'-4075': BadRequest, # output premium fee is less than 0, reject order
'-4076': OperationRejected, # original fee is too much higher than last fee
'-4077': OperationRejected, # place order amount has reached to limit, reject order
'-4078': OperationFailed, # options internal error
'-4079': BadRequest, # invalid options id
'-4080': PermissionDenied, # user not found
'-4081': BadRequest, # options not found
'-4082': BadRequest, # Invalid number of batch place orders.
'-4083': BadRequest, # Fail to place batch orders.
'-4084': NotSupported, # Method is not allowed currently. Upcoming soon.
'-4085': BadRequest, # Invalid notional limit coefficient
'-4086': BadRequest, # Invalid price spread threshold
'-4087': PermissionDenied, # User can only place reduce only order
'-4088': PermissionDenied, # User can not place order currently
'-4104': BadRequest, # Invalid contract type
'-4114': BadRequest, # clientTranId is not valid
'-4115': BadRequest, # clientTranId is duplicated
'-4118': OperationRejected, # ReduceOnly Order Failed. Please check your existing position and open orders
'-4131': OperationRejected, # The counterparty's best price does not meet the PERCENT_PRICE filter limit
'-4135': BadRequest, # Invalid activation price
'-4137': BadRequest, # Quantity must be zero with closePosition equals True
'-4138': BadRequest, # Reduce only must be True with closePosition equals True
'-4139': BadRequest, # Order type can not be market if it's unable to cancel
'-4140': OrderImmediatelyFillable, # Invalid symbol status for opening position
'-4141': BadRequest, # Symbol is closed
'-4142': OrderImmediatelyFillable, # REJECT: take profit or stop order will be triggered immediately
'-4144': BadSymbol, # Invalid pair
'-4161': OperationRejected, # Leverage reduction is not supported in Isolated Margin Mode with open positions
'-4164': InvalidOrder, # Order's notional must be no smaller than 5.0(unless you choose reduce only)
'-4165': BadRequest, # Invalid time interval
'-4183': InvalidOrder, # Price is higher than stop price multiplier cap.
'-4184': InvalidOrder, # Price is lower than stop price multiplier floor.
'-4408': InvalidOrder, # This symbol is in reduce only mode due to regulation requirements. Please upgrade to Binance Credits Trading Mode.
#
# 5xxx Order Execution Issues
#
'-5021': OrderNotFillable, # Due to the order could not be filled immediately, the FOK order has been rejected.
'-5022': OrderNotFillable, # Due to the order could not be executed, the Post Only order will be rejected.
'-5028': OperationFailed, # The requested timestamp is outside the recvWindow of the matching engine
'-5041': RateLimitExceeded, # Time out for too many requests from self account queueing at the same time.
},
},
'exact': {
# error codes to cover ALL market types(however, specific market type might have override)
#
# 1xxx
#
'-1000': OperationFailed, # {"code":-1000,"msg":"An unknown error occured while processing the request."}
'-1001': OperationFailed, # {"code":-1001,"msg":"'Internal error; unable to process your request. Please try again.'"}
'-1002': AuthenticationError, # {"code":-1002,"msg":"'You are not authorized to execute self request.'"}
'-1003': RateLimitExceeded, # {"code":-1003,"msg":"Too much request weight used, current limit is 1200 request weight per 1 MINUTE. Please use the websocket for live updates to avoid polling the API."}
'-1004': OperationRejected, # DUPLICATE_IP : This IP is already on the white list
'-1006': OperationFailed, # {"code":-1006,"msg":"An unexpected response was received from the message bus. Execution status unknown."}
'-1007': RequestTimeout, # {"code":-1007,"msg":"Timeout waiting for response from backend server. Send status unknown; execution status unknown."}
'-1010': OperationFailed, # {"code":-1010,"msg":"ERROR_MSG_RECEIVED."}
'-1013': BadRequest, # INVALID_MESSAGE
'-1014': InvalidOrder, # {"code":-1014,"msg":"Unsupported order combination."}
'-1015': RateLimitExceeded, # {"code":-1015,"msg":"'Too many new orders; current limit is %s orders per %s.'"}
'-1016': BadRequest, # {"code":-1016,"msg":"'This service is no longer available.',"}
'-1020': BadRequest, # {"code":-1020,"msg":"'This operation is not supported.'"}
'-1021': InvalidNonce, # {"code":-1021,"msg":"'your time is ahead of server'"}
'-1022': AuthenticationError, # {"code":-1022,"msg":"Signature for self request is not valid."}
'-1100': BadRequest, # {"code":-1100,"msg":"createOrder(symbol, 1, asdf) -> 'Illegal characters found in parameter 'price'"}
'-1101': BadRequest, # {"code":-1101,"msg":"Too many parameters; expected %s and received %s."}
'-1102': BadRequest, # {"code":-1102,"msg":"Param %s or %s must be sent, but both were empty"}
'-1103': BadRequest, # {"code":-1103,"msg":"An unknown parameter was sent."}
'-1104': BadRequest, # {"code":-1104,"msg":"Not all sent parameters were read, read 8 parameters but was sent 9"}
'-1105': BadRequest, # {"code":-1105,"msg":"Parameter %s was empty."}
'-1106': BadRequest, # {"code":-1106,"msg":"Parameter %s sent when not required."}
'-1108': BadSymbol, # {"code":-1108,"msg":"Invalid asset."}
'-1111': BadRequest, # {"code":-1111,"msg":"Precision is over the maximum defined for self asset."}
'-1112': OperationFailed, # {"code":-1112,"msg":"No orders on book for symbol."}
'-1114': BadRequest, # {"code":-1114,"msg":"TimeInForce parameter sent when not required."}
'-1115': BadRequest, # {"code":-1115,"msg":"Invalid timeInForce."}
'-1116': BadRequest, # {"code":-1116,"msg":"Invalid orderType."}
'-1117': BadRequest, # {"code":-1117,"msg":"Invalid side."}
'-1118': BadRequest, # {"code":-1118,"msg":"New client order ID was empty."}
'-1119': BadRequest, # {"code":-1119,"msg":"Original client order ID was empty."}
'-1120': BadRequest, # {"code":-1120,"msg":"Invalid interval."}
'-1121': BadSymbol, # {"code":-1121,"msg":"Invalid symbol."}
'-1125': AuthenticationError, # {"code":-1125,"msg":"This listenKey does not exist."}
'-1127': BadRequest, # {"code":-1127,"msg":"More than %s hours between startTime and endTime."}
'-1128': BadRequest, # {"code":-1128,"msg":"Combination of optional parameters invalid."}
'-1130': BadRequest, # {"code":-1130,"msg":"Data sent for paramter %s is not valid."}
#
# 2xxx
#
'-2010': InvalidOrder, # NEW_ORDER_REJECTED
'-2011': OrderNotFound, # {"code":-2011,"msg":"cancelOrder(1, 'BTC/USDT') -> 'UNKNOWN_ORDER'"}
'-2013': OrderNotFound, # {"code":-2013,"msg":"fetchOrder(1, 'BTC/USDT') -> 'Order does not exist'"}
'-2014': AuthenticationError, # {"code":-2014,"msg":"API-key format invalid."}
'-2015': AuthenticationError, # {"code":-2015,"msg":"Invalid API-key, IP, or permissions for action."}
#
# 4xxx(common for linear, inverse, pm)
#
'-4000': InvalidOrder, # INVALID_ORDER_STATUS
'-4001': BadRequest, # PRICE_LESS_THAN_ZERO
'-4002': BadRequest, # PRICE_GREATER_THAN_MAX_PRICE
'-4003': BadRequest, # QTY_LESS_THAN_ZERO
'-4004': BadRequest, # QTY_LESS_THAN_MIN_QTY
'-4005': BadRequest, # QTY_GREATER_THAN_MAX_QTY
'-4006': BadRequest, # STOP_PRICE_LESS_THAN_ZERO
'-4007': BadRequest, # STOP_PRICE_GREATER_THAN_MAX_PRICE
'-4008': BadRequest, # TICK SIZE LESS THAN ZERO
'-4009': BadRequest, # MAX_PRICE_LESS_THAN_MIN_PRICE
'-4010': BadRequest, # MAX_QTY_LESS_THAN_MIN_QTY
'-4011': BadRequest, # STEP_SIZE_LESS_THAN_ZERO
'-4012': BadRequest, # MAX_NUM_ORDERS_LESS_THAN_ZERO
'-4013': BadRequest, # PRICE_LESS_THAN_MIN_PRICE
'-4014': BadRequest, # PRICE NOT INCREASED BY TICK SIZE
'-4015': BadRequest, # Client order id is not valid
'-4016': BadRequest, # Price is higher than mark price multiplier cap.
'-4017': BadRequest, # MULTIPLIER_UP_LESS_THAN_ZERO
'-4018': BadRequest, # MULTIPLIER_DOWN_LESS_THAN_ZERO
'-4019': OperationRejected, # COMPOSITE_SCALE_OVERFLOW
'-4020': BadRequest, # TARGET_STRATEGY_INVALID
'-4021': BadRequest, # INVALID_DEPTH_LIMIT
'-4022': BadRequest, # WRONG_MARKET_STATUS
'-4023': BadRequest, # QTY_NOT_INCREASED_BY_STEP_SIZE
'-4024': BadRequest, # PRICE_LOWER_THAN_MULTIPLIER_DOWN
'-4025': BadRequest, # MULTIPLIER_DECIMAL_LESS_THAN_ZERO
'-4026': BadRequest, # COMMISSION_INVALID
'-4027': BadRequest, # INVALID_ACCOUNT_TYPE
'-4028': BadRequest, # INVALID_LEVERAGE
'-4029': BadRequest, # INVALID_TICK SIZE_PRECISION
'-4030': BadRequest, # INVALID_STEP_SIZE_PRECISION
'-4031': BadRequest, # INVALID_WORKING_TYPE
'-4032': OperationRejected, # EXCEED_MAX_CANCEL_ORDER_SIZE(or Invalid parameter working type: %s)
'-4033': BadRequest, # INSURANCE_ACCOUNT_NOT_FOUND
'-4044': BadRequest, # INVALID_BALANCE_TYPE
'-4045': OperationRejected, # MAX_STOP_ORDER_EXCEEDED
'-4046': OperationRejected, # NO_NEED_TO_CHANGE_MARGIN_TYPE
'-4047': OperationRejected, # Margin type cannot be changed if there exists open orders.
'-4048': OperationRejected, # Margin type cannot be changed if there exists position.
'-4049': BadRequest, # Add margin only support for isolated position.
'-4050': InsufficientFunds, # Cross balance insufficient
'-4051': InsufficientFunds, # Isolated balance insufficient.
'-4052': OperationRejected, # No need to change auto add margin.
'-4053': BadRequest, # Auto add margin only support for isolated position.
'-4054': OperationRejected, # Cannot add position margin: position is 0.
'-4055': BadRequest, # Amount must be positive.
'-4056': AuthenticationError, # INVALID_API_KEY_TYPE
'-4057': AuthenticationError, # INVALID_RSA_PUBLIC_KEY: Invalid api public key
'-4058': BadRequest, # MAX_PRICE_TOO_LARGE
'-4059': OperationRejected, # NO_NEED_TO_CHANGE_POSITION_SIDE
'-4060': BadRequest, # INVALID_POSITION_SIDE
'-4061': OperationRejected, # POSITION_SIDE_NOT_MATCH: Order's position side does not match user's setting.
'-4062': BadRequest, # REDUCE_ONLY_CONFLICT: Invalid or improper reduceOnly value.
'-4067': OperationRejected, # Position side cannot be changed if there exists open orders.
'-4068': OperationRejected, # Position side cannot be changed if there exists position.
'-4082': BadRequest, # Invalid number of batch place orders.
'-4083': OperationRejected, # PLACE_BATCH_ORDERS_FAIL : Fail to place batch orders.
'-4084': BadRequest, # UPCOMING_METHOD : Method is not allowed currently. Upcoming soon.
'-4086': BadRequest, # Invalid price spread threshold.
'-4104': BadRequest, # INVALID_CONTRACT_TYPE
'-4135': BadRequest, # Invalid activation price
'-4137': BadRequest, # Quantity must be zero with closePosition equals True
'-4138': BadRequest, # Reduce only must be True with closePosition equals True
'-4139': BadRequest, # Order type can not be market if it's unable to cancel
'-4142': OrderImmediatelyFillable, # REJECT: take profit or stop order will be triggered immediately
#
# 2xxxx
#
# 20xxx - spot & futures algo(TBD for OPTIONS & PORTFOLIO MARGIN)
'-20121': BadSymbol, # Invalid symbol.
'-20124': BadRequest, # Invalid algo id or it has been completed.
'-20130': BadRequest, # Invalid data sent for a parameter
'-20132': BadRequest, # The client algo id is duplicated
'-20194': BadRequest, # Duration is too short to execute all required quantity.
'-20195': BadRequest, # The total size is too small.
'-20196': BadRequest, # The total size is too large.
'-20198': OperationRejected, # Reach the max open orders allowed.
'-20204': BadRequest, # The notional of USD is less or more than the limit.
#
# strings
#
'System is under maintenance.': OnMaintenance, # {"code":1,"msg":"System is under maintenance."}
'System abnormality': OperationFailed, # {"code":-1000,"msg":"System abnormality"}
'You are not authorized to execute self request.': PermissionDenied, # {"msg":"You are not authorized to execute self request."}
'API key does not exist': AuthenticationError,
'Order would trigger immediately.': OrderImmediatelyFillable,
'Stop price would trigger immediately.': OrderImmediatelyFillable, # {"code":-2010,"msg":"Stop price would trigger immediately."}
'Order would immediately match and take.': OrderImmediatelyFillable, # {"code":-2010,"msg":"Order would immediately match and take."}
'Account has insufficient balance for requested action.': InsufficientFunds,
'Rest API trading is not enabled.': PermissionDenied,
'This account may not place or cancel orders.': PermissionDenied,
"You don't have permission.": PermissionDenied, # {"msg":"You don't have permission.","success":false}
'Market is closed.': MarketClosed, # {"code":-1013,"msg":"Market is closed."}
'Too many requests. Please try again later.': RateLimitExceeded, # {"msg":"Too many requests. Please try again later.","success":false}
'This action is disabled on self account.': AccountSuspended, # {"code":-2011,"msg":"This action is disabled on self account."}
'Limit orders require GTC for self phase.': BadRequest,
'This order type is not hasattr(self, possible) trading phase.': BadRequest,
'This type of sub-account exceeds the maximum number limit': OperationRejected, # {"code":-9000,"msg":"This type of sub-account exceeds the maximum number limit"}
'This symbol is restricted for self account.': PermissionDenied,
'This symbol is not permitted for self account.': PermissionDenied, # {"code":-2010,"msg":"This symbol is not permitted for self account."}
},
'broad': {
'has no operation privilege': PermissionDenied,
'MAX_POSITION': BadRequest, # {"code":-2010,"msg":"Filter failure: MAX_POSITION"}
},
},
})
def is_inverse(self, type: str, subType: Str = None) -> bool:
if subType is None:
return(type == 'delivery')
else:
return subType == 'inverse'
def is_linear(self, type: str, subType: Str = None) -> bool:
if subType is None:
return(type == 'future') or (type == 'swap')
else:
return subType == 'linear'
def set_sandbox_mode(self, enable: bool):
super(binance, self).set_sandbox_mode(enable)
self.options['sandboxMode'] = enable
def create_expired_option_market(self, symbol: str):
# support expired option contracts
settle = 'USDT'
optionParts = symbol.split('-')
symbolBase = symbol.split('/')
base = None
if symbol.find('/') > -1:
base = self.safe_string(symbolBase, 0)
else:
base = self.safe_string(optionParts, 0)
expiry = self.safe_string(optionParts, 1)
strike = self.safe_integer(optionParts, 2)
strikeAsString = self.safe_string(optionParts, 2)
optionType = self.safe_string(optionParts, 3)
datetime = self.convert_expire_date(expiry)
timestamp = self.parse8601(datetime)
return {
'id': base + '-' + expiry + '-' + strikeAsString + '-' + optionType,
'symbol': base + '/' + settle + ':' + settle + '-' + expiry + '-' + strikeAsString + '-' + optionType,
'base': base,
'quote': settle,
'baseId': base,
'quoteId': settle,
'active': None,
'type': 'option',
'linear': None,
'inverse': None,
'spot': False,
'swap': False,
'future': False,
'option': True,
'margin': False,
'contract': True,
'contractSize': None,
'expiry': timestamp,
'expiryDatetime': datetime,
'optionType': 'call' if (optionType == 'C') else 'put',
'strike': strike,
'settle': settle,
'settleId': settle,
'precision': {
'amount': None,
'price': None,
},
'limits': {
'amount': {
'min': None,
'max': None,
},
'price': {
'min': None,
'max': None,
},
'cost': {
'min': None,
'max': None,
},
},
'info': None,
}
def market(self, symbol: str) -> MarketInterface:
if self.markets is None:
raise ExchangeError(self.id + ' markets not loaded')
# defaultType has legacy support on binance
defaultType = self.safe_string(self.options, 'defaultType')
defaultSubType = self.safe_string(self.options, 'defaultSubType')
isLegacyLinear = defaultType == 'future'
isLegacyInverse = defaultType == 'delivery'
isLegacy = isLegacyLinear or isLegacyInverse
if isinstance(symbol, str):
if symbol in self.markets:
market = self.markets[symbol]
# begin diff
if isLegacy and market['spot']:
settle = market['quote'] if isLegacyLinear else market['base']
futuresSymbol = symbol + ':' + settle
if futuresSymbol in self.markets:
return self.markets[futuresSymbol]
else:
return market
# end diff
elif symbol in self.markets_by_id:
markets = self.markets_by_id[symbol]
# begin diff
if isLegacyLinear:
defaultType = 'linear'
elif isLegacyInverse:
defaultType = 'inverse'
elif defaultType is None:
defaultType = defaultSubType
# end diff
for i in range(0, len(markets)):
market = markets[i]
if market[defaultType]:
return market
return markets[0]
elif (symbol.find('/') > -1) and (symbol.find(':') < 0):
if (defaultType is not None) and (defaultType != 'spot'):
# support legacy symbols
base, quote = symbol.split('/')
settle = base if (quote == 'USD') else quote
futuresSymbol = symbol + ':' + settle
if futuresSymbol in self.markets:
return self.markets[futuresSymbol]
elif (symbol.find('-C') > -1) or (symbol.find('-P') > -1): # both exchange-id and unified symbols are supported self way regardless of the defaultType
return self.create_expired_option_market(symbol)
raise BadSymbol(self.id + ' does not have market symbol ' + symbol)
def safe_market(self, marketId: Str = None, market: Market = None, delimiter: Str = None, marketType: Str = None) -> MarketInterface:
isOption = (marketId is not None) and ((marketId.find('-C') > -1) or (marketId.find('-P') > -1))
if isOption and not (marketId in self.markets_by_id):
# handle expired option contracts
return self.create_expired_option_market(marketId)
return super(binance, self).safe_market(marketId, market, delimiter, marketType)
def cost_to_precision(self, symbol, cost):
return self.decimal_to_precision(cost, TRUNCATE, self.markets[symbol]['precision']['quote'], self.precisionMode, self.paddingMode)
def nonce(self):
return self.milliseconds() - self.options['timeDifference']
def enable_demo_trading(self, enable: bool):
"""
enables or disables demo trading mode
https://www.binance.com/en/support/faq/detail/9be58f73e5e14338809e3b705b9687dd
https://demo.binance.com/en/my/settings/api-management
:param boolean [enable]: True if demo trading should be enabled, False otherwise
"""
if self.isSandboxModeEnabled:
raise NotSupported(self.id + ' demo trading is not supported in the sandbox environment. Please check https://www.binance.com/en/support/faq/detail/9be58f73e5e14338809e3b705b9687dd to see the differences')
if enable:
self.urls['apiBackupDemoTrading'] = self.urls['api']
self.urls['api'] = self.urls['demo']
elif 'apiBackupDemoTrading' in self.urls:
self.urls['api'] = self.urls['apiBackupDemoTrading']
newUrls = self.omit(self.urls, 'apiBackupDemoTrading')
self.urls = newUrls
self.options['enableDemoTrading'] = enable
def fetch_time(self, params={}) -> Int:
"""
fetches the current integer timestamp in milliseconds from the exchange server
https://developers.binance.com/docs/binance-spot-api-docs/rest-api/general-endpoints#check-server-time # spot
https://developers.binance.com/docs/derivatives/usds-margined-futures/market-data/rest-api/Check-Server-Time # swap
https://developers.binance.com/docs/derivatives/coin-margined-futures/market-data/rest-api/Check-Server-time # future
:param dict [params]: extra parameters specific to the exchange API endpoint
:param str [params.subType]: "linear" or "inverse"
:returns int: the current integer timestamp in milliseconds from the exchange server
"""
defaultType = self.safe_string_2(self.options, 'fetchTime', 'defaultType', 'spot')
type = self.safe_string(params, 'type', defaultType)
query = self.omit(params, 'type')
subType = None
subType, params = self.handle_sub_type_and_params('fetchTime', None, params)
response = None
if self.is_linear(type, subType):
response = self.fapiPublicGetTime(query)
elif self.is_inverse(type, subType):
response = self.dapiPublicGetTime(query)
else:
response = self.publicGetTime(query)
return self.safe_integer(response, 'serverTime')
def fetch_currencies(self, params={}) -> Currencies:
"""
fetches all available currencies on an exchange
https://developers.binance.com/docs/wallet/capital/all-coins-info
https://developers.binance.com/docs/margin_trading/market-data/Get-All-Margin-Assets
:param dict [params]: extra parameters specific to the exchange API endpoint
:returns dict: an associative dictionary of currencies
"""
fetchCurrenciesEnabled = self.safe_bool(self.options, 'fetchCurrencies')
if not fetchCurrenciesEnabled:
return {}
# self endpoint requires authentication
# while fetchCurrencies is a public API method by design
# therefore we check the keys here
# and fallback to generating the currencies from the markets
if not self.check_required_credentials(False):
return {}
# sandbox/testnet does not support sapi endpoints
apiBackup = self.safe_value(self.urls, 'apiBackup')
if apiBackup is not None:
return {}
# demotrading does not support sapi endpoints
if self.safe_bool(self.options, 'enableDemoTrading', False):
return {}
promises = [self.sapiGetCapitalConfigGetall(params)]
fetchMargins = self.safe_bool(self.options, 'fetchMargins', False)
if fetchMargins:
promises.append(self.sapiGetMarginAllPairs(params))
results = promises
responseCurrencies = results[0]
marginablesById = None
if fetchMargins:
responseMarginables = results[1]
marginablesById = self.index_by(responseMarginables, 'assetName')
result: dict = {}
for i in range(0, len(responseCurrencies)):
#
# {
# "coin": "LINK",
# "depositAllEnable": True,
# "withdrawAllEnable": True,
# "name": "ChainLink",
# "free": "0",
# "locked": "0",
# "freeze": "0",
# "withdrawing": "0",
# "ipoing": "0",
# "ipoable": "0",
# "storage": "0",
# "isLegalMoney": False,
# "trading": True,
# "networkList": [
# {
# "network": "BSC",
# "coin": "LINK",
# "withdrawIntegerMultiple": "0.00000001",
# "isDefault": False,
# "depositEnable": True,
# "withdrawEnable": True,
# "depositDesc": "",
# "withdrawDesc": "",
# "specialTips": "",
# "specialWithdrawTips": "The network you have selected is BSC. Please ensure that the withdrawal address supports the Binance Smart Chain network. You will lose your assets if the chosen platform does not support retrievals.",
# "name": "BNB Smart Chain(BEP20)",
# "resetAddressStatus": False,
# "addressRegex": "^(0x)[0-9A-Fa-f]{40}$",
# "addressRule": "",
# "memoRegex": "",
# "withdrawFee": "0.012",
# "withdrawMin": "0.024",
# "withdrawMax": "9999999999.99999999",
# "minConfirm": "15",
# "unLockConfirm": "0",
# "sameAddress": False,
# "estimatedArrivalTime": "5",
# "busy": False,
# "country": "AE,BINANCE_BAHRAIN_BSC"
# },
# {
# "network": "BNB",
# "coin": "LINK",
# "withdrawIntegerMultiple": "0.00000001",
# "isDefault": False,
# "depositEnable": True,
# "withdrawEnable": True,
# "depositDesc": "",
# "withdrawDesc": "",
# "specialTips": "Both a MEMO and an Address are required to successfully deposit your LINK BEP2 tokens to Binance.",
# "specialWithdrawTips": "",
# "name": "BNB Beacon Chain(BEP2)",
# "resetAddressStatus": False,
# "addressRegex": "^(bnb1)[0-9a-z]{38}$",
# "addressRule": "",
# "memoRegex": "^[0-9A-Za-z\\-_]{1,120}$",
# "withdrawFee": "0.003",
# "withdrawMin": "0.01",
# "withdrawMax": "10000000000",
# "minConfirm": "1",
# "unLockConfirm": "0",
# "sameAddress": True,
# "estimatedArrivalTime": "5",
# "busy": False,
# "country": "AE,BINANCE_BAHRAIN_BSC"
# },
# {
# "network": "ETH",
# "coin": "LINK",
# "withdrawIntegerMultiple": "0.00000001",
# "isDefault": True,
# "depositEnable": True,
# "withdrawEnable": True,
# "depositDesc": "",
# "withdrawDesc": "",
# "name": "Ethereum(ERC20)",
# "resetAddressStatus": False,
# "addressRegex": "^(0x)[0-9A-Fa-f]{40}$",
# "addressRule": "",
# "memoRegex": "",
# "withdrawFee": "0.55",
# "withdrawMin": "1.1",
# "withdrawMax": "10000000000",
# "minConfirm": "12",
# "unLockConfirm": "0",
# "sameAddress": False,
# "estimatedArrivalTime": "5",
# "busy": False,
# "country": "AE,BINANCE_BAHRAIN_BSC"
# }
# ]
# }
#
entry = responseCurrencies[i]
id = self.safe_string(entry, 'coin')
name = self.safe_string(entry, 'name')
code = self.safe_currency_code(id)
isFiat = self.safe_bool(entry, 'isLegalMoney')
minPrecision = None
isWithdrawEnabled = True
isDepositEnabled = True
networkList = self.safe_list(entry, 'networkList', [])
fees: dict = {}
fee = None
networks: dict = {}
for j in range(0, len(networkList)):
networkItem = networkList[j]
network = self.safe_string(networkItem, 'network')
networkCode = self.network_id_to_code(network)
isETF = (network == 'ETF') # e.g. BTCUP, ETHDOWN
# name = self.safe_string(networkItem, 'name')
withdrawFee = self.safe_number(networkItem, 'withdrawFee')
depositEnable = self.safe_bool(networkItem, 'depositEnable')
withdrawEnable = self.safe_bool(networkItem, 'withdrawEnable')
isDepositEnabled = isDepositEnabled or depositEnable
isWithdrawEnabled = isWithdrawEnabled or withdrawEnable
fees[network] = withdrawFee
isDefault = self.safe_bool(networkItem, 'isDefault')
if isDefault or (fee is None):
fee = withdrawFee
# todo: default networks in "setMarkets" overload
# if isDefault:
# self.options['defaultNetworkCodesForCurrencies'][code] = networkCode
# }
precisionTick = self.safe_string(networkItem, 'withdrawIntegerMultiple')
withdrawPrecision = precisionTick
# avoid zero values, which are mostly from fiat or leveraged tokens or some abandoned coins : https://github.com/ccxt/ccxt/pull/14902#issuecomment-1271636731
if not Precise.string_eq(precisionTick, '0'):
minPrecision = precisionTick if (minPrecision is None) else Precise.string_min(minPrecision, precisionTick)
else:
if not isFiat and not isETF:
# non-fiat and non-ETF currency, there are many cases when precision is set to zero(probably bug, we've reported to binance already)
# in such cases, we can set default precision of 8(which is in UI for such coins)
withdrawPrecision = self.omit_zero(self.safe_string(networkItem, 'withdrawInternalMin'))
if withdrawPrecision is None:
withdrawPrecision = self.safe_string(self.options, 'defaultWithdrawPrecision')
networks[networkCode] = {
'info': networkItem,
'id': network,
'network': networkCode,
'active': depositEnable and withdrawEnable,
'deposit': depositEnable,
'withdraw': withdrawEnable,
'fee': withdrawFee,
'precision': self.parse_number(withdrawPrecision),
'limits': {
'withdraw': {
'min': self.safe_number(networkItem, 'withdrawMin'),
'max': self.safe_number(networkItem, 'withdrawMax'),
},
'deposit': {
'min': self.safe_number(networkItem, 'depositDust'),
'max': None,
},
},
}
trading = self.safe_bool(entry, 'trading')
active = (isWithdrawEnabled and isDepositEnabled and trading)
marginEntry = self.safe_dict(marginablesById, id, {})
#
# {
# assetName: "BTC",
# assetFullName: "Bitcoin",
# isBorrowable: True,
# isMortgageable: True,
# userMinBorrow: "0",
# userMinRepay: "0",
# }
#
result[code] = {
'id': id,
'name': name,
'code': code,
'type': 'fiat' if isFiat else 'crypto',
'precision': self.parse_number(minPrecision),
'info': entry,
'active': active,
'deposit': isDepositEnabled,
'withdraw': isWithdrawEnabled,
'networks': networks,
'fee': fee,
'fees': fees,
'limits': self.limits,
'margin': self.safe_bool(marginEntry, 'isBorrowable'),
}
return result
def fetch_markets(self, params={}) -> List[Market]:
"""
retrieves data on all markets for binance
https://developers.binance.com/docs/binance-spot-api-docs/rest-api/general-endpoints#exchange-information # spot
https://developers.binance.com/docs/derivatives/usds-margined-futures/market-data/rest-api/Exchange-Information # swap
https://developers.binance.com/docs/derivatives/coin-margined-futures/market-data/rest-api/Exchange-Information # future
https://developers.binance.com/docs/derivatives/option/market-data/Exchange-Information # option
https://developers.binance.com/docs/margin_trading/market-data/Get-All-Cross-Margin-Pairs # cross margin
https://developers.binance.com/docs/margin_trading/market-data/Get-All-Isolated-Margin-Symbol # isolated margin
:param dict [params]: extra parameters specific to the exchange API endpoint
:returns dict[]: an array of objects representing market data
"""
promisesRaw = []
rawFetchMarkets = None
defaultTypes = ['spot', 'linear', 'inverse']
fetchMarketsOptions = self.safe_dict(self.options, 'fetchMarkets')
if fetchMarketsOptions is not None:
rawFetchMarkets = self.safe_list(fetchMarketsOptions, 'types', defaultTypes)
else:
# for backward-compatibility
rawFetchMarkets = self.safe_list(self.options, 'fetchMarkets', defaultTypes)
# handle loadAllOptions option
loadAllOptions = self.safe_bool(self.options, 'loadAllOptions', False)
if loadAllOptions:
if not self.in_array('option', rawFetchMarkets):
rawFetchMarkets.append('option')
sandboxMode = self.safe_bool(self.options, 'sandboxMode', False)
demoMode = self.safe_bool(self.options, 'enableDemoTrading', False)
isDemoEnv = demoMode or sandboxMode
fetchMarkets = []
for i in range(0, len(rawFetchMarkets)):
type = rawFetchMarkets[i]
if type == 'option' and isDemoEnv:
continue
fetchMarkets.append(type)
fetchMargins = self.safe_bool(self.options, 'fetchMargins', False)
for i in range(0, len(fetchMarkets)):
marketType = fetchMarkets[i]
if marketType == 'spot':
promisesRaw.append(self.publicGetExchangeInfo(params))
if fetchMargins and self.check_required_credentials(False) and not isDemoEnv:
promisesRaw.append(self.sapiGetMarginAllPairs(params))
promisesRaw.append(self.sapiGetMarginIsolatedAllPairs(params))
elif marketType == 'linear':
promisesRaw.append(self.fapiPublicGetExchangeInfo(params))
elif marketType == 'inverse':
promisesRaw.append(self.dapiPublicGetExchangeInfo(params))
elif marketType == 'option':
promisesRaw.append(self.eapiPublicGetExchangeInfo(params))
else:
raise ExchangeError(self.id + ' fetchMarkets() self.options fetchMarkets "' + marketType + '" is not a supported market type')
results = promisesRaw
markets = []
self.options['crossMarginPairsData'] = []
self.options['isolatedMarginPairsData'] = []
for i in range(0, len(results)):
res = self.safe_value(results, i)
if fetchMargins and isinstance(res, list):
keysList = list(self.index_by(res, 'symbol').keys())
length = len(self.options['crossMarginPairsData'])
# first one is the cross-margin promise
if length == 0:
self.options['crossMarginPairsData'] = keysList
else:
self.options['isolatedMarginPairsData'] = keysList
else:
resultMarkets = self.safe_list_2(res, 'symbols', 'optionSymbols', [])
markets = self.array_concat(markets, resultMarkets)
#
# spot / margin
#
# {
# "timezone":"UTC",
# "serverTime":1575416692969,
# "rateLimits":[
# {"rateLimitType":"REQUEST_WEIGHT","interval":"MINUTE","intervalNum":1,"limit":1200},
# {"rateLimitType":"ORDERS","interval":"SECOND","intervalNum":10,"limit":100},
# {"rateLimitType":"ORDERS","interval":"DAY","intervalNum":1,"limit":200000}
# ],
# "exchangeFilters":[],
# "symbols":[
# {
# "symbol":"ETHBTC",
# "status":"TRADING",
# "baseAsset":"ETH",
# "baseAssetPrecision":8,
# "quoteAsset":"BTC",
# "quotePrecision":8,
# "baseCommissionPrecision":8,
# "quoteCommissionPrecision":8,
# "orderTypes":["LIMIT","LIMIT_MAKER","MARKET","STOP_LOSS_LIMIT","TAKE_PROFIT_LIMIT"],
# "icebergAllowed":true,
# "ocoAllowed":true,
# "quoteOrderQtyMarketAllowed":true,
# "allowTrailingStop":false,
# "isSpotTradingAllowed":true,
# "isMarginTradingAllowed":true,
# "filters":[
# {"filterType":"PRICE_FILTER","minPrice":"0.00000100","maxPrice":"100000.00000000","tickSize":"0.00000100"},
# {"filterType":"PERCENT_PRICE","multiplierUp":"5","multiplierDown":"0.2","avgPriceMins":5},
# {"filterType":"LOT_SIZE","minQty":"0.00100000","maxQty":"100000.00000000","stepSize":"0.00100000"},
# {"filterType":"MIN_NOTIONAL","minNotional":"0.00010000","applyToMarket":true,"avgPriceMins":5},
# {"filterType":"ICEBERG_PARTS","limit":10},
# {"filterType":"MARKET_LOT_SIZE","minQty":"0.00000000","maxQty":"63100.00000000","stepSize":"0.00000000"},
# {"filterType":"MAX_NUM_ORDERS","maxNumOrders":200},
# {"filterType":"MAX_NUM_ALGO_ORDERS","maxNumAlgoOrders":5}
# ],
# "permissions":["SPOT","MARGIN"]}
# },
# ],
# }
#
# cross & isolated pairs response:
#
# [
# {
# symbol: "BTCUSDT",
# base: "BTC",
# quote: "USDT",
# isMarginTrade: True,
# isBuyAllowed: True,
# isSellAllowed: True,
# id: "376870555451677893", # doesn't exist in isolated
# },
# ]
#
# futures/usdt-margined(fapi)
#
# {
# "timezone":"UTC",
# "serverTime":1575417244353,
# "rateLimits":[
# {"rateLimitType":"REQUEST_WEIGHT","interval":"MINUTE","intervalNum":1,"limit":1200},
# {"rateLimitType":"ORDERS","interval":"MINUTE","intervalNum":1,"limit":1200}
# ],
# "exchangeFilters":[],
# "symbols":[
# {
# "symbol":"BTCUSDT",
# "status":"TRADING",
# "maintMarginPercent":"2.5000",
# "requiredMarginPercent":"5.0000",
# "baseAsset":"BTC",
# "quoteAsset":"USDT",
# "pricePrecision":2,
# "quantityPrecision":3,
# "baseAssetPrecision":8,
# "quotePrecision":8,
# "filters":[
# {"minPrice":"0.01","maxPrice":"100000","filterType":"PRICE_FILTER","tickSize":"0.01"},
# {"stepSize":"0.001","filterType":"LOT_SIZE","maxQty":"1000","minQty":"0.001"},
# {"stepSize":"0.001","filterType":"MARKET_LOT_SIZE","maxQty":"1000","minQty":"0.001"},
# {"limit":200,"filterType":"MAX_NUM_ORDERS"},
# {"multiplierDown":"0.8500","multiplierUp":"1.1500","multiplierDecimal":"4","filterType":"PERCENT_PRICE"}
# ],
# "orderTypes":["LIMIT","MARKET","STOP"],
# "timeInForce":["GTC","IOC","FOK","GTX"]
# }
# ]
# }
#
# delivery/coin-margined(dapi)
#
# {
# "timezone": "UTC",
# "serverTime": 1597667052958,
# "rateLimits": [
# {"rateLimitType":"REQUEST_WEIGHT","interval":"MINUTE","intervalNum":1,"limit":6000},
# {"rateLimitType":"ORDERS","interval":"MINUTE","intervalNum":1,"limit":6000}
# ],
# "exchangeFilters": [],
# "symbols": [
# {
# "symbol": "BTCUSD_200925",
# "pair": "BTCUSD",
# "contractType": "CURRENT_QUARTER",
# "deliveryDate": 1601020800000,
# "onboardDate": 1590739200000,
# "contractStatus": "TRADING",
# "contractSize": 100,
# "marginAsset": "BTC",
# "maintMarginPercent": "2.5000",
# "requiredMarginPercent": "5.0000",
# "baseAsset": "BTC",
# "quoteAsset": "USD",
# "pricePrecision": 1,
# "quantityPrecision": 0,
# "baseAssetPrecision": 8,
# "quotePrecision": 8,
# "equalQtyPrecision": 4,
# "filters": [
# {"minPrice":"0.1","maxPrice":"100000","filterType":"PRICE_FILTER","tickSize":"0.1"},
# {"stepSize":"1","filterType":"LOT_SIZE","maxQty":"100000","minQty":"1"},
# {"stepSize":"0","filterType":"MARKET_LOT_SIZE","maxQty":"100000","minQty":"1"},
# {"limit":200,"filterType":"MAX_NUM_ORDERS"},
# {"multiplierDown":"0.9500","multiplierUp":"1.0500","multiplierDecimal":"4","filterType":"PERCENT_PRICE"}
# ],
# "orderTypes": ["LIMIT","MARKET","STOP","STOP_MARKET","TAKE_PROFIT","TAKE_PROFIT_MARKET","TRAILING_STOP_MARKET"],
# "timeInForce": ["GTC","IOC","FOK","GTX"]
# },
# {
# "symbol": "BTCUSD_PERP",
# "pair": "BTCUSD",
# "contractType": "PERPETUAL",
# "deliveryDate": 4133404800000,
# "onboardDate": 1596006000000,
# "contractStatus": "TRADING",
# "contractSize": 100,
# "marginAsset": "BTC",
# "maintMarginPercent": "2.5000",
# "requiredMarginPercent": "5.0000",
# "baseAsset": "BTC",
# "quoteAsset": "USD",
# "pricePrecision": 1,
# "quantityPrecision": 0,
# "baseAssetPrecision": 8,
# "quotePrecision": 8,
# "equalQtyPrecision": 4,
# "filters": [
# {"minPrice":"0.1","maxPrice":"100000","filterType":"PRICE_FILTER","tickSize":"0.1"},
# {"stepSize":"1","filterType":"LOT_SIZE","maxQty":"100000","minQty":"1"},
# {"stepSize":"1","filterType":"MARKET_LOT_SIZE","maxQty":"100000","minQty":"1"},
# {"limit":200,"filterType":"MAX_NUM_ORDERS"},
# {"multiplierDown":"0.8500","multiplierUp":"1.1500","multiplierDecimal":"4","filterType":"PERCENT_PRICE"}
# ],
# "orderTypes": ["LIMIT","MARKET","STOP","STOP_MARKET","TAKE_PROFIT","TAKE_PROFIT_MARKET","TRAILING_STOP_MARKET"],
# "timeInForce": ["GTC","IOC","FOK","GTX"]
# }
# ]
# }
#
# options(eapi)
#
# {
# "timezone": "UTC",
# "serverTime": 1675912490405,
# "optionContracts": [
# {
# "id": 1,
# "baseAsset": "SOL",
# "quoteAsset": "USDT",
# "underlying": "SOLUSDT",
# "settleAsset": "USDT"
# },
# ...
# ],
# "optionAssets": [
# {"id":1,"name":"USDT"}
# ],
# "optionSymbols": [
# {
# "contractId": 3,
# "expiryDate": 1677225600000,
# "filters": [
# {"filterType":"PRICE_FILTER","minPrice":"724.6","maxPrice":"919.2","tickSize":"0.1"},
# {"filterType":"LOT_SIZE","minQty":"0.01","maxQty":"1001","stepSize":"0.01"}
# ],
# "id": 2474,
# "symbol": "ETH-230224-800-C",
# "side": "CALL",
# "strikePrice": "800.00000000",
# "underlying": "ETHUSDT",
# "unit": 1,
# "makerFeeRate": "0.00020000",
# "takerFeeRate": "0.00020000",
# "minQty": "0.01",
# "maxQty": "1000",
# "initialMargin": "0.15000000",
# "maintenanceMargin": "0.07500000",
# "minInitialMargin": "0.10000000",
# "minMaintenanceMargin": "0.05000000",
# "priceScale": 1,
# "quantityScale": 2,
# "quoteAsset": "USDT"
# },
# ...
# ],
# "rateLimits": [
# {"rateLimitType":"REQUEST_WEIGHT","interval":"MINUTE","intervalNum":1,"limit":400},
# {"rateLimitType":"ORDERS","interval":"MINUTE","intervalNum":1,"limit":100},
# {"rateLimitType":"ORDERS","interval":"SECOND","intervalNum":10,"limit":30}
# ]
# }
#
if self.options['adjustForTimeDifference']:
self.load_time_difference()
result = []
for i in range(0, len(markets)):
result.append(self.parse_market(markets[i]))
return result
def parse_market(self, market: dict) -> Market:
swap = False
future = False
option = False
underlying = self.safe_string(market, 'underlying')
id = self.safe_string(market, 'symbol')
optionParts = id.split('-')
optionBase = self.safe_string(optionParts, 0)
lowercaseId = self.safe_string_lower(market, 'symbol')
baseId = self.safe_string(market, 'baseAsset', optionBase)
quoteId = self.safe_string(market, 'quoteAsset')
base = self.safe_currency_code(baseId)
quote = self.safe_currency_code(quoteId)
contractType = self.safe_string(market, 'contractType')
contract = ('contractType' in market)
expiry = self.safe_integer_2(market, 'deliveryDate', 'expiryDate')
settleId = self.safe_string(market, 'marginAsset')
if (contractType == 'PERPETUAL') or (expiry == 4133404800000): # some swap markets do not have contract type, eg: BTCST
expiry = None
swap = True
elif underlying is not None:
contract = True
option = True
settleId = 'USDT' if (settleId is None) else settleId
elif expiry is not None:
future = True
settle = self.safe_currency_code(settleId)
spot = not contract
filters = self.safe_list(market, 'filters', [])
filtersByType = self.index_by(filters, 'filterType')
status = self.safe_string_2(market, 'status', 'contractStatus')
contractSize = None
fees = self.fees
linear = None
inverse = None
symbol = base + '/' + quote
strike = None
if contract:
if swap:
symbol = symbol + ':' + settle
elif future:
symbol = symbol + ':' + settle + '-' + self.yymmdd(expiry)
elif option:
strike = self.number_to_string(self.parse_to_numeric(self.safe_string(market, 'strikePrice')))
symbol = symbol + ':' + settle + '-' + self.yymmdd(expiry) + '-' + strike + '-' + self.safe_string(optionParts, 3)
contractSize = self.safe_number_2(market, 'contractSize', 'unit', self.parse_number('1'))
linear = settle == quote
inverse = settle == base
feesType = 'linear' if linear else 'inverse'
fees = self.safe_dict(self.fees, feesType, {})
active = (status == 'TRADING')
if spot:
permissions = self.safe_list(market, 'permissions', [])
for j in range(0, len(permissions)):
if permissions[j] == 'TRD_GRP_003':
active = False
break
isMarginTradingAllowed = self.safe_bool(market, 'isMarginTradingAllowed', False)
marginModes = None
if spot:
hasCrossMargin = self.in_array(id, self.options['crossMarginPairsData'])
hasIsolatedMargin = self.in_array(id, self.options['isolatedMarginPairsData'])
marginModes = {
'cross': hasCrossMargin,
'isolated': hasIsolatedMargin,
}
elif linear or inverse:
marginModes = {
'cross': True,
'isolated': True,
}
unifiedType = None
if spot:
unifiedType = 'spot'
elif swap:
unifiedType = 'swap'
elif future:
unifiedType = 'future'
elif option:
unifiedType = 'option'
active = None
parsedStrike = None
if strike is not None:
parsedStrike = self.parse_to_numeric(strike)
entry = {
'id': id,
'lowercaseId': lowercaseId,
'symbol': symbol,
'base': base,
'quote': quote,
'settle': settle,
'baseId': baseId,
'quoteId': quoteId,
'settleId': settleId,
'type': unifiedType,
'spot': spot,
'margin': spot and isMarginTradingAllowed,
'marginModes': marginModes,
'swap': swap,
'future': future,
'option': option,
'active': active,
'contract': contract,
'linear': linear,
'inverse': inverse,
'taker': fees['trading']['taker'],
'maker': fees['trading']['maker'],
'contractSize': contractSize,
'expiry': expiry,
'expiryDatetime': self.iso8601(expiry),
'strike': parsedStrike,
'optionType': self.safe_string_lower(market, 'side'),
'precision': {
'amount': self.parse_number(self.parse_precision(self.safe_string_2(market, 'quantityPrecision', 'quantityScale'))),
'price': self.parse_number(self.parse_precision(self.safe_string_2(market, 'pricePrecision', 'priceScale'))),
'base': self.parse_number(self.parse_precision(self.safe_string(market, 'baseAssetPrecision'))),
'quote': self.parse_number(self.parse_precision(self.safe_string(market, 'quotePrecision'))),
},
'limits': {
'leverage': {
'min': None,
'max': None,
},
'amount': {
'min': self.safe_number(market, 'minQty'),
'max': self.safe_number(market, 'maxQty'),
},
'price': {
'min': None,
'max': None,
},
'cost': {
'min': None,
'max': None,
},
},
'info': market,
'created': self.safe_integer(market, 'onboardDate'), # present in inverse & linear apis
}
if 'PRICE_FILTER' in filtersByType:
filter = self.safe_dict(filtersByType, 'PRICE_FILTER', {})
# PRICE_FILTER reports zero values for maxPrice
# since they updated filter types in November 2018
# https://github.com/ccxt/ccxt/issues/4286
# therefore limits['price']['max'] doesn't have any meaningful value except None
entry['limits']['price'] = {
'min': self.safe_number(filter, 'minPrice'),
'max': self.safe_number(filter, 'maxPrice'),
}
entry['precision']['price'] = self.safe_number(filter, 'tickSize')
if 'LOT_SIZE' in filtersByType:
filter = self.safe_dict(filtersByType, 'LOT_SIZE', {})
entry['precision']['amount'] = self.safe_number(filter, 'stepSize')
entry['limits']['amount'] = {
'min': self.safe_number(filter, 'minQty'),
'max': self.safe_number(filter, 'maxQty'),
}
if 'MARKET_LOT_SIZE' in filtersByType:
filter = self.safe_dict(filtersByType, 'MARKET_LOT_SIZE', {})
entry['limits']['market'] = {
'min': self.safe_number(filter, 'minQty'),
'max': self.safe_number(filter, 'maxQty'),
}
if ('MIN_NOTIONAL' in filtersByType) or ('NOTIONAL' in filtersByType): # notional added in 12/04/23 to spot testnet
filter = self.safe_dict_2(filtersByType, 'MIN_NOTIONAL', 'NOTIONAL', {})
entry['limits']['cost']['min'] = self.safe_number_2(filter, 'minNotional', 'notional')
entry['limits']['cost']['max'] = self.safe_number(filter, 'maxNotional')
return entry
def parse_balance_helper(self, entry):
account = self.account()
account['used'] = self.safe_string(entry, 'locked')
account['free'] = self.safe_string(entry, 'free')
interest = self.safe_string(entry, 'interest')
debt = self.safe_string(entry, 'borrowed')
account['debt'] = Precise.string_add(debt, interest)
return account
def parse_balance_custom(self, response, type=None, marginMode=None, isPortfolioMargin=False) -> Balances:
result = {
'info': response,
}
timestamp = None
isolated = marginMode == 'isolated'
cross = (type == 'margin') or (marginMode == 'cross')
if isPortfolioMargin:
for i in range(0, len(response)):
entry = response[i]
account = self.account()
currencyId = self.safe_string(entry, 'asset')
code = self.safe_currency_code(currencyId)
if type == 'linear':
account['free'] = self.safe_string(entry, 'umWalletBalance')
account['used'] = self.safe_string(entry, 'umUnrealizedPNL')
elif type == 'inverse':
account['free'] = self.safe_string(entry, 'cmWalletBalance')
account['used'] = self.safe_string(entry, 'cmUnrealizedPNL')
elif cross:
borrowed = self.safe_string(entry, 'crossMarginBorrowed')
interest = self.safe_string(entry, 'crossMarginInterest')
account['debt'] = Precise.string_add(borrowed, interest)
account['free'] = self.safe_string(entry, 'crossMarginFree')
account['used'] = self.safe_string(entry, 'crossMarginLocked')
account['total'] = self.safe_string(entry, 'crossMarginAsset')
else:
usedLinear = self.safe_string(entry, 'umUnrealizedPNL')
usedInverse = self.safe_string(entry, 'cmUnrealizedPNL')
totalUsed = Precise.string_add(usedLinear, usedInverse)
totalWalletBalance = self.safe_string(entry, 'totalWalletBalance')
account['total'] = Precise.string_add(totalUsed, totalWalletBalance)
result[code] = account
elif not isolated and ((type == 'spot') or cross):
timestamp = self.safe_integer(response, 'updateTime')
balances = self.safe_list_2(response, 'balances', 'userAssets', [])
for i in range(0, len(balances)):
balance = balances[i]
currencyId = self.safe_string(balance, 'asset')
code = self.safe_currency_code(currencyId)
account = self.account()
account['free'] = self.safe_string(balance, 'free')
account['used'] = self.safe_string(balance, 'locked')
if cross:
debt = self.safe_string(balance, 'borrowed')
interest = self.safe_string(balance, 'interest')
account['debt'] = Precise.string_add(debt, interest)
result[code] = account
elif isolated:
assets = self.safe_list(response, 'assets')
for i in range(0, len(assets)):
asset = assets[i]
marketId = self.safe_string(asset, 'symbol')
symbol = self.safe_symbol(marketId, None, None, 'spot')
base = self.safe_dict(asset, 'baseAsset', {})
quote = self.safe_dict(asset, 'quoteAsset', {})
baseCode = self.safe_currency_code(self.safe_string(base, 'asset'))
quoteCode = self.safe_currency_code(self.safe_string(quote, 'asset'))
subResult: dict = {}
subResult[baseCode] = self.parse_balance_helper(base)
subResult[quoteCode] = self.parse_balance_helper(quote)
result[symbol] = self.safe_balance(subResult)
elif type == 'savings':
positionAmountVos = self.safe_list(response, 'positionAmountVos', [])
for i in range(0, len(positionAmountVos)):
entry = positionAmountVos[i]
currencyId = self.safe_string(entry, 'asset')
code = self.safe_currency_code(currencyId)
account = self.account()
usedAndTotal = self.safe_string(entry, 'amount')
account['total'] = usedAndTotal
account['used'] = usedAndTotal
result[code] = account
elif type == 'funding':
for i in range(0, len(response)):
entry = response[i]
account = self.account()
currencyId = self.safe_string(entry, 'asset')
code = self.safe_currency_code(currencyId)
account['free'] = self.safe_string(entry, 'free')
frozen = self.safe_string(entry, 'freeze')
withdrawing = self.safe_string(entry, 'withdrawing')
locked = self.safe_string(entry, 'locked')
account['used'] = Precise.string_add(frozen, Precise.string_add(locked, withdrawing))
result[code] = account
else:
balances = response
if not isinstance(response, list):
balances = self.safe_list(response, 'assets', [])
for i in range(0, len(balances)):
balance = balances[i]
currencyId = self.safe_string(balance, 'asset')
code = self.safe_currency_code(currencyId)
account = self.account()
account['free'] = self.safe_string(balance, 'availableBalance')
account['used'] = self.safe_string(balance, 'initialMargin')
account['total'] = self.safe_string_2(balance, 'marginBalance', 'balance')
result[code] = account
result['timestamp'] = timestamp
result['datetime'] = self.iso8601(timestamp)
return result if isolated else self.safe_balance(result)
def fetch_balance(self, params={}) -> Balances:
"""
query for balance and get the amount of funds available for trading or funds locked in orders
https://developers.binance.com/docs/binance-spot-api-docs/rest-api/account-endpoints#account-information-user_data # spot
https://developers.binance.com/docs/margin_trading/account/Query-Cross-Margin-Account-Details # cross margin
https://developers.binance.com/docs/margin_trading/account/Query-Isolated-Margin-Account-Info # isolated margin
https://developers.binance.com/docs/wallet/asset/funding-wallet # funding
https://developers.binance.com/docs/derivatives/usds-margined-futures/account/rest-api/Futures-Account-Balance-V2 # swap
https://developers.binance.com/docs/derivatives/coin-margined-futures/account/rest-api/Futures-Account-Balance # future
https://developers.binance.com/docs/derivatives/option/account/Option-Account-Information # option
https://developers.binance.com/docs/derivatives/portfolio-margin/account/Account-Balance # portfolio margin
:param dict [params]: extra parameters specific to the exchange API endpoint
:param str [params.type]: 'future', 'delivery', 'savings', 'funding', or 'spot' or 'papi'
:param str [params.marginMode]: 'cross' or 'isolated', for margin trading, uses self.options.defaultMarginMode if not passed, defaults to None/None/None
:param str[]|None [params.symbols]: unified market symbols, only used in isolated margin mode
:param boolean [params.portfolioMargin]: set to True if you would like to fetch the balance for a portfolio margin account
:param str [params.subType]: 'linear' or 'inverse'
:returns dict: a `balance structure <https://docs.ccxt.com/#/?id=balance-structure>`
"""
self.load_markets()
defaultType = self.safe_string_2(self.options, 'fetchBalance', 'defaultType', 'spot')
type = self.safe_string(params, 'type', defaultType)
subType = None
subType, params = self.handle_sub_type_and_params('fetchBalance', None, params)
isPortfolioMargin = None
isPortfolioMargin, params = self.handle_option_and_params_2(params, 'fetchBalance', 'papi', 'portfolioMargin', False)
marginMode = None
query = None
marginMode, query = self.handle_margin_mode_and_params('fetchBalance', params)
query = self.omit(query, 'type')
response = None
request: dict = {}
if isPortfolioMargin or (type == 'papi'):
if self.is_linear(type, subType):
type = 'linear'
elif self.is_inverse(type, subType):
type = 'inverse'
isPortfolioMargin = True
response = self.papiGetBalance(self.extend(request, query))
elif self.is_linear(type, subType):
type = 'linear'
useV2 = None
useV2, params = self.handle_option_and_params(params, 'fetchBalance', 'useV2', False)
params = self.extend(request, query)
if not useV2:
response = self.fapiPrivateV3GetAccount(params)
else:
response = self.fapiPrivateV2GetAccount(params)
elif self.is_inverse(type, subType):
type = 'inverse'
response = self.dapiPrivateGetAccount(self.extend(request, query))
elif marginMode == 'isolated':
paramSymbols = self.safe_list(params, 'symbols')
query = self.omit(query, 'symbols')
if paramSymbols is not None:
symbols = ''
if isinstance(paramSymbols, list):
symbols = self.market_id(paramSymbols[0])
for i in range(1, len(paramSymbols)):
symbol = paramSymbols[i]
id = self.market_id(symbol)
symbols += ',' + id
else:
symbols = paramSymbols
request['symbols'] = symbols
response = self.sapiGetMarginIsolatedAccount(self.extend(request, query))
elif (type == 'margin') or (marginMode == 'cross'):
response = self.sapiGetMarginAccount(self.extend(request, query))
elif type == 'savings':
response = self.sapiGetLendingUnionAccount(self.extend(request, query))
elif type == 'funding':
response = self.sapiPostAssetGetFundingAsset(self.extend(request, query))
else:
response = self.privateGetAccount(self.extend(request, query))
#
# spot
#
# {
# "makerCommission": 10,
# "takerCommission": 10,
# "buyerCommission": 0,
# "sellerCommission": 0,
# "canTrade": True,
# "canWithdraw": True,
# "canDeposit": True,
# "updateTime": 1575357359602,
# "accountType": "MARGIN",
# "balances": [
# {asset: "BTC", free: "0.00219821", locked: "0.00000000" },
# ]
# }
#
# margin(cross)
#
# {
# "borrowEnabled":true,
# "marginLevel":"999.00000000",
# "totalAssetOfBtc":"0.00000000",
# "totalLiabilityOfBtc":"0.00000000",
# "totalNetAssetOfBtc":"0.00000000",
# "tradeEnabled":true,
# "transferEnabled":true,
# "userAssets":[
# {"asset":"MATIC","borrowed":"0.00000000","free":"0.00000000","interest":"0.00000000","locked":"0.00000000","netAsset":"0.00000000"},
# {"asset":"VET","borrowed":"0.00000000","free":"0.00000000","interest":"0.00000000","locked":"0.00000000","netAsset":"0.00000000"},
# {"asset":"USDT","borrowed":"0.00000000","free":"0.00000000","interest":"0.00000000","locked":"0.00000000","netAsset":"0.00000000"}
# ],
# }
#
# margin(isolated)
#
# {
# "info": {
# "assets": [
# {
# "baseAsset": {
# "asset": "1INCH",
# "borrowEnabled": True,
# "borrowed": "0",
# "free": "0",
# "interest": "0",
# "locked": "0",
# "netAsset": "0",
# "netAssetOfBtc": "0",
# "repayEnabled": True,
# "totalAsset": "0"
# },
# "quoteAsset": {
# "asset": "USDT",
# "borrowEnabled": True,
# "borrowed": "0",
# "free": "11",
# "interest": "0",
# "locked": "0",
# "netAsset": "11",
# "netAssetOfBtc": "0.00054615",
# "repayEnabled": True,
# "totalAsset": "11"
# },
# "symbol": "1INCHUSDT",
# "isolatedCreated": True,
# "marginLevel": "999",
# "marginLevelStatus": "EXCESSIVE",
# "marginRatio": "5",
# "indexPrice": "0.59184331",
# "liquidatePrice": "0",
# "liquidateRate": "0",
# "tradeEnabled": True,
# "enabled": True
# },
# ]
# }
# }
#
# futures(fapi)
#
# fapiPrivateV3GetAccount
#
# {
# "feeTier":0,
# "canTrade":true,
# "canDeposit":true,
# "canWithdraw":true,
# "updateTime":0,
# "totalInitialMargin":"0.00000000",
# "totalMaintMargin":"0.00000000",
# "totalWalletBalance":"0.00000000",
# "totalUnrealizedProfit":"0.00000000",
# "totalMarginBalance":"0.00000000",
# "totalPositionInitialMargin":"0.00000000",
# "totalOpenOrderInitialMargin":"0.00000000",
# "totalCrossWalletBalance":"0.00000000",
# "totalCrossUnPnl":"0.00000000",
# "availableBalance":"0.00000000",
# "maxWithdrawAmount":"0.00000000",
# "assets":[
# {
# "asset":"BNB",
# "walletBalance":"0.01000000",
# "unrealizedProfit":"0.00000000",
# "marginBalance":"0.01000000",
# "maintMargin":"0.00000000",
# "initialMargin":"0.00000000",
# "positionInitialMargin":"0.00000000",
# "openOrderInitialMargin":"0.00000000",
# "maxWithdrawAmount":"0.01000000",
# "crossWalletBalance":"0.01000000",
# "crossUnPnl":"0.00000000",
# "availableBalance":"0.01000000"
# }
# ],
# "positions":[
# {
# "symbol":"BTCUSDT",
# "initialMargin":"0",
# "maintMargin":"0",
# "unrealizedProfit":"0.00000000",
# "positionInitialMargin":"0",
# "openOrderInitialMargin":"0",
# "leverage":"21",
# "isolated":false,
# "entryPrice":"0.00000",
# "maxNotional":"5000000",
# "positionSide":"BOTH"
# },
# ]
# }
#
# fapiPrivateV2GetBalance
#
# [
# {
# "accountAlias":"FzFzXquXXqoC",
# "asset":"BNB",
# "balance":"0.01000000",
# "crossWalletBalance":"0.01000000",
# "crossUnPnl":"0.00000000",
# "availableBalance":"0.01000000",
# "maxWithdrawAmount":"0.01000000"
# }
# ]
#
# binance pay
#
# [
# {
# "asset": "BUSD",
# "free": "1129.83",
# "locked": "0",
# "freeze": "0",
# "withdrawing": "0"
# }
# ]
#
# portfolio margin
#
# [
# {
# "asset": "USDT",
# "totalWalletBalance": "66.9923261",
# "crossMarginAsset": "35.9697141",
# "crossMarginBorrowed": "0.0",
# "crossMarginFree": "35.9697141",
# "crossMarginInterest": "0.0",
# "crossMarginLocked": "0.0",
# "umWalletBalance": "31.022612",
# "umUnrealizedPNL": "0.0",
# "cmWalletBalance": "0.0",
# "cmUnrealizedPNL": "0.0",
# "updateTime": 0,
# "negativeBalance": "0.0"
# },
# ]
#
return self.parse_balance_custom(response, type, marginMode, isPortfolioMargin)
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://developers.binance.com/docs/binance-spot-api-docs/rest-api/market-data-endpoints#order-book # spot
https://developers.binance.com/docs/derivatives/usds-margined-futures/market-data/rest-api/Order-Book # swap
https://developers.binance.com/docs/derivatives/coin-margined-futures/market-data/rest-api/Order-Book # future
https://developers.binance.com/docs/derivatives/option/market-data/Order-Book # option
: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
:returns dict: A dictionary of `order book structures <https://docs.ccxt.com/#/?id=order-book-structure>` indexed by market symbols
"""
self.load_markets()
market = self.market(symbol)
request: dict = {
'symbol': market['id'],
}
if limit is not None:
request['limit'] = limit # default 100, max 5000, see https://github.com/binance/binance-spot-api-docs/blob/master/rest-api.md#order-book
response = None
if market['option']:
response = self.eapiPublicGetDepth(self.extend(request, params))
elif market['linear']:
response = self.fapiPublicGetDepth(self.extend(request, params))
elif market['inverse']:
response = self.dapiPublicGetDepth(self.extend(request, params))
else:
response = self.publicGetDepth(self.extend(request, params))
#
# future
#
# {
# "lastUpdateId":333598053905,
# "E":1618631511986,
# "T":1618631511964,
# "bids":[
# ["2493.56","20.189"],
# ["2493.54","1.000"],
# ["2493.51","0.005"]
# ],
# "asks":[
# ["2493.57","0.877"],
# ["2493.62","0.063"],
# ["2493.71","12.054"],
# ]
# }
#
# options(eapi)
#
# {
# "bids": [
# ["108.7","16.08"],
# ["106","21.29"],
# ["82.4","0.02"]
# ],
# "asks": [
# ["111.4","19.52"],
# ["119.9","17.6"],
# ["141.2","31"]
# ],
# "T": 1676771382078,
# "u": 1015939
# }
#
timestamp = self.safe_integer(response, 'T')
orderbook = self.parse_order_book(response, symbol, timestamp)
orderbook['nonce'] = self.safe_integer_2(response, 'lastUpdateId', 'u')
return orderbook
def parse_ticker(self, ticker: dict, market: Market = None) -> Ticker:
# markPrices
#
# {
# "symbol": "BTCUSDT",
# "markPrice": "11793.63104561", # mark price
# "indexPrice": "11781.80495970", # index price
# "estimatedSettlePrice": "11781.16138815", # Estimated Settle Price, only useful in the last hour before the settlement starts
# "lastFundingRate": "0.00038246", # This is the lastest estimated funding rate
# "nextFundingTime": 1597392000000,
# "interestRate": "0.00010000",
# "time": 1597370495002
# }
#
# spot - ticker
#
# {
# "symbol": "BTCUSDT",
# "priceChange": "-188.18000000",
# "priceChangePercent": "-0.159",
# "weightedAvgPrice": "118356.64734074",
# "lastPrice": "118449.03000000",
# "prevClosePrice": "118637.22000000", # field absent in rolling ticker
# "lastQty": "0.00731000", # field absent in rolling ticker
# "bidPrice": "118449.02000000", # field absent in rolling ticker
# "bidQty": "7.15931000", # field absent in rolling ticker
# "askPrice": "118449.03000000", # field absent in rolling ticker
# "askQty": "0.09592000", # field absent in rolling ticker
# "openPrice": "118637.21000000",
# "highPrice": "119273.36000000",
# "lowPrice": "117427.50000000",
# "volume": "14741.41491000",
# "quoteVolume": "1744744445.80640740",
# "openTime": "1753701474013",
# "closeTime": "1753787874013",
# "firstId": "5116031635",
# "lastId": "5117964946",
# "count": "1933312"
# }
#
# usdm tickers
#
# {
# "symbol": "SUSDT",
# "priceChange": "-0.0229000",
# "priceChangePercent": "-6.777",
# "weightedAvgPrice": "0.3210035",
# "lastPrice": "0.3150000",
# "lastQty": "16",
# "openPrice": "0.3379000",
# "highPrice": "0.3411000",
# "lowPrice": "0.3071000",
# "volume": "120588225",
# "quoteVolume": "38709237.2289000",
# "openTime": "1753701720000",
# "closeTime": "1753788172414",
# "firstId": "72234973",
# "lastId": "72423677",
# "count": "188700"
# }
#
# coinm
#
# {
# "baseVolume": "214549.95171161",
# "closeTime": "1621965286847",
# "count": "1283779",
# "firstId": "152560106",
# "highPrice": "39938.3",
# "lastId": "153843955",
# "lastPrice": "37993.4",
# "lastQty": "1",
# "lowPrice": "36457.2",
# "openPrice": "37783.4",
# "openTime": "1621878840000",
# "pair": "BTCUSD",
# "priceChange": "210.0",
# "priceChangePercent": "0.556",
# "symbol": "BTCUSD_PERP",
# "volume": "81990451",
# "weightedAvgPrice": "38215.08713747"
# }
#
# eapi: fetchTicker, fetchTickers
#
# {
# "symbol": "ETH-230510-1825-C",
# "priceChange": "-5.1",
# "priceChangePercent": "-0.1854",
# "lastPrice": "22.4",
# "lastQty": "0",
# "open": "27.5",
# "high": "34.1",
# "low": "22.4",
# "volume": "6.83",
# "amount": "201.44",
# "bidPrice": "21.9",
# "askPrice": "22.4",
# "openTime": 1683614771898,
# "closeTime": 1683695017784,
# "firstTradeId": 12,
# "tradeCount": 22,
# "strikePrice": "1825",
# "exercisePrice": "1845.95341176"
# }
#
# spot bidsAsks
#
# {
# "symbol":"ETHBTC",
# "bidPrice":"0.07466800",
# "bidQty":"5.31990000",
# "askPrice":"0.07466900",
# "askQty":"10.93540000"
# }
#
# usdm bidsAsks
#
# {
# "symbol":"BTCUSDT",
# "bidPrice":"21321.90",
# "bidQty":"33.592",
# "askPrice":"21322.00",
# "askQty":"1.427",
# "time":"1673899207538"
# }
#
# coinm bidsAsks
#
# {
# "symbol":"BTCUSD_PERP",
# "pair":"BTCUSD",
# "bidPrice":"21301.2",
# "bidQty":"188",
# "askPrice":"21301.3",
# "askQty":"10302",
# "time":"1673899278514"
# }
#
timestamp = self.safe_integer_2(ticker, 'closeTime', 'time')
marketType = None
if ('time' in ticker):
marketType = 'contract'
if marketType is None:
marketType = 'spot' if ('bidQty' in ticker) else 'contract'
marketId = self.safe_string(ticker, 'symbol')
symbol = self.safe_symbol(marketId, market, None, marketType)
last = self.safe_string(ticker, 'lastPrice')
wAvg = self.safe_string(ticker, 'weightedAvgPrice')
isCoinm = ('baseVolume' in ticker)
baseVolume = None
quoteVolume = None
if isCoinm:
baseVolume = self.safe_string(ticker, 'baseVolume')
# 'volume' field in inverse markets is not quoteVolume, but traded amount(per contracts)
quoteVolume = Precise.string_mul(baseVolume, wAvg)
else:
baseVolume = self.safe_string(ticker, 'volume')
quoteVolume = self.safe_string_2(ticker, 'quoteVolume', 'amount')
return self.safe_ticker({
'symbol': symbol,
'timestamp': timestamp,
'datetime': self.iso8601(timestamp),
'high': self.safe_string_2(ticker, 'highPrice', 'high'),
'low': self.safe_string_2(ticker, 'lowPrice', 'low'),
'bid': self.safe_string(ticker, 'bidPrice'),
'bidVolume': self.safe_string(ticker, 'bidQty'),
'ask': self.safe_string(ticker, 'askPrice'),
'askVolume': self.safe_string(ticker, 'askQty'),
'vwap': wAvg,
'open': self.safe_string_2(ticker, 'openPrice', 'open'),
'close': last,
'last': last,
'previousClose': self.safe_string(ticker, 'prevClosePrice'), # previous day close
'change': self.safe_string(ticker, 'priceChange'),
'percentage': self.safe_string(ticker, 'priceChangePercent'),
'average': None,
'baseVolume': baseVolume,
'quoteVolume': quoteVolume,
'markPrice': self.safe_string(ticker, 'markPrice'),
'indexPrice': self.safe_string(ticker, 'indexPrice'),
'info': ticker,
}, market)
def fetch_status(self, params={}):
"""
the latest known information on the availability of the exchange API
https://developers.binance.com/docs/wallet/others/system-status
:param dict [params]: extra parameters specific to the exchange API endpoint
:returns dict: a `status structure <https://docs.ccxt.com/#/?id=exchange-status-structure>`
"""
response = self.sapiGetSystemStatus(params)
#
# {
# "status": 0, # 0: normal1system maintenance
# "msg": "normal" # "normal", "system_maintenance"
# }
#
statusRaw = self.safe_string(response, 'status')
return {
'status': self.safe_string({'0': 'ok', '1': 'maintenance'}, statusRaw, statusRaw),
'updated': None,
'eta': None,
'url': None,
'info': response,
}
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://developers.binance.com/docs/binance-spot-api-docs/rest-api/market-data-endpoints#24hr-ticker-price-change-statistics # spot
https://developers.binance.com/docs/binance-spot-api-docs/rest-api/market-data-endpoints#rolling-window-price-change-statistics # spot
https://developers.binance.com/docs/derivatives/usds-margined-futures/market-data/rest-api/24hr-Ticker-Price-Change-Statistics # swap
https://developers.binance.com/docs/derivatives/coin-margined-futures/market-data/rest-api/24hr-Ticker-Price-Change-Statistics # future
https://developers.binance.com/docs/derivatives/option/market-data/24hr-Ticker-Price-Change-Statistics # option
: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.rolling]:(spot only) default False, if True, uses the rolling 24 hour ticker endpoint /api/v3/ticker
:returns dict: a `ticker structure <https://docs.ccxt.com/#/?id=ticker-structure>`
"""
self.load_markets()
market = self.market(symbol)
request: dict = {
'symbol': market['id'],
}
response = None
if market['option']:
response = self.eapiPublicGetTicker(self.extend(request, params))
elif market['linear']:
response = self.fapiPublicGetTicker24hr(self.extend(request, params))
elif market['inverse']:
response = self.dapiPublicGetTicker24hr(self.extend(request, params))
else:
rolling = self.safe_bool(params, 'rolling', False)
params = self.omit(params, 'rolling')
if rolling:
response = self.publicGetTicker(self.extend(request, params))
else:
response = self.publicGetTicker24hr(self.extend(request, params))
if isinstance(response, list):
firstTicker = self.safe_dict(response, 0, {})
return self.parse_ticker(firstTicker, market)
return self.parse_ticker(response, market)
def fetch_bids_asks(self, symbols: Strings = None, params={}):
"""
fetches the bid and ask price and volume for multiple markets
https://developers.binance.com/docs/binance-spot-api-docs/rest-api/market-data-endpoints#symbol-order-book-ticker # spot
https://developers.binance.com/docs/derivatives/usds-margined-futures/market-data/rest-api/Symbol-Order-Book-Ticker # swap
https://developers.binance.com/docs/derivatives/coin-margined-futures/market-data/rest-api/Symbol-Order-Book-Ticker # future
:param str[]|None symbols: unified symbols of the markets to fetch the bids and asks for, all markets are returned if not assigned
:param dict [params]: extra parameters specific to the exchange API endpoint
:param str [params.subType]: "linear" or "inverse"
:returns dict: a dictionary of `ticker structures <https://docs.ccxt.com/#/?id=ticker-structure>`
"""
self.load_markets()
symbols = self.market_symbols(symbols, None, True, True, True)
market = self.get_market_from_symbols(symbols)
type = None
type, params = self.handle_market_type_and_params('fetchBidsAsks', market, params)
subType = None
subType, params = self.handle_sub_type_and_params('fetchBidsAsks', market, params)
response = None
if self.is_linear(type, subType):
response = self.fapiPublicGetTickerBookTicker(params)
elif self.is_inverse(type, subType):
response = self.dapiPublicGetTickerBookTicker(params)
elif type == 'spot':
request: dict = {}
if symbols is not None:
request['symbols'] = self.json(self.market_ids(symbols))
response = self.publicGetTickerBookTicker(self.extend(request, params))
else:
raise NotSupported(self.id + ' fetchBidsAsks() does not support ' + type + ' markets yet')
return self.parse_tickers(response, symbols)
def fetch_last_prices(self, symbols: Strings = None, params={}):
"""
fetches the last price for multiple markets
https://developers.binance.com/docs/binance-spot-api-docs/rest-api/market-data-endpoints#symbol-price-ticker # spot
https://developers.binance.com/docs/derivatives/usds-margined-futures/market-data/rest-api/Symbol-Price-Ticker # swap
https://developers.binance.com/docs/derivatives/coin-margined-futures/market-data/rest-api/Symbol-Price-Ticker # future
:param str[]|None symbols: unified symbols of the markets to fetch the last prices
:param dict [params]: extra parameters specific to the exchange API endpoint
:param str [params.subType]: "linear" or "inverse"
:returns dict: a dictionary of lastprices structures
"""
self.load_markets()
symbols = self.market_symbols(symbols, None, True, True, True)
market = self.get_market_from_symbols(symbols)
type = None
type, params = self.handle_market_type_and_params('fetchLastPrices', market, params)
subType = None
subType, params = self.handle_sub_type_and_params('fetchLastPrices', market, params)
response = None
if self.is_linear(type, subType):
response = self.fapiPublicV2GetTickerPrice(params)
#
# [
# {
# "symbol": "LTCBTC",
# "price": "4.00000200"
# "time": 1589437530011
# },
# ...
# ]
#
elif self.is_inverse(type, subType):
response = self.dapiPublicGetTickerPrice(params)
#
# [
# {
# "symbol": "BTCUSD_200626",
# "ps": "9647.8",
# "price": "9647.8",
# "time": 1591257246176
# }
# ]
#
elif type == 'spot':
response = self.publicGetTickerPrice(params)
#
# [
# {
# "symbol": "LTCBTC",
# "price": "4.00000200"
# },
# ...
# ]
#
else:
raise NotSupported(self.id + ' fetchLastPrices() does not support ' + type + ' markets yet')
return self.parse_last_prices(response, symbols)
def parse_last_price(self, entry, market: Market = None):
#
# spot
#
# {
# "symbol": "LTCBTC",
# "price": "4.00000200"
# }
#
# usdm(swap/future)
#
# {
# "symbol": "BTCUSDT",
# "price": "6000.01",
# "time": 1589437530011 # Transaction time
# }
#
#
# coinm(swap/future)
#
# {
# "symbol": "BTCUSD_200626", # symbol("BTCUSD_200626", "BTCUSD_PERP", etc..)
# "ps": "BTCUSD", # pair
# "price": "9647.8",
# "time": 1591257246176
# }
#
timestamp = self.safe_integer(entry, 'time')
type = 'spot' if (timestamp is None) else 'swap'
marketId = self.safe_string(entry, 'symbol')
market = self.safe_market(marketId, market, None, type)
return {
'symbol': market['symbol'],
'timestamp': timestamp,
'datetime': self.iso8601(timestamp),
'price': self.safe_number_omit_zero(entry, 'price'),
'side': None,
'info': entry,
}
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://developers.binance.com/docs/binance-spot-api-docs/rest-api/market-data-endpoints#24hr-ticker-price-change-statistics # spot
https://developers.binance.com/docs/derivatives/usds-margined-futures/market-data/rest-api/24hr-Ticker-Price-Change-Statistics # swap
https://developers.binance.com/docs/derivatives/coin-margined-futures/market-data/rest-api/24hr-Ticker-Price-Change-Statistics # future
https://developers.binance.com/docs/derivatives/option/market-data/24hr-Ticker-Price-Change-Statistics # option
:param str[] [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 str [params.subType]: "linear" or "inverse"
:param str [params.type]: 'spot', 'option', use params["subType"] for swap and future markets
:returns dict: a dictionary of `ticker structures <https://docs.ccxt.com/#/?id=ticker-structure>`
"""
self.load_markets()
symbols = self.market_symbols(symbols, None, True, True, True)
market = self.get_market_from_symbols(symbols)
type = None
type, params = self.handle_market_type_and_params('fetchTickers', market, params)
subType = None
subType, params = self.handle_sub_type_and_params('fetchTickers', market, params)
response = None
if self.is_linear(type, subType):
response = self.fapiPublicGetTicker24hr(params)
elif self.is_inverse(type, subType):
response = self.dapiPublicGetTicker24hr(params)
elif type == 'spot':
rolling = self.safe_bool(params, 'rolling', False)
params = self.omit(params, 'rolling')
if rolling:
symbols = self.market_symbols(symbols)
request: dict = {
'symbols': self.json(self.market_ids(symbols)),
}
response = self.publicGetTicker(self.extend(request, params))
# parseTicker is not able to handle marketType for spot-rolling ticker fields, so we need custom parsing
return self.parse_tickers_for_rolling(response, symbols)
else:
request: dict = {}
if symbols is not None:
request['symbols'] = self.json(self.market_ids(symbols))
response = self.publicGetTicker24hr(self.extend(request, params))
elif type == 'option':
response = self.eapiPublicGetTicker(params)
else:
raise NotSupported(self.id + ' fetchTickers() does not support ' + type + ' markets yet')
return self.parse_tickers(response, symbols)
def parse_tickers_for_rolling(self, response, symbols):
results = []
for i in range(0, len(response)):
marketId = self.safe_string(response[i], 'symbol')
tickerMarket = self.safe_market(marketId, None, None, 'spot')
parsedTicker = self.parse_ticker(response[i])
parsedTicker['symbol'] = tickerMarket['symbol']
results.append(parsedTicker)
return self.filter_by_array(results, 'symbol', symbols)
def fetch_mark_price(self, symbol: str, params={}) -> Ticker:
"""
fetches mark price for the market
https://developers.binance.com/docs/derivatives/coin-margined-futures/market-data/rest-api/Index-Price-and-Mark-Price
https://developers.binance.com/docs/derivatives/usds-margined-futures/market-data/rest-api/Mark-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
:param str [params.subType]: "linear" or "inverse"
:returns dict: a dictionary of `ticker structures <https://docs.ccxt.com/#/?id=ticker-structure>`
"""
self.load_markets()
market = self.market(symbol)
type = None
type, params = self.handle_market_type_and_params('fetchMarkPrice', market, params, 'swap')
subType = None
subType, params = self.handle_sub_type_and_params('fetchMarkPrice', market, params, 'linear')
request = {
'symbol': market['id'],
}
response = None
if self.is_linear(type, subType):
response = self.fapiPublicGetPremiumIndex(self.extend(request, params))
elif self.is_inverse(type, subType):
response = self.dapiPublicGetPremiumIndex(self.extend(request, params))
else:
raise NotSupported(self.id + ' fetchMarkPrice() does not support ' + type + ' markets yet')
if isinstance(response, list):
return self.parse_ticker(self.safe_dict(response, 0, {}), market)
return self.parse_ticker(response, market)
def fetch_mark_prices(self, symbols: Strings = None, params={}) -> Tickers:
"""
fetches mark prices for multiple markets
https://developers.binance.com/docs/derivatives/coin-margined-futures/market-data/rest-api/Index-Price-and-Mark-Price
https://developers.binance.com/docs/derivatives/usds-margined-futures/market-data/rest-api/Mark-Price
:param str[] [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 str [params.subType]: "linear" or "inverse"
:returns dict: a dictionary of `ticker structures <https://docs.ccxt.com/#/?id=ticker-structure>`
"""
self.load_markets()
symbols = self.market_symbols(symbols, None, True, True, True)
market = self.get_market_from_symbols(symbols)
type = None
type, params = self.handle_market_type_and_params('fetchMarkPrices', market, params, 'swap')
subType = None
subType, params = self.handle_sub_type_and_params('fetchMarkPrices', market, params, 'linear')
response = None
if self.is_linear(type, subType):
response = self.fapiPublicGetPremiumIndex(params)
elif self.is_inverse(type, subType):
response = self.dapiPublicGetPremiumIndex(params)
else:
raise NotSupported(self.id + ' fetchMarkPrices() does not support ' + type + ' markets yet')
return self.parse_tickers(response, symbols)
def parse_ohlcv(self, ohlcv, market: Market = None) -> list:
# when api method = publicGetKlines or fapiPublicGetKlines or dapiPublicGetKlines
# [
# 1591478520000, # open time
# "0.02501300", # open
# "0.02501800", # high
# "0.02500000", # low
# "0.02500000", # close
# "22.19000000", # volume
# 1591478579999, # close time
# "0.55490906", # quote asset volume, base asset volume for dapi
# 40, # number of trades
# "10.92900000", # taker buy base asset volume
# "0.27336462", # taker buy quote asset volume
# "0" # ignore
# ]
#
# when api method = fapiPublicGetMarkPriceKlines or fapiPublicGetIndexPriceKlines
# [
# [
# 1591256460000, # Open time
# "9653.29201333", # Open
# "9654.56401333", # High
# "9653.07367333", # Low
# "9653.07367333", # Close(or latest price)
# "0", # Ignore
# 1591256519999, # Close time
# "0", # Ignore
# 60, # Number of bisic data
# "0", # Ignore
# "0", # Ignore
# "0" # Ignore
# ]
# ]
#
# options
#
# {
# "open": "32.2",
# "high": "32.2",
# "low": "32.2",
# "close": "32.2",
# "volume": "0",
# "interval": "5m",
# "tradeCount": 0,
# "takerVolume": "0",
# "takerAmount": "0",
# "amount": "0",
# "openTime": 1677096900000,
# "closeTime": 1677097200000
# }
#
inverse = self.safe_bool(market, 'inverse')
volumeIndex = 7 if inverse else 5
return [
self.safe_integer_2(ohlcv, 0, 'openTime'),
self.safe_number_2(ohlcv, 1, 'open'),
self.safe_number_2(ohlcv, 2, 'high'),
self.safe_number_2(ohlcv, 3, 'low'),
self.safe_number_2(ohlcv, 4, 'close'),
self.safe_number_2(ohlcv, volumeIndex, 'volume'),
]
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://developers.binance.com/docs/binance-spot-api-docs/rest-api/market-data-endpoints#klinecandlestick-data
https://developers.binance.com/docs/derivatives/option/market-data/Kline-Candlestick-Data
https://developers.binance.com/docs/derivatives/usds-margined-futures/market-data/rest-api/Kline-Candlestick-Data
https://developers.binance.com/docs/derivatives/usds-margined-futures/market-data/rest-api/Index-Price-Kline-Candlestick-Data
https://developers.binance.com/docs/derivatives/usds-margined-futures/market-data/rest-api/Mark-Price-Kline-Candlestick-Data
https://developers.binance.com/docs/derivatives/usds-margined-futures/market-data/rest-api/Premium-Index-Kline-Data
https://developers.binance.com/docs/derivatives/coin-margined-futures/market-data/rest-api/Kline-Candlestick-Data
https://developers.binance.com/docs/derivatives/coin-margined-futures/market-data/rest-api/Index-Price-Kline-Candlestick-Data
https://developers.binance.com/docs/derivatives/coin-margined-futures/market-data/rest-api/Mark-Price-Kline-Candlestick-Data
https://developers.binance.com/docs/derivatives/coin-margined-futures/market-data/rest-api/Premium-Index-Kline-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 str [params.price]: "mark" or "index" for mark price and index price candles
:param int [params.until]: timestamp in ms of the latest candle to fetch
: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 int[][]: A list of candles ordered, open, high, low, close, volume
"""
self.load_markets()
paginate = False
paginate, params = self.handle_option_and_params(params, 'fetchOHLCV', 'paginate', False)
if paginate:
return self.fetch_paginated_call_deterministic('fetchOHLCV', symbol, since, limit, timeframe, params, 1000)
market = self.market(symbol)
# binance docs say that the default limit 500, max 1500 for futures, max 1000 for spot markets
# the reality is that the time range wider than 500 candles won't work right
defaultLimit = 500
maxLimit = 1500
price = self.safe_string(params, 'price')
until = self.safe_integer(params, 'until')
params = self.omit(params, ['price', 'until'])
if since is not None and until is not None and limit is None:
limit = maxLimit
limit = defaultLimit if (limit is None) else min(limit, maxLimit)
request: dict = {
'interval': self.safe_string(self.timeframes, timeframe, timeframe),
'limit': limit,
}
marketId = market['id']
if price == 'index':
parts = marketId.split('_')
pair = self.safe_string(parts, 0)
request['pair'] = pair # Index price takes self argument instead of symbol
else:
request['symbol'] = marketId
# duration = self.parse_timeframe(timeframe)
if since is not None:
request['startTime'] = since
#
# It didn't work before without the endTime
# https://github.com/ccxt/ccxt/issues/8454
#
if market['inverse']:
if since > 0:
duration = self.parse_timeframe(timeframe)
endTime = self.sum(since, limit * duration * 1000 - 1)
now = self.milliseconds()
request['endTime'] = min(now, endTime)
if until is not None:
request['endTime'] = until
response = None
if market['option']:
response = self.eapiPublicGetKlines(self.extend(request, params))
elif price == 'mark':
if market['inverse']:
response = self.dapiPublicGetMarkPriceKlines(self.extend(request, params))
else:
response = self.fapiPublicGetMarkPriceKlines(self.extend(request, params))
elif price == 'index':
if market['inverse']:
response = self.dapiPublicGetIndexPriceKlines(self.extend(request, params))
else:
response = self.fapiPublicGetIndexPriceKlines(self.extend(request, params))
elif price == 'premiumIndex':
if market['inverse']:
response = self.dapiPublicGetPremiumIndexKlines(self.extend(request, params))
else:
response = self.fapiPublicGetPremiumIndexKlines(self.extend(request, params))
elif market['linear']:
response = self.fapiPublicGetKlines(self.extend(request, params))
elif market['inverse']:
response = self.dapiPublicGetKlines(self.extend(request, params))
else:
response = self.publicGetKlines(self.extend(request, params))
#
# [
# [1591478520000,"0.02501300","0.02501800","0.02500000","0.02500000","22.19000000",1591478579999,"0.55490906",40,"10.92900000","0.27336462","0"],
# [1591478580000,"0.02499600","0.02500900","0.02499400","0.02500300","21.34700000",1591478639999,"0.53370468",24,"7.53800000","0.18850725","0"],
# [1591478640000,"0.02500800","0.02501100","0.02500300","0.02500800","154.14200000",1591478699999,"3.85405839",97,"5.32300000","0.13312641","0"],
# ]
#
# options(eapi)
#
# [
# {
# "open": "32.2",
# "high": "32.2",
# "low": "32.2",
# "close": "32.2",
# "volume": "0",
# "interval": "5m",
# "tradeCount": 0,
# "takerVolume": "0",
# "takerAmount": "0",
# "amount": "0",
# "openTime": 1677096900000,
# "closeTime": 1677097200000
# }
# ]
#
candles = self.parse_ohlcvs(response, market, timeframe, since, limit)
return candles
def parse_trade(self, trade: dict, market: Market = None) -> Trade:
if 'isDustTrade' in trade:
return self.parse_dust_trade(trade, market)
#
# aggregate trades
# https://github.com/binance-exchange/binance-official-api-docs/blob/master/rest-api.md#compressedaggregate-trades-list
#
# {
# "a": 26129, # Aggregate tradeId
# "p": "0.01633102", # Price
# "q": "4.70443515", # Quantity
# "f": 27781, # First tradeId
# "l": 27781, # Last tradeId
# "T": 1498793709153, # Timestamp
# "m": True, # Was the buyer the maker?
# "M": True # Was the trade the best price match?
# }
#
# REST: aggregate trades for swap & future(both linear and inverse)
#
# {
# "a": "269772814",
# "p": "25864.1",
# "q": "3",
# "f": "662149354",
# "l": "662149355",
# "T": "1694209776022",
# "m": False,
# }
#
# recent public trades and old public trades
# https://github.com/binance-exchange/binance-official-api-docs/blob/master/rest-api.md#recent-trades-list
# https://github.com/binance-exchange/binance-official-api-docs/blob/master/rest-api.md#old-trade-lookup-market_data
#
# {
# "id": 28457,
# "price": "4.00000100",
# "qty": "12.00000000",
# "time": 1499865549590,
# "isBuyerMaker": True,
# "isBestMatch": True
# }
#
# private trades
# https://github.com/binance-exchange/binance-official-api-docs/blob/master/rest-api.md#account-trade-list-user_data
#
# {
# "symbol": "BNBBTC",
# "id": 28457,
# "orderId": 100234,
# "price": "4.00000100",
# "qty": "12.00000000",
# "commission": "10.10000000",
# "commissionAsset": "BNB",
# "time": 1499865549590,
# "isBuyer": True,
# "isMaker": False,
# "isBestMatch": True
# }
#
# futures trades
#
# {
# "accountId": 20,
# "buyer": False,
# "commission": "-0.07819010",
# "commissionAsset": "USDT",
# "counterPartyId": 653,
# "id": 698759,
# "maker": False,
# "orderId": 25851813,
# "price": "7819.01",
# "qty": "0.002",
# "quoteQty": "0.01563",
# "realizedPnl": "-0.91539999",
# "side": "SELL",
# "symbol": "BTCUSDT",
# "time": 1569514978020
# }
# {
# "symbol": "BTCUSDT",
# "id": 477128891,
# "orderId": 13809777875,
# "side": "SELL",
# "price": "38479.55",
# "qty": "0.001",
# "realizedPnl": "-0.00009534",
# "marginAsset": "USDT",
# "quoteQty": "38.47955",
# "commission": "-0.00076959",
# "commissionAsset": "USDT",
# "time": 1612733566708,
# "positionSide": "BOTH",
# "maker": True,
# "buyer": False
# }
#
# {respType: FULL}
#
# {
# "price": "4000.00000000",
# "qty": "1.00000000",
# "commission": "4.00000000",
# "commissionAsset": "USDT",
# "tradeId": "1234",
# }
#
# options: fetchMyTrades
#
# {
# "id": 1125899906844226012,
# "tradeId": 73,
# "orderId": 4638761100843040768,
# "symbol": "ETH-230211-1500-C",
# "price": "18.70000000",
# "quantity": "-0.57000000",
# "fee": "0.17305890",
# "realizedProfit": "-3.53400000",
# "side": "SELL",
# "type": "LIMIT",
# "volatility": "0.30000000",
# "liquidity": "MAKER",
# "time": 1676085216845,
# "priceScale": 1,
# "quantityScale": 2,
# "optionSide": "CALL",
# "quoteAsset": "USDT"
# }
#
# options: fetchTrades
#
# {
# "id": 1,
# "symbol": "ETH-230216-1500-C",
# "price": "35.5",
# "qty": "0.03",
# "quoteQty": "1.065",
# "side": 1,
# "time": 1676366446072
# }
#
# fetchMyTrades: linear portfolio margin
#
# {
# "symbol": "BTCUSDT",
# "id": 4575108247,
# "orderId": 261942655610,
# "side": "SELL",
# "price": "47263.40",
# "qty": "0.010",
# "realizedPnl": "27.38400000",
# "marginAsset": "USDT",
# "quoteQty": "472.63",
# "commission": "0.18905360",
# "commissionAsset": "USDT",
# "time": 1707530039409,
# "buyer": False,
# "maker": False,
# "positionSide": "LONG"
# }
#
# fetchMyTrades: inverse portfolio margin
#
# {
# "symbol": "ETHUSD_PERP",
# "id": 701907838,
# "orderId": 71548909034,
# "pair": "ETHUSD",
# "side": "SELL",
# "price": "2498.15",
# "qty": "1",
# "realizedPnl": "0.00012517",
# "marginAsset": "ETH",
# "baseQty": "0.00400296",
# "commission": "0.00000160",
# "commissionAsset": "ETH",
# "time": 1707530317519,
# "positionSide": "LONG",
# "buyer": False,
# "maker": False
# }
#
# fetchMyTrades: spot margin portfolio margin
#
# {
# "symbol": "ADAUSDT",
# "id": 470227543,
# "orderId": 4421170947,
# "price": "0.53880000",
# "qty": "10.00000000",
# "quoteQty": "5.38800000",
# "commission": "0.00538800",
# "commissionAsset": "USDT",
# "time": 1707545780522,
# "isBuyer": False,
# "isMaker": False,
# "isBestMatch": True
# }
#
timestamp = self.safe_integer_2(trade, 'T', 'time')
amount = self.safe_string_2(trade, 'q', 'qty')
amount = self.safe_string(trade, 'quantity', amount)
marketId = self.safe_string(trade, 'symbol')
isSpotTrade = ('isIsolated' in trade) or ('M' in trade) or ('orderListId' in trade) or ('isMaker' in trade)
marketType = 'spot' if isSpotTrade else 'contract'
market = self.safe_market(marketId, market, None, marketType)
symbol = market['symbol']
side = None
buyerMaker = self.safe_bool_2(trade, 'm', 'isBuyerMaker')
takerOrMaker = None
if buyerMaker is not None:
side = 'sell' if buyerMaker else 'buy' # self is reversed intentionally
elif 'side' in trade:
side = self.safe_string_lower(trade, 'side')
else:
if 'isBuyer' in trade:
side = 'buy' if trade['isBuyer'] else 'sell' # self is a True side
fee = None
if 'commission' in trade:
fee = {
'cost': self.safe_string(trade, 'commission'),
'currency': self.safe_currency_code(self.safe_string(trade, 'commissionAsset')),
}
if 'isMaker' in trade:
takerOrMaker = 'maker' if trade['isMaker'] else 'taker'
if 'maker' in trade:
takerOrMaker = 'maker' if trade['maker'] else 'taker'
if ('optionSide' in trade) or market['option']:
settle = self.safe_currency_code(self.safe_string(trade, 'quoteAsset', 'USDT'))
takerOrMaker = self.safe_string_lower(trade, 'liquidity')
if 'fee' in trade:
fee = {
'cost': self.safe_string(trade, 'fee'),
'currency': settle,
}
if (side != 'buy') and (side != 'sell'):
side = 'buy' if (side == '1') else 'sell'
if 'optionSide' in trade:
if side != 'buy':
amount = Precise.string_mul('-1', amount)
return self.safe_trade({
'info': trade,
'timestamp': timestamp,
'datetime': self.iso8601(timestamp),
'symbol': symbol,
'id': self.safe_string_n(trade, ['t', 'a', 'tradeId', 'id']),
'order': self.safe_string(trade, 'orderId'),
'type': self.safe_string_lower(trade, 'type'),
'side': side,
'takerOrMaker': takerOrMaker,
'price': self.safe_string_2(trade, 'p', 'price'),
'amount': amount,
'cost': self.safe_string_2(trade, 'quoteQty', 'baseQty'),
'fee': fee,
}, market)
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
Default fetchTradesMethod
https://developers.binance.com/docs/binance-spot-api-docs/rest-api/market-data-endpoints#compressedaggregate-trades-list # publicGetAggTrades(spot)
https://developers.binance.com/docs/derivatives/usds-margined-futures/market-data/rest-api/Compressed-Aggregate-Trades-List # fapiPublicGetAggTrades(swap)
https://developers.binance.com/docs/derivatives/coin-margined-futures/market-data/rest-api/Compressed-Aggregate-Trades-List # dapiPublicGetAggTrades(future)
https://developers.binance.com/docs/derivatives/option/market-data/Recent-Trades-List # eapiPublicGetTrades(option)
Other fetchTradesMethod
https://developers.binance.com/docs/binance-spot-api-docs/rest-api/market-data-endpoints#recent-trades-list # publicGetTrades(spot)
https://developers.binance.com/docs/derivatives/usds-margined-futures/market-data/rest-api/Recent-Trades-List # fapiPublicGetTrades(swap)
https://developers.binance.com/docs/derivatives/coin-margined-futures/market-data/rest-api/Recent-Trades-List # dapiPublicGetTrades(future)
https://developers.binance.com/docs/binance-spot-api-docs/rest-api/market-data-endpoints#old-trade-lookup # publicGetHistoricalTrades(spot)
https://developers.binance.com/docs/derivatives/usds-margined-futures/market-data/rest-api/Old-Trades-Lookup # fapiPublicGetHistoricalTrades(swap)
https://developers.binance.com/docs/derivatives/coin-margined-futures/market-data/rest-api/Old-Trades-Lookup # dapiPublicGetHistoricalTrades(future)
https://developers.binance.com/docs/derivatives/option/market-data/Old-Trades-Lookup # eapiPublicGetHistoricalTrades(option)
:param str symbol: unified symbol of the market to fetch trades for
:param int [since]: only used when fetchTradesMethod is 'publicGetAggTrades', 'fapiPublicGetAggTrades', or 'dapiPublicGetAggTrades'
:param int [limit]: default 500, max 1000
:param dict [params]: extra parameters specific to the exchange API endpoint
:param int [params.until]: only used when fetchTradesMethod is 'publicGetAggTrades', 'fapiPublicGetAggTrades', or 'dapiPublicGetAggTrades'
:param int [params.fetchTradesMethod]: 'publicGetAggTrades'(spot default), 'fapiPublicGetAggTrades'(swap default), 'dapiPublicGetAggTrades'(future default), 'eapiPublicGetTrades'(option default), 'publicGetTrades', 'fapiPublicGetTrades', 'dapiPublicGetTrades', 'publicGetHistoricalTrades', 'fapiPublicGetHistoricalTrades', 'dapiPublicGetHistoricalTrades', 'eapiPublicGetHistoricalTrades'
: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)
EXCHANGE SPECIFIC PARAMETERS
:param int [params.fromId]: trade id to fetch from, default gets most recent trades, not used when fetchTradesMethod is 'publicGetTrades', 'fapiPublicGetTrades', 'dapiPublicGetTrades', or 'eapiPublicGetTrades'
:returns Trade[]: a list of `trade structures <https://docs.ccxt.com/#/?id=public-trades>`
"""
self.load_markets()
paginate = False
paginate, params = self.handle_option_and_params(params, 'fetchTrades', 'paginate')
if paginate:
return self.fetch_paginated_call_dynamic('fetchTrades', symbol, since, limit, params)
market = self.market(symbol)
request: dict = {
'symbol': market['id'],
# 'fromId': 123, # ID to get aggregate trades from INCLUSIVE.
# 'startTime': 456, # Timestamp in ms to get aggregate trades from INCLUSIVE.
# 'endTime': 789, # Timestamp in ms to get aggregate trades until INCLUSIVE.
# 'limit': 500, # default = 500, maximum = 1000
}
if not market['option']:
if since is not None:
request['startTime'] = since
# https://github.com/ccxt/ccxt/issues/6400
# https://github.com/binance-exchange/binance-official-api-docs/blob/master/rest-api.md#compressedaggregate-trades-list
request['endTime'] = self.sum(since, 3600000)
until = self.safe_integer(params, 'until')
if until is not None:
request['endTime'] = until
method = self.safe_string(self.options, 'fetchTradesMethod')
method = self.safe_string_2(params, 'fetchTradesMethod', 'method', method)
if limit is not None:
isFutureOrSwap = (market['swap'] or market['future'])
isHistoricalEndpoint = (method is not None) and (method.find('GetHistoricalTrades') >= 0)
maxLimitForContractHistorical = 500 if isHistoricalEndpoint else 1000
request['limit'] = min(limit, maxLimitForContractHistorical) if isFutureOrSwap else limit # default = 500, maximum = 1000
params = self.omit(params, ['until', 'fetchTradesMethod'])
response = None
if market['option'] or method == 'eapiPublicGetTrades':
response = self.eapiPublicGetTrades(self.extend(request, params))
elif market['linear'] or method == 'fapiPublicGetAggTrades':
response = self.fapiPublicGetAggTrades(self.extend(request, params))
elif market['inverse'] or method == 'dapiPublicGetAggTrades':
response = self.dapiPublicGetAggTrades(self.extend(request, params))
else:
response = self.publicGetAggTrades(self.extend(request, params))
#
# Caveats:
# - default limit(500) applies only if no other parameters set, trades up
# to the maximum limit may be returned to satisfy other parameters
# - if both limit and time window is set and time window contains more
# trades than the limit then the last trades from the window are returned
# - "tradeId" accepted and returned by self method is "aggregate" trade id
# which is different from actual trade id
# - setting both fromId and time window results in error
#
# aggregate trades
#
# [
# {
# "a": 26129, # Aggregate tradeId
# "p": "0.01633102", # Price
# "q": "4.70443515", # Quantity
# "f": 27781, # First tradeId
# "l": 27781, # Last tradeId
# "T": 1498793709153, # Timestamp
# "m": True, # Was the buyer the maker?
# "M": True # Was the trade the best price match?
# }
# ]
#
# inverse(swap & future)
#
# [
# {
# "a": "269772814",
# "p": "25864.1",
# "q": "3",
# "f": "662149354",
# "l": "662149355",
# "T": "1694209776022",
# "m": False,
# },
# ]
#
# recent public trades and historical public trades
#
# [
# {
# "id": 28457,
# "price": "4.00000100",
# "qty": "12.00000000",
# "time": 1499865549590,
# "isBuyerMaker": True,
# "isBestMatch": True
# }
# ]
#
# options(eapi)
#
# [
# {
# "id": 1,
# "symbol": "ETH-230216-1500-C",
# "price": "35.5",
# "qty": "0.03",
# "quoteQty": "1.065",
# "side": 1,
# "time": 1676366446072
# },
# ]
#
return self.parse_trades(response, market, since, limit)
def edit_spot_order(self, id: str, symbol: str, type: OrderType, side: OrderSide, amount: float, price: Num = None, params={}):
"""
@ignore
edit a trade order
https://developers.binance.com/docs/binance-spot-api-docs/rest-api/trading-endpoints#cancel-an-existing-order-and-send-a-new-order-trade
: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' or 'STOP_LOSS' or 'STOP_LOSS_LIMIT' or 'TAKE_PROFIT' or 'TAKE_PROFIT_LIMIT' or 'STOP'
:param str side: 'buy' or 'sell'
:param float amount: how much of currency you want to trade in units of 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 str [params.marginMode]: 'cross' or 'isolated', for spot margin trading
:returns dict: an `order structure <https://docs.ccxt.com/#/?id=order-structure>`
"""
self.load_markets()
market = self.market(symbol)
if not market['spot']:
raise NotSupported(self.id + ' editSpotOrder() does not support ' + market['type'] + ' orders')
payload = self.edit_spot_order_request(id, symbol, type, side, amount, price, params)
response = self.privatePostOrderCancelReplace(payload)
#
# spot
#
# {
# "cancelResult": "SUCCESS",
# "newOrderResult": "SUCCESS",
# "cancelResponse": {
# "symbol": "BTCUSDT",
# "origClientOrderId": "web_3f6286480b194b079870ac75fb6978b7",
# "orderId": 16383156620,
# "orderListId": -1,
# "clientOrderId": "Azt6foVTTgHPNhqBf41TTt",
# "price": "14000.00000000",
# "origQty": "0.00110000",
# "executedQty": "0.00000000",
# "cummulativeQuoteQty": "0.00000000",
# "status": "CANCELED",
# "timeInForce": "GTC",
# "type": "LIMIT",
# "side": "BUY"
# },
# "newOrderResponse": {
# "symbol": "BTCUSDT",
# "orderId": 16383176297,
# "orderListId": -1,
# "clientOrderId": "x-TKT5PX2F22ecb58eb9074fb1be018c",
# "transactTime": 1670891847932,
# "price": "13500.00000000",
# "origQty": "0.00085000",
# "executedQty": "0.00000000",
# "cummulativeQuoteQty": "0.00000000",
# "status": "NEW",
# "timeInForce": "GTC",
# "type": "LIMIT",
# "side": "BUY",
# "fills": []
# }
# }
#
data = self.safe_dict(response, 'newOrderResponse')
return self.parse_order(data, market)
def edit_spot_order_request(self, id: str, symbol: str, type: OrderType, side: OrderSide, amount: float, price: Num = None, params={}):
"""
@ignore
helper function to build request for editSpotOrder
:param str id: order id to be edited
:param str symbol: unified symbol of the market to create an order in
:param str type: 'market' or 'limit' or 'STOP_LOSS' or 'STOP_LOSS_LIMIT' or 'TAKE_PROFIT' or 'TAKE_PROFIT_LIMIT' or 'STOP'
:param str side: 'buy' or 'sell'
:param float amount: how much of currency you want to trade in units of 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 str [params.marginMode]: 'cross' or 'isolated', for spot margin trading
:returns dict: request to be sent to the exchange
"""
market = self.market(symbol)
clientOrderId = self.safe_string_n(params, ['newClientOrderId', 'clientOrderId', 'origClientOrderId'])
request: dict = {
'symbol': market['id'],
'side': side.upper(),
}
initialUppercaseType = type.upper()
uppercaseType = initialUppercaseType
postOnly = self.is_post_only(initialUppercaseType == 'MARKET', initialUppercaseType == 'LIMIT_MAKER', params)
if postOnly:
uppercaseType = 'LIMIT_MAKER'
triggerPrice = self.safe_number_2(params, 'stopPrice', 'triggerPrice')
if triggerPrice is not None:
if uppercaseType == 'MARKET':
uppercaseType = 'STOP_LOSS'
elif uppercaseType == 'LIMIT':
uppercaseType = 'STOP_LOSS_LIMIT'
request['type'] = uppercaseType
validOrderTypes = self.safe_list(market['info'], 'orderTypes')
if not self.in_array(uppercaseType, validOrderTypes):
if initialUppercaseType != uppercaseType:
raise InvalidOrder(self.id + ' triggerPrice parameter is not allowed for ' + symbol + ' ' + type + ' orders')
else:
raise InvalidOrder(self.id + ' ' + type + ' is not a valid order type for the ' + symbol + ' market')
if clientOrderId is None:
broker = self.safe_dict(self.options, 'broker')
if broker is not None:
brokerId = self.safe_string(broker, 'spot')
if brokerId is not None:
request['newClientOrderId'] = brokerId + self.uuid22()
else:
request['newClientOrderId'] = clientOrderId
request['newOrderRespType'] = self.safe_value(self.options['newOrderRespType'], type, 'RESULT') # 'ACK' for order id, 'RESULT' for full order or 'FULL' for order with fills
timeInForceIsRequired = False
priceIsRequired = False
triggerPriceIsRequired = False
quantityIsRequired = False
if uppercaseType == 'MARKET':
quoteOrderQty = self.safe_bool(self.options, 'quoteOrderQty', True)
if quoteOrderQty:
quoteOrderQtyNew = self.safe_value_2(params, 'quoteOrderQty', 'cost')
precision = market['precision']['price']
if quoteOrderQtyNew is not None:
request['quoteOrderQty'] = self.decimal_to_precision(quoteOrderQtyNew, TRUNCATE, precision, self.precisionMode)
elif price is not None:
amountString = self.number_to_string(amount)
priceString = self.number_to_string(price)
quoteOrderQuantity = Precise.string_mul(amountString, priceString)
request['quoteOrderQty'] = self.decimal_to_precision(quoteOrderQuantity, TRUNCATE, precision, self.precisionMode)
else:
quantityIsRequired = True
else:
quantityIsRequired = True
elif uppercaseType == 'LIMIT':
priceIsRequired = True
timeInForceIsRequired = True
quantityIsRequired = True
elif (uppercaseType == 'STOP_LOSS') or (uppercaseType == 'TAKE_PROFIT'):
triggerPriceIsRequired = True
quantityIsRequired = True
elif (uppercaseType == 'STOP_LOSS_LIMIT') or (uppercaseType == 'TAKE_PROFIT_LIMIT'):
quantityIsRequired = True
triggerPriceIsRequired = True
priceIsRequired = True
timeInForceIsRequired = True
elif uppercaseType == 'LIMIT_MAKER':
priceIsRequired = True
quantityIsRequired = True
if quantityIsRequired:
request['quantity'] = self.amount_to_precision(symbol, amount)
if priceIsRequired:
if price is None:
raise InvalidOrder(self.id + ' editOrder() requires a price argument for a ' + type + ' order')
request['price'] = self.price_to_precision(symbol, price)
if timeInForceIsRequired and (self.safe_string(params, 'timeInForce') is None):
request['timeInForce'] = self.options['defaultTimeInForce'] # 'GTC' = Good To Cancel(default), 'IOC' = Immediate Or Cancel
if triggerPriceIsRequired:
if triggerPrice is None:
raise InvalidOrder(self.id + ' editOrder() requires a triggerPrice extra param for a ' + type + ' order')
else:
request['stopPrice'] = self.price_to_precision(symbol, triggerPrice)
request['cancelReplaceMode'] = 'STOP_ON_FAILURE' # If the cancel request fails, the new order placement will not be attempted.
cancelId = self.safe_string_2(params, 'cancelNewClientOrderId', 'cancelOrigClientOrderId')
if cancelId is None:
request['cancelOrderId'] = id # user can provide either cancelOrderId, cancelOrigClientOrderId or cancelOrigClientOrderId
# remove timeInForce from params because PO is only used by self.is_post_only and it's not a valid value for Binance
if self.safe_string(params, 'timeInForce') == 'PO':
params = self.omit(params, ['timeInForce'])
params = self.omit(params, ['quoteOrderQty', 'cost', 'stopPrice', 'newClientOrderId', 'clientOrderId', 'postOnly'])
return self.extend(request, params)
def edit_contract_order_request(self, id: str, symbol: str, type: OrderType, side: OrderSide, amount: float, price: Num = None, params={}):
market = self.market(symbol)
if not market['contract']:
raise NotSupported(self.id + ' editContractOrder() does not support ' + market['type'] + ' orders')
request: dict = {
'symbol': market['id'],
'side': side.upper(),
'orderId': id,
'quantity': self.amount_to_precision(symbol, amount),
}
clientOrderId = self.safe_string_n(params, ['newClientOrderId', 'clientOrderId', 'origClientOrderId'])
if price is not None:
request['price'] = self.price_to_precision(symbol, price)
if clientOrderId is not None:
request['origClientOrderId'] = clientOrderId
params = self.omit(params, ['clientOrderId', 'newClientOrderId'])
return request
def edit_contract_order(self, id: str, symbol: str, type: OrderType, side: OrderSide, amount: float, price: Num = None, params={}):
"""
edit a trade order
https://developers.binance.com/docs/derivatives/usds-margined-futures/trade/rest-api/Modify-Order
https://developers.binance.com/docs/derivatives/coin-margined-futures/trade/rest-api/Modify-Order
https://developers.binance.com/docs/derivatives/portfolio-margin/trade/Modify-UM-Order
https://developers.binance.com/docs/derivatives/portfolio-margin/trade/Modify-CM-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 of currency you want to trade in units of 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 boolean [params.portfolioMargin]: set to True if you would like to edit an order in a portfolio margin account
:returns dict: an `order structure <https://docs.ccxt.com/#/?id=order-structure>`
"""
self.load_markets()
market = self.market(symbol)
isPortfolioMargin = None
isPortfolioMargin, params = self.handle_option_and_params_2(params, 'editContractOrder', 'papi', 'portfolioMargin', False)
if market['linear'] or isPortfolioMargin:
if (price is None) and not ('priceMatch' in params):
raise ArgumentsRequired(self.id + ' editOrder() requires a price argument for portfolio margin and linear orders')
request = self.edit_contract_order_request(id, symbol, type, side, amount, price, params)
response = None
if market['linear']:
if isPortfolioMargin:
response = self.papiPutUmOrder(self.extend(request, params))
else:
response = self.fapiPrivatePutOrder(self.extend(request, params))
elif market['inverse']:
if isPortfolioMargin:
response = self.papiPutCmOrder(self.extend(request, params))
else:
response = self.dapiPrivatePutOrder(self.extend(request, params))
#
# swap and future
#
# {
# "orderId": 151007482392,
# "symbol": "BTCUSDT",
# "status": "NEW",
# "clientOrderId": "web_pCCGp9AIHjziKLlpGpXI",
# "price": "25000",
# "avgPrice": "0.00000",
# "origQty": "0.001",
# "executedQty": "0",
# "cumQty": "0",
# "cumQuote": "0",
# "timeInForce": "GTC",
# "type": "LIMIT",
# "reduceOnly": False,
# "closePosition": False,
# "side": "BUY",
# "positionSide": "BOTH",
# "stopPrice": "0",
# "workingType": "CONTRACT_PRICE",
# "priceProtect": False,
# "origType": "LIMIT",
# "updateTime": 1684300587845
# }
#
return self.parse_order(response, market)
def edit_order(self, id: str, symbol: str, type: OrderType, side: OrderSide, amount: Num = None, price: Num = None, params={}):
"""
edit a trade order
https://developers.binance.com/docs/binance-spot-api-docs/rest-api/trading-endpoints#cancel-an-existing-order-and-send-a-new-order-trade
https://developers.binance.com/docs/derivatives/usds-margined-futures/trade/rest-api/Modify-Order
https://developers.binance.com/docs/derivatives/coin-margined-futures/trade/rest-api/Modify-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 of currency you want to trade in units of 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
:returns dict: an `order structure <https://docs.ccxt.com/#/?id=order-structure>`
"""
self.load_markets()
market = self.market(symbol)
if market['option']:
raise NotSupported(self.id + ' editOrder() does not support ' + market['type'] + ' orders')
if market['spot']:
return self.edit_spot_order(id, symbol, type, side, amount, price, params)
else:
return self.edit_contract_order(id, symbol, type, side, amount, price, params)
def edit_orders(self, orders: List[OrderRequest], params={}):
"""
edit a list of trade orders
https://developers.binance.com/docs/derivatives/usds-margined-futures/trade/rest-api/Modify-Multiple-Orders
https://developers.binance.com/docs/derivatives/coin-margined-futures/trade/rest-api/Modify-Multiple-Orders
: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 exchange API endpoint
:returns dict: an `order structure <https://docs.ccxt.com/#/?id=order-structure>`
"""
self.load_markets()
ordersRequests = []
orderSymbols = []
for i in range(0, len(orders)):
rawOrder = orders[i]
marketId = self.safe_string(rawOrder, 'symbol')
orderSymbols.append(marketId)
id = self.safe_string(rawOrder, 'id')
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_dict(rawOrder, 'params', {})
isPortfolioMargin = None
isPortfolioMargin, orderParams = self.handle_option_and_params_2(orderParams, 'editOrders', 'papi', 'portfolioMargin', False)
if isPortfolioMargin:
raise NotSupported(self.id + ' editOrders() does not support portfolio margin orders')
orderRequest = self.edit_contract_order_request(id, marketId, type, side, amount, price, orderParams)
ordersRequests.append(orderRequest)
orderSymbols = self.market_symbols(orderSymbols, None, False, True, True)
market = self.market(orderSymbols[0])
if market['spot'] or market['option']:
raise NotSupported(self.id + ' editOrders() does not support ' + market['type'] + ' orders')
response = None
request: dict = {
'batchOrders': ordersRequests,
}
request = self.extend(request, params)
if market['linear']:
response = self.fapiPrivatePutBatchOrders(request)
elif market['inverse']:
response = self.dapiPrivatePutBatchOrders(request)
#
# [
# {
# "code": -4005,
# "msg": "Quantity greater than max quantity."
# },
# {
# "orderId": 650640530,
# "symbol": "LTCUSDT",
# "status": "NEW",
# "clientOrderId": "x-xcKtGhcu32184eb13585491289bbaf",
# "price": "54.00",
# "avgPrice": "0.00",
# "origQty": "0.100",
# "executedQty": "0.000",
# "cumQty": "0.000",
# "cumQuote": "0.00000",
# "timeInForce": "GTC",
# "type": "LIMIT",
# "reduceOnly": False,
# "closePosition": False,
# "side": "BUY",
# "positionSide": "BOTH",
# "stopPrice": "0.00",
# "workingType": "CONTRACT_PRICE",
# "priceProtect": False,
# "origType": "LIMIT",
# "priceMatch": "NONE",
# "selfTradePreventionMode": "NONE",
# "goodTillDate": 0,
# "updateTime": 1698073926929
# }
# ]
#
return self.parse_orders(response)
def parse_order_status(self, status: Str):
statuses: dict = {
'NEW': 'open',
'PARTIALLY_FILLED': 'open',
'ACCEPTED': 'open',
'FILLED': 'closed',
'CANCELED': 'canceled',
'CANCELLED': 'canceled',
'PENDING_CANCEL': 'canceling', # currently unused
'REJECTED': 'rejected',
'EXPIRED': 'expired',
'EXPIRED_IN_MATCH': 'expired',
}
return self.safe_string(statuses, status, status)
def parse_order(self, order: dict, market: Market = None) -> Order:
#
# spot
#
# {
# "symbol": "LTCBTC",
# "orderId": 1,
# "clientOrderId": "myOrder1",
# "price": "0.1",
# "origQty": "1.0",
# "executedQty": "0.0",
# "cummulativeQuoteQty": "0.0",
# "status": "NEW",
# "timeInForce": "GTC",
# "type": "LIMIT",
# "side": "BUY",
# "stopPrice": "0.0",
# "icebergQty": "0.0",
# "time": 1499827319559,
# "updateTime": 1499827319559,
# "isWorking": True
# }
#
# spot: editOrder
#
# {
# "symbol": "BTCUSDT",
# "orderId": 16383176297,
# "orderListId": -1,
# "clientOrderId": "x-TKT5PX2F22ecb58eb9074fb1be018c",
# "transactTime": 1670891847932,
# "price": "13500.00000000",
# "origQty": "0.00085000",
# "executedQty": "0.00000000",
# "cummulativeQuoteQty": "0.00000000",
# "status": "NEW",
# "timeInForce": "GTC",
# "type": "LIMIT",
# "side": "BUY",
# "fills": []
# }
#
# swap and future: editOrder
#
# {
# "orderId": 151007482392,
# "symbol": "BTCUSDT",
# "status": "NEW",
# "clientOrderId": "web_pCCGp9AIHjziKLlpGpXI",
# "price": "25000",
# "avgPrice": "0.00000",
# "origQty": "0.001",
# "executedQty": "0",
# "cumQty": "0",
# "cumQuote": "0",
# "timeInForce": "GTC",
# "type": "LIMIT",
# "reduceOnly": False,
# "closePosition": False,
# "side": "BUY",
# "positionSide": "BOTH",
# "stopPrice": "0",
# "workingType": "CONTRACT_PRICE",
# "priceProtect": False,
# "origType": "LIMIT",
# "updateTime": 1684300587845
# }
#
# futures
#
# {
# "symbol": "BTCUSDT",
# "orderId": 1,
# "clientOrderId": "myOrder1",
# "price": "0.1",
# "origQty": "1.0",
# "executedQty": "1.0",
# "cumQuote": "10.0",
# "status": "NEW",
# "timeInForce": "GTC",
# "type": "LIMIT",
# "side": "BUY",
# "stopPrice": "0.0",
# "updateTime": 1499827319559
# }
#
# createOrder with {"newOrderRespType": "FULL"}
#
# {
# "symbol": "BTCUSDT",
# "orderId": 5403233939,
# "orderListId": -1,
# "clientOrderId": "x-TKT5PX2F5e669e75b6c14f69a2c43e",
# "transactTime": 1617151923742,
# "price": "0.00000000",
# "origQty": "0.00050000",
# "executedQty": "0.00050000",
# "cummulativeQuoteQty": "29.47081500",
# "status": "FILLED",
# "timeInForce": "GTC",
# "type": "MARKET",
# "side": "BUY",
# "fills": [
# {
# "price": "58941.63000000",
# "qty": "0.00050000",
# "commission": "0.00007050",
# "commissionAsset": "BNB",
# "tradeId": 737466631
# }
# ]
# }
#
# delivery
#
# {
# "orderId": "18742727411",
# "symbol": "ETHUSD_PERP",
# "pair": "ETHUSD",
# "status": "FILLED",
# "clientOrderId": "x-xcKtGhcu3e2d1503fdd543b3b02419",
# "price": "0",
# "avgPrice": "4522.14",
# "origQty": "1",
# "executedQty": "1",
# "cumBase": "0.00221134",
# "timeInForce": "GTC",
# "type": "MARKET",
# "reduceOnly": False,
# "closePosition": False,
# "side": "SELL",
# "positionSide": "BOTH",
# "stopPrice": "0",
# "workingType": "CONTRACT_PRICE",
# "priceProtect": False,
# "origType": "MARKET",
# "time": "1636061952660",
# "updateTime": "1636061952660"
# }
#
# option: createOrder, fetchOrder, fetchOpenOrders, fetchOrders
#
# {
# "orderId": 4728833085436977152,
# "symbol": "ETH-230211-1500-C",
# "price": "10.0",
# "quantity": "1.00",
# "executedQty": "0.00",
# "fee": "0",
# "side": "BUY",
# "type": "LIMIT",
# "timeInForce": "GTC",
# "reduceOnly": False,
# "postOnly": False,
# "createTime": 1676083034462,
# "updateTime": 1676083034462,
# "status": "ACCEPTED",
# "avgPrice": "0",
# "source": "API",
# "clientOrderId": "",
# "priceScale": 1,
# "quantityScale": 2,
# "optionSide": "CALL",
# "quoteAsset": "USDT",
# "lastTrade": {"id":"69","time":"1676084430567","price":"24.9","qty":"1.00"},
# "mmp": False
# }
#
# cancelOrders/createOrders
#
# {
# "code": -4005,
# "msg": "Quantity greater than max quantity."
# }
#
# createOrder, fetchOpenOrders, fetchOrder, cancelOrder, fetchOrders: portfolio margin linear swap and future
#
# {
# "symbol": "BTCUSDT",
# "side": "BUY",
# "executedQty": "0.000",
# "orderId": 258649539704,
# "goodTillDate": 0,
# "avgPrice": "0",
# "origQty": "0.010",
# "clientOrderId": "x-xcKtGhcu02573c6f15e544e990057b",
# "positionSide": "BOTH",
# "cumQty": "0.000",
# "updateTime": 1707110415436,
# "type": "LIMIT",
# "reduceOnly": False,
# "price": "35000.00",
# "cumQuote": "0.00000",
# "selfTradePreventionMode": "NONE",
# "timeInForce": "GTC",
# "status": "NEW"
# }
#
# createOrder, fetchOpenOrders, fetchOrder, cancelOrder, fetchOrders: portfolio margin inverse swap and future
#
# {
# "symbol": "ETHUSD_PERP",
# "side": "BUY",
# "cumBase": "0",
# "executedQty": "0",
# "orderId": 71275227732,
# "avgPrice": "0.00",
# "origQty": "1",
# "clientOrderId": "x-xcKtGhcuca5af3acfb5044198c5398",
# "positionSide": "BOTH",
# "cumQty": "0",
# "updateTime": 1707110994334,
# "type": "LIMIT",
# "pair": "ETHUSD",
# "reduceOnly": False,
# "price": "2000",
# "timeInForce": "GTC",
# "status": "NEW"
# }
#
# createOrder, fetchOpenOrders, fetchOpenOrder: portfolio margin linear swap and future conditional
#
# {
# "newClientStrategyId": "x-xcKtGhcu27f109953d6e4dc0974006",
# "strategyId": 3645916,
# "strategyStatus": "NEW",
# "strategyType": "STOP",
# "origQty": "0.010",
# "price": "35000.00",
# "reduceOnly": False,
# "side": "BUY",
# "positionSide": "BOTH",
# "stopPrice": "45000.00",
# "symbol": "BTCUSDT",
# "timeInForce": "GTC",
# "bookTime": 1707112625879,
# "updateTime": 1707112625879,
# "workingType": "CONTRACT_PRICE",
# "priceProtect": False,
# "goodTillDate": 0,
# "selfTradePreventionMode": "NONE"
# }
#
# createOrder, fetchOpenOrders: portfolio margin inverse swap and future conditional
#
# {
# "newClientStrategyId": "x-xcKtGhcuc6b86f053bb34933850739",
# "strategyId": 1423462,
# "strategyStatus": "NEW",
# "strategyType": "STOP",
# "origQty": "1",
# "price": "2000",
# "reduceOnly": False,
# "side": "BUY",
# "positionSide": "BOTH",
# "stopPrice": "3000",
# "symbol": "ETHUSD_PERP",
# "timeInForce": "GTC",
# "bookTime": 1707113098840,
# "updateTime": 1707113098840,
# "workingType": "CONTRACT_PRICE",
# "priceProtect": False
# }
#
# createOrder, cancelAllOrders, cancelOrder: portfolio margin spot margin
#
# {
# "clientOrderId": "x-TKT5PX2Fe9ef29d8346440f0b28b86",
# "cummulativeQuoteQty": "0.00000000",
# "executedQty": "0.00000000",
# "fills": [],
# "orderId": 24684460474,
# "origQty": "0.00100000",
# "price": "35000.00000000",
# "selfTradePreventionMode": "EXPIRE_MAKER",
# "side": "BUY",
# "status": "NEW",
# "symbol": "BTCUSDT",
# "timeInForce": "GTC",
# "transactTime": 1707113538870,
# "type": "LIMIT"
# }
#
# fetchOpenOrders, fetchOrder, fetchOrders: portfolio margin spot margin
#
# {
# "symbol": "BTCUSDT",
# "orderId": 24700763749,
# "clientOrderId": "x-TKT5PX2F6f724c2a4af6425f98c7b6",
# "price": "35000.00000000",
# "origQty": "0.00100000",
# "executedQty": "0.00000000",
# "cummulativeQuoteQty": "0.00000000",
# "status": "NEW",
# "timeInForce": "GTC",
# "type": "LIMIT",
# "side": "BUY",
# "stopPrice": "0.00000000",
# "icebergQty": "0.00000000",
# "time": 1707199187679,
# "updateTime": 1707199187679,
# "isWorking": True,
# "accountId": 200180970,
# "selfTradePreventionMode": "EXPIRE_MAKER",
# "preventedMatchId": null,
# "preventedQuantity": null
# }
#
# cancelOrder: portfolio margin linear and inverse swap conditional
#
# {
# "strategyId": 3733211,
# "newClientStrategyId": "x-xcKtGhcuaf166172ed504cd1bc0396",
# "strategyType": "STOP",
# "strategyStatus": "CANCELED",
# "origQty": "0.010",
# "price": "35000.00",
# "reduceOnly": False,
# "side": "BUY",
# "positionSide": "BOTH",
# "stopPrice": "50000.00", # ignored with trailing orders
# "symbol": "BTCUSDT",
# "timeInForce": "GTC",
# "activatePrice": null, # only return with trailing orders
# "priceRate": null, # only return with trailing orders
# "bookTime": 1707270098774,
# "updateTime": 1707270119261,
# "workingType": "CONTRACT_PRICE",
# "priceProtect": False,
# "goodTillDate": 0,
# "selfTradePreventionMode": "NONE"
# }
#
# fetchOrders: portfolio margin linear and inverse swap conditional
#
# {
# "newClientStrategyId": "x-xcKtGhcuaf166172ed504cd1bc0396",
# "strategyId": 3733211,
# "strategyStatus": "CANCELLED",
# "strategyType": "STOP",
# "origQty": "0.010",
# "price": "35000",
# "orderId": 0,
# "reduceOnly": False,
# "side": "BUY",
# "positionSide": "BOTH",
# "stopPrice": "50000",
# "symbol": "BTCUSDT",
# "type": "LIMIT",
# "bookTime": 1707270098774,
# "updateTime": 1707270119261,
# "timeInForce": "GTC",
# "triggerTime": 0,
# "workingType": "CONTRACT_PRICE",
# "priceProtect": False,
# "goodTillDate": 0,
# "selfTradePreventionMode": "NONE"
# }
#
# fetchOpenOrder: linear swap
#
# {
# "orderId": 3697213934,
# "symbol": "BTCUSDT",
# "status": "NEW",
# "clientOrderId": "x-xcKtGhcufb20c5a7761a4aa09aa156",
# "price": "33000.00",
# "avgPrice": "0.00000",
# "origQty": "0.010",
# "executedQty": "0.000",
# "cumQuote": "0.00000",
# "timeInForce": "GTC",
# "type": "LIMIT",
# "reduceOnly": False,
# "closePosition": False,
# "side": "BUY",
# "positionSide": "BOTH",
# "stopPrice": "0.00",
# "workingType": "CONTRACT_PRICE",
# "priceProtect": False,
# "origType": "LIMIT",
# "priceMatch": "NONE",
# "selfTradePreventionMode": "NONE",
# "goodTillDate": 0,
# "time": 1707892893502,
# "updateTime": 1707892893515
# }
#
# fetchOpenOrder: inverse swap
#
# {
# "orderId": 597368542,
# "symbol": "BTCUSD_PERP",
# "pair": "BTCUSD",
# "status": "NEW",
# "clientOrderId": "x-xcKtGhcubbde7ba93b1a4ab881eff3",
# "price": "35000",
# "avgPrice": "0",
# "origQty": "1",
# "executedQty": "0",
# "cumBase": "0",
# "timeInForce": "GTC",
# "type": "LIMIT",
# "reduceOnly": False,
# "closePosition": False,
# "side": "BUY",
# "positionSide": "BOTH",
# "stopPrice": "0",
# "workingType": "CONTRACT_PRICE",
# "priceProtect": False,
# "origType": "LIMIT",
# "time": 1707893453199,
# "updateTime": 1707893453199
# }
#
# fetchOpenOrder: linear portfolio margin
#
# {
# "orderId": 264895013409,
# "symbol": "BTCUSDT",
# "status": "NEW",
# "clientOrderId": "x-xcKtGhcu6278f1adbdf14f74ab432e",
# "price": "35000",
# "avgPrice": "0",
# "origQty": "0.010",
# "executedQty": "0",
# "cumQuote": "0",
# "timeInForce": "GTC",
# "type": "LIMIT",
# "reduceOnly": False,
# "side": "BUY",
# "positionSide": "LONG",
# "origType": "LIMIT",
# "time": 1707893839364,
# "updateTime": 1707893839364,
# "goodTillDate": 0,
# "selfTradePreventionMode": "NONE"
# }
#
# fetchOpenOrder: inverse portfolio margin
#
# {
# "orderId": 71790316950,
# "symbol": "ETHUSD_PERP",
# "pair": "ETHUSD",
# "status": "NEW",
# "clientOrderId": "x-xcKtGhcuec11030474204ab08ba2c2",
# "price": "2500",
# "avgPrice": "0",
# "origQty": "1",
# "executedQty": "0",
# "cumBase": "0",
# "timeInForce": "GTC",
# "type": "LIMIT",
# "reduceOnly": False,
# "side": "BUY",
# "positionSide": "LONG",
# "origType": "LIMIT",
# "time": 1707894181694,
# "updateTime": 1707894181694
# }
#
# fetchOpenOrder: inverse portfolio margin conditional
#
# {
# "newClientStrategyId": "x-xcKtGhcu2da9c765294b433994ffce",
# "strategyId": 1423501,
# "strategyStatus": "NEW",
# "strategyType": "STOP",
# "origQty": "1",
# "price": "2500",
# "reduceOnly": False,
# "side": "BUY",
# "positionSide": "LONG",
# "stopPrice": "4000",
# "symbol": "ETHUSD_PERP",
# "bookTime": 1707894782679,
# "updateTime": 1707894782679,
# "timeInForce": "GTC",
# "workingType": "CONTRACT_PRICE",
# "priceProtect": False
# }
#
code = self.safe_string(order, 'code')
if code is not None:
# cancelOrders/createOrders might have a partial success
return self.safe_order({'info': order, 'status': 'rejected'}, market)
status = self.parse_order_status(self.safe_string_2(order, 'status', 'strategyStatus'))
marketId = self.safe_string(order, 'symbol')
isContract = ('positionSide' in order) or ('cumQuote' in order)
marketType = 'contract' if isContract else 'spot'
symbol = self.safe_symbol(marketId, market, None, marketType)
filled = self.safe_string(order, 'executedQty', '0')
timestamp = self.safe_integer_n(order, ['time', 'createTime', 'workingTime', 'transactTime', 'updateTime']) # order of the keys matters here
lastTradeTimestamp = None
if ('transactTime' in order) or ('updateTime' in order):
timestampValue = self.safe_integer_2(order, 'updateTime', 'transactTime')
if status == 'open':
if Precise.string_gt(filled, '0'):
lastTradeTimestamp = timestampValue
elif status == 'closed':
lastTradeTimestamp = timestampValue
lastUpdateTimestamp = self.safe_integer_2(order, 'transactTime', 'updateTime')
average = self.safe_string(order, 'avgPrice')
price = self.safe_string(order, 'price')
amount = self.safe_string_2(order, 'origQty', 'quantity')
# - Spot/Margin market: cummulativeQuoteQty
# - Futures market: cumQuote.
# Note self is not the actual cost, since Binance futures uses leverage to calculate margins.
cost = self.safe_string_2(order, 'cummulativeQuoteQty', 'cumQuote')
cost = self.safe_string(order, 'cumBase', cost)
type = self.safe_string_lower(order, 'type')
side = self.safe_string_lower(order, 'side')
fills = self.safe_list(order, 'fills', [])
timeInForce = self.safe_string(order, 'timeInForce')
if timeInForce == 'GTX':
# GTX means "Good Till Crossing" and is an equivalent way of saying Post Only
timeInForce = 'PO'
postOnly = (type == 'limit_maker') or (timeInForce == 'PO')
if type == 'limit_maker':
type = 'limit'
stopPriceString = self.safe_string(order, 'stopPrice')
triggerPrice = self.parse_number(self.omit_zero(stopPriceString))
feeCost = self.safe_number(order, 'fee')
fee = None
if feeCost is not None:
fee = {
'currency': self.safe_string(order, 'quoteAsset'),
'cost': feeCost,
'rate': None,
}
return self.safe_order({
'info': order,
'id': self.safe_string_2(order, 'strategyId', 'orderId'),
'clientOrderId': self.safe_string_2(order, 'clientOrderId', 'newClientStrategyId'),
'timestamp': timestamp,
'datetime': self.iso8601(timestamp),
'lastTradeTimestamp': lastTradeTimestamp,
'lastUpdateTimestamp': lastUpdateTimestamp,
'symbol': symbol,
'type': type,
'timeInForce': timeInForce,
'postOnly': postOnly,
'reduceOnly': self.safe_bool(order, 'reduceOnly'),
'side': side,
'price': price,
'triggerPrice': triggerPrice,
'amount': amount,
'cost': cost,
'average': average,
'filled': filled,
'remaining': None,
'status': status,
'fee': fee,
'trades': fills,
}, market)
def create_orders(self, orders: List[OrderRequest], params={}):
"""
*contract only* create a list of trade orders
https://developers.binance.com/docs/derivatives/coin-margined-futures/trade/rest-api/Place-Multiple-Orders
https://developers.binance.com/docs/derivatives/usds-margined-futures/trade/rest-api/Place-Multiple-Orders
https://developers.binance.com/docs/derivatives/option/trade/Place-Multiple-Orders
: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 exchange API endpoint
:returns dict: an `order structure <https://docs.ccxt.com/#/?id=order-structure>`
"""
self.load_markets()
ordersRequests = []
orderSymbols = []
for i in range(0, len(orders)):
rawOrder = orders[i]
marketId = self.safe_string(rawOrder, 'symbol')
orderSymbols.append(marketId)
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_dict(rawOrder, 'params', {})
orderRequest = self.create_order_request(marketId, type, side, amount, price, orderParams)
ordersRequests.append(orderRequest)
orderSymbols = self.market_symbols(orderSymbols, None, False, True, True)
market = self.market(orderSymbols[0])
if market['spot']:
raise NotSupported(self.id + ' createOrders() does not support ' + market['type'] + ' orders')
response = None
request: dict = {
'batchOrders': ordersRequests,
}
request = self.extend(request, params)
if market['linear']:
response = self.fapiPrivatePostBatchOrders(request)
elif market['option']:
response = self.eapiPrivatePostBatchOrders(request)
else:
response = self.dapiPrivatePostBatchOrders(request)
#
# [
# {
# "code": -4005,
# "msg": "Quantity greater than max quantity."
# },
# {
# "orderId": 650640530,
# "symbol": "LTCUSDT",
# "status": "NEW",
# "clientOrderId": "x-xcKtGhcu32184eb13585491289bbaf",
# "price": "54.00",
# "avgPrice": "0.00",
# "origQty": "0.100",
# "executedQty": "0.000",
# "cumQty": "0.000",
# "cumQuote": "0.00000",
# "timeInForce": "GTC",
# "type": "LIMIT",
# "reduceOnly": False,
# "closePosition": False,
# "side": "BUY",
# "positionSide": "BOTH",
# "stopPrice": "0.00",
# "workingType": "CONTRACT_PRICE",
# "priceProtect": False,
# "origType": "LIMIT",
# "priceMatch": "NONE",
# "selfTradePreventionMode": "NONE",
# "goodTillDate": 0,
# "updateTime": 1698073926929
# }
# ]
#
return self.parse_orders(response)
def create_order(self, symbol: str, type: OrderType, side: OrderSide, amount: float, price: Num = None, params={}):
"""
create a trade order
https://developers.binance.com/docs/binance-spot-api-docs/rest-api/trading-endpoints#new-order-trade
https://developers.binance.com/docs/binance-spot-api-docs/testnet/rest-api/trading-endpoints#test-new-order-trade
https://developers.binance.com/docs/derivatives/usds-margined-futures/trade/rest-api/New-Order
https://developers.binance.com/docs/derivatives/coin-margined-futures/trade/rest-api
https://developers.binance.com/docs/derivatives/option/trade/New-Order
https://developers.binance.com/docs/binance-spot-api-docs/rest-api/trading-endpoints#sor
https://developers.binance.com/docs/binance-spot-api-docs/testnet/rest-api/trading-endpoints#sor
https://developers.binance.com/docs/derivatives/portfolio-margin/trade/New-UM-Order
https://developers.binance.com/docs/derivatives/portfolio-margin/trade/New-CM-Order
https://developers.binance.com/docs/derivatives/portfolio-margin/trade/New-Margin-Order
https://developers.binance.com/docs/derivatives/portfolio-margin/trade/New-UM-Conditional-Order
https://developers.binance.com/docs/derivatives/portfolio-margin/trade/New-CM-Conditional-Order
:param str symbol: unified symbol of the market to create an order in
:param str type: 'market' or 'limit' or 'STOP_LOSS' or 'STOP_LOSS_LIMIT' or 'TAKE_PROFIT' or 'TAKE_PROFIT_LIMIT' or 'STOP'
:param str side: 'buy' or 'sell'
:param float amount: how much of you want to trade in units of the base currency
:param float [price]: the price that 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 str [params.reduceOnly]: for swap and future reduceOnly is a string 'true' or 'false' that cant be sent with close position set to True or in hedge mode. For spot margin and option reduceOnly is a boolean.
:param str [params.marginMode]: 'cross' or 'isolated', for spot margin trading
:param boolean [params.sor]: *spot only* whether to use SOR(Smart Order Routing) or not, default is False
:param boolean [params.test]: *spot only* whether to use the test endpoint or not, default is False
:param float [params.trailingPercent]: the percent to trail away from the current market price
:param float [params.trailingTriggerPrice]: the price to trigger a trailing order, default uses the price argument
:param float [params.triggerPrice]: the price that a trigger order is triggered at
:param float [params.stopLossPrice]: the price that a stop loss order is triggered at
:param float [params.takeProfitPrice]: the price that a take profit order is triggered at
:param boolean [params.portfolioMargin]: set to True if you would like to create an order in a portfolio margin account
:param str [params.selfTradePrevention]: set unified value for stp, one of NONE, EXPIRE_MAKER, EXPIRE_TAKER or EXPIRE_BOTH
:param float [params.icebergAmount]: set iceberg amount for limit orders
:param str [params.stopLossOrTakeProfit]: 'stopLoss' or 'takeProfit', required for spot trailing orders
:param str [params.positionSide]: *swap and portfolio margin only* "BOTH" for one-way mode, "LONG" for buy side of hedged mode, "SHORT" for sell side of hedged mode
:param bool [params.hedged]: *swap and portfolio margin only* True for hedged mode, False for one way mode, default is False
:returns dict: an `order structure <https://docs.ccxt.com/#/?id=order-structure>`
"""
self.load_markets()
market = self.market(symbol)
# don't handle/omit params here, omitting happens inside createOrderRequest
marketType = self.safe_string(params, 'type', market['type'])
marginMode = self.safe_string(params, 'marginMode')
porfolioOptionsValue = self.safe_bool_2(self.options, 'papi', 'portfolioMargin', False)
isPortfolioMargin = self.safe_bool_2(params, 'papi', 'portfolioMargin', porfolioOptionsValue)
triggerPrice = self.safe_string_2(params, 'triggerPrice', 'stopPrice')
stopLossPrice = self.safe_string(params, 'stopLossPrice')
takeProfitPrice = self.safe_string(params, 'takeProfitPrice')
trailingPercent = self.safe_string_2(params, 'trailingPercent', 'callbackRate')
isTrailingPercentOrder = trailingPercent is not None
isStopLoss = stopLossPrice is not None
isTakeProfit = takeProfitPrice is not None
isConditional = (triggerPrice is not None) or isTrailingPercentOrder or isStopLoss or isTakeProfit
sor = self.safe_bool_2(params, 'sor', 'SOR', False)
test = self.safe_bool(params, 'test', False)
params = self.omit(params, ['sor', 'SOR', 'test'])
# if isPortfolioMargin:
# params['portfolioMargin'] = isPortfolioMargin
# }
request = self.create_order_request(symbol, type, side, amount, price, params)
response = None
if market['option']:
response = self.eapiPrivatePostOrder(request)
elif sor:
if test:
response = self.privatePostSorOrderTest(request)
else:
response = self.privatePostSorOrder(request)
elif market['linear']:
if isPortfolioMargin:
if isConditional:
response = self.papiPostUmConditionalOrder(request)
else:
response = self.papiPostUmOrder(request)
else:
response = self.fapiPrivatePostOrder(request)
elif market['inverse']:
if isPortfolioMargin:
if isConditional:
response = self.papiPostCmConditionalOrder(request)
else:
response = self.papiPostCmOrder(request)
else:
response = self.dapiPrivatePostOrder(request)
elif marketType == 'margin' or marginMode is not None or isPortfolioMargin:
if isPortfolioMargin:
response = self.papiPostMarginOrder(request)
else:
response = self.sapiPostMarginOrder(request)
else:
if test:
response = self.privatePostOrderTest(request)
else:
response = self.privatePostOrder(request)
return self.parse_order(response, market)
def create_order_request(self, symbol: str, type: OrderType, side: OrderSide, amount: float, price: Num = None, params={}):
"""
@ignore
helper function to build the request
: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 that 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
:returns dict: request to be sent to the exchange
"""
market = self.market(symbol)
marketType = self.safe_string(params, 'type', market['type'])
clientOrderId = self.safe_string_2(params, 'newClientOrderId', 'clientOrderId')
initialUppercaseType = type.upper()
isMarketOrder = initialUppercaseType == 'MARKET'
isLimitOrder = initialUppercaseType == 'LIMIT'
request: dict = {
'symbol': market['id'],
'side': side.upper(),
}
isPortfolioMargin = None
isPortfolioMargin, params = self.handle_option_and_params_2(params, 'createOrder', 'papi', 'portfolioMargin', False)
marginMode = None
marginMode, params = self.handle_margin_mode_and_params('createOrder', params)
reduceOnly = self.safe_bool(params, 'reduceOnly', False)
if reduceOnly:
if marketType == 'margin' or (not market['contract'] and (marginMode is not None)):
params = self.omit(params, 'reduceOnly')
request['sideEffectType'] = 'AUTO_REPAY'
triggerPrice = self.safe_string_2(params, 'triggerPrice', 'stopPrice')
stopLossPrice = self.safe_string(params, 'stopLossPrice', triggerPrice) # fallback to stopLoss
takeProfitPrice = self.safe_string(params, 'takeProfitPrice')
trailingDelta = self.safe_string(params, 'trailingDelta')
trailingTriggerPrice = self.safe_string_2(params, 'trailingTriggerPrice', 'activationPrice')
trailingPercent = self.safe_string_n(params, ['trailingPercent', 'callbackRate', 'trailingDelta'])
priceMatch = self.safe_string(params, 'priceMatch')
isTrailingPercentOrder = trailingPercent is not None
isStopLoss = stopLossPrice is not None or trailingDelta is not None
isTakeProfit = takeProfitPrice is not None
isTriggerOrder = triggerPrice is not None
isConditional = isTriggerOrder or isTrailingPercentOrder or isStopLoss or isTakeProfit
isPortfolioMarginConditional = (isPortfolioMargin and isConditional)
isPriceMatch = priceMatch is not None
priceRequiredForTrailing = True
uppercaseType = type.upper()
stopPrice = None
if isTrailingPercentOrder:
if market['swap']:
uppercaseType = 'TRAILING_STOP_MARKET'
request['callbackRate'] = trailingPercent
if trailingTriggerPrice is not None:
request['activationPrice'] = self.price_to_precision(symbol, trailingTriggerPrice)
else:
if (uppercaseType != 'STOP_LOSS') and (uppercaseType != 'TAKE_PROFIT') and (uppercaseType != 'STOP_LOSS_LIMIT') and (uppercaseType != 'TAKE_PROFIT_LIMIT'):
stopLossOrTakeProfit = self.safe_string(params, 'stopLossOrTakeProfit')
params = self.omit(params, 'stopLossOrTakeProfit')
if (stopLossOrTakeProfit != 'stopLoss') and (stopLossOrTakeProfit != 'takeProfit'):
raise InvalidOrder(self.id + symbol + ' trailingPercent orders require a stopLossOrTakeProfit parameter of either stopLoss or takeProfit')
if isMarketOrder:
if stopLossOrTakeProfit == 'stopLoss':
uppercaseType = 'STOP_LOSS'
elif stopLossOrTakeProfit == 'takeProfit':
uppercaseType = 'TAKE_PROFIT'
else:
if stopLossOrTakeProfit == 'stopLoss':
uppercaseType = 'STOP_LOSS_LIMIT'
elif stopLossOrTakeProfit == 'takeProfit':
uppercaseType = 'TAKE_PROFIT_LIMIT'
if (uppercaseType == 'STOP_LOSS') or (uppercaseType == 'TAKE_PROFIT'):
priceRequiredForTrailing = False
if trailingTriggerPrice is not None:
stopPrice = self.price_to_precision(symbol, trailingTriggerPrice)
trailingPercentConverted = Precise.string_mul(trailingPercent, '100')
request['trailingDelta'] = trailingPercentConverted
elif isStopLoss:
stopPrice = stopLossPrice
if isMarketOrder:
# spot STOP_LOSS market orders are not a valid order type
uppercaseType = 'STOP_MARKET' if market['contract'] else 'STOP_LOSS'
elif isLimitOrder:
uppercaseType = 'STOP' if market['contract'] else 'STOP_LOSS_LIMIT'
elif isTakeProfit:
stopPrice = takeProfitPrice
if isMarketOrder:
# spot TAKE_PROFIT market orders are not a valid order type
uppercaseType = 'TAKE_PROFIT_MARKET' if market['contract'] else 'TAKE_PROFIT'
elif isLimitOrder:
uppercaseType = 'TAKE_PROFIT' if market['contract'] else 'TAKE_PROFIT_LIMIT'
if market['option']:
if type == 'market':
raise InvalidOrder(self.id + ' ' + type + ' is not a valid order type for the ' + symbol + ' market')
else:
validOrderTypes = self.safe_list(market['info'], 'orderTypes')
if not self.in_array(uppercaseType, validOrderTypes):
if initialUppercaseType != uppercaseType:
raise InvalidOrder(self.id + ' triggerPrice parameter is not allowed for ' + symbol + ' ' + type + ' orders')
else:
raise InvalidOrder(self.id + ' ' + type + ' is not a valid order type for the ' + symbol + ' market')
clientOrderIdRequest = 'newClientStrategyId' if isPortfolioMarginConditional else 'newClientOrderId'
if clientOrderId is None:
broker = self.safe_dict(self.options, 'broker', {})
defaultId = 'x-xcKtGhcu' if (market['contract']) else 'x-TKT5PX2F'
idMarketType = 'spot'
if market['contract']:
idMarketType = 'swap' if (market['swap'] and market['linear']) else 'inverse'
brokerId = self.safe_string(broker, idMarketType, defaultId)
request[clientOrderIdRequest] = brokerId + self.uuid22()
else:
request[clientOrderIdRequest] = clientOrderId
postOnly = None
if not isPortfolioMargin:
postOnly = self.is_post_only(isMarketOrder, initialUppercaseType == 'LIMIT_MAKER', params)
if market['spot'] or marketType == 'margin':
# only supported for spot/margin api(all margin markets are spot markets)
if postOnly:
uppercaseType = 'LIMIT_MAKER'
if marginMode == 'isolated':
request['isIsolated'] = True
else:
postOnly = self.is_post_only(isMarketOrder, initialUppercaseType == 'LIMIT_MAKER', params)
if postOnly:
if not market['contract']:
uppercaseType = 'LIMIT_MAKER'
else:
request['timeInForce'] = 'GTX'
# handle newOrderRespType response type
if ((marketType == 'spot') or (marketType == 'margin')) and not isPortfolioMargin:
request['newOrderRespType'] = self.safe_string(self.options['newOrderRespType'], type, 'FULL') # 'ACK' for order id, 'RESULT' for full order or 'FULL' for order with fills
else:
# swap, futures and options
request['newOrderRespType'] = 'RESULT' # "ACK", "RESULT", default "ACK"
typeRequest = 'strategyType' if isPortfolioMarginConditional else 'type'
request[typeRequest] = uppercaseType
# additional required fields depending on the order type
closePosition = self.safe_bool(params, 'closePosition', False)
timeInForceIsRequired = False
priceIsRequired = False
triggerPriceIsRequired = False
quantityIsRequired = False
#
# spot/margin
#
# LIMIT timeInForce, quantity, price
# MARKET quantity or quoteOrderQty
# STOP_LOSS quantity, stopPrice
# STOP_LOSS_LIMIT timeInForce, quantity, price, stopPrice
# TAKE_PROFIT quantity, stopPrice
# TAKE_PROFIT_LIMIT timeInForce, quantity, price, stopPrice
# LIMIT_MAKER quantity, price
#
# futures
#
# LIMIT timeInForce, quantity, price
# MARKET quantity
# STOP/TAKE_PROFIT quantity, price, stopPrice
# STOP_MARKET stopPrice
# TAKE_PROFIT_MARKET stopPrice
# TRAILING_STOP_MARKET callbackRate
#
if uppercaseType == 'MARKET':
if market['spot']:
quoteOrderQty = self.safe_bool(self.options, 'quoteOrderQty', True)
if quoteOrderQty:
quoteOrderQtyNew = self.safe_string_2(params, 'quoteOrderQty', 'cost')
precision = market['precision']['price']
if quoteOrderQtyNew is not None:
request['quoteOrderQty'] = self.decimal_to_precision(quoteOrderQtyNew, TRUNCATE, precision, self.precisionMode)
elif price is not None:
amountString = self.number_to_string(amount)
priceString = self.number_to_string(price)
quoteOrderQuantity = Precise.string_mul(amountString, priceString)
request['quoteOrderQty'] = self.decimal_to_precision(quoteOrderQuantity, TRUNCATE, precision, self.precisionMode)
else:
quantityIsRequired = True
else:
quantityIsRequired = True
else:
quantityIsRequired = True
elif uppercaseType == 'LIMIT':
priceIsRequired = True
timeInForceIsRequired = True
quantityIsRequired = True
elif (uppercaseType == 'STOP_LOSS') or (uppercaseType == 'TAKE_PROFIT'):
triggerPriceIsRequired = True
quantityIsRequired = True
if (market['linear'] or market['inverse']) and priceRequiredForTrailing:
priceIsRequired = True
elif (uppercaseType == 'STOP_LOSS_LIMIT') or (uppercaseType == 'TAKE_PROFIT_LIMIT'):
quantityIsRequired = True
triggerPriceIsRequired = True
priceIsRequired = True
timeInForceIsRequired = True
elif uppercaseType == 'LIMIT_MAKER':
priceIsRequired = True
quantityIsRequired = True
elif uppercaseType == 'STOP':
quantityIsRequired = True
triggerPriceIsRequired = True
priceIsRequired = True
elif (uppercaseType == 'STOP_MARKET') or (uppercaseType == 'TAKE_PROFIT_MARKET'):
if not closePosition:
quantityIsRequired = True
triggerPriceIsRequired = True
elif uppercaseType == 'TRAILING_STOP_MARKET':
if not closePosition:
quantityIsRequired = True
if trailingPercent is None:
raise InvalidOrder(self.id + ' createOrder() requires a trailingPercent param for a ' + type + ' order')
if quantityIsRequired:
marketAmountPrecision = self.safe_string(market['precision'], 'amount')
isPrecisionAvailable = (marketAmountPrecision is not None)
if isPrecisionAvailable:
request['quantity'] = self.amount_to_precision(symbol, amount)
else:
request['quantity'] = self.parse_to_numeric(amount) # some options don't have the precision available
if priceIsRequired and not isPriceMatch:
if price is None:
raise InvalidOrder(self.id + ' createOrder() requires a price argument for a ' + type + ' order')
pricePrecision = self.safe_string(market['precision'], 'price')
isPricePrecisionAvailable = (pricePrecision is not None)
if isPricePrecisionAvailable:
request['price'] = self.price_to_precision(symbol, price)
else:
request['price'] = self.parse_to_numeric(price) # some options don't have the precision available
if triggerPriceIsRequired:
if market['contract']:
if stopPrice is None:
raise InvalidOrder(self.id + ' createOrder() requires a triggerPrice extra param for a ' + type + ' order')
else:
# check for delta price
if trailingDelta is None and stopPrice is None and trailingPercent is None:
raise InvalidOrder(self.id + ' createOrder() requires a triggerPrice, trailingDelta or trailingPercent param for a ' + type + ' order')
if stopPrice is not None:
request['stopPrice'] = self.price_to_precision(symbol, stopPrice)
if timeInForceIsRequired and (self.safe_string(params, 'timeInForce') is None) and (self.safe_string(request, 'timeInForce') is None):
request['timeInForce'] = self.safe_string(self.options, 'defaultTimeInForce') # 'GTC' = Good To Cancel(default), 'IOC' = Immediate Or Cancel
if not isPortfolioMargin and market['contract'] and postOnly:
request['timeInForce'] = 'GTX'
# remove timeInForce from params because PO is only used by self.is_post_only and it's not a valid value for Binance
if self.safe_string(params, 'timeInForce') == 'PO':
params = self.omit(params, 'timeInForce')
hedged = self.safe_bool(params, 'hedged', False)
if not market['spot'] and not market['option'] and hedged:
if reduceOnly:
params = self.omit(params, 'reduceOnly')
side = 'sell' if (side == 'buy') else 'buy'
request['positionSide'] = 'LONG' if (side == 'buy') else 'SHORT'
# unified stp
selfTradePrevention = None
selfTradePrevention, params = self.handle_option_and_params(params, 'createOrder', 'selfTradePrevention')
if selfTradePrevention is not None:
if market['spot']:
request['selfTradePreventionMode'] = selfTradePrevention.upper() # binance enums exactly match the unified ccxt enums(but needs uppercase)
# unified iceberg
icebergAmount = self.safe_number(params, 'icebergAmount')
if icebergAmount is not None:
if market['spot']:
request['icebergQty'] = self.amount_to_precision(symbol, icebergAmount)
requestParams = self.omit(params, ['type', 'newClientOrderId', 'clientOrderId', 'postOnly', 'stopLossPrice', 'takeProfitPrice', 'stopPrice', 'triggerPrice', 'trailingTriggerPrice', 'trailingPercent', 'quoteOrderQty', 'cost', 'test', 'hedged', 'icebergAmount'])
return self.extend(request, requestParams)
def create_market_order_with_cost(self, symbol: str, side: OrderSide, cost: float, params={}):
"""
create a market order by providing the symbol, side and cost
https://developers.binance.com/docs/binance-spot-api-docs/rest-api/trading-endpoints#new-order-trade
:param str symbol: unified symbol of the market to create an order in
:param str side: 'buy' or 'sell'
: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>`
"""
self.load_markets()
market = self.market(symbol)
if not market['spot']:
raise NotSupported(self.id + ' createMarketOrderWithCost() supports spot orders only')
req = {
'cost': cost,
}
return self.create_order(symbol, 'market', side, cost, None, self.extend(req, params))
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://developers.binance.com/docs/binance-spot-api-docs/rest-api/trading-endpoints#new-order-trade
: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>`
"""
self.load_markets()
market = self.market(symbol)
if not market['spot']:
raise NotSupported(self.id + ' createMarketBuyOrderWithCost() supports spot orders only')
req = {
'cost': cost,
}
return self.create_order(symbol, 'market', 'buy', cost, None, self.extend(req, params))
def create_market_sell_order_with_cost(self, symbol: str, cost: float, params={}):
"""
create a market sell order by providing the symbol and cost
https://developers.binance.com/docs/binance-spot-api-docs/rest-api/trading-endpoints#new-order-trade
: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>`
"""
self.load_markets()
market = self.market(symbol)
if not market['spot']:
raise NotSupported(self.id + ' createMarketSellOrderWithCost() supports spot orders only')
params['quoteOrderQty'] = cost
return self.create_order(symbol, 'market', 'sell', cost, None, params)
def fetch_order(self, id: str, symbol: Str = None, params={}):
"""
fetches information on an order made by the user
https://developers.binance.com/docs/binance-spot-api-docs/rest-api/trading-endpoints#query-order-user_data
https://developers.binance.com/docs/derivatives/usds-margined-futures/trade/rest-api/Query-Order
https://developers.binance.com/docs/derivatives/coin-margined-futures/trade/rest-api/Query-Order
https://developers.binance.com/docs/derivatives/option/trade/Query-Single-Order
https://developers.binance.com/docs/margin_trading/trade/Query-Margin-Account-Order
https://developers.binance.com/docs/derivatives/portfolio-margin/trade/Query-UM-Order
https://developers.binance.com/docs/derivatives/portfolio-margin/trade/Query-CM-Order
: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 str [params.marginMode]: 'cross' or 'isolated', for spot margin trading
:param boolean [params.portfolioMargin]: set to True if you would like to fetch an order in a portfolio margin account
: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')
self.load_markets()
market = self.market(symbol)
defaultType = self.safe_string_2(self.options, 'fetchOrder', 'defaultType', 'spot')
type = self.safe_string(params, 'type', defaultType)
marginMode = None
marginMode, params = self.handle_margin_mode_and_params('fetchOrder', params)
isPortfolioMargin = None
isPortfolioMargin, params = self.handle_option_and_params_2(params, 'fetchOrder', 'papi', 'portfolioMargin', False)
request: dict = {
'symbol': market['id'],
}
clientOrderId = self.safe_string_2(params, 'origClientOrderId', 'clientOrderId')
if clientOrderId is not None:
if market['option']:
request['clientOrderId'] = clientOrderId
else:
request['origClientOrderId'] = clientOrderId
else:
request['orderId'] = id
params = self.omit(params, ['type', 'clientOrderId', 'origClientOrderId'])
response = None
if market['option']:
response = self.eapiPrivateGetOrder(self.extend(request, params))
elif market['linear']:
if isPortfolioMargin:
response = self.papiGetUmOrder(self.extend(request, params))
else:
response = self.fapiPrivateGetOrder(self.extend(request, params))
elif market['inverse']:
if isPortfolioMargin:
response = self.papiGetCmOrder(self.extend(request, params))
else:
response = self.dapiPrivateGetOrder(self.extend(request, params))
elif (type == 'margin') or (marginMode is not None) or isPortfolioMargin:
if isPortfolioMargin:
response = self.papiGetMarginOrder(self.extend(request, params))
else:
if marginMode == 'isolated':
request['isIsolated'] = True
response = self.sapiGetMarginOrder(self.extend(request, params))
else:
response = self.privateGetOrder(self.extend(request, params))
return self.parse_order(response, market)
def fetch_orders(self, symbol: Str = None, since: Int = None, limit: Int = None, params={}) -> List[Order]:
"""
fetches information on multiple orders made by the user
https://developers.binance.com/docs/binance-spot-api-docs/rest-api/trading-endpoints#all-orders-user_data
https://developers.binance.com/docs/derivatives/usds-margined-futures/trade/rest-api/All-Orders
https://developers.binance.com/docs/derivatives/coin-margined-futures/trade/rest-api/All-Orders
https://developers.binance.com/docs/derivatives/option/trade/Query-Option-Order-History
https://developers.binance.com/docs/margin_trading/trade/Query-Margin-Account-All-Orders
https://developers.binance.com/docs/derivatives/portfolio-margin/trade/Query-All-UM-Orders
https://developers.binance.com/docs/derivatives/portfolio-margin/trade/Query-All-CM-Orders
https://developers.binance.com/docs/derivatives/portfolio-margin/trade/Query-All-UM-Conditional-Orders
https://developers.binance.com/docs/derivatives/portfolio-margin/trade/Query-All-CM-Conditional-Orders
: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 str [params.marginMode]: 'cross' or 'isolated', for spot margin trading
:param int [params.until]: the latest time in ms to fetch orders 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)
:param boolean [params.portfolioMargin]: set to True if you would like to fetch orders in a portfolio margin account
:param boolean [params.trigger]: set to True if you would like to fetch portfolio margin account trigger or conditional orders
:returns Order[]: a list of `order structures <https://docs.ccxt.com/#/?id=order-structure>`
"""
if symbol is None:
raise ArgumentsRequired(self.id + ' fetchOrders() requires a symbol argument')
self.load_markets()
paginate = False
paginate, params = self.handle_option_and_params(params, 'fetchOrders', 'paginate')
if paginate:
return self.fetch_paginated_call_dynamic('fetchOrders', symbol, since, limit, params)
market = self.market(symbol)
defaultType = self.safe_string_2(self.options, 'fetchOrders', 'defaultType', market['type'])
type = self.safe_string(params, 'type', defaultType)
marginMode = None
marginMode, params = self.handle_margin_mode_and_params('fetchOrders', params)
isPortfolioMargin = None
isPortfolioMargin, params = self.handle_option_and_params_2(params, 'fetchOrders', 'papi', 'portfolioMargin', False)
isConditional = self.safe_bool_n(params, ['stop', 'trigger', 'conditional'])
params = self.omit(params, ['stop', 'trigger', 'conditional', 'type'])
request: dict = {
'symbol': market['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 market['option']:
response = self.eapiPrivateGetHistoryOrders(self.extend(request, params))
elif market['linear']:
if isPortfolioMargin:
if isConditional:
response = self.papiGetUmConditionalAllOrders(self.extend(request, params))
else:
response = self.papiGetUmAllOrders(self.extend(request, params))
else:
response = self.fapiPrivateGetAllOrders(self.extend(request, params))
elif market['inverse']:
if isPortfolioMargin:
if isConditional:
response = self.papiGetCmConditionalAllOrders(self.extend(request, params))
else:
response = self.papiGetCmAllOrders(self.extend(request, params))
else:
response = self.dapiPrivateGetAllOrders(self.extend(request, params))
else:
if isPortfolioMargin:
response = self.papiGetMarginAllOrders(self.extend(request, params))
elif type == 'margin' or marginMode is not None:
if marginMode == 'isolated':
request['isIsolated'] = True
response = self.sapiGetMarginAllOrders(self.extend(request, params))
else:
response = self.privateGetAllOrders(self.extend(request, params))
#
# spot
#
# [
# {
# "symbol": "LTCBTC",
# "orderId": 1,
# "clientOrderId": "myOrder1",
# "price": "0.1",
# "origQty": "1.0",
# "executedQty": "0.0",
# "cummulativeQuoteQty": "0.0",
# "status": "NEW",
# "timeInForce": "GTC",
# "type": "LIMIT",
# "side": "BUY",
# "stopPrice": "0.0",
# "icebergQty": "0.0",
# "time": 1499827319559,
# "updateTime": 1499827319559,
# "isWorking": True
# }
# ]
#
# futures
#
# [
# {
# "symbol": "BTCUSDT",
# "orderId": 1,
# "clientOrderId": "myOrder1",
# "price": "0.1",
# "origQty": "1.0",
# "executedQty": "1.0",
# "cumQuote": "10.0",
# "status": "NEW",
# "timeInForce": "GTC",
# "type": "LIMIT",
# "side": "BUY",
# "stopPrice": "0.0",
# "updateTime": 1499827319559
# }
# ]
#
# options
#
# [
# {
# "orderId": 4728833085436977152,
# "symbol": "ETH-230211-1500-C",
# "price": "10.0",
# "quantity": "1.00",
# "executedQty": "0.00",
# "fee": "0",
# "side": "BUY",
# "type": "LIMIT",
# "timeInForce": "GTC",
# "reduceOnly": False,
# "postOnly": False,
# "createTime": 1676083034462,
# "updateTime": 1676083034462,
# "status": "ACCEPTED",
# "avgPrice": "0",
# "source": "API",
# "clientOrderId": "",
# "priceScale": 1,
# "quantityScale": 2,
# "optionSide": "CALL",
# "quoteAsset": "USDT",
# "lastTrade": {"id":"69","time":"1676084430567","price":"24.9","qty":"1.00"},
# "mmp": False
# }
# ]
#
# inverse portfolio margin
#
# [
# {
# "orderId": 71328442983,
# "symbol": "ETHUSD_PERP",
# "pair": "ETHUSD",
# "status": "CANCELED",
# "clientOrderId": "x-xcKtGhcu4b3e3d8515dd4dc5ba9ccc",
# "price": "2000",
# "avgPrice": "0.00",
# "origQty": "1",
# "executedQty": "0",
# "cumBase": "0",
# "timeInForce": "GTC",
# "type": "LIMIT",
# "reduceOnly": False,
# "side": "BUY",
# "origType": "LIMIT",
# "time": 1707197843046,
# "updateTime": 1707197941373,
# "positionSide": "BOTH"
# },
# ]
#
# linear portfolio margin
#
# [
# {
# "orderId": 259235347005,
# "symbol": "BTCUSDT",
# "status": "CANCELED",
# "clientOrderId": "x-xcKtGhcu402881c9103f42bdb4183b",
# "price": "35000",
# "avgPrice": "0.00000",
# "origQty": "0.010",
# "executedQty": "0",
# "cumQuote": "0",
# "timeInForce": "GTC",
# "type": "LIMIT",
# "reduceOnly": False,
# "side": "BUY",
# "origType": "LIMIT",
# "time": 1707194702167,
# "updateTime": 1707197804748,
# "positionSide": "BOTH",
# "selfTradePreventionMode": "NONE",
# "goodTillDate": 0
# },
# ]
#
# conditional portfolio margin
#
# [
# {
# "newClientStrategyId": "x-xcKtGhcuaf166172ed504cd1bc0396",
# "strategyId": 3733211,
# "strategyStatus": "CANCELLED",
# "strategyType": "STOP",
# "origQty": "0.010",
# "price": "35000",
# "orderId": 0,
# "reduceOnly": False,
# "side": "BUY",
# "positionSide": "BOTH",
# "stopPrice": "50000",
# "symbol": "BTCUSDT",
# "type": "LIMIT",
# "bookTime": 1707270098774,
# "updateTime": 1707270119261,
# "timeInForce": "GTC",
# "triggerTime": 0,
# "workingType": "CONTRACT_PRICE",
# "priceProtect": False,
# "goodTillDate": 0,
# "selfTradePreventionMode": "NONE"
# },
# ]
#
# spot margin portfolio margin
#
# [
# {
# "symbol": "BTCUSDT",
# "orderId": 24684460474,
# "clientOrderId": "x-TKT5PX2Fe9ef29d8346440f0b28b86",
# "price": "35000.00000000",
# "origQty": "0.00100000",
# "executedQty": "0.00000000",
# "cummulativeQuoteQty": "0.00000000",
# "status": "CANCELED",
# "timeInForce": "GTC",
# "type": "LIMIT",
# "side": "BUY",
# "stopPrice": "0.00000000",
# "icebergQty": "0.00000000",
# "time": 1707113538870,
# "updateTime": 1707113797688,
# "isWorking": True,
# "accountId": 200180970,
# "selfTradePreventionMode": "EXPIRE_MAKER",
# "preventedMatchId": null,
# "preventedQuantity": null
# },
# ]
#
return self.parse_orders(response, market, since, limit)
def fetch_open_orders(self, symbol: Str = None, since: Int = None, limit: Int = None, params={}) -> List[Order]:
"""
fetch all unfilled currently open orders
https://developers.binance.com/docs/binance-spot-api-docs/rest-api/trading-endpoints#current-open-orders-user_data
https://developers.binance.com/docs/derivatives/usds-margined-futures/trade/rest-api/Current-All-Open-Orders
https://developers.binance.com/docs/derivatives/coin-margined-futures/trade/rest-api/Current-All-Open-Orders
https://developers.binance.com/docs/derivatives/option/trade/Query-Current-Open-Option-Orders
https://developers.binance.com/docs/margin_trading/trade/Query-Margin-Account-Open-Orders
https://developers.binance.com/docs/derivatives/portfolio-margin/trade/Query-All-Current-UM-Open-Orders
https://developers.binance.com/docs/derivatives/portfolio-margin/trade/Query-All-Current-UM-Open-Conditional-Orders
https://developers.binance.com/docs/derivatives/portfolio-margin/trade/Query-All-Current-CM-Open-Orders
https://developers.binance.com/docs/derivatives/portfolio-margin/trade/Query-All-Current-CM-Open-Conditional-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 orders structures to retrieve
:param dict [params]: extra parameters specific to the exchange API endpoint
:param str [params.marginMode]: 'cross' or 'isolated', for spot margin trading
:param boolean [params.portfolioMargin]: set to True if you would like to fetch open orders in the portfolio margin account
:param boolean [params.trigger]: set to True if you would like to fetch portfolio margin account conditional orders
:param str [params.subType]: "linear" or "inverse"
:returns Order[]: a list of `order structures <https://docs.ccxt.com/#/?id=order-structure>`
"""
self.load_markets()
market = None
type = None
request: dict = {}
marginMode = None
marginMode, params = self.handle_margin_mode_and_params('fetchOpenOrders', params)
isPortfolioMargin = None
isPortfolioMargin, params = self.handle_option_and_params_2(params, 'fetchOpenOrders', 'papi', 'portfolioMargin', False)
isConditional = self.safe_bool_n(params, ['stop', 'trigger', 'conditional'])
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)
elif self.options['warnOnFetchOpenOrdersWithoutSymbol']:
raise ExchangeError(self.id + ' fetchOpenOrders() WARNING: fetching open orders without specifying a symbol has stricter rate limits(10 times more for spot, 40 times more for other markets) compared to requesting with symbol argument. To acknowledge self warning, set ' + self.id + '.options["warnOnFetchOpenOrdersWithoutSymbol"] = False to suppress self warning message.')
else:
defaultType = self.safe_string_2(self.options, 'fetchOpenOrders', 'defaultType', 'spot')
type = self.safe_string(params, 'type', defaultType)
subType = None
subType, params = self.handle_sub_type_and_params('fetchOpenOrders', market, params)
params = self.omit(params, ['type', 'stop', 'trigger', 'conditional'])
response = None
if type == 'option':
if since is not None:
request['startTime'] = since
if limit is not None:
request['limit'] = limit
response = self.eapiPrivateGetOpenOrders(self.extend(request, params))
elif self.is_linear(type, subType):
if isPortfolioMargin:
if isConditional:
response = self.papiGetUmConditionalOpenOrders(self.extend(request, params))
else:
response = self.papiGetUmOpenOrders(self.extend(request, params))
else:
response = self.fapiPrivateGetOpenOrders(self.extend(request, params))
elif self.is_inverse(type, subType):
if isPortfolioMargin:
if isConditional:
response = self.papiGetCmConditionalOpenOrders(self.extend(request, params))
else:
response = self.papiGetCmOpenOrders(self.extend(request, params))
else:
response = self.dapiPrivateGetOpenOrders(self.extend(request, params))
elif type == 'margin' or marginMode is not None or isPortfolioMargin:
if isPortfolioMargin:
response = self.papiGetMarginOpenOrders(self.extend(request, params))
else:
if marginMode == 'isolated':
request['isIsolated'] = True
if symbol is None:
raise ArgumentsRequired(self.id + ' fetchOpenOrders() requires a symbol argument for isolated markets')
response = self.sapiGetMarginOpenOrders(self.extend(request, params))
else:
response = self.privateGetOpenOrders(self.extend(request, params))
return self.parse_orders(response, market, since, limit)
def fetch_open_order(self, id: str, symbol: Str = None, params={}):
"""
fetch an open order by the id
https://developers.binance.com/docs/derivatives/usds-margined-futures/trade/rest-api/Query-Current-Open-Order
https://developers.binance.com/docs/derivatives/coin-margined-futures/trade/rest-api/Query-Current-Open-Order
https://developers.binance.com/docs/derivatives/portfolio-margin/trade/Query-Current-UM-Open-Order
https://developers.binance.com/docs/derivatives/portfolio-margin/trade/Query-Current-UM-Open-Conditional-Order
https://developers.binance.com/docs/derivatives/portfolio-margin/trade/Query-Current-CM-Open-Order
https://developers.binance.com/docs/derivatives/portfolio-margin/trade/Query-Current-CM-Open-Conditional-Order
:param str id: order id
:param str symbol: unified market symbol
:param dict [params]: extra parameters specific to the exchange API endpoint
:param str [params.trigger]: set to True if you would like to fetch portfolio margin account stop or conditional orders
:param boolean [params.portfolioMargin]: set to True if you would like to fetch for a portfolio margin account
:returns dict: an `order structure <https://docs.ccxt.com/#/?id=order-structure>`
"""
if symbol is None:
raise ArgumentsRequired(self.id + ' fetchOpenOrder() requires a symbol argument')
self.load_markets()
market = self.market(symbol)
request: dict = {
'symbol': market['id'],
}
isPortfolioMargin = None
isPortfolioMargin, params = self.handle_option_and_params_2(params, 'fetchOpenOrder', 'papi', 'portfolioMargin', False)
isConditional = self.safe_bool_n(params, ['stop', 'trigger', 'conditional'])
params = self.omit(params, ['stop', 'trigger', 'conditional'])
isPortfolioMarginConditional = (isPortfolioMargin and isConditional)
orderIdRequest = 'strategyId' if isPortfolioMarginConditional else 'orderId'
request[orderIdRequest] = id
response = None
if market['linear']:
if isPortfolioMargin:
if isConditional:
response = self.papiGetUmConditionalOpenOrder(self.extend(request, params))
else:
response = self.papiGetUmOpenOrder(self.extend(request, params))
else:
response = self.fapiPrivateGetOpenOrder(self.extend(request, params))
elif market['inverse']:
if isPortfolioMargin:
if isConditional:
response = self.papiGetCmConditionalOpenOrder(self.extend(request, params))
else:
response = self.papiGetCmOpenOrder(self.extend(request, params))
else:
response = self.dapiPrivateGetOpenOrder(self.extend(request, params))
else:
if market['option']:
raise NotSupported(self.id + ' fetchOpenOrder() does not support option markets')
elif market['spot']:
raise NotSupported(self.id + ' fetchOpenOrder() does not support spot markets')
#
# linear swap
#
# {
# "orderId": 3697213934,
# "symbol": "BTCUSDT",
# "status": "NEW",
# "clientOrderId": "x-xcKtGhcufb20c5a7761a4aa09aa156",
# "price": "33000.00",
# "avgPrice": "0.00000",
# "origQty": "0.010",
# "executedQty": "0.000",
# "cumQuote": "0.00000",
# "timeInForce": "GTC",
# "type": "LIMIT",
# "reduceOnly": False,
# "closePosition": False,
# "side": "BUY",
# "positionSide": "BOTH",
# "stopPrice": "0.00",
# "workingType": "CONTRACT_PRICE",
# "priceProtect": False,
# "origType": "LIMIT",
# "priceMatch": "NONE",
# "selfTradePreventionMode": "NONE",
# "goodTillDate": 0,
# "time": 1707892893502,
# "updateTime": 1707892893515
# }
#
# inverse swap
#
# {
# "orderId": 597368542,
# "symbol": "BTCUSD_PERP",
# "pair": "BTCUSD",
# "status": "NEW",
# "clientOrderId": "x-xcKtGhcubbde7ba93b1a4ab881eff3",
# "price": "35000",
# "avgPrice": "0",
# "origQty": "1",
# "executedQty": "0",
# "cumBase": "0",
# "timeInForce": "GTC",
# "type": "LIMIT",
# "reduceOnly": False,
# "closePosition": False,
# "side": "BUY",
# "positionSide": "BOTH",
# "stopPrice": "0",
# "workingType": "CONTRACT_PRICE",
# "priceProtect": False,
# "origType": "LIMIT",
# "time": 1707893453199,
# "updateTime": 1707893453199
# }
#
# linear portfolio margin
#
# {
# "orderId": 264895013409,
# "symbol": "BTCUSDT",
# "status": "NEW",
# "clientOrderId": "x-xcKtGhcu6278f1adbdf14f74ab432e",
# "price": "35000",
# "avgPrice": "0",
# "origQty": "0.010",
# "executedQty": "0",
# "cumQuote": "0",
# "timeInForce": "GTC",
# "type": "LIMIT",
# "reduceOnly": False,
# "side": "BUY",
# "positionSide": "LONG",
# "origType": "LIMIT",
# "time": 1707893839364,
# "updateTime": 1707893839364,
# "goodTillDate": 0,
# "selfTradePreventionMode": "NONE"
# }
#
# inverse portfolio margin
#
# {
# "orderId": 71790316950,
# "symbol": "ETHUSD_PERP",
# "pair": "ETHUSD",
# "status": "NEW",
# "clientOrderId": "x-xcKtGhcuec11030474204ab08ba2c2",
# "price": "2500",
# "avgPrice": "0",
# "origQty": "1",
# "executedQty": "0",
# "cumBase": "0",
# "timeInForce": "GTC",
# "type": "LIMIT",
# "reduceOnly": False,
# "side": "BUY",
# "positionSide": "LONG",
# "origType": "LIMIT",
# "time": 1707894181694,
# "updateTime": 1707894181694
# }
#
# linear portfolio margin conditional
#
# {
# "newClientStrategyId": "x-xcKtGhcu2205fde44418483ca21874",
# "strategyId": 4084339,
# "strategyStatus": "NEW",
# "strategyType": "STOP",
# "origQty": "0.010",
# "price": "35000",
# "reduceOnly": False,
# "side": "BUY",
# "positionSide": "LONG",
# "stopPrice": "60000",
# "symbol": "BTCUSDT",
# "bookTime": 1707894490094,
# "updateTime": 1707894490094,
# "timeInForce": "GTC",
# "workingType": "CONTRACT_PRICE",
# "priceProtect": False,
# "goodTillDate": 0,
# "selfTradePreventionMode": "NONE"
# }
#
# inverse portfolio margin conditional
#
# {
# "newClientStrategyId": "x-xcKtGhcu2da9c765294b433994ffce",
# "strategyId": 1423501,
# "strategyStatus": "NEW",
# "strategyType": "STOP",
# "origQty": "1",
# "price": "2500",
# "reduceOnly": False,
# "side": "BUY",
# "positionSide": "LONG",
# "stopPrice": "4000",
# "symbol": "ETHUSD_PERP",
# "bookTime": 1707894782679,
# "updateTime": 1707894782679,
# "timeInForce": "GTC",
# "workingType": "CONTRACT_PRICE",
# "priceProtect": False
# }
#
return self.parse_order(response, market)
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://developers.binance.com/docs/binance-spot-api-docs/rest-api/trading-endpoints#all-orders-user_data
https://developers.binance.com/docs/derivatives/usds-margined-futures/trade/rest-api/All-Orders
https://developers.binance.com/docs/derivatives/coin-margined-futures/trade/rest-api/All-Orders
https://developers.binance.com/docs/derivatives/option/trade/Query-Option-Order-History
https://developers.binance.com/docs/margin_trading/trade/Query-Margin-Account-All-Orders
https://developers.binance.com/docs/derivatives/portfolio-margin/trade/Query-All-UM-Orders
https://developers.binance.com/docs/derivatives/portfolio-margin/trade/Query-All-CM-Orders
https://developers.binance.com/docs/derivatives/portfolio-margin/trade/Query-All-UM-Conditional-Orders
https://developers.binance.com/docs/derivatives/portfolio-margin/trade/Query-All-CM-Conditional-Orders
: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 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.portfolioMargin]: set to True if you would like to fetch orders in a portfolio margin account
:param boolean [params.trigger]: set to True if you would like to fetch portfolio margin account trigger or conditional orders
:returns Order[]: a list of `order structures <https://docs.ccxt.com/#/?id=order-structure>`
"""
if symbol is None:
raise ArgumentsRequired(self.id + ' fetchClosedOrders() requires a symbol argument')
orders = self.fetch_orders(symbol, since, None, params)
filteredOrders = self.filter_by(orders, 'status', 'closed')
return self.filter_by_since_limit(filteredOrders, since, limit)
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://developers.binance.com/docs/binance-spot-api-docs/rest-api/trading-endpoints#all-orders-user_data
https://developers.binance.com/docs/derivatives/usds-margined-futures/trade/rest-api/All-Orders
https://developers.binance.com/docs/derivatives/coin-margined-futures/trade/rest-api/All-Orders
https://developers.binance.com/docs/derivatives/option/trade/Query-Option-Order-History
https://developers.binance.com/docs/margin_trading/trade/Query-Margin-Account-All-Orders
https://developers.binance.com/docs/derivatives/portfolio-margin/trade/Query-All-UM-Orders
https://developers.binance.com/docs/derivatives/portfolio-margin/trade/Query-All-CM-Orders
https://developers.binance.com/docs/derivatives/portfolio-margin/trade/Query-All-UM-Conditional-Orders
https://developers.binance.com/docs/derivatives/portfolio-margin/trade/Query-All-CM-Conditional-Orders
:param str symbol: unified market symbol of the market the 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 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.portfolioMargin]: set to True if you would like to fetch orders in a portfolio margin account
:param boolean [params.trigger]: set to True if you would like to fetch portfolio margin account trigger or conditional orders
:returns dict[]: a list of `order structures <https://docs.ccxt.com/#/?id=order-structure>`
"""
if symbol is None:
raise ArgumentsRequired(self.id + ' fetchCanceledOrders() requires a symbol argument')
orders = self.fetch_orders(symbol, since, None, params)
filteredOrders = self.filter_by(orders, 'status', 'canceled')
return self.filter_by_since_limit(filteredOrders, since, limit)
def fetch_canceled_and_closed_orders(self, symbol: Str = None, since: Int = None, limit: Int = None, params={}) -> List[Order]:
"""
fetches information on multiple canceled orders made by the user
https://developers.binance.com/docs/binance-spot-api-docs/rest-api/trading-endpoints#all-orders-user_data
https://developers.binance.com/docs/derivatives/usds-margined-futures/trade/rest-api/All-Orders
https://developers.binance.com/docs/derivatives/coin-margined-futures/trade/rest-api/All-Orders
https://developers.binance.com/docs/derivatives/option/trade/Query-Option-Order-History
https://developers.binance.com/docs/margin_trading/trade/Query-Margin-Account-All-Orders
https://developers.binance.com/docs/derivatives/portfolio-margin/trade/Query-All-UM-Orders
https://developers.binance.com/docs/derivatives/portfolio-margin/trade/Query-All-CM-Orders
https://developers.binance.com/docs/derivatives/portfolio-margin/trade/Query-All-UM-Conditional-Orders
https://developers.binance.com/docs/derivatives/portfolio-margin/trade/Query-All-CM-Conditional-Orders
:param str symbol: unified market symbol of the market the 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 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.portfolioMargin]: set to True if you would like to fetch orders in a portfolio margin account
:param boolean [params.trigger]: set to True if you would like to fetch portfolio margin account trigger or conditional orders
:returns dict[]: a list of `order structures <https://docs.ccxt.com/#/?id=order-structure>`
"""
if symbol is None:
raise ArgumentsRequired(self.id + ' fetchCanceledAndClosedOrders() requires a symbol argument')
orders = self.fetch_orders(symbol, since, None, params)
canceledOrders = self.filter_by(orders, 'status', 'canceled')
closedOrders = self.filter_by(orders, 'status', 'closed')
filteredOrders = self.array_concat(canceledOrders, closedOrders)
sortedOrders = self.sort_by(filteredOrders, 'timestamp')
return self.filter_by_since_limit(sortedOrders, since, limit)
def cancel_order(self, id: str, symbol: Str = None, params={}):
"""
cancels an open order
https://developers.binance.com/docs/binance-spot-api-docs/rest-api/trading-endpoints#cancel-order-trade
https://developers.binance.com/docs/derivatives/usds-margined-futures/trade/rest-api/Cancel-Order
https://developers.binance.com/docs/derivatives/coin-margined-futures/trade/rest-api/Cancel-Order
https://developers.binance.com/docs/derivatives/option/trade/Cancel-Option-Order
https://developers.binance.com/docs/margin_trading/trade/Margin-Account-Cancel-Order
https://developers.binance.com/docs/derivatives/portfolio-margin/trade/Cancel-UM-Order
https://developers.binance.com/docs/derivatives/portfolio-margin/trade/Cancel-CM-Order
https://developers.binance.com/docs/derivatives/portfolio-margin/trade/Cancel-UM-Conditional-Order
https://developers.binance.com/docs/derivatives/portfolio-margin/trade/Cancel-CM-Conditional-Order
https://developers.binance.com/docs/derivatives/portfolio-margin/trade/Cancel-Margin-Account-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 boolean [params.portfolioMargin]: set to True if you would like to cancel an order in a portfolio margin account
:param boolean [params.trigger]: set to True if you would like to cancel a portfolio margin account conditional order
: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')
self.load_markets()
market = self.market(symbol)
defaultType = self.safe_string_2(self.options, 'cancelOrder', 'defaultType', 'spot')
type = self.safe_string(params, 'type', defaultType)
marginMode = None
marginMode, params = self.handle_margin_mode_and_params('cancelOrder', params)
isPortfolioMargin = None
isPortfolioMargin, params = self.handle_option_and_params_2(params, 'cancelOrder', 'papi', 'portfolioMargin', False)
isConditional = self.safe_bool_n(params, ['stop', 'trigger', 'conditional'])
request: dict = {
'symbol': market['id'],
}
clientOrderId = self.safe_string_n(params, ['origClientOrderId', 'clientOrderId', 'newClientStrategyId'])
if clientOrderId is not None:
if market['option']:
request['clientOrderId'] = clientOrderId
else:
if isPortfolioMargin and isConditional:
request['newClientStrategyId'] = clientOrderId
else:
request['origClientOrderId'] = clientOrderId
else:
if isPortfolioMargin and isConditional:
request['strategyId'] = id
else:
request['orderId'] = id
params = self.omit(params, ['type', 'origClientOrderId', 'clientOrderId', 'newClientStrategyId', 'stop', 'trigger', 'conditional'])
response = None
if market['option']:
response = self.eapiPrivateDeleteOrder(self.extend(request, params))
elif market['linear']:
if isPortfolioMargin:
if isConditional:
response = self.papiDeleteUmConditionalOrder(self.extend(request, params))
else:
response = self.papiDeleteUmOrder(self.extend(request, params))
else:
response = self.fapiPrivateDeleteOrder(self.extend(request, params))
elif market['inverse']:
if isPortfolioMargin:
if isConditional:
response = self.papiDeleteCmConditionalOrder(self.extend(request, params))
else:
response = self.papiDeleteCmOrder(self.extend(request, params))
else:
response = self.dapiPrivateDeleteOrder(self.extend(request, params))
elif (type == 'margin') or (marginMode is not None) or isPortfolioMargin:
if isPortfolioMargin:
response = self.papiDeleteMarginOrder(self.extend(request, params))
else:
if marginMode == 'isolated':
request['isIsolated'] = True
response = self.sapiDeleteMarginOrder(self.extend(request, params))
else:
response = self.privateDeleteOrder(self.extend(request, params))
return self.parse_order(response, market)
def cancel_all_orders(self, symbol: Str = None, params={}):
"""
cancel all open orders in a market
https://developers.binance.com/docs/binance-spot-api-docs/rest-api/trading-endpoints#cancel-all-open-orders-on-a-symbol-trade
https://developers.binance.com/docs/derivatives/usds-margined-futures/trade/rest-api/Cancel-All-Open-Orders
https://developers.binance.com/docs/derivatives/coin-margined-futures/trade/rest-api/Cancel-All-Open-Orders
https://developers.binance.com/docs/derivatives/option/trade/Cancel-all-Option-orders-on-specific-symbol
https://developers.binance.com/docs/margin_trading/trade/Margin-Account-Cancel-All-Open-Orders
https://developers.binance.com/docs/derivatives/portfolio-margin/trade/Cancel-All-UM-Open-Orders
https://developers.binance.com/docs/derivatives/portfolio-margin/trade/Cancel-All-UM-Open-Conditional-Orders
https://developers.binance.com/docs/derivatives/portfolio-margin/trade/Cancel-All-CM-Open-Orders
https://developers.binance.com/docs/derivatives/portfolio-margin/trade/Cancel-All-CM-Open-Conditional-Orders
https://developers.binance.com/docs/derivatives/portfolio-margin/trade/Cancel-Margin-Account-All-Open-Orders-on-a-Symbol
:param str symbol: unified market symbol of the market to cancel orders in
:param dict [params]: extra parameters specific to the exchange API endpoint
:param str [params.marginMode]: 'cross' or 'isolated', for spot margin trading
:param boolean [params.portfolioMargin]: set to True if you would like to cancel orders in a portfolio margin account
:param boolean [params.trigger]: set to True if you would like to cancel portfolio margin account conditional orders
: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')
self.load_markets()
market = self.market(symbol)
request: dict = {
'symbol': market['id'],
}
isPortfolioMargin = None
isPortfolioMargin, params = self.handle_option_and_params_2(params, 'cancelAllOrders', 'papi', 'portfolioMargin', False)
isConditional = self.safe_bool_n(params, ['stop', 'trigger', 'conditional'])
type = self.safe_string(params, 'type', market['type'])
params = self.omit(params, ['type', 'stop', 'trigger', 'conditional'])
marginMode = None
marginMode, params = self.handle_margin_mode_and_params('cancelAllOrders', params)
response = None
if market['option']:
response = self.eapiPrivateDeleteAllOpenOrders(self.extend(request, params))
#
# {
# "code": 0,
# "msg": "success"
# }
#
elif market['linear']:
if isPortfolioMargin:
if isConditional:
response = self.papiDeleteUmConditionalAllOpenOrders(self.extend(request, params))
#
# {
# "code": "200",
# "msg": "The operation of cancel all conditional open order is done."
# }
#
else:
response = self.papiDeleteUmAllOpenOrders(self.extend(request, params))
#
# {
# "code": 200,
# "msg": "The operation of cancel all open order is done."
# }
#
else:
response = self.fapiPrivateDeleteAllOpenOrders(self.extend(request, params))
#
# {
# "code": 200,
# "msg": "The operation of cancel all open order is done."
# }
#
elif market['inverse']:
if isPortfolioMargin:
if isConditional:
response = self.papiDeleteCmConditionalAllOpenOrders(self.extend(request, params))
#
# {
# "code": "200",
# "msg": "The operation of cancel all conditional open order is done."
# }
#
else:
response = self.papiDeleteCmAllOpenOrders(self.extend(request, params))
#
# {
# "code": 200,
# "msg": "The operation of cancel all open order is done."
# }
#
else:
response = self.dapiPrivateDeleteAllOpenOrders(self.extend(request, params))
#
# {
# "code": 200,
# "msg": "The operation of cancel all open order is done."
# }
#
elif (type == 'margin') or (marginMode is not None) or isPortfolioMargin:
if isPortfolioMargin:
response = self.papiDeleteMarginAllOpenOrders(self.extend(request, params))
else:
if marginMode == 'isolated':
request['isIsolated'] = True
response = self.sapiDeleteMarginOpenOrders(self.extend(request, params))
#
# [
# {
# "symbol": "BTCUSDT",
# "isIsolated": True, # if isolated margin
# "origClientOrderId": "E6APeyTJvkMvLMYMqu1KQ4",
# "orderId": 11,
# "orderListId": -1,
# "clientOrderId": "pXLV6Hz6mprAcVYpVMTGgx",
# "price": "0.089853",
# "origQty": "0.178622",
# "executedQty": "0.000000",
# "cummulativeQuoteQty": "0.000000",
# "status": "CANCELED",
# "timeInForce": "GTC",
# "type": "LIMIT",
# "side": "BUY",
# "selfTradePreventionMode": "NONE"
# },
# ...
# ]
#
else:
response = self.privateDeleteOpenOrders(self.extend(request, params))
#
# [
# {
# "symbol": "ADAUSDT",
# "origClientOrderId": "x-TKT5PX2F662cde7a90114475b86e21",
# "orderId": 3935107,
# "orderListId": -1,
# "clientOrderId": "bqM2w1oTlugfRAjnTIFBE8",
# "transactTime": 1720589016657,
# "price": "0.35000000",
# "origQty": "30.00000000",
# "executedQty": "0.00000000",
# "cummulativeQuoteQty": "0.00000000",
# "status": "CANCELED",
# "timeInForce": "GTC",
# "type": "LIMIT",
# "side": "BUY",
# "selfTradePreventionMode": "EXPIRE_MAKER"
# }
# ]
#
if isinstance(response, list):
return self.parse_orders(response, market)
else:
return [
self.safe_order({
'info': response,
}),
]
def cancel_orders(self, ids: List[str], symbol: Str = None, params={}):
"""
cancel multiple orders
https://developers.binance.com/docs/derivatives/usds-margined-futures/trade/rest-api/Cancel-Multiple-Orders
https://developers.binance.com/docs/derivatives/coin-margined-futures/trade/rest-api/Cancel-Multiple-Orders
:param str[] ids: order ids
:param str [symbol]: unified market symbol
:param dict [params]: extra parameters specific to the exchange API endpoint
:param str[] [params.clientOrderIds]: alternative to ids, array of client order ids
EXCHANGE SPECIFIC PARAMETERS
:param str[] [params.origClientOrderIdList]: max length 10 e.g. ["my_id_1","my_id_2"], encode the double quotes. No space after comma
:param int[] [params.recvWindow]:
:returns dict: an list of `order structures <https://docs.ccxt.com/#/?id=order-structure>`
"""
if symbol is None:
raise ArgumentsRequired(self.id + ' cancelOrders() requires a symbol argument')
self.load_markets()
market = self.market(symbol)
if not market['contract']:
raise BadRequest(self.id + ' cancelOrders is only supported for swap markets.')
request: dict = {
'symbol': market['id'],
# 'orderidlist': ids,
}
origClientOrderIdList = self.safe_list_2(params, 'origClientOrderIdList', 'clientOrderIds')
if origClientOrderIdList is not None:
params = self.omit(params, ['clientOrderIds'])
request['origClientOrderIdList'] = origClientOrderIdList
else:
request['orderidlist'] = ids
response = None
if market['linear']:
response = self.fapiPrivateDeleteBatchOrders(self.extend(request, params))
elif market['inverse']:
response = self.dapiPrivateDeleteBatchOrders(self.extend(request, params))
#
# [
# {
# "clientOrderId": "myOrder1",
# "cumQty": "0",
# "cumQuote": "0",
# "executedQty": "0",
# "orderId": 283194212,
# "origQty": "11",
# "origType": "TRAILING_STOP_MARKET",
# "price": "0",
# "reduceOnly": False,
# "side": "BUY",
# "positionSide": "SHORT",
# "status": "CANCELED",
# "stopPrice": "9300", # please ignore when order type is TRAILING_STOP_MARKET
# "closePosition": False, # if Close-All
# "symbol": "BTCUSDT",
# "timeInForce": "GTC",
# "type": "TRAILING_STOP_MARKET",
# "activatePrice": "9020", # activation price, only return with TRAILING_STOP_MARKET order
# "priceRate": "0.3", # callback rate, only return with TRAILING_STOP_MARKET order
# "updateTime": 1571110484038,
# "workingType": "CONTRACT_PRICE",
# "priceProtect": False, # if conditional order trigger is protected
# "priceMatch": "NONE", # price match mode
# "selfTradePreventionMode": "NONE", # self trading preventation mode
# "goodTillDate": 0 # order pre-set auot cancel time for TIF GTD order
# },
# {
# "code": -2011,
# "msg": "Unknown order sent."
# }
# ]
#
return self.parse_orders(response, market)
def fetch_order_trades(self, id: str, symbol: Str = None, since: Int = None, limit: Int = None, params={}):
"""
fetch all the trades made from a single order
https://developers.binance.com/docs/binance-spot-api-docs/rest-api/account-endpoints#account-trade-list-user_data
https://developers.binance.com/docs/derivatives/usds-margined-futures/trade/rest-api/Account-Trade-List
https://developers.binance.com/docs/derivatives/coin-margined-futures/trade/rest-api/Account-Trade-List
https://developers.binance.com/docs/margin_trading/trade/Query-Margin-Account-Trade-List
:param str id: order id
: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 to retrieve
:param dict [params]: extra parameters specific to the exchange API endpoint
:returns dict[]: a list of `trade structures <https://docs.ccxt.com/#/?id=trade-structure>`
"""
if symbol is None:
raise ArgumentsRequired(self.id + ' fetchOrderTrades() requires a symbol argument')
self.load_markets()
market = self.market(symbol)
type = self.safe_string(params, 'type', market['type'])
params = self.omit(params, 'type')
if type != 'spot':
raise NotSupported(self.id + ' fetchOrderTrades() supports spot markets only')
request: dict = {
'orderId': id,
}
return self.fetch_my_trades(symbol, since, limit, self.extend(request, params))
def fetch_my_trades(self, symbol: Str = None, since: Int = None, limit: Int = None, params={}):
"""
fetch all trades made by the user
https://developers.binance.com/docs/binance-spot-api-docs/rest-api/account-endpoints#account-trade-list-user_data
https://developers.binance.com/docs/derivatives/usds-margined-futures/trade/rest-api/Account-Trade-List
https://developers.binance.com/docs/derivatives/coin-margined-futures/trade/rest-api/Account-Trade-List
https://developers.binance.com/docs/margin_trading/trade/Query-Margin-Account-Trade-List
https://developers.binance.com/docs/derivatives/option/trade/Account-Trade-List
https://developers.binance.com/docs/derivatives/portfolio-margin/trade/UM-Account-Trade-List
https://developers.binance.com/docs/derivatives/portfolio-margin/trade/CM-Account-Trade-List
: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 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 int [params.until]: the latest time in ms to fetch entries for
:param boolean [params.portfolioMargin]: set to True if you would like to fetch trades for a portfolio margin account
:returns Trade[]: a list of `trade structures <https://docs.ccxt.com/#/?id=trade-structure>`
"""
self.load_markets()
paginate = False
paginate, params = self.handle_option_and_params(params, 'fetchMyTrades', 'paginate')
if paginate:
return self.fetch_paginated_call_dynamic('fetchMyTrades', symbol, since, limit, params)
request: dict = {}
market = None
type = None
marginMode = None
if symbol is not None:
market = self.market(symbol)
request['symbol'] = market['id']
type, params = self.handle_market_type_and_params('fetchMyTrades', market, params)
endTime = self.safe_integer_2(params, 'until', 'endTime')
if since is not None:
startTime = since
request['startTime'] = startTime
# If startTime and endTime are both not sent, then the last 7 days' data will be returned.
# The time between startTime and endTime cannot be longer than 7 days.
# The parameter fromId cannot be sent with startTime or endTime.
currentTimestamp = self.milliseconds()
oneWeek = 7 * 24 * 60 * 60 * 1000
if (currentTimestamp - startTime) >= oneWeek:
if (endTime is None) and market['linear']:
endTime = self.sum(startTime, oneWeek)
endTime = min(endTime, currentTimestamp)
if endTime is not None:
request['endTime'] = endTime
params = self.omit(params, ['endTime', 'until'])
if limit is not None:
if (type == 'option') or market['contract']:
limit = min(limit, 1000) # above 1000, returns error
request['limit'] = limit
response = None
if type == 'option':
response = self.eapiPrivateGetUserTrades(self.extend(request, params))
else:
if symbol is None:
raise ArgumentsRequired(self.id + ' fetchMyTrades() requires a symbol argument')
marginMode, params = self.handle_margin_mode_and_params('fetchMyTrades', params)
isPortfolioMargin = None
isPortfolioMargin, params = self.handle_option_and_params_2(params, 'fetchMyTrades', 'papi', 'portfolioMargin', False)
if type == 'spot' or type == 'margin':
if isPortfolioMargin:
response = self.papiGetMarginMyTrades(self.extend(request, params))
elif (type == 'margin') or (marginMode is not None):
if marginMode == 'isolated':
request['isIsolated'] = True
response = self.sapiGetMarginMyTrades(self.extend(request, params))
else:
response = self.privateGetMyTrades(self.extend(request, params))
elif market['linear']:
if isPortfolioMargin:
response = self.papiGetUmUserTrades(self.extend(request, params))
else:
response = self.fapiPrivateGetUserTrades(self.extend(request, params))
elif market['inverse']:
if isPortfolioMargin:
response = self.papiGetCmUserTrades(self.extend(request, params))
else:
response = self.dapiPrivateGetUserTrades(self.extend(request, params))
#
# spot trade
#
# [
# {
# "symbol": "BNBBTC",
# "id": 28457,
# "orderId": 100234,
# "price": "4.00000100",
# "qty": "12.00000000",
# "commission": "10.10000000",
# "commissionAsset": "BNB",
# "time": 1499865549590,
# "isBuyer": True,
# "isMaker": False,
# "isBestMatch": True,
# }
# ]
#
# futures trade
#
# [
# {
# "accountId": 20,
# "buyer": False,
# "commission": "-0.07819010",
# "commissionAsset": "USDT",
# "counterPartyId": 653,
# "id": 698759,
# "maker": False,
# "orderId": 25851813,
# "price": "7819.01",
# "qty": "0.002",
# "quoteQty": "0.01563",
# "realizedPnl": "-0.91539999",
# "side": "SELL",
# "symbol": "BTCUSDT",
# "time": 1569514978020
# }
# ]
#
# options(eapi)
#
# [
# {
# "id": 1125899906844226012,
# "tradeId": 73,
# "orderId": 4638761100843040768,
# "symbol": "ETH-230211-1500-C",
# "price": "18.70000000",
# "quantity": "-0.57000000",
# "fee": "0.17305890",
# "realizedProfit": "-3.53400000",
# "side": "SELL",
# "type": "LIMIT",
# "volatility": "0.30000000",
# "liquidity": "MAKER",
# "time": 1676085216845,
# "priceScale": 1,
# "quantityScale": 2,
# "optionSide": "CALL",
# "quoteAsset": "USDT"
# }
# ]
#
# linear portfolio margin
#
# [
# {
# "symbol": "BTCUSDT",
# "id": 4575108247,
# "orderId": 261942655610,
# "side": "SELL",
# "price": "47263.40",
# "qty": "0.010",
# "realizedPnl": "27.38400000",
# "marginAsset": "USDT",
# "quoteQty": "472.63",
# "commission": "0.18905360",
# "commissionAsset": "USDT",
# "time": 1707530039409,
# "buyer": False,
# "maker": False,
# "positionSide": "LONG"
# }
# ]
#
# inverse portfolio margin
#
# [
# {
# "symbol": "ETHUSD_PERP",
# "id": 701907838,
# "orderId": 71548909034,
# "pair": "ETHUSD",
# "side": "SELL",
# "price": "2498.15",
# "qty": "1",
# "realizedPnl": "0.00012517",
# "marginAsset": "ETH",
# "baseQty": "0.00400296",
# "commission": "0.00000160",
# "commissionAsset": "ETH",
# "time": 1707530317519,
# "positionSide": "LONG",
# "buyer": False,
# "maker": False
# }
# ]
#
# spot margin portfolio margin
#
# [
# {
# "symbol": "ADAUSDT",
# "id": 470227543,
# "orderId": 4421170947,
# "price": "0.53880000",
# "qty": "10.00000000",
# "quoteQty": "5.38800000",
# "commission": "0.00538800",
# "commissionAsset": "USDT",
# "time": 1707545780522,
# "isBuyer": False,
# "isMaker": False,
# "isBestMatch": True
# }
# ]
#
return self.parse_trades(response, market, since, limit)
def fetch_my_dust_trades(self, symbol: Str = None, since: Int = None, limit: Int = None, params={}):
"""
fetch all dust trades made by the user
https://developers.binance.com/docs/wallet/asset/dust-log
:param str symbol: not used by binance fetchMyDustTrades()
:param int [since]: the earliest time in ms to fetch my dust trades for
:param int [limit]: the maximum number of dust trades to retrieve
:param dict [params]: extra parameters specific to the exchange API endpoint
:param str [params.type]: 'spot' or 'margin', default spot
:returns dict[]: a list of `trade structures <https://docs.ccxt.com/#/?id=trade-structure>`
"""
#
# Binance provides an opportunity to trade insignificant(i.e. non-tradable and non-withdrawable)
# token leftovers(of any asset) into `BNB` coin which in turn can be used to pay trading fees with it.
# The corresponding trades history is called the `Dust Log` and can be requested via the following end-point:
# https://github.com/binance-exchange/binance-official-api-docs/blob/master/wapi-api.md#dustlog-user_data
#
self.load_markets()
request: dict = {}
if since is not None:
request['startTime'] = since
request['endTime'] = self.sum(since, 7776000000)
accountType = self.safe_string_upper(params, 'type')
params = self.omit(params, 'type')
if accountType is not None:
request['accountType'] = accountType
response = self.sapiGetAssetDribblet(self.extend(request, params))
# {
# "total": "4",
# "userAssetDribblets": [
# {
# "operateTime": "1627575731000",
# "totalServiceChargeAmount": "0.00001453",
# "totalTransferedAmount": "0.00072693",
# "transId": "70899815863",
# "userAssetDribbletDetails": [
# {
# "fromAsset": "LTC",
# "amount": "0.000006",
# "transferedAmount": "0.00000267",
# "serviceChargeAmount": "0.00000005",
# "operateTime": "1627575731000",
# "transId": "70899815863"
# },
# {
# "fromAsset": "GBP",
# "amount": "0.15949157",
# "transferedAmount": "0.00072426",
# "serviceChargeAmount": "0.00001448",
# "operateTime": "1627575731000",
# "transId": "70899815863"
# }
# ]
# },
# ]
# }
results = self.safe_list(response, 'userAssetDribblets', [])
rows = self.safe_integer(response, 'total', 0)
data = []
for i in range(0, rows):
logs = self.safe_list(results[i], 'userAssetDribbletDetails', [])
for j in range(0, len(logs)):
logs[j]['isDustTrade'] = True
data.append(logs[j])
trades = self.parse_trades(data, None, since, limit)
return self.filter_by_since_limit(trades, since, limit)
def parse_dust_trade(self, trade, market: Market = None):
#
# {
# "fromAsset": "USDT",
# "amount": "0.009669",
# "transferedAmount": "0.00002992",
# "serviceChargeAmount": "0.00000059",
# "operateTime": "1628076010000",
# "transId": "71416578712",
# "isDustTrade": True
# }
#
orderId = self.safe_string(trade, 'transId')
timestamp = self.safe_integer(trade, 'operateTime')
currencyId = self.safe_string(trade, 'fromAsset')
tradedCurrency = self.safe_currency_code(currencyId)
bnb = self.currency('BNB')
earnedCurrency = bnb['code']
applicantSymbol = earnedCurrency + '/' + tradedCurrency
tradedCurrencyIsQuote = False
if applicantSymbol in self.markets:
tradedCurrencyIsQuote = True
feeCostString = self.safe_string(trade, 'serviceChargeAmount')
fee = {
'currency': earnedCurrency,
'cost': self.parse_number(feeCostString),
}
symbol = None
amountString = None
costString = None
side = None
if tradedCurrencyIsQuote:
symbol = applicantSymbol
amountString = self.safe_string(trade, 'transferedAmount')
costString = self.safe_string(trade, 'amount')
side = 'buy'
else:
symbol = tradedCurrency + '/' + earnedCurrency
amountString = self.safe_string(trade, 'amount')
costString = self.safe_string(trade, 'transferedAmount')
side = 'sell'
priceString = None
if costString is not None:
if amountString:
priceString = Precise.string_div(costString, amountString)
id = None
amount = self.parse_number(amountString)
price = self.parse_number(priceString)
cost = self.parse_number(costString)
type = None
takerOrMaker = None
return {
'id': id,
'timestamp': timestamp,
'datetime': self.iso8601(timestamp),
'symbol': symbol,
'order': orderId,
'type': type,
'takerOrMaker': takerOrMaker,
'side': side,
'amount': amount,
'price': price,
'cost': cost,
'fee': fee,
'info': trade,
}
def fetch_deposits(self, code: Str = None, since: Int = None, limit: Int = None, params={}) -> List[Transaction]:
"""
fetch all deposits made to an account
https://developers.binance.com/docs/wallet/capital/deposite-history
https://developers.binance.com/docs/fiat/rest-api/Get-Fiat-Deposit-Withdraw-History
: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 bool [params.fiat]: if True, only fiat deposits will be returned
:param int [params.until]: the latest time in ms to fetch entries 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 `transaction structures <https://docs.ccxt.com/#/?id=transaction-structure>`
"""
self.load_markets()
paginate = False
paginate, params = self.handle_option_and_params(params, 'fetchDeposits', 'paginate')
if paginate:
return self.fetch_paginated_call_dynamic('fetchDeposits', code, since, limit, params)
currency = None
response = None
request: dict = {}
legalMoney = self.safe_dict(self.options, 'legalMoney', {})
fiatOnly = self.safe_bool(params, 'fiat', False)
params = self.omit(params, 'fiatOnly')
until = self.safe_integer(params, 'until')
params = self.omit(params, 'until')
if fiatOnly or (code in legalMoney):
if code is not None:
currency = self.currency(code)
request['transactionType'] = 0
if since is not None:
request['beginTime'] = since
if until is not None:
request['endTime'] = until
raw = self.sapiGetFiatOrders(self.extend(request, params))
response = self.safe_list(raw, 'data', [])
# {
# "code": "000000",
# "message": "success",
# "data": [
# {
# "orderNo": "25ced37075c1470ba8939d0df2316e23",
# "fiatCurrency": "EUR",
# "indicatedAmount": "15.00",
# "amount": "15.00",
# "totalFee": "0.00",
# "method": "card",
# "status": "Failed",
# "createTime": 1627501026000,
# "updateTime": 1627501027000
# }
# ],
# "total": 1,
# "success": True
# }
else:
if code is not None:
currency = self.currency(code)
request['coin'] = currency['id']
if since is not None:
request['startTime'] = since
# max 3 months range https://github.com/ccxt/ccxt/issues/6495
endTime = self.sum(since, 7776000000)
if until is not None:
endTime = min(endTime, until)
request['endTime'] = endTime
if limit is not None:
request['limit'] = limit
response = self.sapiGetCapitalDepositHisrec(self.extend(request, params))
# [
# {
# "amount": "0.01844487",
# "coin": "BCH",
# "network": "BCH",
# "status": 1,
# "address": "1NYxAJhW2281HK1KtJeaENBqHeygA88FzR",
# "addressTag": "",
# "txId": "bafc5902504d6504a00b7d0306a41154cbf1d1b767ab70f3bc226327362588af",
# "insertTime": 1610784980000,
# "transferType": 0,
# "confirmTimes": "2/2"
# },
# {
# "amount": "4500",
# "coin": "USDT",
# "network": "BSC",
# "status": 1,
# "address": "0xc9c923c87347ca0f3451d6d308ce84f691b9f501",
# "addressTag": "",
# "txId": "Internal transfer 51376627901",
# "insertTime": 1618394381000,
# "transferType": 1,
# "confirmTimes": "1/15"
# }
# ]
for i in range(0, len(response)):
response[i]['type'] = 'deposit'
return self.parse_transactions(response, currency, since, limit)
def fetch_withdrawals(self, code: Str = None, since: Int = None, limit: Int = None, params={}) -> List[Transaction]:
"""
fetch all withdrawals made from an account
https://developers.binance.com/docs/wallet/capital/withdraw-history
https://developers.binance.com/docs/fiat/rest-api/Get-Fiat-Deposit-Withdraw-History
: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 bool [params.fiat]: if True, only fiat withdrawals will be returned
:param int [params.until]: the latest time in ms to fetch withdrawals 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 `transaction structures <https://docs.ccxt.com/#/?id=transaction-structure>`
"""
self.load_markets()
paginate = False
paginate, params = self.handle_option_and_params(params, 'fetchWithdrawals', 'paginate')
if paginate:
return self.fetch_paginated_call_dynamic('fetchWithdrawals', code, since, limit, params)
legalMoney = self.safe_dict(self.options, 'legalMoney', {})
fiatOnly = self.safe_bool(params, 'fiat', False)
params = self.omit(params, 'fiatOnly')
request: dict = {}
until = self.safe_integer(params, 'until')
if until is not None:
params = self.omit(params, 'until')
request['endTime'] = until
response = None
currency = None
if fiatOnly or (code in legalMoney):
if code is not None:
currency = self.currency(code)
request['transactionType'] = 1
if since is not None:
request['beginTime'] = since
raw = self.sapiGetFiatOrders(self.extend(request, params))
response = self.safe_list(raw, 'data', [])
# {
# "code": "000000",
# "message": "success",
# "data": [
# {
# "orderNo": "CJW706452266115170304",
# "fiatCurrency": "GBP",
# "indicatedAmount": "10001.50",
# "amount": "100.00",
# "totalFee": "1.50",
# "method": "bank transfer",
# "status": "Successful",
# "createTime": 1620037745000,
# "updateTime": 1620038480000
# },
# {
# "orderNo": "CJW706287492781891584",
# "fiatCurrency": "GBP",
# "indicatedAmount": "10001.50",
# "amount": "100.00",
# "totalFee": "1.50",
# "method": "bank transfer",
# "status": "Successful",
# "createTime": 1619998460000,
# "updateTime": 1619998823000
# }
# ],
# "total": 39,
# "success": True
# }
else:
if code is not None:
currency = self.currency(code)
request['coin'] = currency['id']
if since is not None:
request['startTime'] = since
# max 3 months range https://github.com/ccxt/ccxt/issues/6495
request['endTime'] = self.sum(since, 7776000000)
if limit is not None:
request['limit'] = limit
response = self.sapiGetCapitalWithdrawHistory(self.extend(request, params))
# [
# {
# "id": "69e53ad305124b96b43668ceab158a18",
# "amount": "28.75",
# "transactionFee": "0.25",
# "coin": "XRP",
# "status": 6,
# "address": "r3T75fuLjX51mmfb5Sk1kMNuhBgBPJsjza",
# "addressTag": "101286922",
# "txId": "19A5B24ED0B697E4F0E9CD09FCB007170A605BC93C9280B9E6379C5E6EF0F65A",
# "applyTime": "2021-04-15 12:09:16",
# "network": "XRP",
# "transferType": 0
# },
# {
# "id": "9a67628b16ba4988ae20d329333f16bc",
# "amount": "20",
# "transactionFee": "20",
# "coin": "USDT",
# "status": 6,
# "address": "0x0AB991497116f7F5532a4c2f4f7B1784488628e1",
# "txId": "0x77fbf2cf2c85b552f0fd31fd2e56dc95c08adae031d96f3717d8b17e1aea3e46",
# "applyTime": "2021-04-15 12:06:53",
# "network": "ETH",
# "transferType": 0
# },
# {
# "id": "a7cdc0afbfa44a48bd225c9ece958fe2",
# "amount": "51",
# "transactionFee": "1",
# "coin": "USDT",
# "status": 6,
# "address": "TYDmtuWL8bsyjvcauUTerpfYyVhFtBjqyo",
# "txId": "168a75112bce6ceb4823c66726ad47620ad332e69fe92d9cb8ceb76023f9a028",
# "applyTime": "2021-04-13 12:46:59",
# "network": "TRX",
# "transferType": 0
# }
# ]
for i in range(0, len(response)):
response[i]['type'] = 'withdrawal'
return self.parse_transactions(response, currency, since, limit)
def parse_transaction_status_by_type(self, status, type=None):
if type is None:
return status
statusesByType: dict = {
'deposit': {
'0': 'pending',
'1': 'ok',
'6': 'ok',
# Fiat
# Processing, Failed, Successful, Finished, Refunding, Refunded, Refund Failed, Order Partial credit Stopped
'Processing': 'pending',
'Failed': 'failed',
'Successful': 'ok',
'Refunding': 'canceled',
'Refunded': 'canceled',
'Refund Failed': 'failed',
},
'withdrawal': {
'0': 'pending', # Email Sent
'1': 'canceled', # Cancelled(different from 1 = ok in deposits)
'2': 'pending', # Awaiting Approval
'3': 'failed', # Rejected
'4': 'pending', # Processing
'5': 'failed', # Failure
'6': 'ok', # Completed
# Fiat
# Processing, Failed, Successful, Finished, Refunding, Refunded, Refund Failed, Order Partial credit Stopped
'Processing': 'pending',
'Failed': 'failed',
'Successful': 'ok',
'Refunding': 'canceled',
'Refunded': 'canceled',
'Refund Failed': 'failed',
},
}
statuses = self.safe_dict(statusesByType, type, {})
return self.safe_string(statuses, status, status)
def parse_transaction(self, transaction: dict, currency: Currency = None) -> Transaction:
#
# fetchDeposits
#
# {
# "amount": "4500",
# "coin": "USDT",
# "network": "BSC",
# "status": 1,
# "address": "0xc9c923c87347ca0f3451d6d308ce84f691b9f501",
# "addressTag": "",
# "txId": "Internal transfer 51376627901",
# "insertTime": 1618394381000,
# "transferType": 1,
# "confirmTimes": "1/15"
# }
#
# fetchWithdrawals
#
# {
# "id": "69e53ad305124b96b43668ceab158a18",
# "amount": "28.75",
# "transactionFee": "0.25",
# "coin": "XRP",
# "status": 6,
# "address": "r3T75fuLjX51mmfb5Sk1kMNuhBgBPJsjza",
# "addressTag": "101286922",
# "txId": "19A5B24ED0B697E4F0E9CD09FCB007170A605BC93C9280B9E6379C5E6EF0F65A",
# "applyTime": "2021-04-15 12:09:16",
# "network": "XRP",
# "transferType": 0
# }
#
# fiat transaction
# withdraw
# {
# "orderNo": "CJW684897551397171200",
# "fiatCurrency": "GBP",
# "indicatedAmount": "29.99",
# "amount": "28.49",
# "totalFee": "1.50",
# "method": "bank transfer",
# "status": "Successful",
# "createTime": 1614898701000,
# "updateTime": 1614898820000
# }
#
# deposit
# {
# "orderNo": "25ced37075c1470ba8939d0df2316e23",
# "fiatCurrency": "EUR",
# "transactionType": 0,
# "indicatedAmount": "15.00",
# "amount": "15.00",
# "totalFee": "0.00",
# "method": "card",
# "status": "Failed",
# "createTime": "1627501026000",
# "updateTime": "1627501027000"
# }
#
# withdraw
#
# {id: "9a67628b16ba4988ae20d329333f16bc"}
#
id = self.safe_string_2(transaction, 'id', 'orderNo')
address = self.safe_string(transaction, 'address')
tag = self.safe_string(transaction, 'addressTag') # set but unused
if tag is not None:
if len(tag) < 1:
tag = None
txid = self.safe_string(transaction, 'txId')
if (txid is not None) and (txid.find('Internal transfer ') >= 0):
txid = txid[18:]
currencyId = self.safe_string_2(transaction, 'coin', 'fiatCurrency')
code = self.safe_currency_code(currencyId, currency)
timestamp = None
timestamp = self.safe_integer_2(transaction, 'insertTime', 'createTime')
if timestamp is None:
timestamp = self.parse8601(self.safe_string(transaction, 'applyTime'))
updated = self.safe_integer_2(transaction, 'successTime', 'updateTime')
type = self.safe_string(transaction, 'type')
if type is None:
txType = self.safe_string(transaction, 'transactionType')
if txType is not None:
type = 'deposit' if (txType == '0') else 'withdrawal'
legalMoneyCurrenciesById = self.safe_dict(self.options, 'legalMoneyCurrenciesById')
code = self.safe_string(legalMoneyCurrenciesById, code, code)
status = self.parse_transaction_status_by_type(self.safe_string(transaction, 'status'), type)
amount = self.safe_number(transaction, 'amount')
feeCost = self.safe_number_2(transaction, 'transactionFee', 'totalFee')
fee = None
if feeCost is not None:
fee = {'currency': code, 'cost': feeCost}
internalInteger = self.safe_integer(transaction, 'transferType')
internal = None
if internalInteger is not None:
internal = True if (internalInteger != 0) else False
network = self.safe_string(transaction, 'network')
return {
'info': transaction,
'id': id,
'txid': txid,
'timestamp': timestamp,
'datetime': self.iso8601(timestamp),
'network': network,
'address': address,
'addressTo': address,
'addressFrom': None,
'tag': tag,
'tagTo': tag,
'tagFrom': None,
'type': type,
'amount': amount,
'currency': code,
'status': status,
'updated': updated,
'internal': internal,
'comment': None,
'fee': fee,
}
def parse_transfer_status(self, status: Str) -> Str:
statuses: dict = {
'CONFIRMED': 'ok',
}
return self.safe_string(statuses, status, status)
def parse_transfer(self, transfer: dict, currency: Currency = None) -> TransferEntry:
#
# transfer
#
# {
# "tranId":13526853623
# }
#
# fetchTransfers
#
# {
# "timestamp": 1614640878000,
# "asset": "USDT",
# "amount": "25",
# "type": "MAIN_UMFUTURE",
# "status": "CONFIRMED",
# "tranId": 43000126248
# }
#
# {
# "orderType": "C2C", # EnumPAY(C2B Merchant Acquiring Payment), PAY_REFUND(C2B Merchant Acquiring Payment,refund), C2C(C2C Transfer Payment),CRYPTO_BOX(Crypto box), CRYPTO_BOX_RF(Crypto Box, refund), C2C_HOLDING(Transfer to new Binance user), C2C_HOLDING_RF(Transfer to new Binance user,refund), PAYOUT(B2C Disbursement Payment), REMITTANCESend cash)
# "transactionId": "M_P_71505104267788288",
# "transactionTime": 1610090460133, #trade timestamp
# "amount": "23.72469206", #order amount(up to 8 decimal places), positive is income, negative is expenditure
# "currency": "BNB",
# "walletType": 1, #main wallet type, 1 for funding wallet, 2 for spot wallet, 3 for fiat wallet, 4 or 6 for card payment, 5 for earn wallet
# "walletTypes": [1,2], #array formatthere are multiple values when using combination payment
# "fundsDetail": [ # details
# {
# "currency": "USDT", #asset
# "amount": "1.2",
# "walletAssetCost":[ #details of asset cost per wallet
# {"1":"0.6"},
# {"2":"0.6"}
# ]
# },
# {
# "currency": "ETH",
# "amount": "0.0001",
# "walletAssetCost":[
# {"1":"0.00005"},
# {"2":"0.00005"}
# ]
# }
# ],
# "payerInfo":{
# "name":"Jack", #nickname or merchant name
# "type":"USER", #account typeUSER for personalMERCHANT for merchant
# "binanceId":"12345678", #binance uid
# "accountId":"67736251" #binance pay id
# },
# "receiverInfo":{
# "name":"Alan", #nickname or merchant name
# "type":"MERCHANT", #account typeUSER for personalMERCHANT for merchant
# "email":"alan@binance.com", #email
# "binanceId":"34355667", #binance uid
# "accountId":"21326891", #binance pay id
# "countryCode":"1", #International area code
# "phoneNumber":"8057651210",
# "mobileCode":"US", #country code
# "extend":[ #extension field
# "institutionName": "",
# "cardNumber": "",
# "digitalWalletId": ""
# ]
# }
# }
id = self.safe_string_2(transfer, 'tranId', 'transactionId')
currencyId = self.safe_string_2(transfer, 'asset', 'currency')
code = self.safe_currency_code(currencyId, currency)
amount = self.safe_number(transfer, 'amount')
type = self.safe_string(transfer, 'type')
fromAccount = None
toAccount = None
accountsById = self.safe_dict(self.options, 'accountsById', {})
if type is not None:
parts = type.split('_')
fromAccount = self.safe_value(parts, 0)
toAccount = self.safe_value(parts, 1)
fromAccount = self.safe_string(accountsById, fromAccount, fromAccount)
toAccount = self.safe_string(accountsById, toAccount, toAccount)
walletType = self.safe_integer(transfer, 'walletType')
if walletType is not None:
payer = self.safe_dict(transfer, 'payerInfo', {})
receiver = self.safe_dict(transfer, 'receiverInfo', {})
fromAccount = self.safe_string(payer, 'accountId')
toAccount = self.safe_string(receiver, 'accountId')
timestamp = self.safe_integer_2(transfer, 'timestamp', 'transactionTime')
status = self.parse_transfer_status(self.safe_string(transfer, 'status'))
return {
'info': transfer,
'id': id,
'timestamp': timestamp,
'datetime': self.iso8601(timestamp),
'currency': code,
'amount': amount,
'fromAccount': fromAccount,
'toAccount': toAccount,
'status': status,
}
def parse_income(self, income, market: Market = None):
#
# {
# "symbol": "ETHUSDT",
# "incomeType": "FUNDING_FEE",
# "income": "0.00134317",
# "asset": "USDT",
# "time": "1621584000000",
# "info": "FUNDING_FEE",
# "tranId": "4480321991774044580",
# "tradeId": ""
# }
#
marketId = self.safe_string(income, 'symbol')
currencyId = self.safe_string(income, 'asset')
timestamp = self.safe_integer(income, 'time')
return {
'info': income,
'symbol': self.safe_symbol(marketId, market, None, 'swap'),
'code': self.safe_currency_code(currencyId),
'timestamp': timestamp,
'datetime': self.iso8601(timestamp),
'id': self.safe_string(income, 'tranId'),
'amount': self.safe_number(income, 'income'),
}
def transfer(self, code: str, amount: float, fromAccount: str, toAccount: str, params={}) -> TransferEntry:
"""
transfer currency internally between wallets on the same account
https://developers.binance.com/docs/wallet/asset/user-universal-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.type]: exchange specific transfer type
:param str [params.symbol]: the unified symbol, required for isolated margin transfers
:returns dict: a `transfer structure <https://docs.ccxt.com/#/?id=transfer-structure>`
"""
self.load_markets()
currency = self.currency(code)
request: dict = {
'asset': currency['id'],
'amount': self.currency_to_precision(code, amount),
}
request['type'] = self.safe_string(params, 'type')
params = self.omit(params, 'type')
if request['type'] is None:
symbol = self.safe_string(params, 'symbol')
market = None
if symbol is not None:
market = self.market(symbol)
params = self.omit(params, 'symbol')
fromId = self.convert_type_to_account(fromAccount).upper()
toId = self.convert_type_to_account(toAccount).upper()
isolatedSymbol = None
if market is not None:
isolatedSymbol = market['id']
if fromId == 'ISOLATED':
if symbol is None:
raise ArgumentsRequired(self.id + ' transfer() requires params["symbol"] when fromAccount is ' + fromAccount)
if toId == 'ISOLATED':
if symbol is None:
raise ArgumentsRequired(self.id + ' transfer() requires params["symbol"] when toAccount is ' + toAccount)
accountsById = self.safe_dict(self.options, 'accountsById', {})
fromIsolated = not (fromId in accountsById)
toIsolated = not (toId in accountsById)
if fromIsolated and (market is None):
isolatedSymbol = fromId # allow user provide symbol from/to account
if toIsolated and (market is None):
isolatedSymbol = toId
if fromIsolated or toIsolated: # Isolated margin transfer
fromFuture = fromId == 'UMFUTURE' or fromId == 'CMFUTURE'
toFuture = toId == 'UMFUTURE' or toId == 'CMFUTURE'
fromSpot = fromId == 'MAIN'
toSpot = toId == 'MAIN'
funding = fromId == 'FUNDING' or toId == 'FUNDING'
option = fromId == 'OPTION' or toId == 'OPTION'
prohibitedWithIsolated = fromFuture or toFuture or funding or option
if (fromIsolated or toIsolated) and prohibitedWithIsolated:
raise BadRequest(self.id + ' transfer() does not allow transfers between ' + fromAccount + ' and ' + toAccount)
elif toSpot and fromIsolated:
fromId = 'ISOLATED_MARGIN'
request['fromSymbol'] = isolatedSymbol
elif fromSpot and toIsolated:
toId = 'ISOLATED_MARGIN'
request['toSymbol'] = isolatedSymbol
else:
if fromIsolated and toIsolated:
request['fromSymbol'] = fromId
request['toSymbol'] = toId
fromId = 'ISOLATEDMARGIN'
toId = 'ISOLATEDMARGIN'
else:
if fromIsolated:
request['fromSymbol'] = isolatedSymbol
fromId = 'ISOLATEDMARGIN'
if toIsolated:
request['toSymbol'] = isolatedSymbol
toId = 'ISOLATEDMARGIN'
request['type'] = fromId + '_' + toId
else:
request['type'] = fromId + '_' + toId
response = self.sapiPostAssetTransfer(self.extend(request, params))
#
# {
# "tranId":13526853623
# }
#
return self.parse_transfer(response, currency)
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://developers.binance.com/docs/wallet/asset/query-user-universal-transfer
: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 transfers 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)
:param boolean [params.internal]: default False, when True will fetch pay trade history
:returns dict[]: a list of `transfer structures <https://docs.ccxt.com/#/?id=transfer-structure>`
"""
self.load_markets()
internal = self.safe_bool(params, 'internal')
params = self.omit(params, 'internal')
paginate = False
paginate, params = self.handle_option_and_params(params, 'fetchTransfers', 'paginate')
if paginate and not internal:
return self.fetch_paginated_call_dynamic('fetchTransfers', code, since, limit, params)
currency = None
if code is not None:
currency = self.currency(code)
request: dict = {}
limitKey = 'limit'
if not internal:
defaultType = self.safe_string_2(self.options, 'fetchTransfers', 'defaultType', 'spot')
fromAccount = self.safe_string(params, 'fromAccount', defaultType)
defaultTo = 'spot' if (fromAccount == 'future') else 'future'
toAccount = self.safe_string(params, 'toAccount', defaultTo)
type = self.safe_string(params, 'type')
accountsByType = self.safe_dict(self.options, 'accountsByType', {})
fromId = self.safe_string(accountsByType, fromAccount)
toId = self.safe_string(accountsByType, toAccount)
if type is None:
if fromId is None:
keys = list(accountsByType.keys())
raise ExchangeError(self.id + ' fromAccount parameter must be one of ' + ', '.join(keys))
if toId is None:
keys = list(accountsByType.keys())
raise ExchangeError(self.id + ' toAccount parameter must be one of ' + ', '.join(keys))
type = fromId + '_' + toId
request['type'] = type
limitKey = 'size'
if limit is not None:
request[limitKey] = limit
if since is not None:
request['startTime'] = since
until = self.safe_integer(params, 'until')
if until is not None:
params = self.omit(params, 'until')
request['endTime'] = until
response = None
if internal:
response = self.sapiGetPayTransactions(self.extend(request, params))
#
# {
# "code": "000000",
# "message": "success",
# "data": [
# {
# "orderType": "C2C", # EnumPAY(C2B Merchant Acquiring Payment), PAY_REFUND(C2B Merchant Acquiring Payment,refund), C2C(C2C Transfer Payment),CRYPTO_BOX(Crypto box), CRYPTO_BOX_RF(Crypto Box, refund), C2C_HOLDING(Transfer to new Binance user), C2C_HOLDING_RF(Transfer to new Binance user,refund), PAYOUT(B2C Disbursement Payment), REMITTANCESend cash)
# "transactionId": "M_P_71505104267788288",
# "transactionTime": 1610090460133, #trade timestamp
# "amount": "23.72469206", #order amount(up to 8 decimal places), positive is income, negative is expenditure
# "currency": "BNB",
# "walletType": 1, #main wallet type, 1 for funding wallet, 2 for spot wallet, 3 for fiat wallet, 4 or 6 for card payment, 5 for earn wallet
# "walletTypes": [1,2], #array formatthere are multiple values when using combination payment
# "fundsDetail": [ # details
# {
# "currency": "USDT", #asset
# "amount": "1.2",
# "walletAssetCost":[ #details of asset cost per wallet
# {"1":"0.6"},
# {"2":"0.6"}
# ]
# },
# {
# "currency": "ETH",
# "amount": "0.0001",
# "walletAssetCost":[
# {"1":"0.00005"},
# {"2":"0.00005"}
# ]
# }
# ],
# "payerInfo":{
# "name":"Jack", #nickname or merchant name
# "type":"USER", #account typeUSER for personalMERCHANT for merchant
# "binanceId":"12345678", #binance uid
# "accountId":"67736251" #binance pay id
# },
# "receiverInfo":{
# "name":"Alan", #nickname or merchant name
# "type":"MERCHANT", #account typeUSER for personalMERCHANT for merchant
# "email":"alan@binance.com", #email
# "binanceId":"34355667", #binance uid
# "accountId":"21326891", #binance pay id
# "countryCode":"1", #International area code
# "phoneNumber":"8057651210",
# "mobileCode":"US", #country code
# "extend":[ #extension field
# "institutionName": "",
# "cardNumber": "",
# "digitalWalletId": ""
# ]
# }
# }
# ],
# "success": True
# }
#
else:
response = self.sapiGetAssetTransfer(self.extend(request, params))
#
# {
# "total": 3,
# "rows": [
# {
# "timestamp": 1614640878000,
# "asset": "USDT",
# "amount": "25",
# "type": "MAIN_UMFUTURE",
# "status": "CONFIRMED",
# "tranId": 43000126248
# },
# ]
# }
#
rows = self.safe_list_2(response, 'rows', 'data', [])
return self.parse_transfers(rows, currency, since, limit)
def fetch_deposit_address(self, code: str, params={}) -> DepositAddress:
"""
fetch the deposit address for a currency associated with self account
https://developers.binance.com/docs/wallet/capital/deposite-address
:param str code: unified currency code
:param dict [params]: extra parameters specific to the exchange API endpoint
:param str [params.network]: network for fetch deposit address
:returns dict: an `address structure <https://docs.ccxt.com/#/?id=address-structure>`
"""
self.load_markets()
currency = self.currency(code)
request: dict = {
'coin': currency['id'],
# 'network': 'ETH', # 'BSC', 'XMR', you can get network and isDefault in networkList in the response of sapiGetCapitalConfigDetail
}
networks = self.safe_dict(self.options, 'networks', {})
network = self.safe_string_upper(params, 'network') # self line allows the user to specify either ERC20 or ETH
network = self.safe_string(networks, network, network) # handle ERC20>ETH alias
if network is not None:
request['network'] = network
params = self.omit(params, 'network')
# has support for the 'network' parameter
response = self.sapiGetCapitalDepositAddress(self.extend(request, params))
#
# {
# "currency": "XRP",
# "address": "rEb8TK3gBgk5auZkwc6sHnwrGVJH8DuaLh",
# "tag": "108618262",
# "info": {
# "coin": "XRP",
# "address": "rEb8TK3gBgk5auZkwc6sHnwrGVJH8DuaLh",
# "tag": "108618262",
# "url": "https://bithomp.com/explorer/rEb8TK3gBgk5auZkwc6sHnwrGVJH8DuaLh"
# }
# }
#
return self.parse_deposit_address(response, currency)
def parse_deposit_address(self, response, currency: Currency = None) -> DepositAddress:
#
# {
# "coin": "XRP",
# "address": "rEb8TK3gBgk5auZkwc6sHnwrGVJH8DuaLh",
# "tag": "108618262",
# "url": "https://bithomp.com/explorer/rEb8TK3gBgk5auZkwc6sHnwrGVJH8DuaLh"
# }
#
url = self.safe_string(response, 'url')
address = self.safe_string(response, 'address')
currencyId = self.safe_string(response, 'currency')
code = self.safe_currency_code(currencyId, currency)
# deposit-address endpoint provides only network url(not network ID/CODE)
# so we should map the url to network(their data is inside currencies)
networkCode = self.get_network_code_by_network_url(code, url)
tag = self.safe_string(response, 'tag', '')
if len(tag) == 0:
tag = None
self.check_address(address)
return {
'info': response,
'currency': code,
'network': networkCode,
'address': address,
'tag': tag,
}
def fetch_transaction_fees(self, codes: Strings = None, params={}):
"""
@deprecated
please use fetchDepositWithdrawFees instead
https://developers.binance.com/docs/wallet/capital/all-coins-info
:param str[]|None codes: not used by binance fetchTransactionFees()
: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>`
"""
self.load_markets()
response = self.sapiGetCapitalConfigGetall(params)
#
# [
# {
# "coin": "BAT",
# "depositAllEnable": True,
# "withdrawAllEnable": True,
# "name": "Basic Attention Token",
# "free": "0",
# "locked": "0",
# "freeze": "0",
# "withdrawing": "0",
# "ipoing": "0",
# "ipoable": "0",
# "storage": "0",
# "isLegalMoney": False,
# "trading": True,
# "networkList": [
# {
# "network": "BNB",
# "coin": "BAT",
# "withdrawIntegerMultiple": "0.00000001",
# "isDefault": False,
# "depositEnable": True,
# "withdrawEnable": True,
# "depositDesc": '',
# "withdrawDesc": '',
# "specialTips": "The name of self asset is Basic Attention Token(BAT). Both a MEMO and an Address are required to successfully deposit your BEP2 tokens to Binance.",
# "name": "BEP2",
# "resetAddressStatus": False,
# "addressRegex": "^(bnb1)[0-9a-z]{38}$",
# "memoRegex": "^[0-9A-Za-z\\-_]{1,120}$",
# "withdrawFee": "0.27",
# "withdrawMin": "0.54",
# "withdrawMax": "10000000000",
# "minConfirm": "1",
# "unLockConfirm": "0"
# },
# {
# "network": "BSC",
# "coin": "BAT",
# "withdrawIntegerMultiple": "0.00000001",
# "isDefault": False,
# "depositEnable": True,
# "withdrawEnable": True,
# "depositDesc": '',
# "withdrawDesc": '',
# "specialTips": "The name of self asset is Basic Attention Token. Please ensure you are depositing Basic Attention Token(BAT) tokens under the contract address ending in 9766e.",
# "name": "BEP20(BSC)",
# "resetAddressStatus": False,
# "addressRegex": "^(0x)[0-9A-Fa-f]{40}$",
# "memoRegex": '',
# "withdrawFee": "0.27",
# "withdrawMin": "0.54",
# "withdrawMax": "10000000000",
# "minConfirm": "15",
# "unLockConfirm": "0"
# },
# {
# "network": "ETH",
# "coin": "BAT",
# "withdrawIntegerMultiple": "0.00000001",
# "isDefault": True,
# "depositEnable": True,
# "withdrawEnable": True,
# "depositDesc": '',
# "withdrawDesc": '',
# "specialTips": "The name of self asset is Basic Attention Token. Please ensure you are depositing Basic Attention Token(BAT) tokens under the contract address ending in 887ef.",
# "name": "ERC20",
# "resetAddressStatus": False,
# "addressRegex": "^(0x)[0-9A-Fa-f]{40}$",
# "memoRegex": '',
# "withdrawFee": "27",
# "withdrawMin": "54",
# "withdrawMax": "10000000000",
# "minConfirm": "12",
# "unLockConfirm": "0"
# }
# ]
# }
# ]
#
withdrawFees: dict = {}
for i in range(0, len(response)):
entry = response[i]
currencyId = self.safe_string(entry, 'coin')
code = self.safe_currency_code(currencyId)
networkList = self.safe_list(entry, 'networkList', [])
withdrawFees[code] = {}
for j in range(0, len(networkList)):
networkEntry = networkList[j]
networkId = self.safe_string(networkEntry, 'network')
networkCode = self.safe_currency_code(networkId)
fee = self.safe_number(networkEntry, 'withdrawFee')
withdrawFees[code][networkCode] = fee
return {
'withdraw': withdrawFees,
'deposit': {},
'info': response,
}
def fetch_deposit_withdraw_fees(self, codes: Strings = None, params={}):
"""
fetch deposit and withdraw fees
https://developers.binance.com/docs/wallet/capital/all-coins-info
:param str[]|None codes: not used by binance fetchDepositWithdrawFees()
: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>`
"""
self.load_markets()
response = self.sapiGetCapitalConfigGetall(params)
#
# [
# {
# "coin": "BAT",
# "depositAllEnable": True,
# "withdrawAllEnable": True,
# "name": "Basic Attention Token",
# "free": "0",
# "locked": "0",
# "freeze": "0",
# "withdrawing": "0",
# "ipoing": "0",
# "ipoable": "0",
# "storage": "0",
# "isLegalMoney": False,
# "trading": True,
# "networkList": [
# {
# "network": "BNB",
# "coin": "BAT",
# "withdrawIntegerMultiple": "0.00000001",
# "isDefault": False,
# "depositEnable": True,
# "withdrawEnable": True,
# "depositDesc": '',
# "withdrawDesc": '',
# "specialTips": "The name of self asset is Basic Attention Token(BAT). Both a MEMO and an Address are required to successfully deposit your BEP2 tokens to Binance.",
# "name": "BEP2",
# "resetAddressStatus": False,
# "addressRegex": "^(bnb1)[0-9a-z]{38}$",
# "memoRegex": "^[0-9A-Za-z\\-_]{1,120}$",
# "withdrawFee": "0.27",
# "withdrawMin": "0.54",
# "withdrawMax": "10000000000",
# "minConfirm": "1",
# "unLockConfirm": "0"
# },
# ...
# ]
# }
# ]
#
return self.parse_deposit_withdraw_fees(response, codes, 'coin')
def parse_deposit_withdraw_fee(self, fee, currency: Currency = None):
#
# {
# "coin": "BAT",
# "depositAllEnable": True,
# "withdrawAllEnable": True,
# "name": "Basic Attention Token",
# "free": "0",
# "locked": "0",
# "freeze": "0",
# "withdrawing": "0",
# "ipoing": "0",
# "ipoable": "0",
# "storage": "0",
# "isLegalMoney": False,
# "trading": True,
# "networkList": [
# {
# "network": "BNB",
# "coin": "BAT",
# "withdrawIntegerMultiple": "0.00000001",
# "isDefault": False,
# "depositEnable": True,
# "withdrawEnable": True,
# "depositDesc": '',
# "withdrawDesc": '',
# "specialTips": "The name of self asset is Basic Attention Token(BAT). Both a MEMO and an Address are required to successfully deposit your BEP2 tokens to Binance.",
# "name": "BEP2",
# "resetAddressStatus": False,
# "addressRegex": "^(bnb1)[0-9a-z]{38}$",
# "memoRegex": "^[0-9A-Za-z\\-_]{1,120}$",
# "withdrawFee": "0.27",
# "withdrawMin": "0.54",
# "withdrawMax": "10000000000",
# "minConfirm": "1",
# "unLockConfirm": "0"
# },
# ...
# ]
# }
#
networkList = self.safe_list(fee, 'networkList', [])
result = self.deposit_withdraw_fee(fee)
for j in range(0, len(networkList)):
networkEntry = networkList[j]
networkId = self.safe_string(networkEntry, 'network')
networkCode = self.network_id_to_code(networkId)
withdrawFee = self.safe_number(networkEntry, 'withdrawFee')
isDefault = self.safe_bool(networkEntry, 'isDefault')
if isDefault is True:
result['withdraw'] = {
'fee': withdrawFee,
'percentage': None,
}
result['networks'][networkCode] = {
'withdraw': {
'fee': withdrawFee,
'percentage': None,
},
'deposit': {
'fee': None,
'percentage': None,
},
}
return result
def withdraw(self, code: str, amount: float, address: str, tag: Str = None, params={}) -> Transaction:
"""
make a withdrawal
https://developers.binance.com/docs/wallet/capital/withdraw
: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
:returns dict: a `transaction structure <https://docs.ccxt.com/#/?id=transaction-structure>`
"""
tag, params = self.handle_withdraw_tag_and_params(tag, params)
self.check_address(address)
self.load_markets()
currency = self.currency(code)
request: dict = {
'coin': currency['id'],
'address': address,
# issue sapiGetCapitalConfigGetall() to get networks for withdrawing USDT ERC20 vs USDT Omni
# 'network': 'ETH', # 'BTC', 'TRX', etc, optional
}
if tag is not None:
request['addressTag'] = tag
networks = self.safe_dict(self.options, 'networks', {})
network = self.safe_string_upper(params, 'network') # self line allows the user to specify either ERC20 or ETH
network = self.safe_string(networks, network, network) # handle ERC20>ETH alias
if network is not None:
request['network'] = network
params = self.omit(params, 'network')
request['amount'] = self.currency_to_precision(code, amount, network)
response = self.sapiPostCapitalWithdrawApply(self.extend(request, params))
# {id: '9a67628b16ba4988ae20d329333f16bc'}
return self.parse_transaction(response, currency)
def parse_trading_fee(self, fee: dict, market: Market = None) -> TradingFeeInterface:
#
# spot
# [
# {
# "symbol": "BTCUSDT",
# "makerCommission": "0.001",
# "takerCommission": "0.001"
# }
# ]
#
# swap
# {
# "symbol": "BTCUSD_PERP",
# "makerCommissionRate": "0.00015", # 0.015%
# "takerCommissionRate": "0.00040" # 0.040%
# }
#
marketId = self.safe_string(fee, 'symbol')
symbol = self.safe_symbol(marketId, market, None, 'spot')
return {
'info': fee,
'symbol': symbol,
'maker': self.safe_number_2(fee, 'makerCommission', 'makerCommissionRate'),
'taker': self.safe_number_2(fee, 'takerCommission', 'takerCommissionRate'),
'percentage': None,
'tierBased': None,
}
def fetch_trading_fee(self, symbol: str, params={}) -> TradingFeeInterface:
"""
fetch the trading fees for a market
https://developers.binance.com/docs/wallet/asset/trade-fee
https://developers.binance.com/docs/derivatives/usds-margined-futures/account/rest-api/User-Commission-Rate
https://developers.binance.com/docs/derivatives/coin-margined-futures/account/rest-api/User-Commission-Rate
https://developers.binance.com/docs/derivatives/portfolio-margin/account/Get-User-Commission-Rate-for-UM
https://developers.binance.com/docs/derivatives/portfolio-margin/account/Get-User-Commission-Rate-for-CM
:param str symbol: unified market symbol
:param dict [params]: extra parameters specific to the exchange API endpoint
:param boolean [params.portfolioMargin]: set to True if you would like to fetch trading fees in a portfolio margin account
:param str [params.subType]: "linear" or "inverse"
:returns dict: a `fee structure <https://docs.ccxt.com/#/?id=fee-structure>`
"""
self.load_markets()
market = self.market(symbol)
type = market['type']
subType = None
subType, params = self.handle_sub_type_and_params('fetchTradingFee', market, params)
isPortfolioMargin = None
isPortfolioMargin, params = self.handle_option_and_params_2(params, 'fetchTradingFee', 'papi', 'portfolioMargin', False)
isLinear = self.is_linear(type, subType)
isInverse = self.is_inverse(type, subType)
request: dict = {
'symbol': market['id'],
}
response = None
if isLinear:
if isPortfolioMargin:
response = self.papiGetUmCommissionRate(self.extend(request, params))
else:
response = self.fapiPrivateGetCommissionRate(self.extend(request, params))
elif isInverse:
if isPortfolioMargin:
response = self.papiGetCmCommissionRate(self.extend(request, params))
else:
response = self.dapiPrivateGetCommissionRate(self.extend(request, params))
else:
response = self.sapiGetAssetTradeFee(self.extend(request, params))
#
# spot
#
# [
# {
# "symbol": "BTCUSDT",
# "makerCommission": "0.001",
# "takerCommission": "0.001"
# }
# ]
#
# swap
#
# {
# "symbol": "BTCUSD_PERP",
# "makerCommissionRate": "0.00015", # 0.015%
# "takerCommissionRate": "0.00040" # 0.040%
# }
#
data = response
if isinstance(data, list):
data = self.safe_dict(data, 0, {})
return self.parse_trading_fee(data, market)
def fetch_trading_fees(self, params={}) -> TradingFees:
"""
fetch the trading fees for multiple markets
https://developers.binance.com/docs/wallet/asset/trade-fee
https://developers.binance.com/docs/derivatives/usds-margined-futures/account/rest-api/Account-Information-V2
https://developers.binance.com/docs/derivatives/coin-margined-futures/account/rest-api/Account-Information
https://developers.binance.com/docs/derivatives/usds-margined-futures/account/rest-api/Account-Config
:param dict [params]: extra parameters specific to the exchange API endpoint
:param str [params.subType]: "linear" or "inverse"
:returns dict: a dictionary of `fee structures <https://docs.ccxt.com/#/?id=fee-structure>` indexed by market symbols
"""
self.load_markets()
type = None
type, params = self.handle_market_type_and_params('fetchTradingFees', None, params)
subType = None
subType, params = self.handle_sub_type_and_params('fetchTradingFees', None, params, 'linear')
isSpotOrMargin = (type == 'spot') or (type == 'margin')
isLinear = self.is_linear(type, subType)
isInverse = self.is_inverse(type, subType)
response = None
if isSpotOrMargin:
response = self.sapiGetAssetTradeFee(params)
elif isLinear:
response = self.fapiPrivateGetAccountConfig(params)
elif isInverse:
response = self.dapiPrivateGetAccount(params)
#
# sapi / spot
#
# [
# {
# "symbol": "ZRXBNB",
# "makerCommission": "0.001",
# "takerCommission": "0.001"
# },
# {
# "symbol": "ZRXBTC",
# "makerCommission": "0.001",
# "takerCommission": "0.001"
# },
# ]
#
# fapi / future / linear
#
# {
# "feeTier": 0, # account commisssion tier
# "canTrade": True, # if can trade
# "canDeposit": True, # if can transfer in asset
# "canWithdraw": True, # if can transfer out asset
# "updateTime": 0,
# "totalInitialMargin": "0.00000000", # total initial margin required with current mark price(useless with isolated positions), only for USDT asset
# "totalMaintMargin": "0.00000000", # total maintenance margin required, only for USDT asset
# "totalWalletBalance": "23.72469206", # total wallet balance, only for USDT asset
# "totalUnrealizedProfit": "0.00000000", # total unrealized profit, only for USDT asset
# "totalMarginBalance": "23.72469206", # total margin balance, only for USDT asset
# "totalPositionInitialMargin": "0.00000000", # initial margin required for positions with current mark price, only for USDT asset
# "totalOpenOrderInitialMargin": "0.00000000", # initial margin required for open orders with current mark price, only for USDT asset
# "totalCrossWalletBalance": "23.72469206", # crossed wallet balance, only for USDT asset
# "totalCrossUnPnl": "0.00000000", # unrealized profit of crossed positions, only for USDT asset
# "availableBalance": "23.72469206", # available balance, only for USDT asset
# "maxWithdrawAmount": "23.72469206" # maximum amount for transfer out, only for USDT asset
# ...
# }
#
# dapi / delivery / inverse
#
# {
# "canDeposit": True,
# "canTrade": True,
# "canWithdraw": True,
# "feeTier": 2,
# "updateTime": 0
# }
#
if isSpotOrMargin:
#
# [
# {
# "symbol": "ZRXBNB",
# "makerCommission": "0.001",
# "takerCommission": "0.001"
# },
# {
# "symbol": "ZRXBTC",
# "makerCommission": "0.001",
# "takerCommission": "0.001"
# },
# ]
#
result: dict = {}
for i in range(0, len(response)):
fee = self.parse_trading_fee(response[i])
symbol = fee['symbol']
result[symbol] = fee
return result
elif isLinear:
#
# {
# "feeTier": 0, # account commisssion tier
# "canTrade": True, # if can trade
# "canDeposit": True, # if can transfer in asset
# "canWithdraw": True, # if can transfer out asset
# "updateTime": 0,
# "totalInitialMargin": "0.00000000", # total initial margin required with current mark price(useless with isolated positions), only for USDT asset
# "totalMaintMargin": "0.00000000", # total maintenance margin required, only for USDT asset
# "totalWalletBalance": "23.72469206", # total wallet balance, only for USDT asset
# "totalUnrealizedProfit": "0.00000000", # total unrealized profit, only for USDT asset
# "totalMarginBalance": "23.72469206", # total margin balance, only for USDT asset
# "totalPositionInitialMargin": "0.00000000", # initial margin required for positions with current mark price, only for USDT asset
# "totalOpenOrderInitialMargin": "0.00000000", # initial margin required for open orders with current mark price, only for USDT asset
# "totalCrossWalletBalance": "23.72469206", # crossed wallet balance, only for USDT asset
# "totalCrossUnPnl": "0.00000000", # unrealized profit of crossed positions, only for USDT asset
# "availableBalance": "23.72469206", # available balance, only for USDT asset
# "maxWithdrawAmount": "23.72469206" # maximum amount for transfer out, only for USDT asset
# ...
# }
#
symbols = list(self.markets.keys())
result: dict = {}
feeTier = self.safe_integer(response, 'feeTier')
feeTiers = self.fees['linear']['trading']['tiers']
maker = feeTiers['maker'][feeTier][1]
taker = feeTiers['taker'][feeTier][1]
for i in range(0, len(symbols)):
symbol = symbols[i]
market = self.markets[symbol]
if market['linear']:
result[symbol] = {
'info': {
'feeTier': feeTier,
},
'symbol': symbol,
'maker': maker,
'taker': taker,
}
return result
elif isInverse:
#
# {
# "canDeposit": True,
# "canTrade": True,
# "canWithdraw": True,
# "feeTier": 2,
# "updateTime": 0
# }
#
symbols = list(self.markets.keys())
result: dict = {}
feeTier = self.safe_integer(response, 'feeTier')
feeTiers = self.fees['inverse']['trading']['tiers']
maker = feeTiers['maker'][feeTier][1]
taker = feeTiers['taker'][feeTier][1]
for i in range(0, len(symbols)):
symbol = symbols[i]
market = self.markets[symbol]
if market['inverse']:
result[symbol] = {
'info': {
'feeTier': feeTier,
},
'symbol': symbol,
'maker': maker,
'taker': taker,
}
return result
return None
def futures_transfer(self, code: str, amount, type, params={}):
"""
@ignore
transfer between futures account
https://developers.binance.com/docs/derivatives/usds-margined-futures/account/rest-api/New-Future-Account-Transfer
:param str code: unified currency code
:param float amount: the amount to transfer
:param str type: 1 - transfer from spot account to USDT-Ⓜ futures account, 2 - transfer from USDT-Ⓜ futures account to spot account, 3 - transfer from spot account to COIN-Ⓜ futures account, 4 - transfer from COIN-Ⓜ futures account to spot account
:param dict [params]: extra parameters specific to the exchange API endpoint
:param float params.recvWindow:
:returns dict: a `transfer structure <https://docs.ccxt.com/#/?id=futures-transfer-structure>`
"""
if (type < 1) or (type > 4):
raise ArgumentsRequired(self.id + ' type must be between 1 and 4')
self.load_markets()
currency = self.currency(code)
request: dict = {
'asset': currency['id'],
'amount': amount,
'type': type,
}
response = self.sapiPostFuturesTransfer(self.extend(request, params))
#
# {
# "tranId": 100000001
# }
#
return self.parse_transfer(response, currency)
def fetch_funding_rate(self, symbol: str, params={}) -> FundingRate:
"""
fetch the current funding rate
https://developers.binance.com/docs/derivatives/usds-margined-futures/market-data/rest-api/Mark-Price
https://developers.binance.com/docs/derivatives/coin-margined-futures/market-data/rest-api/Index-Price-and-Mark-Price
:param str symbol: unified market symbol
:param dict [params]: extra parameters specific to the exchange API endpoint
:returns dict: a `funding rate structure <https://docs.ccxt.com/#/?id=funding-rate-structure>`
"""
self.load_markets()
market = self.market(symbol)
request: dict = {
'symbol': market['id'],
}
response = None
if market['linear']:
response = self.fapiPublicGetPremiumIndex(self.extend(request, params))
elif market['inverse']:
response = self.dapiPublicGetPremiumIndex(self.extend(request, params))
else:
raise NotSupported(self.id + ' fetchFundingRate() supports linear and inverse contracts only')
if market['inverse']:
response = response[0]
#
# {
# "symbol": "BTCUSDT",
# "markPrice": "45802.81129892",
# "indexPrice": "45745.47701915",
# "estimatedSettlePrice": "45133.91753671",
# "lastFundingRate": "0.00063521",
# "interestRate": "0.00010000",
# "nextFundingTime": "1621267200000",
# "time": "1621252344001"
# }
#
return self.parse_funding_rate(response, market)
def fetch_funding_rate_history(self, symbol: Str = None, since: Int = None, limit: Int = None, params={}):
"""
fetches historical funding rate prices
https://developers.binance.com/docs/derivatives/usds-margined-futures/market-data/rest-api/Get-Funding-Rate-History
https://developers.binance.com/docs/derivatives/coin-margined-futures/market-data/rest-api/Get-Funding-Rate-History-of-Perpetual-Futures
: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 <https://docs.ccxt.com/#/?id=funding-rate-history-structure>` to fetch
:param dict [params]: extra parameters specific to the exchange API endpoint
:param int [params.until]: timestamp in ms of the latest funding rate
: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.subType]: "linear" or "inverse"
:returns dict[]: a list of `funding rate structures <https://docs.ccxt.com/#/?id=funding-rate-history-structure>`
"""
self.load_markets()
request: dict = {}
paginate = False
paginate, params = self.handle_option_and_params(params, 'fetchFundingRateHistory', 'paginate')
if paginate:
return self.fetch_paginated_call_deterministic('fetchFundingRateHistory', symbol, since, limit, '8h', params)
defaultType = self.safe_string_2(self.options, 'fetchFundingRateHistory', 'defaultType', 'future')
type = self.safe_string(params, 'type', defaultType)
market = None
if symbol is not None:
market = self.market(symbol)
symbol = market['symbol']
request['symbol'] = market['id']
subType = None
subType, params = self.handle_sub_type_and_params('fetchFundingRateHistory', market, params, 'linear')
params = self.omit(params, 'type')
if since is not None:
request['startTime'] = since
until = self.safe_integer(params, 'until') # unified in milliseconds
endTime = self.safe_integer(params, 'endTime', until) # exchange-specific in milliseconds
params = self.omit(params, ['endTime', 'until'])
if endTime is not None:
request['endTime'] = endTime
if limit is not None:
request['limit'] = limit
response = None
if self.is_linear(type, subType):
response = self.fapiPublicGetFundingRate(self.extend(request, params))
elif self.is_inverse(type, subType):
response = self.dapiPublicGetFundingRate(self.extend(request, params))
else:
raise NotSupported(self.id + ' fetchFundingRateHistory() is not supported for ' + type + ' markets')
#
# {
# "symbol": "BTCUSDT",
# "fundingRate": "0.00063521",
# "fundingTime": "1621267200000",
# }
#
return self.parse_funding_rate_histories(response, market, since, limit)
def parse_funding_rate_history(self, contract, market: Market = None):
#
# {
# "symbol": "BTCUSDT",
# "fundingRate": "0.00063521",
# "fundingTime": "1621267200000",
# }
#
timestamp = self.safe_integer(contract, 'fundingTime')
return {
'info': contract,
'symbol': self.safe_symbol(self.safe_string(contract, 'symbol'), None, None, 'swap'),
'fundingRate': self.safe_number(contract, 'fundingRate'),
'timestamp': timestamp,
'datetime': self.iso8601(timestamp),
}
def fetch_funding_rates(self, symbols: Strings = None, params={}) -> FundingRates:
"""
fetch the funding rate for multiple markets
https://developers.binance.com/docs/derivatives/usds-margined-futures/market-data/rest-api/Mark-Price
https://developers.binance.com/docs/derivatives/coin-margined-futures/market-data/rest-api/Index-Price-and-Mark-Price
:param str[]|None symbols: list of unified market symbols
:param dict [params]: extra parameters specific to the exchange API endpoint
:param str [params.subType]: "linear" or "inverse"
:returns dict[]: a list of `funding rate structures <https://docs.ccxt.com/#/?id=funding-rates-structure>`, indexed by market symbols
"""
self.load_markets()
symbols = self.market_symbols(symbols)
defaultType = self.safe_string_2(self.options, 'fetchFundingRates', 'defaultType', 'future')
type = self.safe_string(params, 'type', defaultType)
subType = None
subType, params = self.handle_sub_type_and_params('fetchFundingRates', None, params, 'linear')
query = self.omit(params, 'type')
response = None
if self.is_linear(type, subType):
response = self.fapiPublicGetPremiumIndex(query)
elif self.is_inverse(type, subType):
response = self.dapiPublicGetPremiumIndex(query)
else:
raise NotSupported(self.id + ' fetchFundingRates() supports linear and inverse contracts only')
return self.parse_funding_rates(response, symbols)
def parse_funding_rate(self, contract, market: Market = None) -> FundingRate:
# ensure it matches with https://www.binance.com/en/futures/funding-history/0
#
# fetchFundingRate, fetchFundingRates
#
# {
# "symbol": "BTCUSDT",
# "markPrice": "45802.81129892",
# "indexPrice": "45745.47701915",
# "estimatedSettlePrice": "45133.91753671",
# "lastFundingRate": "0.00063521",
# "interestRate": "0.00010000",
# "nextFundingTime": "1621267200000",
# "time": "1621252344001"
# }
#
# fetchFundingInterval, fetchFundingIntervals
#
# {
# "symbol": "BLZUSDT",
# "adjustedFundingRateCap": "0.03000000",
# "adjustedFundingRateFloor": "-0.03000000",
# "fundingIntervalHours": 4,
# "disclaimer": False
# }
#
timestamp = self.safe_integer(contract, 'time')
marketId = self.safe_string(contract, 'symbol')
symbol = self.safe_symbol(marketId, market, None, 'contract')
markPrice = self.safe_number(contract, 'markPrice')
indexPrice = self.safe_number(contract, 'indexPrice')
interestRate = self.safe_number(contract, 'interestRate')
estimatedSettlePrice = self.safe_number(contract, 'estimatedSettlePrice')
fundingRate = self.safe_number(contract, 'lastFundingRate')
fundingTime = self.safe_integer(contract, 'nextFundingTime')
interval = self.safe_string(contract, 'fundingIntervalHours')
intervalString = None
if interval is not None:
intervalString = interval + 'h'
return {
'info': contract,
'symbol': symbol,
'markPrice': markPrice,
'indexPrice': indexPrice,
'interestRate': interestRate,
'estimatedSettlePrice': estimatedSettlePrice,
'timestamp': timestamp,
'datetime': self.iso8601(timestamp),
'fundingRate': fundingRate,
'fundingTimestamp': fundingTime,
'fundingDatetime': self.iso8601(fundingTime),
'nextFundingRate': None,
'nextFundingTimestamp': None,
'nextFundingDatetime': None,
'previousFundingRate': None,
'previousFundingTimestamp': None,
'previousFundingDatetime': None,
'interval': intervalString,
}
def parse_account_positions(self, account, filterClosed=False):
positions = self.safe_list(account, 'positions')
assets = self.safe_list(account, 'assets', [])
balances: dict = {}
for i in range(0, len(assets)):
entry = assets[i]
currencyId = self.safe_string(entry, 'asset')
code = self.safe_currency_code(currencyId)
crossWalletBalance = self.safe_string(entry, 'crossWalletBalance')
crossUnPnl = self.safe_string(entry, 'crossUnPnl')
balances[code] = {
'crossMargin': Precise.string_add(crossWalletBalance, crossUnPnl),
'crossWalletBalance': crossWalletBalance,
}
result = []
for i in range(0, len(positions)):
position = positions[i]
marketId = self.safe_string(position, 'symbol')
market = self.safe_market(marketId, None, None, 'contract')
code = market['quote'] if market['linear'] else market['base']
maintenanceMargin = self.safe_string(position, 'maintMargin')
# check for maintenance margin so empty positions are not returned
isPositionOpen = (maintenanceMargin != '0') and (maintenanceMargin != '0.00000000')
if not filterClosed or isPositionOpen:
# sometimes not all the codes are correctly returned...
if code in balances:
parsed = self.parse_account_position(self.extend(position, {
'crossMargin': balances[code]['crossMargin'],
'crossWalletBalance': balances[code]['crossWalletBalance'],
}), market)
result.append(parsed)
return result
def parse_account_position(self, position, market: Market = None):
#
# usdm
#
# v3(similar for cross & isolated)
#
# {
# "symbol": "WLDUSDT",
# "positionSide": "BOTH",
# "positionAmt": "-849",
# "unrealizedProfit": "11.17920750",
# "notional": "-1992.46079250",
# "isolatedMargin": "0",
# "isolatedWallet": "0",
# "initialMargin": "99.62303962",
# "maintMargin": "11.95476475",
# "updateTime": "1721995760449"
# "leverage": "50", # in v2
# "entryPrice": "2.34", # in v2
# "positionInitialMargin": "118.82116614", # in v2
# "openOrderInitialMargin": "0", # in v2
# "isolated": False, # in v2
# "breakEvenPrice": "2.3395788", # in v2
# "maxNotional": "25000", # in v2
# "bidNotional": "0", # in v2
# "askNotional": "0" # in v2
# }
#
# coinm
#
# {
# "symbol": "BTCUSD_210625",
# "initialMargin": "0.00024393",
# "maintMargin": "0.00002439",
# "unrealizedProfit": "-0.00000163",
# "positionInitialMargin": "0.00024393",
# "openOrderInitialMargin": "0",
# "leverage": "10",
# "isolated": False,
# "positionSide": "BOTH",
# "entryPrice": "41021.20000069",
# "maxQty": "100",
# "notionalValue": "0.00243939",
# "isolatedWallet": "0",
# "crossMargin": "0.314"
# "crossWalletBalance": "34",
# }
#
# linear portfolio margin
#
# {
# "symbol": "CTSIUSDT",
# "initialMargin": "0",
# "maintMargin": "0",
# "unrealizedProfit": "0.00000000",
# "positionInitialMargin": "0",
# "openOrderInitialMargin": "0",
# "leverage": "20",
# "entryPrice": "0.0",
# "maxNotional": "25000",
# "bidNotional": "0",
# "askNotional": "0",
# "positionSide": "SHORT",
# "positionAmt": "0",
# "updateTime": 0,
# "notional": "0",
# "breakEvenPrice": "0.0"
# }
#
# inverse portoflio margin
#
# {
# "symbol": "TRXUSD_PERP",
# "initialMargin": "0",
# "maintMargin": "0",
# "unrealizedProfit": "0.00000000",
# "positionInitialMargin": "0",
# "openOrderInitialMargin": "0",
# "leverage": "20",
# "entryPrice": "0.00000000",
# "positionSide": "SHORT",
# "positionAmt": "0",
# "maxQty": "5000000",
# "updateTime": 0,
# "notionalValue": "0",
# "breakEvenPrice": "0.00000000"
# }
#
marketId = self.safe_string(position, 'symbol')
market = self.safe_market(marketId, market, None, 'contract')
symbol = self.safe_string(market, 'symbol')
leverageString = self.safe_string(position, 'leverage')
leverage = int(leverageString) if (leverageString is not None) else None
initialMarginString = self.safe_string(position, 'initialMargin')
initialMargin = self.parse_number(initialMarginString)
initialMarginPercentageString = None
if leverageString is not None:
initialMarginPercentageString = Precise.string_div('1', leverageString, 8)
rational = self.is_round_number(1000 % leverage)
if not rational:
initialMarginPercentageString = Precise.string_div(Precise.string_add(initialMarginPercentageString, '1e-8'), '1', 8)
# to notionalValue
usdm = ('notional' in position)
maintenanceMarginString = self.safe_string(position, 'maintMargin')
maintenanceMargin = self.parse_number(maintenanceMarginString)
entryPriceString = self.safe_string(position, 'entryPrice')
entryPrice = self.parse_number(entryPriceString)
notionalString = self.safe_string_2(position, 'notional', 'notionalValue')
notionalStringAbs = Precise.string_abs(notionalString)
notional = self.parse_number(notionalStringAbs)
contractsString = self.safe_string(position, 'positionAmt')
contractsStringAbs = Precise.string_abs(contractsString)
if contractsString is None:
entryNotional = Precise.string_mul(Precise.string_mul(leverageString, initialMarginString), entryPriceString)
contractSizeNew = self.safe_string(market, 'contractSize')
contractsString = Precise.string_div(entryNotional, contractSizeNew)
contractsStringAbs = Precise.string_div(Precise.string_add(contractsString, '0.5'), '1', 0)
contracts = self.parse_number(contractsStringAbs)
leverageBrackets = self.safe_dict(self.options, 'leverageBrackets', {})
leverageBracket = self.safe_list(leverageBrackets, symbol, [])
maintenanceMarginPercentageString = None
for i in range(0, len(leverageBracket)):
bracket = leverageBracket[i]
if Precise.string_lt(notionalStringAbs, bracket[0]):
break
maintenanceMarginPercentageString = bracket[1]
maintenanceMarginPercentage = self.parse_number(maintenanceMarginPercentageString)
unrealizedPnlString = self.safe_string(position, 'unrealizedProfit')
unrealizedPnl = self.parse_number(unrealizedPnlString)
timestamp = self.safe_integer(position, 'updateTime')
if timestamp == 0:
timestamp = None
isolated = self.safe_bool(position, 'isolated')
if isolated is None:
isolatedMarginRaw = self.safe_string(position, 'isolatedMargin')
isolated = not Precise.string_eq(isolatedMarginRaw, '0')
marginMode = None
collateralString = None
walletBalance = None
if isolated:
marginMode = 'isolated'
walletBalance = self.safe_string(position, 'isolatedWallet')
collateralString = Precise.string_add(walletBalance, unrealizedPnlString)
else:
marginMode = 'cross'
walletBalance = self.safe_string(position, 'crossWalletBalance')
collateralString = self.safe_string(position, 'crossMargin')
collateral = self.parse_number(collateralString)
marginRatio = None
side = None
percentage = None
liquidationPriceStringRaw = None
liquidationPrice = None
contractSize = self.safe_value(market, 'contractSize')
contractSizeString = self.number_to_string(contractSize)
if Precise.string_equals(notionalString, '0'):
entryPrice = None
else:
side = 'short' if Precise.string_lt(notionalString, '0') else 'long'
marginRatio = self.parse_number(Precise.string_div(Precise.string_add(Precise.string_div(maintenanceMarginString, collateralString), '5e-5'), '1', 4))
percentage = self.parse_number(Precise.string_mul(Precise.string_div(unrealizedPnlString, initialMarginString, 4), '100'))
if usdm:
# calculate liquidation price
#
# liquidationPrice = (walletBalance / (contracts * (±1 + mmp))) + (±entryPrice / (±1 + mmp))
#
# mmp = maintenanceMarginPercentage
# where ± is negative for long and positive for short
# TODO: calculate liquidation price for coinm contracts
onePlusMaintenanceMarginPercentageString = None
entryPriceSignString = entryPriceString
if side == 'short':
onePlusMaintenanceMarginPercentageString = Precise.string_add('1', maintenanceMarginPercentageString)
else:
onePlusMaintenanceMarginPercentageString = Precise.string_add('-1', maintenanceMarginPercentageString)
entryPriceSignString = Precise.string_mul('-1', entryPriceSignString)
leftSide = Precise.string_div(walletBalance, Precise.string_mul(contractsStringAbs, onePlusMaintenanceMarginPercentageString))
rightSide = Precise.string_div(entryPriceSignString, onePlusMaintenanceMarginPercentageString)
liquidationPriceStringRaw = Precise.string_add(leftSide, rightSide)
else:
# calculate liquidation price
#
# liquidationPrice = (contracts * contractSize(±1 - mmp)) / (±1/entryPrice * contracts * contractSize - walletBalance)
#
onePlusMaintenanceMarginPercentageString = None
entryPriceSignString = entryPriceString
if side == 'short':
onePlusMaintenanceMarginPercentageString = Precise.string_sub('1', maintenanceMarginPercentageString)
else:
onePlusMaintenanceMarginPercentageString = Precise.string_sub('-1', maintenanceMarginPercentageString)
entryPriceSignString = Precise.string_mul('-1', entryPriceSignString)
size = Precise.string_mul(contractsStringAbs, contractSizeString)
leftSide = Precise.string_mul(size, onePlusMaintenanceMarginPercentageString)
rightSide = Precise.string_sub(Precise.string_mul(Precise.string_div('1', entryPriceSignString), size), walletBalance)
liquidationPriceStringRaw = Precise.string_div(leftSide, rightSide)
pricePrecision = self.precision_from_string(self.safe_string(market['precision'], 'price'))
pricePrecisionPlusOne = pricePrecision + 1
pricePrecisionPlusOneString = str(pricePrecisionPlusOne)
# round half up
rounder = Precise('5e-' + pricePrecisionPlusOneString)
rounderString = str(rounder)
liquidationPriceRoundedString = Precise.string_add(rounderString, liquidationPriceStringRaw)
truncatedLiquidationPrice = Precise.string_div(liquidationPriceRoundedString, '1', pricePrecision)
if truncatedLiquidationPrice[0] == '-':
# user cannot be liquidated
# since he has more collateral than the size of the position
truncatedLiquidationPrice = None
liquidationPrice = self.parse_number(truncatedLiquidationPrice)
positionSide = self.safe_string(position, 'positionSide')
hedged = positionSide != 'BOTH'
return {
'info': position,
'id': None,
'symbol': symbol,
'timestamp': timestamp,
'datetime': self.iso8601(timestamp),
'initialMargin': initialMargin,
'initialMarginPercentage': self.parse_number(initialMarginPercentageString),
'maintenanceMargin': maintenanceMargin,
'maintenanceMarginPercentage': maintenanceMarginPercentage,
'entryPrice': entryPrice,
'notional': notional,
'leverage': self.parse_number(leverageString),
'unrealizedPnl': unrealizedPnl,
'contracts': contracts,
'contractSize': contractSize,
'marginRatio': marginRatio,
'liquidationPrice': liquidationPrice,
'markPrice': None,
'collateral': collateral,
'marginMode': marginMode,
'side': side,
'hedged': hedged,
'percentage': percentage,
}
def parse_position_risk(self, position, market: Market = None):
#
# usdm
#
# {
# symbol: "WLDUSDT",
# positionSide: "BOTH",
# positionAmt: "5",
# entryPrice: "2.3483",
# breakEvenPrice: "2.349356735",
# markPrice: "2.39560000",
# unRealizedProfit: "0.23650000",
# liquidationPrice: "0",
# isolatedMargin: "0",
# notional: "11.97800000",
# isolatedWallet: "0",
# updateTime: "1722062678998",
# initialMargin: "2.39560000", # not in v2
# maintMargin: "0.07186800", # not in v2
# positionInitialMargin: "2.39560000", # not in v2
# openOrderInitialMargin: "0", # not in v2
# adl: "2", # not in v2
# bidNotional: "0", # not in v2
# askNotional: "0", # not in v2
# marginAsset: "USDT", # not in v2
# # the below fields are only in v2
# leverage: "5",
# maxNotionalValue: "6000000",
# marginType: "cross",
# isAutoAddMargin: "false",
# isolated: False,
# adlQuantile: "2",
#
# coinm
#
# {
# "symbol": "BTCUSD_PERP",
# "positionAmt": "2",
# "entryPrice": "37643.10000021",
# "markPrice": "38103.05510455",
# "unRealizedProfit": "0.00006413",
# "liquidationPrice": "25119.97445760",
# "leverage": "2",
# "maxQty": "1500",
# "marginType": "isolated",
# "isolatedMargin": "0.00274471",
# "isAutoAddMargin": "false",
# "positionSide": "BOTH",
# "notionalValue": "0.00524892",
# "isolatedWallet": "0.00268058"
# }
#
# inverse portfolio margin
#
# {
# "symbol": "ETHUSD_PERP",
# "positionAmt": "1",
# "entryPrice": "2422.400000007",
# "markPrice": "2424.51267823",
# "unRealizedProfit": "0.0000036",
# "liquidationPrice": "293.57678898",
# "leverage": "100",
# "positionSide": "LONG",
# "updateTime": 1707371941861,
# "maxQty": "15",
# "notionalValue": "0.00412454",
# "breakEvenPrice": "2423.368960034"
# }
#
# linear portfolio margin
#
# {
# "symbol": "BTCUSDT",
# "positionAmt": "0.01",
# "entryPrice": "44525.0",
# "markPrice": "45464.1735922",
# "unRealizedProfit": "9.39173592",
# "liquidationPrice": "38007.16308568",
# "leverage": "100",
# "positionSide": "LONG",
# "updateTime": 1707371879042,
# "maxNotionalValue": "500000.0",
# "notional": "454.64173592",
# "breakEvenPrice": "44542.81"
# }
#
marketId = self.safe_string(position, 'symbol')
market = self.safe_market(marketId, market, None, 'contract')
symbol = self.safe_string(market, 'symbol')
isolatedMarginString = self.safe_string(position, 'isolatedMargin')
leverageBrackets = self.safe_dict(self.options, 'leverageBrackets', {})
leverageBracket = self.safe_list(leverageBrackets, symbol, [])
notionalString = self.safe_string_2(position, 'notional', 'notionalValue')
notionalStringAbs = Precise.string_abs(notionalString)
maintenanceMarginPercentageString = None
for i in range(0, len(leverageBracket)):
bracket = leverageBracket[i]
if Precise.string_lt(notionalStringAbs, bracket[0]):
break
maintenanceMarginPercentageString = bracket[1]
notional = self.parse_number(notionalStringAbs)
contractsAbs = Precise.string_abs(self.safe_string(position, 'positionAmt'))
contracts = self.parse_number(contractsAbs)
unrealizedPnlString = self.safe_string(position, 'unRealizedProfit')
unrealizedPnl = self.parse_number(unrealizedPnlString)
liquidationPriceString = self.omit_zero(self.safe_string(position, 'liquidationPrice'))
liquidationPrice = self.parse_number(liquidationPriceString)
collateralString = None
marginMode = self.safe_string(position, 'marginType')
if marginMode is None and isolatedMarginString is not None:
marginMode = 'cross' if Precise.string_eq(isolatedMarginString, '0') else 'isolated'
side = None
if Precise.string_gt(notionalString, '0'):
side = 'long'
elif Precise.string_lt(notionalString, '0'):
side = 'short'
entryPriceString = self.safe_string(position, 'entryPrice')
entryPrice = self.parse_number(entryPriceString)
contractSize = self.safe_value(market, 'contractSize')
contractSizeString = self.number_to_string(contractSize)
# to notionalValue
linear = ('notional' in position)
if marginMode == 'cross':
# calculate collateral
precision = self.safe_dict(market, 'precision', {})
basePrecisionValue = self.safe_string(precision, 'base')
quotePrecisionValue = self.safe_string_2(precision, 'quote', 'price')
precisionIsUndefined = (basePrecisionValue is None) and (quotePrecisionValue is None)
if not precisionIsUndefined:
if linear:
# walletBalance = (liquidationPrice * (±1 + mmp) ± entryPrice) * contracts
onePlusMaintenanceMarginPercentageString = None
entryPriceSignString = entryPriceString
if side == 'short':
onePlusMaintenanceMarginPercentageString = Precise.string_add('1', maintenanceMarginPercentageString)
entryPriceSignString = Precise.string_mul('-1', entryPriceSignString)
else:
onePlusMaintenanceMarginPercentageString = Precise.string_add('-1', maintenanceMarginPercentageString)
inner = Precise.string_mul(liquidationPriceString, onePlusMaintenanceMarginPercentageString)
leftSide = Precise.string_add(inner, entryPriceSignString)
quotePrecision = self.precision_from_string(self.safe_string_2(precision, 'quote', 'price'))
if quotePrecision is not None:
collateralString = Precise.string_div(Precise.string_mul(leftSide, contractsAbs), '1', quotePrecision)
else:
# walletBalance = (contracts * contractSize) * (±1/entryPrice - (±1 - mmp) / liquidationPrice)
onePlusMaintenanceMarginPercentageString = None
entryPriceSignString = entryPriceString
if side == 'short':
onePlusMaintenanceMarginPercentageString = Precise.string_sub('1', maintenanceMarginPercentageString)
else:
onePlusMaintenanceMarginPercentageString = Precise.string_sub('-1', maintenanceMarginPercentageString)
entryPriceSignString = Precise.string_mul('-1', entryPriceSignString)
leftSide = Precise.string_mul(contractsAbs, contractSizeString)
rightSide = Precise.string_sub(Precise.string_div('1', entryPriceSignString), Precise.string_div(onePlusMaintenanceMarginPercentageString, liquidationPriceString))
basePrecision = self.precision_from_string(self.safe_string(precision, 'base'))
if basePrecision is not None:
collateralString = Precise.string_div(Precise.string_mul(leftSide, rightSide), '1', basePrecision)
else:
collateralString = self.safe_string(position, 'isolatedMargin')
collateralString = '0' if (collateralString is None) else collateralString
collateral = self.parse_number(collateralString)
markPrice = self.parse_number(self.omit_zero(self.safe_string(position, 'markPrice')))
timestamp = self.safe_integer(position, 'updateTime')
if timestamp == 0:
timestamp = None
maintenanceMarginPercentage = self.parse_number(maintenanceMarginPercentageString)
maintenanceMarginString = Precise.string_mul(maintenanceMarginPercentageString, notionalStringAbs)
if maintenanceMarginString is None:
# for a while, self new value was a backup to the existing calculations, but in future we might prioritize self
maintenanceMarginString = self.safe_string(position, 'maintMargin')
maintenanceMargin = self.parse_number(maintenanceMarginString)
initialMarginString = None
initialMarginPercentageString = None
leverageString = self.safe_string(position, 'leverage')
if leverageString is not None:
leverage = int(leverageString)
rational = self.is_round_number(1000 % leverage)
initialMarginPercentageString = Precise.string_div('1', leverageString, 8)
if not rational:
initialMarginPercentageString = Precise.string_add(initialMarginPercentageString, '1e-8')
unrounded = Precise.string_mul(notionalStringAbs, initialMarginPercentageString)
initialMarginString = Precise.string_div(unrounded, '1', 8)
else:
initialMarginString = self.safe_string(position, 'initialMargin')
unrounded = Precise.string_mul(initialMarginString, '1')
initialMarginPercentageString = Precise.string_div(unrounded, notionalStringAbs, 8)
marginRatio = None
percentage = None
if not Precise.string_equals(collateralString, '0'):
marginRatio = self.parse_number(Precise.string_div(Precise.string_add(Precise.string_div(maintenanceMarginString, collateralString), '5e-5'), '1', 4))
percentage = self.parse_number(Precise.string_mul(Precise.string_div(unrealizedPnlString, initialMarginString, 4), '100'))
positionSide = self.safe_string(position, 'positionSide')
hedged = positionSide != 'BOTH'
return {
'info': position,
'id': None,
'symbol': symbol,
'contracts': contracts,
'contractSize': contractSize,
'unrealizedPnl': unrealizedPnl,
'leverage': self.parse_number(leverageString),
'liquidationPrice': liquidationPrice,
'collateral': collateral,
'notional': notional,
'markPrice': markPrice,
'entryPrice': entryPrice,
'timestamp': timestamp,
'initialMargin': self.parse_number(initialMarginString),
'initialMarginPercentage': self.parse_number(initialMarginPercentageString),
'maintenanceMargin': maintenanceMargin,
'maintenanceMarginPercentage': maintenanceMarginPercentage,
'marginRatio': marginRatio,
'datetime': self.iso8601(timestamp),
'marginMode': marginMode,
'marginType': marginMode, # deprecated
'side': side,
'hedged': hedged,
'percentage': percentage,
'stopLossPrice': None,
'takeProfitPrice': None,
}
def load_leverage_brackets(self, reload=False, params={}):
self.load_markets()
# by default cache the leverage bracket
# it contains useful stuff like the maintenance margin and initial margin for positions
leverageBrackets = self.safe_dict(self.options, 'leverageBrackets')
if (leverageBrackets is None) or (reload):
defaultType = self.safe_string(self.options, 'defaultType', 'future')
type = self.safe_string(params, 'type', defaultType)
query = self.omit(params, 'type')
subType = None
subType, params = self.handle_sub_type_and_params('loadLeverageBrackets', None, params, 'linear')
isPortfolioMargin = None
isPortfolioMargin, params = self.handle_option_and_params_2(params, 'loadLeverageBrackets', 'papi', 'portfolioMargin', False)
response = None
if self.is_linear(type, subType):
if isPortfolioMargin:
response = self.papiGetUmLeverageBracket(query)
else:
response = self.fapiPrivateGetLeverageBracket(query)
elif self.is_inverse(type, subType):
if isPortfolioMargin:
response = self.papiGetCmLeverageBracket(query)
else:
response = self.dapiPrivateV2GetLeverageBracket(query)
else:
raise NotSupported(self.id + ' loadLeverageBrackets() supports linear and inverse contracts only')
self.options['leverageBrackets'] = self.create_safe_dictionary()
for i in range(0, len(response)):
entry = response[i]
marketId = self.safe_string(entry, 'symbol')
symbol = self.safe_symbol(marketId, None, None, 'contract')
brackets = self.safe_list(entry, 'brackets', [])
result = []
for j in range(0, len(brackets)):
bracket = brackets[j]
floorValue = self.safe_string_2(bracket, 'notionalFloor', 'qtyFloor')
maintenanceMarginPercentage = self.safe_string(bracket, 'maintMarginRatio')
result.append([floorValue, maintenanceMarginPercentage])
self.options['leverageBrackets'][symbol] = result
return self.options['leverageBrackets']
def fetch_leverage_tiers(self, symbols: Strings = None, params={}) -> LeverageTiers:
"""
retrieve information on the maximum leverage, and maintenance margin for trades of varying trade sizes
https://developers.binance.com/docs/derivatives/usds-margined-futures/account/rest-api/Notional-and-Leverage-Brackets
https://developers.binance.com/docs/derivatives/coin-margined-futures/account/rest-api/Notional-Bracket-for-Pair
https://developers.binance.com/docs/derivatives/portfolio-margin/account/UM-Notional-and-Leverage-Brackets
https://developers.binance.com/docs/derivatives/portfolio-margin/account/CM-Notional-and-Leverage-Brackets
:param str[]|None symbols: list of unified market symbols
:param dict [params]: extra parameters specific to the exchange API endpoint
:param boolean [params.portfolioMargin]: set to True if you would like to fetch the leverage tiers for a portfolio margin account
:param str [params.subType]: "linear" or "inverse"
:returns dict: a dictionary of `leverage tiers structures <https://docs.ccxt.com/#/?id=leverage-tiers-structure>`, indexed by market symbols
"""
self.load_markets()
type = None
type, params = self.handle_market_type_and_params('fetchLeverageTiers', None, params)
subType = None
subType, params = self.handle_sub_type_and_params('fetchLeverageTiers', None, params, 'linear')
isPortfolioMargin = None
isPortfolioMargin, params = self.handle_option_and_params_2(params, 'fetchLeverageTiers', 'papi', 'portfolioMargin', False)
response = None
if self.is_linear(type, subType):
if isPortfolioMargin:
response = self.papiGetUmLeverageBracket(params)
else:
response = self.fapiPrivateGetLeverageBracket(params)
elif self.is_inverse(type, subType):
if isPortfolioMargin:
response = self.papiGetCmLeverageBracket(params)
else:
response = self.dapiPrivateV2GetLeverageBracket(params)
else:
raise NotSupported(self.id + ' fetchLeverageTiers() supports linear and inverse contracts only')
#
# usdm
#
# [
# {
# "symbol": "SUSHIUSDT",
# "brackets": [
# {
# "bracket": 1,
# "initialLeverage": 50,
# "notionalCap": 50000,
# "notionalFloor": 0,
# "maintMarginRatio": 0.01,
# "cum": 0.0
# },
# ...
# ]
# }
# ]
#
# coinm
#
# [
# {
# "symbol":"XRPUSD_210326",
# "brackets":[
# {
# "bracket":1,
# "initialLeverage":20,
# "qtyCap":500000,
# "qtyFloor":0,
# "maintMarginRatio":0.0185,
# "cum":0.0
# }
# ]
# }
# ]
#
return self.parse_leverage_tiers(response, symbols, 'symbol')
def parse_market_leverage_tiers(self, info, market: Market = None) -> List[LeverageTier]:
"""
@ignore
:param dict info: Exchange response for 1 market
:param dict market: CCXT market
"""
#
# {
# "symbol": "SUSHIUSDT",
# "brackets": [
# {
# "bracket": 1,
# "initialLeverage": 50,
# "notionalCap": 50000,
# "notionalFloor": 0,
# "maintMarginRatio": 0.01,
# "cum": 0.0
# },
# ...
# ]
# }
#
marketId = self.safe_string(info, 'symbol')
market = self.safe_market(marketId, market, None, 'contract')
brackets = self.safe_list(info, 'brackets', [])
tiers = []
for j in range(0, len(brackets)):
bracket = brackets[j]
tiers.append({
'tier': self.safe_number(bracket, 'bracket'),
'symbol': self.safe_symbol(marketId, market),
'currency': market['quote'],
'minNotional': self.safe_number_2(bracket, 'notionalFloor', 'qtyFloor'),
'maxNotional': self.safe_number_2(bracket, 'notionalCap', 'qtyCap'),
'maintenanceMarginRate': self.safe_number(bracket, 'maintMarginRatio'),
'maxLeverage': self.safe_number(bracket, 'initialLeverage'),
'info': bracket,
})
return tiers
def fetch_position(self, symbol: str, params={}):
"""
fetch data on an open position
https://developers.binance.com/docs/derivatives/option/trade/Option-Position-Information
: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
:returns dict: a `position structure <https://docs.ccxt.com/#/?id=position-structure>`
"""
self.load_markets()
market = self.market(symbol)
if not market['option']:
raise NotSupported(self.id + ' fetchPosition() supports option markets only')
request: dict = {
'symbol': market['id'],
}
response = self.eapiPrivateGetPosition(self.extend(request, params))
#
# [
# {
# "entryPrice": "27.70000000",
# "symbol": "ETH-230426-1850-C",
# "side": "LONG",
# "quantity": "0.50000000",
# "reducibleQty": "0.50000000",
# "markValue": "10.250000000",
# "ror": "-0.2599",
# "unrealizedPNL": "-3.600000000",
# "markPrice": "20.5",
# "strikePrice": "1850.00000000",
# "positionCost": "13.85000000",
# "expiryDate": 1682496000000,
# "priceScale": 1,
# "quantityScale": 2,
# "optionSide": "CALL",
# "quoteAsset": "USDT",
# "time": 1682492427106
# }
# ]
#
return self.parse_position(response[0], market)
def fetch_option_positions(self, symbols: Strings = None, params={}):
"""
fetch data on open options positions
https://developers.binance.com/docs/derivatives/option/trade/Option-Position-Information
:param str[]|None symbols: list of unified market symbols
:param dict [params]: extra parameters specific to the exchange API endpoint
:returns dict[]: a list of `position structures <https://docs.ccxt.com/#/?id=position-structure>`
"""
self.load_markets()
symbols = self.market_symbols(symbols)
request: dict = {}
market = None
if symbols is not None:
symbol = None
if isinstance(symbols, list):
symbolsLength = len(symbols)
if symbolsLength > 1:
raise BadRequest(self.id + ' fetchPositions() symbols argument cannot contain more than 1 symbol')
symbol = symbols[0]
else:
symbol = symbols
market = self.market(symbol)
request['symbol'] = market['id']
response = self.eapiPrivateGetPosition(self.extend(request, params))
#
# [
# {
# "entryPrice": "27.70000000",
# "symbol": "ETH-230426-1850-C",
# "side": "LONG",
# "quantity": "0.50000000",
# "reducibleQty": "0.50000000",
# "markValue": "10.250000000",
# "ror": "-0.2599",
# "unrealizedPNL": "-3.600000000",
# "markPrice": "20.5",
# "strikePrice": "1850.00000000",
# "positionCost": "13.85000000",
# "expiryDate": 1682496000000,
# "priceScale": 1,
# "quantityScale": 2,
# "optionSide": "CALL",
# "quoteAsset": "USDT",
# "time": 1682492427106
# }
# ]
#
result = []
for i in range(0, len(response)):
result.append(self.parse_position(response[i], market))
return self.filter_by_array_positions(result, 'symbol', symbols, False)
def parse_position(self, position: dict, market: Market = None):
#
# {
# "entryPrice": "27.70000000",
# "symbol": "ETH-230426-1850-C",
# "side": "LONG",
# "quantity": "0.50000000",
# "reducibleQty": "0.50000000",
# "markValue": "10.250000000",
# "ror": "-0.2599",
# "unrealizedPNL": "-3.600000000",
# "markPrice": "20.5",
# "strikePrice": "1850.00000000",
# "positionCost": "13.85000000",
# "expiryDate": 1682496000000,
# "priceScale": 1,
# "quantityScale": 2,
# "optionSide": "CALL",
# "quoteAsset": "USDT",
# "time": 1682492427106
# }
#
marketId = self.safe_string(position, 'symbol')
market = self.safe_market(marketId, market, None, 'swap')
symbol = market['symbol']
side = self.safe_string_lower(position, 'side')
quantity = self.safe_string(position, 'quantity')
if side != 'long':
quantity = Precise.string_mul('-1', quantity)
timestamp = self.safe_integer(position, 'time')
return self.safe_position({
'info': position,
'id': None,
'symbol': symbol,
'entryPrice': self.safe_number(position, 'entryPrice'),
'markPrice': self.safe_number(position, 'markPrice'),
'notional': self.safe_number(position, 'markValue'),
'collateral': self.safe_number(position, 'positionCost'),
'unrealizedPnl': self.safe_number(position, 'unrealizedPNL'),
'side': side,
'contracts': self.parse_number(quantity),
'contractSize': None,
'timestamp': timestamp,
'datetime': self.iso8601(timestamp),
'hedged': None,
'maintenanceMargin': None,
'maintenanceMarginPercentage': None,
'initialMargin': None,
'initialMarginPercentage': None,
'leverage': None,
'liquidationPrice': None,
'marginRatio': None,
'marginMode': None,
'percentage': None,
})
def fetch_positions(self, symbols: Strings = None, params={}) -> List[Position]:
"""
fetch all open positions
https://developers.binance.com/docs/derivatives/usds-margined-futures/account/rest-api/Account-Information-V2
https://developers.binance.com/docs/derivatives/coin-margined-futures/account/rest-api/Account-Information
https://developers.binance.com/docs/derivatives/usds-margined-futures/trade/rest-api/Position-Information-V2
https://developers.binance.com/docs/derivatives/coin-margined-futures/trade/rest-api/Position-Information
https://developers.binance.com/docs/derivatives/option/trade/Option-Position-Information
:param str[] [symbols]: list of unified market symbols
:param dict [params]: extra parameters specific to the exchange API endpoint
:param dict [params.params]: extra parameters specific to the exchange API endpoint
:param str [params.method]: method name to call, "positionRisk", "account" or "option", default is "positionRisk"
:param bool [params.useV2]: set to True if you want to use the obsolete endpoint, where some more additional fields were provided
:returns dict[]: a list of `position structure <https://docs.ccxt.com/#/?id=position-structure>`
"""
defaultMethod = None
defaultMethod, params = self.handle_option_and_params(params, 'fetchPositions', 'method')
if defaultMethod is None:
options = self.safe_dict(self.options, 'fetchPositions')
if options is None:
defaultMethod = self.safe_string(self.options, 'fetchPositions', 'positionRisk')
else:
defaultMethod = 'positionRisk'
if defaultMethod == 'positionRisk':
return self.fetch_positions_risk(symbols, params)
elif defaultMethod == 'account':
return self.fetch_account_positions(symbols, params)
elif defaultMethod == 'option':
return self.fetch_option_positions(symbols, params)
else:
raise NotSupported(self.id + '.options["fetchPositions"]["method"] or params["method"] = "' + defaultMethod + '" is invalid, please choose between "account", "positionRisk" and "option"')
def fetch_account_positions(self, symbols: Strings = None, params={}):
"""
@ignore
fetch account positions
https://developers.binance.com/docs/derivatives/usds-margined-futures/account/rest-api/Account-Information-V2
https://developers.binance.com/docs/derivatives/coin-margined-futures/account/rest-api/Account-Information
https://developers.binance.com/docs/derivatives/usds-margined-futures/trade/rest-api/Position-Information-V2
https://developers.binance.com/docs/derivatives/coin-margined-futures/trade/rest-api/Position-Information
https://developers.binance.com/docs/derivatives/usds-margined-futures/account/rest-api/Account-Information-V3
:param str[] [symbols]: list of unified market symbols
:param dict [params]: extra parameters specific to the exchange API endpoint
:param boolean [params.portfolioMargin]: set to True if you would like to fetch positions in a portfolio margin account
:param str [params.subType]: "linear" or "inverse"
:param boolean [params.filterClosed]: set to True if you would like to filter out closed positions, default is False
:param boolean [params.useV2]: set to True if you want to use obsolete endpoint, where some more additional fields were provided
:returns dict: data on account positions
"""
if symbols is not None:
if not isinstance(symbols, list):
raise ArgumentsRequired(self.id + ' fetchPositions() requires an array argument for symbols')
self.load_markets()
self.load_leverage_brackets(False, params)
defaultType = self.safe_string(self.options, 'defaultType', 'future')
type = self.safe_string(params, 'type', defaultType)
params = self.omit(params, 'type')
subType = None
subType, params = self.handle_sub_type_and_params('fetchAccountPositions', None, params, 'linear')
isPortfolioMargin = None
isPortfolioMargin, params = self.handle_option_and_params_2(params, 'fetchAccountPositions', 'papi', 'portfolioMargin', False)
response = None
if self.is_linear(type, subType):
if isPortfolioMargin:
response = self.papiV2GetUmAccount(params)
else:
useV2 = None
useV2, params = self.handle_option_and_params(params, 'fetchAccountPositions', 'useV2', False)
if not useV2:
response = self.fapiPrivateV3GetAccount(params)
else:
response = self.fapiPrivateV2GetAccount(params)
#
# {
# "totalInitialMargin": "99.62112386",
# "totalMaintMargin": "11.95453485",
# "totalWalletBalance": "99.84331553",
# "totalUnrealizedProfit": "11.17675690",
# "totalMarginBalance": "111.02007243",
# "totalPositionInitialMargin": "99.62112386",
# "totalOpenOrderInitialMargin": "0.00000000",
# "totalCrossWalletBalance": "99.84331553",
# "totalCrossUnPnl": "11.17675690",
# "availableBalance": "11.39894857",
# "maxWithdrawAmount": "11.39894857",
# "feeTier": "0", # in v2
# "canTrade": True, # in v2
# "canDeposit": True, # in v2
# "canWithdraw": True, # in v2
# "feeBurn": True, # in v2
# "tradeGroupId": "-1",// in v2
# "updateTime": "0", # in v2
# "multiAssetsMargin": True # in v2
# "assets": [
# {
# "asset": "USDT",
# "walletBalance": "72.72317863",
# "unrealizedProfit": "11.17920750",
# "marginBalance": "83.90238613",
# "maintMargin": "11.95476475",
# "initialMargin": "99.62303962",
# "positionInitialMargin": "99.62303962",
# "openOrderInitialMargin": "0.00000000",
# "crossWalletBalance": "72.72317863",
# "crossUnPnl": "11.17920750",
# "availableBalance": "11.39916777",
# "maxWithdrawAmount": "11.39916777",
# "updateTime": "1721995605338",
# "marginAvailable": True # in v2
# },
# ... and some few supported settle currencies: USDC, BTC, ETH, BNB ..
# ],
# "positions": [
# {
# "symbol": "WLDUSDT",
# "positionSide": "BOTH",
# "positionAmt": "-849",
# "unrealizedProfit": "11.17920750",
# "isolatedMargin": "0",
# "isolatedWallet": "0",
# "notional": "-1992.46079250",
# "initialMargin": "99.62303962",
# "maintMargin": "11.95476475",
# "updateTime": "1721995760449"
# "leverage": "50", # in v2
# "entryPrice": "2.34", # in v2
# "positionInitialMargin": "118.82116614", # in v2
# "openOrderInitialMargin": "0", # in v2
# "isolated": False, # in v2
# "breakEvenPrice": "2.3395788", # in v2
# "maxNotional": "25000", # in v2
# "bidNotional": "0", # in v2
# "askNotional": "0" # in v2
# },
# ...
# ]
# }
#
elif self.is_inverse(type, subType):
if isPortfolioMargin:
response = self.papiGetCmAccount(params)
else:
response = self.dapiPrivateGetAccount(params)
else:
raise NotSupported(self.id + ' fetchPositions() supports linear and inverse contracts only')
filterClosed = None
filterClosed, params = self.handle_option_and_params(params, 'fetchAccountPositions', 'filterClosed', False)
result = self.parse_account_positions(response, filterClosed)
symbols = self.market_symbols(symbols)
return self.filter_by_array_positions(result, 'symbol', symbols, False)
def fetch_positions_risk(self, symbols: Strings = None, params={}):
"""
@ignore
fetch positions risk
https://developers.binance.com/docs/derivatives/usds-margined-futures/trade/rest-api/Position-Information-V2
https://developers.binance.com/docs/derivatives/coin-margined-futures/trade/rest-api/Position-Information
https://developers.binance.com/docs/derivatives/portfolio-margin/account/Query-UM-Position-Information
https://developers.binance.com/docs/derivatives/portfolio-margin/account/Query-CM-Position-Information
https://developers.binance.com/docs/derivatives/usds-margined-futures/trade/rest-api/Position-Information-V3
:param str[]|None symbols: list of unified market symbols
:param dict [params]: extra parameters specific to the exchange API endpoint
:param boolean [params.portfolioMargin]: set to True if you would like to fetch positions for a portfolio margin account
:param str [params.subType]: "linear" or "inverse"
:param bool [params.useV2]: set to True if you want to use the obsolete endpoint, where some more additional fields were provided
:returns dict: data on the positions risk
"""
if symbols is not None:
if not isinstance(symbols, list):
raise ArgumentsRequired(self.id + ' fetchPositionsRisk() requires an array argument for symbols')
self.load_markets()
self.load_leverage_brackets(False, params)
request: dict = {}
defaultType = 'future'
defaultType = self.safe_string(self.options, 'defaultType', defaultType)
type = self.safe_string(params, 'type', defaultType)
subType = None
subType, params = self.handle_sub_type_and_params('fetchPositionsRisk', None, params, 'linear')
isPortfolioMargin = None
isPortfolioMargin, params = self.handle_option_and_params_2(params, 'fetchPositionsRisk', 'papi', 'portfolioMargin', False)
params = self.omit(params, 'type')
response = None
if self.is_linear(type, subType):
if isPortfolioMargin:
response = self.papiGetUmPositionRisk(self.extend(request, params))
else:
useV2 = None
useV2, params = self.handle_option_and_params(params, 'fetchPositionsRisk', 'useV2', False)
params = self.extend(request, params)
if not useV2:
response = self.fapiPrivateV3GetPositionRisk(params)
else:
response = self.fapiPrivateV2GetPositionRisk(params)
#
# [
# {
# symbol: "WLDUSDT",
# positionSide: "BOTH",
# positionAmt: "5",
# entryPrice: "2.3483",
# breakEvenPrice: "2.349356735",
# markPrice: "2.39560000",
# unRealizedProfit: "0.23650000",
# liquidationPrice: "0",
# isolatedMargin: "0",
# notional: "11.97800000",
# isolatedWallet: "0",
# updateTime: "1722062678998",
# initialMargin: "2.39560000", # added in v3
# maintMargin: "0.07186800", # added in v3
# positionInitialMargin: "2.39560000", # added in v3
# openOrderInitialMargin: "0", # added in v3
# adl: "2", # added in v3
# bidNotional: "0", # added in v3
# askNotional: "0", # added in v3
# marginAsset: "USDT", # added in v3
# },
# ]
#
elif self.is_inverse(type, subType):
if isPortfolioMargin:
response = self.papiGetCmPositionRisk(self.extend(request, params))
else:
response = self.dapiPrivateGetPositionRisk(self.extend(request, params))
else:
raise NotSupported(self.id + ' fetchPositionsRisk() supports linear and inverse contracts only')
# ### Response examples ###
#
# For One-way position mode:
#
# [
# {
# "symbol": "BTCUSDT",
# "positionSide": "BOTH",
# "positionAmt": "0.000",
# "entryPrice": "0.00000",
# "markPrice": "6679.50671178",
# "unRealizedProfit": "0.00000000",
# "liquidationPrice": "0",
# "isolatedMargin": "0.00000000",
# "marginType": "isolated",
# "isAutoAddMargin": "false",
# "leverage": "10",
# "maxNotionalValue": "20000000",
# "updateTime": 0
# }
# ]
#
# For Hedge position mode:
#
# [
# {
# "entryPrice": "6563.66500",
# "marginType": "isolated",
# "isAutoAddMargin": "false",
# "isolatedMargin": "15517.54150468",
# "leverage": "10",
# "liquidationPrice": "5930.78",
# "markPrice": "6679.50671178",
# "maxNotionalValue": "20000000",
# "positionSide": "LONG",
# "positionAmt": "20.000", # negative value for 'SHORT'
# "symbol": "BTCUSDT",
# "unRealizedProfit": "2316.83423560"
# "updateTime": 1625474304765
# },
# .. second dict is similar, but with `positionSide: SHORT`
# ]
#
# inverse portfolio margin:
#
# [
# {
# "symbol": "ETHUSD_PERP",
# "positionAmt": "1",
# "entryPrice": "2422.400000007",
# "markPrice": "2424.51267823",
# "unRealizedProfit": "0.0000036",
# "liquidationPrice": "293.57678898",
# "leverage": "100",
# "positionSide": "LONG",
# "updateTime": 1707371941861,
# "maxQty": "15",
# "notionalValue": "0.00412454",
# "breakEvenPrice": "2423.368960034"
# }
# ]
#
# linear portfolio margin:
#
# [
# {
# "symbol": "BTCUSDT",
# "positionAmt": "0.01",
# "entryPrice": "44525.0",
# "markPrice": "45464.1735922",
# "unRealizedProfit": "9.39173592",
# "liquidationPrice": "38007.16308568",
# "leverage": "100",
# "positionSide": "LONG",
# "updateTime": 1707371879042,
# "maxNotionalValue": "500000.0",
# "notional": "454.64173592",
# "breakEvenPrice": "44542.81"
# }
# ]
#
result = []
for i in range(0, len(response)):
rawPosition = response[i]
entryPriceString = self.safe_string(rawPosition, 'entryPrice')
if Precise.string_gt(entryPriceString, '0'):
result.append(self.parse_position_risk(response[i]))
symbols = self.market_symbols(symbols)
return self.filter_by_array_positions(result, 'symbol', symbols, False)
def fetch_funding_history(self, symbol: Str = None, since: Int = None, limit: Int = None, params={}):
"""
fetch the history of funding payments paid and received on self account
https://developers.binance.com/docs/derivatives/usds-margined-futures/account/rest-api/Get-Income-History
https://developers.binance.com/docs/derivatives/coin-margined-futures/account/rest-api/Get-Income-History
https://developers.binance.com/docs/derivatives/portfolio-margin/account/Get-UM-Income-History
https://developers.binance.com/docs/derivatives/portfolio-margin/account/Get-CM-Income-History
:param str symbol: unified market symbol
:param int [since]: the earliest time in ms to fetch funding history for
:param int [limit]: the maximum number of funding history structures to retrieve
:param dict [params]: extra parameters specific to the exchange API endpoint
:param int [params.until]: timestamp in ms of the latest funding history entry
:param boolean [params.portfolioMargin]: set to True if you would like to fetch the funding history for a portfolio margin account
:param str [params.subType]: "linear" or "inverse"
:returns dict: a `funding history structure <https://docs.ccxt.com/#/?id=funding-history-structure>`
"""
self.load_markets()
market = None
request: dict = {
'incomeType': 'FUNDING_FEE', # "TRANSFER""WELCOME_BONUS", "REALIZED_PNL""FUNDING_FEE", "COMMISSION" and "INSURANCE_CLEAR"
}
if symbol is not None:
market = self.market(symbol)
request['symbol'] = market['id']
if not market['swap']:
raise NotSupported(self.id + ' fetchFundingHistory() supports swap contracts only')
subType = None
subType, params = self.handle_sub_type_and_params('fetchFundingHistory', market, params, 'linear')
isPortfolioMargin = None
isPortfolioMargin, params = self.handle_option_and_params_2(params, 'fetchFundingHistory', 'papi', 'portfolioMargin', False)
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
defaultType = self.safe_string_2(self.options, 'fetchFundingHistory', 'defaultType', 'future')
type = self.safe_string(params, 'type', defaultType)
params = self.omit(params, 'type')
response = None
if self.is_linear(type, subType):
if isPortfolioMargin:
response = self.papiGetUmIncome(self.extend(request, params))
else:
response = self.fapiPrivateGetIncome(self.extend(request, params))
elif self.is_inverse(type, subType):
if isPortfolioMargin:
response = self.papiGetCmIncome(self.extend(request, params))
else:
response = self.dapiPrivateGetIncome(self.extend(request, params))
else:
raise NotSupported(self.id + ' fetchFundingHistory() supports linear and inverse contracts only')
return self.parse_incomes(response, market, since, limit)
def set_leverage(self, leverage: int, symbol: Str = None, params={}):
"""
set the level of leverage for a market
https://developers.binance.com/docs/derivatives/usds-margined-futures/trade/rest-api/Change-Initial-Leverage
https://developers.binance.com/docs/derivatives/coin-margined-futures/trade/rest-api/Change-Initial-Leverage
https://developers.binance.com/docs/derivatives/portfolio-margin/account/Change-UM-Initial-Leverage
https://developers.binance.com/docs/derivatives/portfolio-margin/account/Change-CM-Initial-Leverage
:param float leverage: the rate of leverage
:param str symbol: unified market symbol
:param dict [params]: extra parameters specific to the exchange API endpoint
:param boolean [params.portfolioMargin]: set to True if you would like to set the leverage for a trading pair in a portfolio margin account
:returns dict: response from the exchange
"""
if symbol is None:
raise ArgumentsRequired(self.id + ' setLeverage() requires a symbol argument')
# WARNING: THIS WILL INCREASE LIQUIDATION PRICE FOR OPEN ISOLATED LONG POSITIONS
# AND DECREASE LIQUIDATION PRICE FOR OPEN ISOLATED SHORT POSITIONS
if (leverage < 1) or (leverage > 125):
raise BadRequest(self.id + ' leverage should be between 1 and 125')
self.load_markets()
market = self.market(symbol)
request: dict = {
'symbol': market['id'],
'leverage': leverage,
}
isPortfolioMargin = None
isPortfolioMargin, params = self.handle_option_and_params_2(params, 'setLeverage', 'papi', 'portfolioMargin', False)
response = None
if market['linear']:
if isPortfolioMargin:
response = self.papiPostUmLeverage(self.extend(request, params))
else:
response = self.fapiPrivatePostLeverage(self.extend(request, params))
elif market['inverse']:
if isPortfolioMargin:
response = self.papiPostCmLeverage(self.extend(request, params))
else:
response = self.dapiPrivatePostLeverage(self.extend(request, params))
else:
raise NotSupported(self.id + ' setLeverage() supports linear and inverse contracts only')
return response
def set_margin_mode(self, marginMode: str, symbol: Str = None, params={}):
"""
set margin mode to 'cross' or 'isolated'
https://developers.binance.com/docs/derivatives/usds-margined-futures/trade/rest-api/Change-Margin-Type
https://developers.binance.com/docs/derivatives/coin-margined-futures/trade/rest-api/Change-Margin-Type
: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')
#
# {"code": -4048 , "msg": "Margin type cannot be changed if there exists position."}
#
# or
#
# {"code": 200, "msg": "success"}
#
marginMode = marginMode.upper()
if marginMode == 'CROSS':
marginMode = 'CROSSED'
if (marginMode != 'ISOLATED') and (marginMode != 'CROSSED'):
raise BadRequest(self.id + ' marginMode must be either isolated or cross')
self.load_markets()
market = self.market(symbol)
request: dict = {
'symbol': market['id'],
'marginType': marginMode,
}
response = None
try:
if market['linear']:
response = self.fapiPrivatePostMarginType(self.extend(request, params))
elif market['inverse']:
response = self.dapiPrivatePostMarginType(self.extend(request, params))
else:
raise NotSupported(self.id + ' setMarginMode() supports linear and inverse contracts only')
except Exception as e:
# not an error
# https://github.com/ccxt/ccxt/issues/11268
# https://github.com/ccxt/ccxt/pull/11624
# POST https://fapi.binance.com/fapi/v1/marginType 400 Bad Request
# binanceusdm
if isinstance(e, MarginModeAlreadySet):
throwMarginModeAlreadySet = self.safe_bool(self.options, 'throwMarginModeAlreadySet', False)
if throwMarginModeAlreadySet:
raise e
else:
response = {'code': -4046, 'msg': 'No need to change margin type.'}
else:
raise e
return response
def set_position_mode(self, hedged: bool, symbol: Str = None, params={}):
"""
set hedged to True or False for a market
https://developers.binance.com/docs/derivatives/usds-margined-futures/trade/rest-api/Change-Position-Mode
https://developers.binance.com/docs/derivatives/coin-margined-futures/trade/rest-api/Change-Position-Mode
https://developers.binance.com/docs/derivatives/portfolio-margin/account/Get-UM-Current-Position-Mode
https://developers.binance.com/docs/derivatives/portfolio-margin/account/Get-CM-Current-Position-Mode
:param bool hedged: set to True to use dualSidePosition
:param str symbol: not used by binance setPositionMode()
:param dict [params]: extra parameters specific to the exchange API endpoint
:param boolean [params.portfolioMargin]: set to True if you would like to set the position mode for a portfolio margin account
:param str [params.subType]: "linear" or "inverse"
:returns dict: response from the exchange
"""
market = None
if symbol is not None:
market = self.market(symbol)
type = None
type, params = self.handle_market_type_and_params('setPositionMode', market, params)
subType = None
subType, params = self.handle_sub_type_and_params('setPositionMode', market, params)
isPortfolioMargin = None
isPortfolioMargin, params = self.handle_option_and_params_2(params, 'setPositionMode', 'papi', 'portfolioMargin', False)
dualSidePosition = None
if hedged:
dualSidePosition = 'true'
else:
dualSidePosition = 'false'
request: dict = {
'dualSidePosition': dualSidePosition,
}
response = None
if self.is_inverse(type, subType):
if isPortfolioMargin:
response = self.papiPostCmPositionSideDual(self.extend(request, params))
else:
response = self.dapiPrivatePostPositionSideDual(self.extend(request, params))
elif self.is_linear(type, subType):
if isPortfolioMargin:
response = self.papiPostUmPositionSideDual(self.extend(request, params))
else:
response = self.fapiPrivatePostPositionSideDual(self.extend(request, params))
else:
raise BadRequest(self.id + ' setPositionMode() supports linear and inverse contracts only')
#
# {
# "code": 200,
# "msg": "success"
# }
#
return response
def fetch_leverages(self, symbols: Strings = None, params={}) -> Leverages:
"""
fetch the set leverage for all markets
https://developers.binance.com/docs/derivatives/usds-margined-futures/account/rest-api/Account-Information-V2
https://developers.binance.com/docs/derivatives/coin-margined-futures/account/rest-api/Account-Information
https://developers.binance.com/docs/derivatives/portfolio-margin/account/Get-UM-Account-Detail
https://developers.binance.com/docs/derivatives/portfolio-margin/account/Get-CM-Account-Detail
https://developers.binance.com/docs/derivatives/usds-margined-futures/account/rest-api/Symbol-Config
:param str[] [symbols]: a list of unified market symbols
:param dict [params]: extra parameters specific to the exchange API endpoint
:param str [params.subType]: "linear" or "inverse"
:returns dict: a list of `leverage structures <https://docs.ccxt.com/#/?id=leverage-structure>`
"""
self.load_markets()
self.load_leverage_brackets(False, params)
type = None
type, params = self.handle_market_type_and_params('fetchLeverages', None, params)
subType = None
subType, params = self.handle_sub_type_and_params('fetchLeverages', None, params, 'linear')
isPortfolioMargin = None
isPortfolioMargin, params = self.handle_option_and_params_2(params, 'fetchLeverages', 'papi', 'portfolioMargin', False)
response = None
if self.is_linear(type, subType):
if isPortfolioMargin:
response = self.papiGetUmAccount(params)
else:
response = self.fapiPrivateGetSymbolConfig(params)
elif self.is_inverse(type, subType):
if isPortfolioMargin:
response = self.papiGetCmAccount(params)
else:
response = self.dapiPrivateGetAccount(params)
else:
raise NotSupported(self.id + ' fetchLeverages() supports linear and inverse contracts only')
leverages = self.safe_list(response, 'positions', [])
if isinstance(response, list):
leverages = response
return self.parse_leverages(leverages, symbols, 'symbol')
def parse_leverage(self, leverage: dict, market: Market = None) -> Leverage:
marketId = self.safe_string(leverage, 'symbol')
marginModeRaw = self.safe_bool(leverage, 'isolated')
marginMode = None
if marginModeRaw is not None:
marginMode = 'isolated' if marginModeRaw else 'cross'
marginTypeRaw = self.safe_string_lower(leverage, 'marginType')
if marginTypeRaw is not None:
marginMode = 'cross' if (marginTypeRaw == 'crossed') else 'isolated'
side = self.safe_string_lower(leverage, 'positionSide')
longLeverage = None
shortLeverage = None
leverageValue = self.safe_integer(leverage, 'leverage')
if (side is None) or (side == 'both'):
longLeverage = leverageValue
shortLeverage = leverageValue
elif side == 'long':
longLeverage = leverageValue
elif side == 'short':
shortLeverage = leverageValue
return {
'info': leverage,
'symbol': self.safe_symbol(marketId, market),
'marginMode': marginMode,
'longLeverage': longLeverage,
'shortLeverage': shortLeverage,
}
def fetch_settlement_history(self, symbol: Str = None, since: Int = None, limit: Int = None, params={}):
"""
fetches historical settlement records
https://developers.binance.com/docs/derivatives/option/market-data/Historical-Exercise-Records
:param str symbol: unified market symbol of the settlement history
:param int [since]: timestamp in ms
:param int [limit]: number of records, default 100, max 100
:param dict [params]: exchange specific params
:returns dict[]: a list of `settlement history objects <https://docs.ccxt.com/#/?id=settlement-history-structure>`
"""
self.load_markets()
market = None if (symbol is None) else self.market(symbol)
type = None
type, params = self.handle_market_type_and_params('fetchSettlementHistory', market, params)
if type != 'option':
raise NotSupported(self.id + ' fetchSettlementHistory() supports option markets only')
request: dict = {}
if symbol is not None:
symbol = market['symbol']
request['underlying'] = market['baseId'] + market['quoteId']
if since is not None:
request['startTime'] = since
if limit is not None:
request['limit'] = limit
response = self.eapiPublicGetExerciseHistory(self.extend(request, params))
#
# [
# {
# "symbol": "ETH-230223-1900-P",
# "strikePrice": "1900",
# "realStrikePrice": "1665.5897334",
# "expiryDate": 1677139200000,
# "strikeResult": "REALISTIC_VALUE_STRICKEN"
# }
# ]
#
settlements = self.parse_settlements(response, market)
sorted = self.sort_by(settlements, 'timestamp')
return self.filter_by_symbol_since_limit(sorted, symbol, since, limit)
def fetch_my_settlement_history(self, symbol: Str = None, since: Int = None, limit: Int = None, params={}):
"""
fetches historical settlement records of the user
https://developers.binance.com/docs/derivatives/option/trade/User-Exercise-Record
:param str symbol: unified market symbol of the settlement history
:param int [since]: timestamp in ms
:param int [limit]: number of records
:param dict [params]: exchange specific params
:returns dict[]: a list of [settlement history objects]
"""
self.load_markets()
market = None if (symbol is None) else self.market(symbol)
type = None
type, params = self.handle_market_type_and_params('fetchMySettlementHistory', market, params)
if type != 'option':
raise NotSupported(self.id + ' fetchMySettlementHistory() supports option markets only')
request: dict = {}
if symbol is not None:
request['symbol'] = market['id']
symbol = market['symbol']
if since is not None:
request['startTime'] = since
if limit is not None:
request['limit'] = limit
response = self.eapiPrivateGetExerciseRecord(self.extend(request, params))
#
# [
# {
# "id": "1125899906842897036",
# "currency": "USDT",
# "symbol": "BTC-230728-30000-C",
# "exercisePrice": "30000.00000000",
# "markPrice": "29160.71284993",
# "quantity": "1.00000000",
# "amount": "0.00000000",
# "fee": "0.00000000",
# "createDate": 1690531200000,
# "priceScale": 0,
# "quantityScale": 2,
# "optionSide": "CALL",
# "positionSide": "LONG",
# "quoteAsset": "USDT"
# }
# ]
#
settlements = self.parse_settlements(response, market)
sorted = self.sort_by(settlements, 'timestamp')
return self.filter_by_symbol_since_limit(sorted, symbol, since, limit)
def parse_settlement(self, settlement, market):
#
# fetchSettlementHistory
#
# {
# "symbol": "ETH-230223-1900-P",
# "strikePrice": "1900",
# "realStrikePrice": "1665.5897334",
# "expiryDate": 1677139200000,
# "strikeResult": "REALISTIC_VALUE_STRICKEN"
# }
#
# fetchMySettlementHistory
#
# {
# "id": "1125899906842897036",
# "currency": "USDT",
# "symbol": "BTC-230728-30000-C",
# "exercisePrice": "30000.00000000",
# "markPrice": "29160.71284993",
# "quantity": "1.00000000",
# "amount": "0.00000000",
# "fee": "0.00000000",
# "createDate": 1690531200000,
# "priceScale": 0,
# "quantityScale": 2,
# "optionSide": "CALL",
# "positionSide": "LONG",
# "quoteAsset": "USDT"
# }
#
timestamp = self.safe_integer_2(settlement, 'expiryDate', 'createDate')
marketId = self.safe_string(settlement, 'symbol')
return {
'info': settlement,
'symbol': self.safe_symbol(marketId, market),
'price': self.safe_number_2(settlement, 'realStrikePrice', 'exercisePrice'),
'timestamp': timestamp,
'datetime': self.iso8601(timestamp),
}
def parse_settlements(self, settlements, market):
#
# fetchSettlementHistory
#
# [
# {
# "symbol": "ETH-230223-1900-P",
# "strikePrice": "1900",
# "realStrikePrice": "1665.5897334",
# "expiryDate": 1677139200000,
# "strikeResult": "EXTRINSIC_VALUE_EXPIRED"
# }
# ]
#
# fetchMySettlementHistory
#
# [
# {
# "id": "1125899906842897036",
# "currency": "USDT",
# "symbol": "BTC-230728-30000-C",
# "exercisePrice": "30000.00000000",
# "markPrice": "29160.71284993",
# "quantity": "1.00000000",
# "amount": "0.00000000",
# "fee": "0.00000000",
# "createDate": 1690531200000,
# "priceScale": 0,
# "quantityScale": 2,
# "optionSide": "CALL",
# "positionSide": "LONG",
# "quoteAsset": "USDT"
# }
# ]
#
result = []
for i in range(0, len(settlements)):
result.append(self.parse_settlement(settlements[i], market))
return result
def fetch_ledger_entry(self, id: str, code: Str = None, params={}) -> LedgerEntry:
"""
fetch the history of changes, actions done by the user or operations that altered the balance of the user
https://developers.binance.com/docs/derivatives/option/account/Account-Funding-Flow
:param str id: the identification number of the ledger entry
:param str code: unified currency code
:param dict [params]: extra parameters specific to the exchange API endpoint
:returns dict: a `ledger structure <https://docs.ccxt.com/#/?id=ledger>`
"""
self.load_markets()
type = None
type, params = self.handle_market_type_and_params('fetchLedgerEntry', None, params)
if type != 'option':
raise BadRequest(self.id + ' fetchLedgerEntry() can only be used for type option')
self.check_required_argument('fetchLedgerEntry', code, 'code')
currency = self.currency(code)
request: dict = {
'recordId': id,
'currency': currency['id'],
}
response = self.eapiPrivateGetBill(self.extend(request, params))
#
# [
# {
# "id": "1125899906845701870",
# "asset": "USDT",
# "amount": "-0.16518203",
# "type": "FEE",
# "createDate": 1676621042489
# }
# ]
#
first = self.safe_dict(response, 0, response)
return self.parse_ledger_entry(first, currency)
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://developers.binance.com/docs/derivatives/option/account/Account-Funding-Flow
https://developers.binance.com/docs/derivatives/usds-margined-futures/account/rest-api/Get-Income-History
https://developers.binance.com/docs/derivatives/coin-margined-futures/account/rest-api/Get-Income-History
https://developers.binance.com/docs/derivatives/portfolio-margin/account/Get-UM-Income-History
https://developers.binance.com/docs/derivatives/portfolio-margin/account/Get-CM-Income-History
:param str [code]: unified currency code
:param int [since]: timestamp in ms of the earliest ledger entry
:param int [limit]: max number of ledger entries to return
:param dict [params]: extra parameters specific to the exchange API endpoint
:param int [params.until]: timestamp in ms of the latest ledger entry
: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.portfolioMargin]: set to True if you would like to fetch the ledger for a portfolio margin account
:param str [params.subType]: "linear" or "inverse"
:returns dict: a `ledger structure <https://docs.ccxt.com/#/?id=ledger>`
"""
self.load_markets()
paginate = False
paginate, params = self.handle_option_and_params(params, 'fetchLedger', 'paginate')
if paginate:
return self.fetch_paginated_call_dynamic('fetchLedger', code, since, limit, params, None, False)
type = None
subType = None
currency = None
if code is not None:
currency = self.currency(code)
request: dict = {}
type, params = self.handle_market_type_and_params('fetchLedger', None, params)
subType, params = self.handle_sub_type_and_params('fetchLedger', None, params)
if since is not None:
request['startTime'] = since
if limit is not None:
request['limit'] = limit
until = self.safe_integer(params, 'until')
if until is not None:
params = self.omit(params, 'until')
request['endTime'] = until
isPortfolioMargin = None
isPortfolioMargin, params = self.handle_option_and_params_2(params, 'fetchLedger', 'papi', 'portfolioMargin', False)
response = None
if type == 'option':
self.check_required_argument('fetchLedger', code, 'code')
request['currency'] = currency['id']
response = self.eapiPrivateGetBill(self.extend(request, params))
elif self.is_linear(type, subType):
if isPortfolioMargin:
response = self.papiGetUmIncome(self.extend(request, params))
else:
response = self.fapiPrivateGetIncome(self.extend(request, params))
elif self.is_inverse(type, subType):
if isPortfolioMargin:
response = self.papiGetCmIncome(self.extend(request, params))
else:
response = self.dapiPrivateGetIncome(self.extend(request, params))
else:
raise NotSupported(self.id + ' fetchLedger() supports contract wallets only')
#
# options(eapi)
#
# [
# {
# "id": "1125899906845701870",
# "asset": "USDT",
# "amount": "-0.16518203",
# "type": "FEE",
# "createDate": 1676621042489
# }
# ]
#
# futures(fapi, dapi, papi)
#
# [
# {
# "symbol": "",
# "incomeType": "TRANSFER",
# "income": "10.00000000",
# "asset": "USDT",
# "time": 1677645250000,
# "info": "TRANSFER",
# "tranId": 131001573082,
# "tradeId": ""
# }
# ]
#
return self.parse_ledger(response, currency, since, limit)
def parse_ledger_entry(self, item: dict, currency: Currency = None) -> LedgerEntry:
#
# options(eapi)
#
# {
# "id": "1125899906845701870",
# "asset": "USDT",
# "amount": "-0.16518203",
# "type": "FEE",
# "createDate": 167662104241
# }
#
# futures(fapi, dapi, papi)
#
# {
# "symbol": "",
# "incomeType": "TRANSFER",
# "income": "10.00000000",
# "asset": "USDT",
# "time": 1677645250000,
# "info": "TRANSFER",
# "tranId": 131001573082,
# "tradeId": ""
# }
#
amount = self.safe_string_2(item, 'amount', 'income')
direction = None
if Precise.string_le(amount, '0'):
direction = 'out'
amount = Precise.string_mul('-1', amount)
else:
direction = 'in'
currencyId = self.safe_string(item, 'asset')
code = self.safe_currency_code(currencyId, currency)
currency = self.safe_currency(currencyId, currency)
timestamp = self.safe_integer_2(item, 'createDate', 'time')
type = self.safe_string_2(item, 'type', 'incomeType')
return self.safe_ledger_entry({
'info': item,
'id': self.safe_string_2(item, 'id', 'tranId'),
'direction': direction,
'account': None,
'referenceAccount': None,
'referenceId': self.safe_string(item, 'tradeId'),
'type': self.parse_ledger_entry_type(type),
'currency': code,
'amount': self.parse_number(amount),
'timestamp': timestamp,
'datetime': self.iso8601(timestamp),
'before': None,
'after': None,
'status': None,
'fee': None,
}, currency)
def parse_ledger_entry_type(self, type):
ledgerType: dict = {
'FEE': 'fee',
'FUNDING_FEE': 'fee',
'OPTIONS_PREMIUM_FEE': 'fee',
'POSITION_LIMIT_INCREASE_FEE': 'fee',
'CONTRACT': 'trade',
'REALIZED_PNL': 'trade',
'TRANSFER': 'transfer',
'CROSS_COLLATERAL_TRANSFER': 'transfer',
'INTERNAL_TRANSFER': 'transfer',
'COIN_SWAP_DEPOSIT': 'deposit',
'COIN_SWAP_WITHDRAW': 'withdrawal',
'OPTIONS_SETTLE_PROFIT': 'settlement',
'DELIVERED_SETTELMENT': 'settlement',
'WELCOME_BONUS': 'cashback',
'CONTEST_REWARD': 'cashback',
'COMMISSION_REBATE': 'rebate',
'API_REBATE': 'rebate',
'REFERRAL_KICKBACK': 'referral',
'COMMISSION': 'commission',
}
return self.safe_string(ledgerType, type, type)
def get_network_code_by_network_url(self, currencyCode: str, depositUrl: Str = None) -> Str:
# depositUrl is like : https://bscscan.com/address/0xEF238AB229342849..
if depositUrl is None:
return None
networkCode = None
currency = self.currency(currencyCode)
networks = self.safe_dict(currency, 'networks', {})
networkCodes = list(networks.keys())
for i in range(0, len(networkCodes)):
currentNetworkCode = networkCodes[i]
info = self.safe_dict(networks[currentNetworkCode], 'info', {})
siteUrl = self.safe_string(info, 'contractAddressUrl')
# check if url matches the field's value
if siteUrl is not None and depositUrl.startswith(self.get_base_domain_from_url(siteUrl)):
networkCode = currentNetworkCode
return networkCode
def get_base_domain_from_url(self, url: Str) -> Str:
if url is None:
return None
urlParts = url.split('/')
scheme = self.safe_string(urlParts, 0)
if scheme is None:
return None
domain = self.safe_string(urlParts, 2)
if domain is None:
return None
return scheme + '//' + domain + '/'
def sign(self, path, api='public', method='GET', params={}, headers=None, body=None):
urls = self.urls
if not (api in urls['api']):
raise NotSupported(self.id + ' does not have a testnet/sandbox URL for ' + api + ' endpoints')
url = self.urls['api'][api]
url += '/' + path
if path == 'historicalTrades':
if self.apiKey:
headers = {
'X-MBX-APIKEY': self.apiKey,
}
else:
raise AuthenticationError(self.id + ' historicalTrades endpoint requires `apiKey` credential')
userDataStream = (path == 'userDataStream') or (path == 'listenKey')
if userDataStream:
if self.apiKey:
# v1 special case for userDataStream
headers = {
'X-MBX-APIKEY': self.apiKey,
'Content-Type': 'application/x-www-form-urlencoded',
}
if method != 'GET':
body = self.urlencode(params)
else:
raise AuthenticationError(self.id + ' userDataStream endpoint requires `apiKey` credential')
elif (api == 'private') or (api == 'eapiPrivate') or (api == 'sapi' and path != 'system/status') or (api == 'sapiV2') or (api == 'sapiV3') or (api == 'sapiV4') or (api == 'dapiPrivate') or (api == 'dapiPrivateV2') or (api == 'fapiPrivate') or (api == 'fapiPrivateV2') or (api == 'fapiPrivateV3') or (api == 'papiV2' or api == 'papi' and path != 'ping'):
self.check_required_credentials()
if (url.find('testnet.binancefuture.com') > -1) and self.isSandboxModeEnabled and (not self.safe_bool(self.options, 'disableFuturesSandboxWarning')):
raise NotSupported(self.id + ' testnet/sandbox mode is not supported for futures anymore, please check the deprecation announcement https://t.me/ccxt_announcements/92 and consider using the demo trading instead.')
if method == 'POST' and ((path == 'order') or (path == 'sor/order')):
# inject in implicit API calls
newClientOrderId = self.safe_string(params, 'newClientOrderId')
if newClientOrderId is None:
isSpotOrMargin = (api.find('sapi') > -1 or api == 'private')
marketType = 'spot' if isSpotOrMargin else 'future'
defaultId = 'x-xcKtGhcu' if (not isSpotOrMargin) else 'x-TKT5PX2F'
broker = self.safe_dict(self.options, 'broker', {})
brokerId = self.safe_string(broker, marketType, defaultId)
params['newClientOrderId'] = brokerId + self.uuid22()
query = None
# handle batchOrders
if (path == 'batchOrders') and ((method == 'POST') or (method == 'PUT')):
batchOrders = self.safe_list(params, 'batchOrders')
checkedBatchOrders = batchOrders
if method == 'POST' and api == 'fapiPrivate':
# check broker id if batchOrders are called with fapiPrivatePostBatchOrders
checkedBatchOrders = []
for i in range(0, len(batchOrders)):
batchOrder = batchOrders[i]
newClientOrderId = self.safe_string(batchOrder, 'newClientOrderId')
if newClientOrderId is None:
defaultId = 'x-xcKtGhcu' # batchOrders can not be spot or margin
broker = self.safe_dict(self.options, 'broker', {})
brokerId = self.safe_string(broker, 'future', defaultId)
newClientOrderId = brokerId + self.uuid22()
batchOrder['newClientOrderId'] = newClientOrderId
checkedBatchOrders.append(batchOrder)
queryBatch = (self.json(checkedBatchOrders))
params['batchOrders'] = queryBatch
defaultRecvWindow = self.safe_integer(self.options, 'recvWindow')
extendedParams = self.extend({
'timestamp': self.nonce(),
}, params)
if defaultRecvWindow is not None:
extendedParams['recvWindow'] = defaultRecvWindow
recvWindow = self.safe_integer(params, 'recvWindow')
if recvWindow is not None:
extendedParams['recvWindow'] = recvWindow
if (api == 'sapi') and (path == 'asset/dust'):
query = self.urlencode_with_array_repeat(extendedParams)
elif (path == 'batchOrders') or (path.find('sub-account') >= 0) or (path == 'capital/withdraw/apply') or (path.find('staking') >= 0) or (path.find('simple-earn') >= 0):
if (method == 'DELETE') and (path == 'batchOrders'):
orderidlist = self.safe_list(extendedParams, 'orderidlist', [])
origclientorderidlist = self.safe_list_2(extendedParams, 'origclientorderidlist', 'origClientOrderIdList', [])
extendedParams = self.omit(extendedParams, ['orderidlist', 'origclientorderidlist', 'origClientOrderIdList'])
if 'symbol' in extendedParams:
extendedParams['symbol'] = self.encode_uri_component(extendedParams['symbol'])
query = self.rawencode(extendedParams)
orderidlistLength = len(orderidlist)
origclientorderidlistLength = len(origclientorderidlist)
if orderidlistLength > 0:
query = query + '&' + 'orderidlist=%5B' + '%2C'.join(orderidlist) + '%5D'
if origclientorderidlistLength > 0:
# wrap clientOrderids around ""
newClientOrderIds = []
for i in range(0, origclientorderidlistLength):
newClientOrderIds.append('%22' + origclientorderidlist[i] + '%22')
query = query + '&' + 'origclientorderidlist=%5B' + '%2C'.join(newClientOrderIds) + '%5D'
else:
query = self.rawencode(extendedParams)
else:
query = self.urlencode(extendedParams)
signature = None
if self.secret.find('PRIVATE KEY') > -1:
if len(self.secret) > 120:
signature = self.encode_uri_component(self.rsa(query, self.secret, 'sha256'))
else:
signature = self.encode_uri_component(self.eddsa(self.encode(query), self.secret, 'ed25519'))
else:
signature = self.hmac(self.encode(query), self.encode(self.secret), hashlib.sha256)
query += '&' + 'signature=' + signature
headers = {
'X-MBX-APIKEY': self.apiKey,
}
if (method == 'GET') or (method == 'DELETE'):
url += '?' + query
else:
body = query
headers['Content-Type'] = 'application/x-www-form-urlencoded'
else:
if params:
url += '?' + self.urlencode(params)
return {'url': url, 'method': method, 'body': body, 'headers': headers}
def get_exceptions_by_url(self, url: str, exactOrBroad: str):
marketType = None
hostname = self.hostname if (self.hostname is not None) else 'binance.com'
if url.startswith('https://api.' + hostname + '/') or url.startswith('https://testnet.binance.vision'):
marketType = 'spot'
elif url.startswith('https://dapi.' + hostname + '/') or url.startswith('https://testnet.binancefuture.com/dapi'):
marketType = 'inverse'
elif url.startswith('https://fapi.' + hostname + '/') or url.startswith('https://testnet.binancefuture.com/fapi'):
marketType = 'linear'
elif url.startswith('https://eapi.' + hostname + '/'):
marketType = 'option'
elif url.startswith('https://papi.' + hostname + '/'):
marketType = 'portfolioMargin'
if marketType is not None:
exceptionsForMarketType = self.safe_dict(self.exceptions, marketType, {})
return self.safe_dict(exceptionsForMarketType, exactOrBroad, {})
return {}
def handle_errors(self, code: int, reason: str, url: str, method: str, headers: dict, body: str, response, requestHeaders, requestBody):
if (code == 418) or (code == 429):
raise DDoSProtection(self.id + ' ' + str(code) + ' ' + reason + ' ' + body)
# error response in a form: {"code": -1013, "msg": "Invalid quantity."}
# following block cointains legacy checks against message patterns in "msg" property
# will switch "code" checks eventually, when we know all of them
if code >= 400:
if body.find('Price * QTY is zero or less') >= 0:
raise InvalidOrder(self.id + ' order cost = amount * price is zero or less ' + body)
if body.find('LOT_SIZE') >= 0:
raise InvalidOrder(self.id + ' order amount should be evenly divisible by lot size ' + body)
if body.find('PRICE_FILTER') >= 0:
raise InvalidOrder(self.id + ' order price is invalid, i.e. exceeds allowed price precision, exceeds min price or max price limits or is invalid value in general, use self.price_to_precision(symbol, amount) ' + body)
if response is None:
return None # fallback to default error handler
# response in format {'msg': 'The coin does not exist.', 'success': True/false}
success = self.safe_bool(response, 'success', True)
if not success:
messageNew = self.safe_string(response, 'msg')
parsedMessage = None
if messageNew is not None:
try:
parsedMessage = json.loads(messageNew)
except Exception as e:
# do nothing
parsedMessage = None
if parsedMessage is not None:
response = parsedMessage
message = self.safe_string(response, 'msg')
if message is not None:
self.throw_exactly_matched_exception(self.get_exceptions_by_url(url, 'exact'), message, self.id + ' ' + message)
self.throw_exactly_matched_exception(self.exceptions['exact'], message, self.id + ' ' + message)
self.throw_broadly_matched_exception(self.get_exceptions_by_url(url, 'broad'), message, self.id + ' ' + message)
self.throw_broadly_matched_exception(self.exceptions['broad'], message, self.id + ' ' + message)
# checks against error codes
error = self.safe_string(response, 'code')
if error is not None:
# https://github.com/ccxt/ccxt/issues/6501
# https://github.com/ccxt/ccxt/issues/7742
if (error == '200') or Precise.string_equals(error, '0'):
return None
# a workaround for {"code":-2015,"msg":"Invalid API-key, IP, or permissions for action."}
# despite that their message is very confusing, it is raised by Binance
# on a temporary ban, the API key is valid, but disabled for a while
if (error == '-2015') and self.options['hasAlreadyAuthenticatedSuccessfully']:
raise DDoSProtection(self.id + ' ' + body)
feedback = self.id + ' ' + body
if message == 'No need to change margin type.':
# not an error
# https://github.com/ccxt/ccxt/issues/11268
# https://github.com/ccxt/ccxt/pull/11624
# POST https://fapi.binance.com/fapi/v1/marginType 400 Bad Request
# binanceusdm {"code":-4046,"msg":"No need to change margin type."}
raise MarginModeAlreadySet(feedback)
self.throw_exactly_matched_exception(self.get_exceptions_by_url(url, 'exact'), error, feedback)
self.throw_exactly_matched_exception(self.exceptions['exact'], error, feedback)
raise ExchangeError(feedback)
if not success:
raise ExchangeError(self.id + ' ' + body)
if isinstance(response, list):
# cancelOrders returns an array like self: [{"code":-2011,"msg":"Unknown order sent."}]
arrayLength = len(response)
if arrayLength == 1: # when there's a single error we can throw, otherwise we have a partial success
element = response[0]
errorCode = self.safe_string(element, 'code')
if errorCode is not None:
self.throw_exactly_matched_exception(self.get_exceptions_by_url(url, 'exact'), errorCode, self.id + ' ' + body)
self.throw_exactly_matched_exception(self.exceptions['exact'], errorCode, self.id + ' ' + body)
return None
def calculate_rate_limiter_cost(self, api, method, path, params, config={}):
if ('noCoin' in config) and not ('coin' in params):
return config['noCoin']
elif ('noSymbol' in config) and not ('symbol' in params):
return config['noSymbol']
elif ('noPoolId' in config) and not ('poolId' in params):
return config['noPoolId']
elif ('byLimit' in config) and ('limit' in params):
limit = params['limit']
byLimit = config['byLimit']
for i in range(0, len(byLimit)):
entry = byLimit[i]
if limit <= entry[0]:
return entry[1]
return self.safe_value(config, 'cost', 1)
def request(self, path, api='public', method='GET', params={}, headers=None, body=None, config={}):
response = self.fetch2(path, api, method, params, headers, body, config)
# a workaround for {"code":-2015,"msg":"Invalid API-key, IP, or permissions for action."}
if api == 'private':
self.options['hasAlreadyAuthenticatedSuccessfully'] = True
return response
def modify_margin_helper(self, symbol: str, amount, addOrReduce, params={}):
# used to modify isolated positions
defaultType = self.safe_string(self.options, 'defaultType', 'future')
if defaultType == 'spot':
defaultType = 'future'
type = self.safe_string(params, 'type', defaultType)
if (type == 'margin') or (type == 'spot'):
raise NotSupported(self.id + ' add / reduce margin only supported with type future or delivery')
self.load_markets()
market = self.market(symbol)
amount = self.amount_to_precision(symbol, amount)
request: dict = {
'type': addOrReduce,
'symbol': market['id'],
'amount': amount,
}
response = None
code = None
if market['linear']:
code = market['quote']
response = self.fapiPrivatePostPositionMargin(self.extend(request, params))
else:
code = market['base']
response = self.dapiPrivatePostPositionMargin(self.extend(request, params))
#
# {
# "code": 200,
# "msg": "Successfully modify position margin.",
# "amount": 0.001,
# "type": 1
# }
#
return self.extend(self.parse_margin_modification(response, market), {
'code': code,
})
def parse_margin_modification(self, data: dict, market: Market = None) -> MarginModification:
#
# add/reduce margin
#
# {
# "code": 200,
# "msg": "Successfully modify position margin.",
# "amount": 0.001,
# "type": 1
# }
#
# fetchMarginAdjustmentHistory
#
# {
# symbol: "XRPUSDT",
# type: "1",
# deltaType: "TRADE",
# amount: "2.57148240",
# asset: "USDT",
# time: "1711046271555",
# positionSide: "BOTH",
# clientTranId: ""
# }
#
rawType = self.safe_integer(data, 'type')
errorCode = self.safe_string(data, 'code')
marketId = self.safe_string(data, 'symbol')
timestamp = self.safe_integer(data, 'time')
market = self.safe_market(marketId, market, None, 'swap')
noErrorCode = errorCode is None
success = errorCode == '200'
return {
'info': data,
'symbol': market['symbol'],
'type': 'add' if (rawType == 1) else 'reduce',
'marginMode': 'isolated',
'amount': self.safe_number(data, 'amount'),
'code': self.safe_string(data, 'asset'),
'total': None,
'status': 'ok' if (success or noErrorCode) else 'failed',
'timestamp': timestamp,
'datetime': self.iso8601(timestamp),
}
def reduce_margin(self, symbol: str, amount: float, params={}) -> MarginModification:
"""
remove margin from a position
https://developers.binance.com/docs/derivatives/usds-margined-futures/trade/rest-api/Modify-Isolated-Position-Margin
https://developers.binance.com/docs/derivatives/coin-margined-futures/trade/rest-api/Modify-Isolated-Position-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>`
"""
return self.modify_margin_helper(symbol, amount, 2, params)
def add_margin(self, symbol: str, amount: float, params={}) -> MarginModification:
"""
add margin
https://developers.binance.com/docs/derivatives/usds-margined-futures/trade/rest-api/Modify-Isolated-Position-Margin
https://developers.binance.com/docs/derivatives/coin-margined-futures/trade/rest-api/Modify-Isolated-Position-Margin
:param str symbol: unified market symbol
:param float amount: 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>`
"""
return self.modify_margin_helper(symbol, amount, 1, params)
def fetch_cross_borrow_rate(self, code: str, params={}) -> CrossBorrowRate:
"""
fetch the rate of interest to borrow a currency for margin trading
https://developers.binance.com/docs/margin_trading/borrow-and-repay/Query-Margin-Interest-Rate-History
:param str code: unified currency code
:param dict [params]: extra parameters specific to the exchange API endpoint
:returns dict: a `borrow rate structure <https://docs.ccxt.com/#/?id=borrow-rate-structure>`
"""
self.load_markets()
currency = self.currency(code)
request: dict = {
'asset': currency['id'],
# 'vipLevel': self.safe_integer(params, 'vipLevel'),
}
response = self.sapiGetMarginInterestRateHistory(self.extend(request, params))
#
# [
# {
# "asset": "USDT",
# "timestamp": 1638230400000,
# "dailyInterestRate": "0.0006",
# "vipLevel": 0
# },
# ]
#
rate = self.safe_dict(response, 0)
return self.parse_borrow_rate(rate)
def fetch_isolated_borrow_rate(self, symbol: str, params={}) -> IsolatedBorrowRate:
"""
fetch the rate of interest to borrow a currency for margin trading
https://developers.binance.com/docs/margin_trading/account/Query-Isolated-Margin-Fee-Data
:param str symbol: unified market symbol
:param dict [params]: extra parameters specific to the exchange API endpoint
EXCHANGE SPECIFIC PARAMETERS
:param dict [params.vipLevel]: user's current specific margin data will be returned if viplevel is omitted
:returns dict: an `isolated borrow rate structure <https://docs.ccxt.com/#/?id=isolated-borrow-rate-structure>`
"""
request: dict = {
'symbol': symbol,
}
borrowRates = self.fetch_isolated_borrow_rates(self.extend(request, params))
return self.safe_dict(borrowRates, symbol)
def fetch_isolated_borrow_rates(self, params={}) -> IsolatedBorrowRates:
"""
fetch the borrow interest rates of all currencies
https://developers.binance.com/docs/margin_trading/account/Query-Isolated-Margin-Fee-Data
:param dict [params]: extra parameters specific to the exchange API endpoint
:param dict [params.symbol]: unified market symbol
EXCHANGE SPECIFIC PARAMETERS
:param dict [params.vipLevel]: user's current specific margin data will be returned if viplevel is omitted
:returns dict: a `borrow rate structure <https://docs.ccxt.com/#/?id=borrow-rate-structure>`
"""
self.load_markets()
request: dict = {}
symbol = self.safe_string(params, 'symbol')
params = self.omit(params, 'symbol')
if symbol is not None:
market = self.market(symbol)
request['symbol'] = market['id']
response = self.sapiGetMarginIsolatedMarginData(self.extend(request, params))
#
# [
# {
# "vipLevel": 0,
# "symbol": "BTCUSDT",
# "leverage": "10",
# "data": [
# {
# "coin": "BTC",
# "dailyInterest": "0.00026125",
# "borrowLimit": "270"
# },
# {
# "coin": "USDT",
# "dailyInterest": "0.000475",
# "borrowLimit": "2100000"
# }
# ]
# }
# ]
#
return self.parse_isolated_borrow_rates(response)
def fetch_borrow_rate_history(self, code: str, since: Int = None, limit: Int = None, params={}):
"""
retrieves a history of a currencies borrow interest rate at specific time slots
https://developers.binance.com/docs/margin_trading/borrow-and-repay/Query-Margin-Interest-Rate-History
:param str code: unified currency code
:param int [since]: timestamp for the earliest borrow rate
:param int [limit]: the maximum number of `borrow rate structures <https://docs.ccxt.com/#/?id=borrow-rate-structure>` to retrieve
:param dict [params]: extra parameters specific to the exchange API endpoint
:returns dict[]: an array of `borrow rate structures <https://docs.ccxt.com/#/?id=borrow-rate-structure>`
"""
self.load_markets()
if limit is None:
limit = 93
elif limit > 93:
# Binance API says the limit is 100, but "Illegal characters found in a parameter." is returned when limit is > 93
raise BadRequest(self.id + ' fetchBorrowRateHistory() limit parameter cannot exceed 92')
currency = self.currency(code)
request: dict = {
'asset': currency['id'],
'limit': limit,
}
if since is not None:
request['startTime'] = since
endTime = self.sum(since, limit * 86400000) - 1 # required when startTime is further than 93 days in the past
now = self.milliseconds()
request['endTime'] = min(endTime, now) # cannot have an endTime later than current time
response = self.sapiGetMarginInterestRateHistory(self.extend(request, params))
#
# [
# {
# "asset": "USDT",
# "timestamp": 1638230400000,
# "dailyInterestRate": "0.0006",
# "vipLevel": 0
# },
# ]
#
return self.parse_borrow_rate_history(response, code, since, limit)
def parse_borrow_rate(self, info, currency: Currency = None):
#
# {
# "asset": "USDT",
# "timestamp": 1638230400000,
# "dailyInterestRate": "0.0006",
# "vipLevel": 0
# }
#
timestamp = self.safe_integer(info, 'timestamp')
currencyId = self.safe_string(info, 'asset')
return {
'currency': self.safe_currency_code(currencyId, currency),
'rate': self.safe_number(info, 'dailyInterestRate'),
'period': 86400000,
'timestamp': timestamp,
'datetime': self.iso8601(timestamp),
'info': info,
}
def parse_isolated_borrow_rate(self, info: dict, market: Market = None) -> IsolatedBorrowRate:
#
# {
# "vipLevel": 0,
# "symbol": "BTCUSDT",
# "leverage": "10",
# "data": [
# {
# "coin": "BTC",
# "dailyInterest": "0.00026125",
# "borrowLimit": "270"
# },
# {
# "coin": "USDT",
# "dailyInterest": "0.000475",
# "borrowLimit": "2100000"
# }
# ]
# }
#
marketId = self.safe_string(info, 'symbol')
market = self.safe_market(marketId, market, None, 'spot')
data = self.safe_list(info, 'data')
baseInfo = self.safe_dict(data, 0)
quoteInfo = self.safe_dict(data, 1)
return {
'info': info,
'symbol': self.safe_string(market, 'symbol'),
'base': self.safe_string(baseInfo, 'coin'),
'baseRate': self.safe_number(baseInfo, 'dailyInterest'),
'quote': self.safe_string(quoteInfo, 'coin'),
'quoteRate': self.safe_number(quoteInfo, 'dailyInterest'),
'period': 86400000,
'timestamp': None,
'datetime': None,
}
def create_gift_code(self, code: str, amount, params={}):
"""
create gift code
https://developers.binance.com/docs/gift_card/market-data/Create-a-single-token-gift-card
:param str code: gift code
:param float amount: amount of currency for the gift
:param dict [params]: extra parameters specific to the exchange API endpoint
:returns dict: The gift code id, code, currency and amount
"""
self.load_markets()
currency = self.currency(code)
# ensure you have enough token in your funding account before calling self code
request: dict = {
'token': currency['id'],
'amount': amount,
}
response = self.sapiPostGiftcardCreateCode(self.extend(request, params))
#
# {
# "code": "000000",
# "message": "success",
# "data": {referenceNo: "0033002404219823", code: "AP6EXTLKNHM6CEX7"},
# "success": True
# }
#
data = self.safe_dict(response, 'data')
giftcardCode = self.safe_string(data, 'code')
id = self.safe_string(data, 'referenceNo')
return {
'info': response,
'id': id,
'code': giftcardCode,
'currency': code,
'amount': amount,
}
def redeem_gift_code(self, giftcardCode, params={}):
"""
redeem gift code
https://developers.binance.com/docs/gift_card/market-data/Redeem-a-Binance-Gift-Card
:param str giftcardCode:
:param dict [params]: extra parameters specific to the exchange API endpoint
:returns dict: response from the exchange
"""
request: dict = {
'code': giftcardCode,
}
response = self.sapiPostGiftcardRedeemCode(self.extend(request, params))
#
# {
# "code": "000000",
# "message": "success",
# "data": {
# "referenceNo": "0033002404219823",
# "identityNo": "10316431732801474560"
# },
# "success": True
# }
#
return response
def verify_gift_code(self, id: str, params={}):
"""
verify gift code
https://developers.binance.com/docs/gift_card/market-data/Verify-Binance-Gift-Card-by-Gift-Card-Number
:param str id: reference number id
:param dict [params]: extra parameters specific to the exchange API endpoint
:returns dict: response from the exchange
"""
request: dict = {
'referenceNo': id,
}
response = self.sapiGetGiftcardVerify(self.extend(request, params))
#
# {
# "code": "000000",
# "message": "success",
# "data": {valid: True},
# "success": True
# }
#
return response
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://developers.binance.com/docs/margin_trading/borrow-and-repay/Get-Interest-History
https://developers.binance.com/docs/derivatives/portfolio-margin/account/Get-Margin-BorrowLoan-Interest-History
:param str [code]: unified currency code
:param str [symbol]: unified market symbol when fetch interest in isolated markets
:param int [since]: the earliest time in ms to fetch borrrow 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.portfolioMargin]: set to True if you would like to fetch the borrow interest in a portfolio margin account
:returns dict[]: a list of `borrow interest structures <https://docs.ccxt.com/#/?id=borrow-interest-structure>`
"""
self.load_markets()
isPortfolioMargin = None
isPortfolioMargin, params = self.handle_option_and_params_2(params, 'fetchBorrowInterest', 'papi', 'portfolioMargin', False)
request: dict = {}
market = None
if code is not None:
currency = self.currency(code)
request['asset'] = currency['id']
if since is not None:
request['startTime'] = since
if limit is not None:
request['size'] = limit
request, params = self.handle_until_option('endTime', request, params)
response = None
if isPortfolioMargin:
response = self.papiGetMarginMarginInterestHistory(self.extend(request, params))
else:
if symbol is not None:
market = self.market(symbol)
request['isolatedSymbol'] = market['id']
response = self.sapiGetMarginInterestHistory(self.extend(request, params))
#
# spot margin
#
# {
# "rows":[
# {
# "isolatedSymbol": "BNBUSDT", # isolated symbol, will not be returned for crossed margin
# "asset": "BNB",
# "interest": "0.02414667",
# "interestAccuredTime": 1566813600000,
# "interestRate": "0.01600000",
# "principal": "36.22000000",
# "type": "ON_BORROW"
# }
# ],
# "total": 1
# }
#
# spot margin portfolio margin
#
# {
# "total": 49,
# "rows": [
# {
# "txId": 1656187724899910076,
# "interestAccuredTime": 1707541200000,
# "asset": "USDT",
# "rawAsset": "USDT",
# "principal": "0.00011146",
# "interest": "0.00000001",
# "interestRate": "0.00089489",
# "type": "PERIODIC"
# },
# ]
# }
#
rows = self.safe_list(response, 'rows')
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:
symbol = self.safe_string(info, 'isolatedSymbol')
timestamp = self.safe_integer(info, 'interestAccuredTime')
marginMode = 'cross' if (symbol is None) else 'isolated'
return {
'info': info,
'symbol': symbol,
'currency': self.safe_currency_code(self.safe_string(info, 'asset')),
'interest': self.safe_number(info, 'interest'),
'interestRate': self.safe_number(info, 'interestRate'),
'amountBorrowed': self.safe_number(info, 'principal'),
'marginMode': marginMode,
'timestamp': timestamp,
'datetime': self.iso8601(timestamp),
}
def repay_cross_margin(self, code: str, amount, params={}):
"""
repay borrowed margin and interest
https://developers.binance.com/docs/derivatives/portfolio-margin/trade/Margin-Account-Repay
https://developers.binance.com/docs/margin_trading/borrow-and-repay/Margin-Account-Borrow-Repay
https://developers.binance.com/docs/derivatives/portfolio-margin/trade/Margin-Account-Repay-Debt
:param str code: unified currency code of the currency to repay
:param float amount: the amount to repay
:param dict [params]: extra parameters specific to the exchange API endpoint
:param boolean [params.portfolioMargin]: set to True if you would like to repay margin in a portfolio margin account
:param str [params.repayCrossMarginMethod]: *portfolio margin only* 'papiPostRepayLoan'(default), 'papiPostMarginRepayDebt'(alternative)
:param str [params.specifyRepayAssets]: *portfolio margin papiPostMarginRepayDebt only* specific asset list to repay debt
:returns dict: a `margin loan structure <https://docs.ccxt.com/#/?id=margin-loan-structure>`
"""
self.load_markets()
currency = self.currency(code)
request: dict = {
'asset': currency['id'],
'amount': self.currency_to_precision(code, amount),
}
response = None
isPortfolioMargin = None
isPortfolioMargin, params = self.handle_option_and_params_2(params, 'repayCrossMargin', 'papi', 'portfolioMargin', False)
if isPortfolioMargin:
method = None
method, params = self.handle_option_and_params_2(params, 'repayCrossMargin', 'repayCrossMarginMethod', 'method')
if method == 'papiPostMarginRepayDebt':
response = self.papiPostMarginRepayDebt(self.extend(request, params))
#
# {
# "asset": "USDC",
# "amount": 10,
# "specifyRepayAssets": null,
# "updateTime": 1727170761267,
# "success": True
# }
#
else:
response = self.papiPostRepayLoan(self.extend(request, params))
#
# {
# "tranId": 108988250265,
# "clientTag":""
# }
#
else:
request['isIsolated'] = 'FALSE'
request['type'] = 'REPAY'
response = self.sapiPostMarginBorrowRepay(self.extend(request, params))
#
# {
# "tranId": 108988250265,
# "clientTag":""
# }
#
return self.parse_margin_loan(response, currency)
def repay_isolated_margin(self, symbol: str, code: str, amount, params={}):
"""
repay borrowed margin and interest
https://developers.binance.com/docs/margin_trading/borrow-and-repay/Margin-Account-Borrow-Repay
:param str symbol: unified market symbol, required for isolated margin
:param str code: unified currency code of the currency to repay
:param float 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>`
"""
self.load_markets()
currency = self.currency(code)
market = self.market(symbol)
request: dict = {
'asset': currency['id'],
'amount': self.currency_to_precision(code, amount),
'symbol': market['id'],
'isIsolated': 'TRUE',
'type': 'REPAY',
}
response = self.sapiPostMarginBorrowRepay(self.extend(request, params))
#
# {
# "tranId": 108988250265,
# "clientTag":""
# }
#
return self.parse_margin_loan(response, currency)
def borrow_cross_margin(self, code: str, amount: float, params={}):
"""
create a loan to borrow margin
https://developers.binance.com/docs/margin_trading/borrow-and-repay/Margin-Account-Borrow-Repay
https://developers.binance.com/docs/derivatives/portfolio-margin/trade/Margin-Account-Borrow
:param str code: unified currency code of the currency to borrow
:param float amount: the amount to borrow
:param dict [params]: extra parameters specific to the exchange API endpoint
:param boolean [params.portfolioMargin]: set to True if you would like to borrow margin in a portfolio margin account
:returns dict: a `margin loan structure <https://docs.ccxt.com/#/?id=margin-loan-structure>`
"""
self.load_markets()
currency = self.currency(code)
request: dict = {
'asset': currency['id'],
'amount': self.currency_to_precision(code, amount),
}
response = None
isPortfolioMargin = None
isPortfolioMargin, params = self.handle_option_and_params_2(params, 'borrowCrossMargin', 'papi', 'portfolioMargin', False)
if isPortfolioMargin:
response = self.papiPostMarginLoan(self.extend(request, params))
else:
request['isIsolated'] = 'FALSE'
request['type'] = 'BORROW'
response = self.sapiPostMarginBorrowRepay(self.extend(request, params))
#
# {
# "tranId": 108988250265,
# "clientTag":""
# }
#
return self.parse_margin_loan(response, currency)
def borrow_isolated_margin(self, symbol: str, code: str, amount: float, params={}):
"""
create a loan to borrow margin
https://developers.binance.com/docs/margin_trading/borrow-and-repay/Margin-Account-Borrow-Repay
:param str symbol: unified market symbol, required for isolated margin
:param str code: unified currency code of the currency to borrow
:param float 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>`
"""
self.load_markets()
currency = self.currency(code)
market = self.market(symbol)
request: dict = {
'asset': currency['id'],
'amount': self.currency_to_precision(code, amount),
'symbol': market['id'],
'isIsolated': 'TRUE',
'type': 'BORROW',
}
response = self.sapiPostMarginBorrowRepay(self.extend(request, params))
#
# {
# "tranId": 108988250265,
# "clientTag":""
# }
#
return self.parse_margin_loan(response, currency)
def parse_margin_loan(self, info, currency: Currency = None):
#
# {
# "tranId": 108988250265,
# "clientTag":""
# }
#
# repayCrossMargin alternative endpoint
#
# {
# "asset": "USDC",
# "amount": 10,
# "specifyRepayAssets": null,
# "updateTime": 1727170761267,
# "success": True
# }
#
currencyId = self.safe_string(info, 'asset')
timestamp = self.safe_integer(info, 'updateTime')
return {
'id': self.safe_integer(info, 'tranId'),
'currency': self.safe_currency_code(currencyId, currency),
'amount': self.safe_number(info, 'amount'),
'symbol': None,
'timestamp': timestamp,
'datetime': self.iso8601(timestamp),
'info': info,
}
def fetch_open_interest_history(self, symbol: str, timeframe='5m', since: Int = None, limit: Int = None, params={}):
"""
Retrieves the open interest history of a currency
https://developers.binance.com/docs/derivatives/usds-margined-futures/market-data/rest-api/Open-Interest-Statistics
https://developers.binance.com/docs/derivatives/coin-margined-futures/market-data/rest-api/Open-Interest-Statistics
:param str symbol: Unified CCXT market symbol
:param str timeframe: "5m","15m","30m","1h","2h","4h","6h","12h", or "1d"
:param int [since]: the time(ms) of the earliest record to retrieve unix timestamp
:param int [limit]: default 30, max 500
:param dict [params]: exchange specific parameters
:param int [params.until]: the time(ms) of the latest record to retrieve unix timestamp
: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: an array of `open interest structure <https://docs.ccxt.com/#/?id=open-interest-structure>`
"""
if timeframe == '1m':
raise BadRequest(self.id + ' fetchOpenInterestHistory cannot use the 1m timeframe')
self.load_markets()
paginate = False
paginate, params = self.handle_option_and_params(params, 'fetchOpenInterestHistory', 'paginate', False)
if paginate:
return self.fetch_paginated_call_deterministic('fetchOpenInterestHistory', symbol, since, limit, timeframe, params, 500)
market = self.market(symbol)
request: dict = {
'period': self.safe_string(self.timeframes, timeframe, timeframe),
}
if limit is not None:
request['limit'] = limit
symbolKey = 'symbol' if market['linear'] else 'pair'
request[symbolKey] = market['id']
if market['inverse']:
request['contractType'] = self.safe_string(params, 'contractType', 'CURRENT_QUARTER')
if since is not None:
request['startTime'] = since
until = self.safe_integer(params, 'until') # unified in milliseconds
endTime = self.safe_integer(params, 'endTime', until) # exchange-specific in milliseconds
params = self.omit(params, ['endTime', 'until'])
if endTime:
request['endTime'] = endTime
elif since:
if limit is None:
limit = 30 # Exchange default
duration = self.parse_timeframe(timeframe)
request['endTime'] = self.sum(since, duration * limit * 1000)
response = None
if market['inverse']:
response = self.dapiDataGetOpenInterestHist(self.extend(request, params))
else:
response = self.fapiDataGetOpenInterestHist(self.extend(request, params))
#
# [
# {
# "symbol":"BTCUSDT",
# "sumOpenInterest":"75375.61700000",
# "sumOpenInterestValue":"3248828883.71251440",
# "timestamp":1642179900000
# },
# ...
# ]
#
return self.parse_open_interests_history(response, market, since, limit)
def fetch_open_interest(self, symbol: str, params={}):
"""
retrieves the open interest of a contract trading pair
https://developers.binance.com/docs/derivatives/usds-margined-futures/market-data/rest-api/Open-Interest
https://developers.binance.com/docs/derivatives/coin-margined-futures/market-data/rest-api/Open-Interest
https://developers.binance.com/docs/derivatives/option/market-data/Open-Interest
:param str symbol: unified CCXT market symbol
:param dict [params]: exchange specific parameters
:returns dict} an open interest structure{@link https://docs.ccxt.com/#/?id=open-interest-structure:
"""
self.load_markets()
market = self.market(symbol)
request: dict = {}
if market['option']:
request['underlyingAsset'] = market['baseId']
if market['expiry'] is None:
raise NotSupported(self.id + ' fetchOpenInterest does not support ' + symbol)
request['expiration'] = self.yymmdd(market['expiry'])
else:
request['symbol'] = market['id']
response = None
if market['option']:
response = self.eapiPublicGetOpenInterest(self.extend(request, params))
elif market['inverse']:
response = self.dapiPublicGetOpenInterest(self.extend(request, params))
else:
response = self.fapiPublicGetOpenInterest(self.extend(request, params))
#
# futures(fapi)
#
# {
# "symbol": "ETHUSDT_230331",
# "openInterest": "23581.677",
# "time": 1677356872265
# }
#
# futures(dapi)
#
# {
# "symbol": "ETHUSD_PERP",
# "pair": "ETHUSD",
# "openInterest": "26542436",
# "contractType": "PERPETUAL",
# "time": 1677360272224
# }
#
# options(eapi)
#
# [
# {
# "symbol": "ETH-230225-1625-C",
# "sumOpenInterest": "460.50",
# "sumOpenInterestUsd": "734957.4358092150",
# "timestamp": "1677304860000"
# }
# ]
#
if market['option']:
symbol = market['symbol']
result = self.parse_open_interests_history(response, market)
for i in range(0, len(result)):
item = result[i]
if item['symbol'] == symbol:
return item
else:
return self.parse_open_interest(response, market)
return None
def parse_open_interest(self, interest, market: Market = None):
timestamp = self.safe_integer_2(interest, 'timestamp', 'time')
id = self.safe_string(interest, 'symbol')
amount = self.safe_number_2(interest, 'sumOpenInterest', 'openInterest')
value = self.safe_number_2(interest, 'sumOpenInterestValue', 'sumOpenInterestUsd')
# Inverse returns the number of contracts different from the base or quote hasattr(self, volume) case
# compared with https://www.binance.com/en/futures/funding-history/quarterly/4
return self.safe_open_interest({
'symbol': self.safe_symbol(id, market, None, 'contract'),
'baseVolume': None if market['inverse'] else amount, # deprecated
'quoteVolume': value, # deprecated
'openInterestAmount': amount,
'openInterestValue': value,
'timestamp': timestamp,
'datetime': self.iso8601(timestamp),
'info': interest,
}, market)
def fetch_my_liquidations(self, symbol: Str = None, since: Int = None, limit: Int = None, params={}):
"""
retrieves the users liquidated positions
https://developers.binance.com/docs/margin_trading/trade/Get-Force-Liquidation-Record
https://developers.binance.com/docs/derivatives/usds-margined-futures/trade/rest-api/Users-Force-Orders
https://developers.binance.com/docs/derivatives/coin-margined-futures/trade/rest-api/Users-Force-Orders
https://developers.binance.com/docs/derivatives/portfolio-margin/trade/Query-Users-UM-Force-Orders
https://developers.binance.com/docs/derivatives/portfolio-margin/trade/Query-Users-CM-Force-Orders
: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 binance api endpoint
:param int [params.until]: timestamp in ms of the latest liquidation
:param boolean [params.paginate]: *spot only* 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.portfolioMargin]: set to True if you would like to fetch liquidations in a portfolio margin account
:param str [params.type]: "spot"
:param str [params.subType]: "linear" or "inverse"
:returns dict: an array of `liquidation structures <https://docs.ccxt.com/#/?id=liquidation-structure>`
"""
self.load_markets()
paginate = False
paginate, params = self.handle_option_and_params(params, 'fetchMyLiquidations', 'paginate')
if paginate:
return self.fetch_paginated_call_incremental('fetchMyLiquidations', symbol, since, limit, params, 'current', 100)
market = None
if symbol is not None:
market = self.market(symbol)
type = None
type, params = self.handle_market_type_and_params('fetchMyLiquidations', market, params)
subType = None
subType, params = self.handle_sub_type_and_params('fetchMyLiquidations', market, params, 'linear')
isPortfolioMargin = None
isPortfolioMargin, params = self.handle_option_and_params_2(params, 'fetchMyLiquidations', 'papi', 'portfolioMargin', False)
request: dict = {}
if type != 'spot':
request['autoCloseType'] = 'LIQUIDATION'
if market is not None:
symbolKey = 'isolatedSymbol' if market['spot'] else 'symbol'
if not isPortfolioMargin:
request[symbolKey] = market['id']
if since is not None:
request['startTime'] = since
if limit is not None:
if type == 'spot':
request['size'] = limit
else:
request['limit'] = limit
request, params = self.handle_until_option('endTime', request, params)
response = None
if type == 'spot':
if isPortfolioMargin:
response = self.papiGetMarginForceOrders(self.extend(request, params))
else:
response = self.sapiGetMarginForceLiquidationRec(self.extend(request, params))
elif subType == 'linear':
if isPortfolioMargin:
response = self.papiGetUmForceOrders(self.extend(request, params))
else:
response = self.fapiPrivateGetForceOrders(self.extend(request, params))
elif subType == 'inverse':
if isPortfolioMargin:
response = self.papiGetCmForceOrders(self.extend(request, params))
else:
response = self.dapiPrivateGetForceOrders(self.extend(request, params))
else:
raise NotSupported(self.id + ' fetchMyLiquidations() does not support ' + market['type'] + ' markets')
#
# margin
#
# {
# "rows": [
# {
# "avgPrice": "0.00388359",
# "executedQty": "31.39000000",
# "orderId": 180015097,
# "price": "0.00388110",
# "qty": "31.39000000",
# "side": "SELL",
# "symbol": "BNBBTC",
# "timeInForce": "GTC",
# "isIsolated": True,
# "updatedTime": 1558941374745
# }
# ],
# "total": 1
# }
#
# linear
#
# [
# {
# "orderId": 6071832819,
# "symbol": "BTCUSDT",
# "status": "FILLED",
# "clientOrderId": "autoclose-1596107620040000020",
# "price": "10871.09",
# "avgPrice": "10913.21000",
# "origQty": "0.001",
# "executedQty": "0.001",
# "cumQuote": "10.91321",
# "timeInForce": "IOC",
# "type": "LIMIT",
# "reduceOnly": False,
# "closePosition": False,
# "side": "SELL",
# "positionSide": "BOTH",
# "stopPrice": "0",
# "workingType": "CONTRACT_PRICE",
# "origType": "LIMIT",
# "time": 1596107620044,
# "updateTime": 1596107620087
# },
# ]
#
# inverse
#
# [
# {
# "orderId": 165123080,
# "symbol": "BTCUSD_200925",
# "pair": "BTCUSD",
# "status": "FILLED",
# "clientOrderId": "autoclose-1596542005017000006",
# "price": "11326.9",
# "avgPrice": "11326.9",
# "origQty": "1",
# "executedQty": "1",
# "cumBase": "0.00882854",
# "timeInForce": "IOC",
# "type": "LIMIT",
# "reduceOnly": False,
# "closePosition": False,
# "side": "SELL",
# "positionSide": "BOTH",
# "stopPrice": "0",
# "workingType": "CONTRACT_PRICE",
# "priceProtect": False,
# "origType": "LIMIT",
# "time": 1596542005019,
# "updateTime": 1596542005050
# },
# ]
#
liquidations = self.safe_list(response, 'rows', response)
return self.parse_liquidations(liquidations, market, since, limit)
def parse_liquidation(self, liquidation, market: Market = None):
#
# margin
#
# {
# "avgPrice": "0.00388359",
# "executedQty": "31.39000000",
# "orderId": 180015097,
# "price": "0.00388110",
# "qty": "31.39000000",
# "side": "SELL",
# "symbol": "BNBBTC",
# "timeInForce": "GTC",
# "isIsolated": True,
# "updatedTime": 1558941374745
# }
#
# linear
#
# {
# "orderId": 6071832819,
# "symbol": "BTCUSDT",
# "status": "FILLED",
# "clientOrderId": "autoclose-1596107620040000020",
# "price": "10871.09",
# "avgPrice": "10913.21000",
# "origQty": "0.001",
# "executedQty": "0.002",
# "cumQuote": "10.91321",
# "timeInForce": "IOC",
# "type": "LIMIT",
# "reduceOnly": False,
# "closePosition": False,
# "side": "SELL",
# "positionSide": "BOTH",
# "stopPrice": "0",
# "workingType": "CONTRACT_PRICE",
# "origType": "LIMIT",
# "time": 1596107620044,
# "updateTime": 1596107620087
# }
#
# inverse
#
# {
# "orderId": 165123080,
# "symbol": "BTCUSD_200925",
# "pair": "BTCUSD",
# "status": "FILLED",
# "clientOrderId": "autoclose-1596542005017000006",
# "price": "11326.9",
# "avgPrice": "11326.9",
# "origQty": "1",
# "executedQty": "1",
# "cumBase": "0.00882854",
# "timeInForce": "IOC",
# "type": "LIMIT",
# "reduceOnly": False,
# "closePosition": False,
# "side": "SELL",
# "positionSide": "BOTH",
# "stopPrice": "0",
# "workingType": "CONTRACT_PRICE",
# "priceProtect": False,
# "origType": "LIMIT",
# "time": 1596542005019,
# "updateTime": 1596542005050
# }
#
marketId = self.safe_string(liquidation, 'symbol')
timestamp = self.safe_integer_2(liquidation, 'updatedTime', 'updateTime')
return self.safe_liquidation({
'info': liquidation,
'symbol': self.safe_symbol(marketId, market),
'contracts': self.safe_number(liquidation, 'executedQty'),
'contractSize': self.safe_number(market, 'contractSize'),
'price': self.safe_number(liquidation, 'avgPrice'),
'side': self.safe_string_lower(liquidation, 'side'),
'baseValue': self.safe_number(liquidation, 'cumBase'),
'quoteValue': self.safe_number(liquidation, 'cumQuote'),
'timestamp': timestamp,
'datetime': self.iso8601(timestamp),
})
def fetch_greeks(self, symbol: str, params={}) -> Greeks:
"""
fetches an option contracts greeks, financial metrics used to measure the factors that affect the price of an options contract
https://developers.binance.com/docs/derivatives/option/market-data/Option-Mark-Price
:param str symbol: unified symbol of the market to fetch greeks for
:param dict [params]: extra parameters specific to the exchange API endpoint
:returns dict: a `greeks structure <https://docs.ccxt.com/#/?id=greeks-structure>`
"""
self.load_markets()
market = self.market(symbol)
request: dict = {
'symbol': market['id'],
}
response = self.eapiPublicGetMark(self.extend(request, params))
#
# [
# {
# "symbol": "BTC-231229-40000-C",
# "markPrice": "2012",
# "bidIV": "0.60236275",
# "askIV": "0.62267244",
# "markIV": "0.6125176",
# "delta": "0.39111646",
# "theta": "-32.13948531",
# "gamma": "0.00004656",
# "vega": "51.70062218",
# "highPriceLimit": "6474",
# "lowPriceLimit": "5"
# }
# ]
#
return self.parse_greeks(response[0], market)
def fetch_all_greeks(self, symbols: Strings = None, params={}) -> List[Greeks]:
"""
fetches all option contracts greeks, financial metrics used to measure the factors that affect the price of an options contract
https://developers.binance.com/docs/derivatives/option/market-data/Option-Mark-Price
:param str[] [symbols]: unified symbols of the markets to fetch greeks for, all markets are returned if not assigned
:param dict [params]: extra parameters specific to the exchange API endpoint
:returns dict: a `greeks structure <https://docs.ccxt.com/#/?id=greeks-structure>`
"""
self.load_markets()
symbols = self.market_symbols(symbols, None, True, True, True)
request: dict = {}
market = None
if symbols is not None:
symbolsLength = len(symbols)
if symbolsLength == 1:
market = self.market(symbols[0])
request['symbol'] = market['id']
response = self.eapiPublicGetMark(self.extend(request, params))
#
# [
# {
# "symbol": "BTC-231229-40000-C",
# "markPrice": "2012",
# "bidIV": "0.60236275",
# "askIV": "0.62267244",
# "markIV": "0.6125176",
# "delta": "0.39111646",
# "theta": "-32.13948531",
# "gamma": "0.00004656",
# "vega": "51.70062218",
# "highPriceLimit": "6474",
# "lowPriceLimit": "5"
# }
# ]
#
return self.parse_all_greeks(response, symbols)
def parse_greeks(self, greeks: dict, market: Market = None) -> Greeks:
#
# {
# "symbol": "BTC-231229-40000-C",
# "markPrice": "2012",
# "bidIV": "0.60236275",
# "askIV": "0.62267244",
# "markIV": "0.6125176",
# "delta": "0.39111646",
# "theta": "-32.13948531",
# "gamma": "0.00004656",
# "vega": "51.70062218",
# "highPriceLimit": "6474",
# "lowPriceLimit": "5"
# }
#
marketId = self.safe_string(greeks, 'symbol')
symbol = self.safe_symbol(marketId, market)
return {
'symbol': symbol,
'timestamp': None,
'datetime': None,
'delta': self.safe_number(greeks, 'delta'),
'gamma': self.safe_number(greeks, 'gamma'),
'theta': self.safe_number(greeks, 'theta'),
'vega': self.safe_number(greeks, 'vega'),
'rho': None,
'bidSize': None,
'askSize': None,
'bidImpliedVolatility': self.safe_number(greeks, 'bidIV'),
'askImpliedVolatility': self.safe_number(greeks, 'askIV'),
'markImpliedVolatility': self.safe_number(greeks, 'markIV'),
'bidPrice': None,
'askPrice': None,
'markPrice': self.safe_number(greeks, 'markPrice'),
'lastPrice': None,
'underlyingPrice': None,
'info': greeks,
}
def fetch_trading_limits(self, symbols: Strings = None, params={}):
# self method should not be called directly, use loadTradingLimits() instead
markets = self.fetch_markets()
tradingLimits: dict = {}
for i in range(0, len(markets)):
market = markets[i]
symbol = market['symbol']
if (symbols is None) or (self.in_array(symbol, symbols)):
tradingLimits[symbol] = market['limits']['amount']
return tradingLimits
def fetch_position_mode(self, symbol: Str = None, params={}):
"""
fetchs the position mode, hedged or one way, hedged for binance is set identically for all linear markets or all inverse markets
https://developers.binance.com/docs/derivatives/usds-margined-futures/account/rest-api/Get-Current-Position-Mode
https://developers.binance.com/docs/derivatives/coin-margined-futures/account/rest-api/Get-Current-Position-Mode
:param str symbol: unified symbol of the market to fetch the order book for
:param dict [params]: extra parameters specific to the exchange API endpoint
:param str [params.subType]: "linear" or "inverse"
:returns dict: an object detailing whether the market is in hedged or one-way mode
"""
market = None
if symbol is not None:
market = self.market(symbol)
subType = None
subType, params = self.handle_sub_type_and_params('fetchPositionMode', market, params)
response = None
if subType == 'linear':
response = self.fapiPrivateGetPositionSideDual(params)
elif subType == 'inverse':
response = self.dapiPrivateGetPositionSideDual(params)
else:
raise BadRequest(self.id + ' fetchPositionMode requires either a symbol argument or params["subType"]')
#
# {
# dualSidePosition: False
# }
#
dualSidePosition = self.safe_bool(response, 'dualSidePosition')
return {
'info': response,
'hedged': dualSidePosition,
}
def fetch_margin_modes(self, symbols: Strings = None, params={}) -> MarginModes:
"""
fetches margin modes("isolated" or "cross") that the market for the symbol in in, with symbol=None all markets for a subType(linear/inverse) are returned
https://developers.binance.com/docs/derivatives/coin-margined-futures/account/rest-api/Account-Information
https://developers.binance.com/docs/derivatives/usds-margined-futures/account/rest-api/Account-Information-V2
https://developers.binance.com/docs/derivatives/usds-margined-futures/account/rest-api/Symbol-Config
:param str[] symbols: unified market symbols
:param dict [params]: extra parameters specific to the exchange API endpoint
:param str [params.subType]: "linear" or "inverse"
:returns dict: a list of `margin mode structures <https://docs.ccxt.com/#/?id=margin-mode-structure>`
"""
self.load_markets()
market = None
if symbols is not None:
symbols = self.market_symbols(symbols)
market = self.market(symbols[0])
subType = None
subType, params = self.handle_sub_type_and_params('fetchMarginMode', market, params)
response = None
if subType == 'linear':
response = self.fapiPrivateGetSymbolConfig(params)
#
# [
# {
# "symbol": "BTCUSDT",
# "marginType": "CROSSED",
# "isAutoAddMargin": "false",
# "leverage": 21,
# "maxNotionalValue": "1000000",
# }
# ]
#
elif subType == 'inverse':
response = self.dapiPrivateGetAccount(params)
#
# {
# feeTier: '0',
# canTrade: True,
# canDeposit: True,
# canWithdraw: True,
# updateTime: '0',
# assets: [
# {
# asset: 'APT',
# walletBalance: '0.00000000',
# unrealizedProfit: '0.00000000',
# marginBalance: '0.00000000',
# maintMargin: '0.00000000',
# initialMargin: '0.00000000',
# positionInitialMargin: '0.00000000',
# openOrderInitialMargin: '0.00000000',
# maxWithdrawAmount: '0.00000000',
# crossWalletBalance: '0.00000000',
# crossUnPnl: '0.00000000',
# availableBalance: '0.00000000',
# updateTime: '0'
# },
# ...
# ],
# positions: [
# {
# symbol: 'BCHUSD_240329',
# initialMargin: '0',
# maintMargin: '0',
# unrealizedProfit: '0.00000000',
# positionInitialMargin: '0',
# openOrderInitialMargin: '0',
# leverage: '20',
# isolated: False,
# positionSide: 'BOTH',
# entryPrice: '0.00000000',
# maxQty: '1000',
# notionalValue: '0',
# isolatedWallet: '0',
# updateTime: '0',
# positionAmt: '0',
# breakEvenPrice: '0.00000000'
# },
# ...
# ]
# }
#
else:
raise BadRequest(self.id + ' fetchMarginModes() supports linear and inverse subTypes only')
assets = self.safe_list(response, 'positions', [])
if isinstance(response, list):
assets = response
return self.parse_margin_modes(assets, symbols, 'symbol', 'swap')
def fetch_margin_mode(self, symbol: str, params={}) -> MarginMode:
"""
fetches the margin mode of a specific symbol
https://developers.binance.com/docs/derivatives/usds-margined-futures/account/rest-api/Symbol-Config
https://developers.binance.com/docs/derivatives/coin-margined-futures/account/rest-api/Account-Information
: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.subType]: "linear" or "inverse"
:returns dict: a `margin mode structure <https://docs.ccxt.com/#/?id=margin-mode-structure>`
"""
self.load_markets()
market = self.market(symbol)
subType = None
subType, params = self.handle_sub_type_and_params('fetchMarginMode', market, params)
response = None
if subType == 'linear':
request: dict = {
'symbol': market['id'],
}
response = self.fapiPrivateGetSymbolConfig(self.extend(request, params))
#
# [
# {
# "symbol": "BTCUSDT",
# "marginType": "CROSSED",
# "isAutoAddMargin": "false",
# "leverage": 21,
# "maxNotionalValue": "1000000",
# }
# ]
#
elif subType == 'inverse':
fetchMarginModesResponse = self.fetch_margin_modes([symbol], params)
return fetchMarginModesResponse[symbol]
else:
raise BadRequest(self.id + ' fetchMarginMode() supports linear and inverse subTypes only')
return self.parse_margin_mode(response[0], market)
def parse_margin_mode(self, marginMode: dict, market=None) -> MarginMode:
marketId = self.safe_string(marginMode, 'symbol')
market = self.safe_market(marketId, market)
marginModeRaw = self.safe_bool(marginMode, 'isolated')
reMarginMode = None
if marginModeRaw is not None:
reMarginMode = 'isolated' if marginModeRaw else 'cross'
marginTypeRaw = self.safe_string_lower(marginMode, 'marginType')
if marginTypeRaw is not None:
reMarginMode = 'cross' if (marginTypeRaw == 'crossed') else 'isolated'
return {
'info': marginMode,
'symbol': market['symbol'],
'marginMode': reMarginMode,
}
def fetch_option(self, symbol: str, params={}) -> Option:
"""
fetches option data that is commonly found in an option chain
https://developers.binance.com/docs/derivatives/option/market-data/24hr-Ticker-Price-Change-Statistics
:param str symbol: unified market symbol
:param dict [params]: extra parameters specific to the exchange API endpoint
:returns dict: an `option chain structure <https://docs.ccxt.com/#/?id=option-chain-structure>`
"""
self.load_markets()
market = self.market(symbol)
request: dict = {
'symbol': market['id'],
}
response = self.eapiPublicGetTicker(self.extend(request, params))
#
# [
# {
# "symbol": "BTC-241227-80000-C",
# "priceChange": "0",
# "priceChangePercent": "0",
# "lastPrice": "2750",
# "lastQty": "0",
# "open": "2750",
# "high": "2750",
# "low": "2750",
# "volume": "0",
# "amount": "0",
# "bidPrice": "4880",
# "askPrice": "0",
# "openTime": 0,
# "closeTime": 0,
# "firstTradeId": 0,
# "tradeCount": 0,
# "strikePrice": "80000",
# "exercisePrice": "63944.09893617"
# }
# ]
#
chain = self.safe_dict(response, 0, {})
return self.parse_option(chain, None, market)
def parse_option(self, chain: dict, currency: Currency = None, market: Market = None) -> Option:
#
# {
# "symbol": "BTC-241227-80000-C",
# "priceChange": "0",
# "priceChangePercent": "0",
# "lastPrice": "2750",
# "lastQty": "0",
# "open": "2750",
# "high": "2750",
# "low": "2750",
# "volume": "0",
# "amount": "0",
# "bidPrice": "4880",
# "askPrice": "0",
# "openTime": 0,
# "closeTime": 0,
# "firstTradeId": 0,
# "tradeCount": 0,
# "strikePrice": "80000",
# "exercisePrice": "63944.09893617"
# }
#
marketId = self.safe_string(chain, 'symbol')
market = self.safe_market(marketId, market)
return {
'info': chain,
'currency': None,
'symbol': market['symbol'],
'timestamp': None,
'datetime': None,
'impliedVolatility': None,
'openInterest': None,
'bidPrice': self.safe_number(chain, 'bidPrice'),
'askPrice': self.safe_number(chain, 'askPrice'),
'midPrice': None,
'markPrice': None,
'lastPrice': self.safe_number(chain, 'lastPrice'),
'underlyingPrice': self.safe_number(chain, 'exercisePrice'),
'change': self.safe_number(chain, 'priceChange'),
'percentage': self.safe_number(chain, 'priceChangePercent'),
'baseVolume': self.safe_number(chain, 'volume'),
'quoteVolume': None,
}
def fetch_margin_adjustment_history(self, symbol: Str = None, type: Str = None, since: Num = None, limit: Num = None, params={}) -> List[MarginModification]:
"""
fetches the history of margin added or reduced from contract isolated positions
https://developers.binance.com/docs/derivatives/usds-margined-futures/trade/rest-api/Get-Position-Margin-Change-History
https://developers.binance.com/docs/derivatives/coin-margined-futures/trade/rest-api/Get-Position-Margin-Change-History
:param str symbol: unified market symbol
:param str [type]: "add" or "reduce"
:param int [since]: timestamp in ms of the earliest change to fetch
:param int [limit]: the maximum amount of changes to fetch
:param dict params: extra parameters specific to the exchange api endpoint
:param int [params.until]: timestamp in ms of the latest change to fetch
:returns dict[]: a list of `margin structures <https://docs.ccxt.com/#/?id=margin-loan-structure>`
"""
self.load_markets()
if symbol is None:
raise ArgumentsRequired(self.id + ' fetchMarginAdjustmentHistory() requires a symbol argument')
market = self.market(symbol)
until = self.safe_integer(params, 'until')
params = self.omit(params, 'until')
request: dict = {
'symbol': market['id'],
}
if type is not None:
request['type'] = 1 if (type == 'add') else 2
if since is not None:
request['startTime'] = since
if limit is not None:
request['limit'] = limit
if until is not None:
request['endTime'] = until
response = None
if market['linear']:
response = self.fapiPrivateGetPositionMarginHistory(self.extend(request, params))
elif market['inverse']:
response = self.dapiPrivateGetPositionMarginHistory(self.extend(request, params))
else:
raise BadRequest(self.id + ' fetchMarginAdjustmentHistory() is not supported for markets of type ' + market['type'])
#
# [
# {
# symbol: "XRPUSDT",
# type: "1",
# deltaType: "TRADE",
# amount: "2.57148240",
# asset: "USDT",
# time: "1711046271555",
# positionSide: "BOTH",
# clientTranId: ""
# }
# ...
# ]
#
modifications = self.parse_margin_modifications(response)
return self.filter_by_symbol_since_limit(modifications, symbol, since, limit)
def fetch_convert_currencies(self, params={}) -> Currencies:
"""
fetches all available currencies that can be converted
https://developers.binance.com/docs/convert/market-data/Query-order-quantity-precision-per-asset
:param dict [params]: extra parameters specific to the exchange API endpoint
:returns dict: an associative dictionary of currencies
"""
self.load_markets()
response = self.sapiGetConvertAssetInfo(params)
#
# [
# {
# "asset": "BTC",
# "fraction": 8
# },
# ]
#
result: dict = {}
for i in range(0, len(response)):
entry = response[i]
id = self.safe_string(entry, 'asset')
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': None,
'fee': None,
'precision': self.parse_number(self.parse_precision(self.safe_string(entry, 'fraction'))),
'limits': {
'amount': {
'min': None,
'max': None,
},
'withdraw': {
'min': None,
'max': None,
},
'deposit': {
'min': None,
'max': None,
},
},
'created': None,
}
return result
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://developers.binance.com/docs/convert/trade/Send-quote-request
: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.walletType]: either 'SPOT' or 'FUNDING', the default is 'SPOT'
:returns dict: a `conversion structure <https://docs.ccxt.com/#/?id=conversion-structure>`
"""
if amount is None:
raise ArgumentsRequired(self.id + ' fetchConvertQuote() requires an amount argument')
self.load_markets()
request: dict = {
'fromAsset': fromCode,
'toAsset': toCode,
'fromAmount': amount,
}
response = self.sapiPostConvertGetQuote(self.extend(request, params))
#
# {
# "quoteId":"12415572564",
# "ratio":"38163.7",
# "inverseRatio":"0.0000262",
# "validTimestamp":1623319461670,
# "toAmount":"3816.37",
# "fromAmount":"0.1"
# }
#
fromCurrency = self.currency(fromCode)
toCurrency = self.currency(toCode)
return self.parse_conversion(response, fromCurrency, toCurrency)
def create_convert_trade(self, id: str, fromCode: str, toCode: str, amount: Num = None, params={}) -> Conversion:
"""
convert from one currency to another
https://developers.binance.com/docs/convert/trade/Accept-Quote
: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
:returns dict: a `conversion structure <https://docs.ccxt.com/#/?id=conversion-structure>`
"""
self.load_markets()
request: dict = {}
response = None
if (fromCode == 'BUSD') or (toCode == 'BUSD'):
if amount is None:
raise ArgumentsRequired(self.id + ' createConvertTrade() requires an amount argument')
request['clientTranId'] = id
request['asset'] = fromCode
request['targetAsset'] = toCode
request['amount'] = amount
response = self.sapiPostAssetConvertTransfer(self.extend(request, params))
#
# {
# "tranId": 118263407119,
# "status": "S"
# }
#
else:
request['quoteId'] = id
response = self.sapiPostConvertAcceptQuote(self.extend(request, params))
#
# {
# "orderId":"933256278426274426",
# "createTime":1623381330472,
# "orderStatus":"PROCESS"
# }
#
fromCurrency = self.currency(fromCode)
toCurrency = self.currency(toCode)
return self.parse_conversion(response, fromCurrency, toCurrency)
def fetch_convert_trade(self, id: str, code: Str = None, params={}) -> Conversion:
"""
fetch the data for a conversion trade
https://developers.binance.com/docs/convert/trade/Order-Status
:param str id: the id of the trade that you want to fetch
:param str [code]: the unified currency code of the conversion trade
:param dict [params]: extra parameters specific to the exchange API endpoint
:returns dict: a `conversion structure <https://docs.ccxt.com/#/?id=conversion-structure>`
"""
self.load_markets()
request: dict = {}
response = None
if code == 'BUSD':
msInDay = 86400000
now = self.milliseconds()
if code is not None:
currency = self.currency(code)
request['asset'] = currency['id']
request['tranId'] = id
request['startTime'] = now - msInDay
request['endTime'] = now
response = self.sapiGetAssetConvertTransferQueryByPage(self.extend(request, params))
#
# {
# "total": 3,
# "rows": [
# {
# "tranId": 118263615991,
# "type": 244,
# "time": 1664442078000,
# "deductedAsset": "BUSD",
# "deductedAmount": "1",
# "targetAsset": "USDC",
# "targetAmount": "1",
# "status": "S",
# "accountType": "MAIN"
# },
# ]
# }
#
else:
request['orderId'] = id
response = self.sapiGetConvertOrderStatus(self.extend(request, params))
#
# {
# "orderId":933256278426274426,
# "orderStatus":"SUCCESS",
# "fromAsset":"BTC",
# "fromAmount":"0.00054414",
# "toAsset":"USDT",
# "toAmount":"20",
# "ratio":"36755",
# "inverseRatio":"0.00002721",
# "createTime":1623381330472
# }
#
data = response
if code == 'BUSD':
rows = self.safe_list(response, 'rows', [])
data = self.safe_dict(rows, 0, {})
fromCurrencyId = self.safe_string_2(data, 'deductedAsset', 'fromAsset')
toCurrencyId = self.safe_string_2(data, 'targetAsset', 'toAsset')
fromCurrency = None
toCurrency = None
if fromCurrencyId is not None:
fromCurrency = self.currency(fromCurrencyId)
if toCurrencyId is not None:
toCurrency = self.currency(toCurrencyId)
return self.parse_conversion(data, fromCurrency, toCurrency)
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://developers.binance.com/docs/convert/trade/Get-Convert-Trade-History
: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
:param int [params.until]: timestamp in ms of the latest conversion to fetch
:returns dict[]: a list of `conversion structures <https://docs.ccxt.com/#/?id=conversion-structure>`
"""
self.load_markets()
request: dict = {}
msInThirtyDays = 2592000000
now = self.milliseconds()
if since is not None:
request['startTime'] = since
else:
request['startTime'] = now - msInThirtyDays
endTime = self.safe_integer_2(params, 'endTime', 'until')
if endTime is not None:
request['endTime'] = endTime
else:
request['endTime'] = now
params = self.omit(params, 'until')
response = None
responseQuery = None
fromCurrencyKey = None
toCurrencyKey = None
if code == 'BUSD':
currency = self.currency(code)
request['asset'] = currency['id']
if limit is not None:
request['size'] = limit
fromCurrencyKey = 'deductedAsset'
toCurrencyKey = 'targetAsset'
responseQuery = 'rows'
response = self.sapiGetAssetConvertTransferQueryByPage(self.extend(request, params))
#
# {
# "total": 3,
# "rows": [
# {
# "tranId": 118263615991,
# "type": 244,
# "time": 1664442078000,
# "deductedAsset": "BUSD",
# "deductedAmount": "1",
# "targetAsset": "USDC",
# "targetAmount": "1",
# "status": "S",
# "accountType": "MAIN"
# },
# ]
# }
#
else:
if (request['endTime'] - request['startTime']) > msInThirtyDays:
raise BadRequest(self.id + ' fetchConvertTradeHistory() the max interval between startTime and endTime is 30 days.')
if limit is not None:
request['limit'] = limit
fromCurrencyKey = 'fromAsset'
toCurrencyKey = 'toAsset'
responseQuery = 'list'
response = self.sapiGetConvertTradeFlow(self.extend(request, params))
#
# {
# "list": [
# {
# "quoteId": "f3b91c525b2644c7bc1e1cd31b6e1aa6",
# "orderId": 940708407462087195,
# "orderStatus": "SUCCESS",
# "fromAsset": "USDT",
# "fromAmount": "20",
# "toAsset": "BNB",
# "toAmount": "0.06154036",
# "ratio": "0.00307702",
# "inverseRatio": "324.99",
# "createTime": 1624248872184
# }
# ],
# "startTime": 1623824139000,
# "endTime": 1626416139000,
# "limit": 100,
# "moreData": False
# }
#
rows = self.safe_list(response, responseQuery, [])
return self.parse_conversions(rows, code, fromCurrencyKey, toCurrencyKey, since, limit)
def parse_conversion(self, conversion: dict, fromCurrency: Currency = None, toCurrency: Currency = None) -> Conversion:
#
# fetchConvertQuote
#
# {
# "quoteId":"12415572564",
# "ratio":"38163.7",
# "inverseRatio":"0.0000262",
# "validTimestamp":1623319461670,
# "toAmount":"3816.37",
# "fromAmount":"0.1"
# }
#
# createConvertTrade
#
# {
# "orderId":"933256278426274426",
# "createTime":1623381330472,
# "orderStatus":"PROCESS"
# }
#
# createConvertTrade BUSD
#
# {
# "tranId": 118263407119,
# "status": "S"
# }
#
# fetchConvertTrade, fetchConvertTradeHistory BUSD
#
# {
# "tranId": 118263615991,
# "type": 244,
# "time": 1664442078000,
# "deductedAsset": "BUSD",
# "deductedAmount": "1",
# "targetAsset": "USDC",
# "targetAmount": "1",
# "status": "S",
# "accountType": "MAIN"
# }
#
# fetchConvertTrade
#
# {
# "orderId":933256278426274426,
# "orderStatus":"SUCCESS",
# "fromAsset":"BTC",
# "fromAmount":"0.00054414",
# "toAsset":"USDT",
# "toAmount":"20",
# "ratio":"36755",
# "inverseRatio":"0.00002721",
# "createTime":1623381330472
# }
#
# fetchConvertTradeHistory
#
# {
# "quoteId": "f3b91c525b2644c7bc1e1cd31b6e1aa6",
# "orderId": 940708407462087195,
# "orderStatus": "SUCCESS",
# "fromAsset": "USDT",
# "fromAmount": "20",
# "toAsset": "BNB",
# "toAmount": "0.06154036",
# "ratio": "0.00307702",
# "inverseRatio": "324.99",
# "createTime": 1624248872184
# }
#
timestamp = self.safe_integer_n(conversion, ['time', 'validTimestamp', 'createTime'])
fromCur = self.safe_string_2(conversion, 'deductedAsset', 'fromAsset')
fromCode = self.safe_currency_code(fromCur, fromCurrency)
to = self.safe_string_2(conversion, 'targetAsset', 'toAsset')
toCode = self.safe_currency_code(to, toCurrency)
return {
'info': conversion,
'timestamp': timestamp,
'datetime': self.iso8601(timestamp),
'id': self.safe_string_n(conversion, ['tranId', 'orderId', 'quoteId']),
'fromCurrency': fromCode,
'fromAmount': self.safe_number_2(conversion, 'deductedAmount', 'fromAmount'),
'toCurrency': toCode,
'toAmount': self.safe_number_2(conversion, 'targetAmount', 'toAmount'),
'price': None,
'fee': None,
}
def fetch_funding_intervals(self, symbols: Strings = None, params={}) -> FundingRates:
"""
fetch the funding rate interval for multiple markets
https://developers.binance.com/docs/derivatives/usds-margined-futures/market-data/rest-api/Get-Funding-Rate-Info
https://developers.binance.com/docs/derivatives/coin-margined-futures/market-data/rest-api/Get-Funding-Info
:param str[] [symbols]: list of unified market symbols
:param dict [params]: extra parameters specific to the exchange API endpoint
:param str [params.subType]: "linear" or "inverse"
:returns dict[]: a list of `funding rate structures <https://docs.ccxt.com/#/?id=funding-rate-structure>`
"""
self.load_markets()
market = None
if symbols is not None:
symbols = self.market_symbols(symbols)
market = self.market(symbols[0])
type = 'swap'
subType = None
subType, params = self.handle_sub_type_and_params('fetchFundingIntervals', market, params, 'linear')
response = None
if self.is_linear(type, subType):
response = self.fapiPublicGetFundingInfo(params)
elif self.is_inverse(type, subType):
response = self.dapiPublicGetFundingInfo(params)
else:
raise NotSupported(self.id + ' fetchFundingIntervals() supports linear and inverse swap contracts only')
#
# [
# {
# "symbol": "BLZUSDT",
# "adjustedFundingRateCap": "0.03000000",
# "adjustedFundingRateFloor": "-0.03000000",
# "fundingIntervalHours": 4,
# "disclaimer": False
# },
# ]
#
return self.parse_funding_rates(response, symbols)
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://developers.binance.com/docs/derivatives/usds-margined-futures/market-data/rest-api/Long-Short-Ratio
https://developers.binance.com/docs/derivatives/coin-margined-futures/market-data/rest-api/Long-Short-Ratio
:param str symbol: unified symbol of the market to fetch the long short ratio for
:param str [timeframe]: the period for the ratio, default is 24 hours
: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
:param int [params.until]: timestamp in ms of the latest ratio to fetch
:returns dict[]: an array of `long short ratio structures <https://docs.ccxt.com/#/?id=long-short-ratio-structure>`
"""
self.load_markets()
market = self.market(symbol)
if timeframe is None:
timeframe = '1d'
request: dict = {
'period': timeframe,
}
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
subType = None
subType, params = self.handle_sub_type_and_params('fetchLongShortRatioHistory', market, params)
response = None
if subType == 'linear':
request['symbol'] = market['id']
response = self.fapiDataGetGlobalLongShortAccountRatio(self.extend(request, params))
#
# [
# {
# "symbol": "BTCUSDT",
# "longAccount": "0.4558",
# "longShortRatio": "0.8376",
# "shortAccount": "0.5442",
# "timestamp": 1726790400000
# },
# ]
#
elif subType == 'inverse':
request['pair'] = market['info']['pair']
response = self.dapiDataGetGlobalLongShortAccountRatio(self.extend(request, params))
#
# [
# {
# "longAccount": "0.7262",
# "longShortRatio": "2.6523",
# "shortAccount": "0.2738",
# "pair": "BTCUSD",
# "timestamp": 1726790400000
# },
# ]
#
else:
raise BadRequest(self.id + ' fetchLongShortRatioHistory() supports linear and inverse subTypes only')
return self.parse_long_short_ratio_history(response, market)
def parse_long_short_ratio(self, info: dict, market: Market = None) -> LongShortRatio:
#
# linear
#
# {
# "symbol": "BTCUSDT",
# "longAccount": "0.4558",
# "longShortRatio": "0.8376",
# "shortAccount": "0.5442",
# "timestamp": 1726790400000
# }
#
# inverse
#
# {
# "longAccount": "0.7262",
# "longShortRatio": "2.6523",
# "shortAccount": "0.2738",
# "pair": "BTCUSD",
# "timestamp": 1726790400000
# }
#
marketId = self.safe_string(info, 'symbol')
timestamp = self.safe_integer_omit_zero(info, 'timestamp')
return {
'info': info,
'symbol': self.safe_symbol(marketId, market, None, 'contract'),
'timestamp': timestamp,
'datetime': self.iso8601(timestamp),
'timeframe': None,
'longShortRatio': self.safe_number(info, 'longShortRatio'),
}