UNPKG

sfccxt

Version:

A JavaScript / Python / PHP cryptocurrency trading library with support for 130+ exchanges

1,183 lines (1,154 loc) 75.1 kB
'use strict'; // ---------------------------------------------------------------------------- const Exchange = require ('./base/Exchange'); const { InsufficientFunds, ArgumentsRequired, ExchangeError, InvalidOrder, InvalidAddress, AuthenticationError, NotSupported, OrderNotFound, OnMaintenance, PermissionDenied, RateLimitExceeded } = require ('./base/errors'); const { TICK_SIZE } = require ('./base/functions/number'); const Precise = require ('./base/Precise'); // ---------------------------------------------------------------------------- module.exports = class coinbasepro extends Exchange { describe () { return this.deepExtend (super.describe (), { 'id': 'coinbasepro', 'name': 'Coinbase Pro', 'countries': [ 'US' ], 'rateLimit': 100, 'userAgent': this.userAgents['chrome'], 'pro': true, 'has': { 'CORS': true, 'spot': true, 'margin': false, 'swap': false, 'future': false, 'option': false, 'cancelAllOrders': true, 'cancelOrder': true, 'createDepositAddress': true, 'createOrder': true, 'createStopLimitOrder': true, 'createStopMarketOrder': true, 'createStopOrder': true, 'fetchAccounts': true, 'fetchBalance': true, 'fetchClosedOrders': true, 'fetchCurrencies': true, 'fetchDepositAddress': undefined, // the exchange does not have this method, only createDepositAddress, see https://github.com/ccxt/ccxt/pull/7405 'fetchDeposits': true, 'fetchLedger': true, 'fetchMarginMode': false, 'fetchMarkets': true, 'fetchMyTrades': true, 'fetchOHLCV': true, 'fetchOpenOrders': true, 'fetchOrder': true, 'fetchOrderBook': true, 'fetchOrders': true, 'fetchOrderTrades': true, 'fetchPositionMode': false, 'fetchTicker': true, 'fetchTickers': true, 'fetchTime': true, 'fetchTrades': true, 'fetchTradingFee': false, 'fetchTradingFees': true, 'fetchTransactions': true, 'fetchWithdrawals': true, 'withdraw': true, }, 'timeframes': { '1m': 60, '5m': 300, '15m': 900, '1h': 3600, '6h': 21600, '1d': 86400, }, 'hostname': 'pro.coinbase.com', 'urls': { 'test': { 'public': 'https://api-public.sandbox.pro.coinbase.com', 'private': 'https://api-public.sandbox.pro.coinbase.com', }, 'logo': 'https://user-images.githubusercontent.com/1294454/41764625-63b7ffde-760a-11e8-996d-a6328fa9347a.jpg', 'api': { 'public': 'https://api.{hostname}', 'private': 'https://api.{hostname}', }, 'www': 'https://pro.coinbase.com/', 'doc': 'https://docs.pro.coinbase.com', 'fees': [ 'https://docs.pro.coinbase.com/#fees', 'https://support.pro.coinbase.com/customer/en/portal/articles/2945310-fees', ], }, 'requiredCredentials': { 'apiKey': true, 'secret': true, 'password': true, }, 'api': { 'public': { 'get': [ 'currencies', 'products', 'products/{id}', 'products/{id}/book', 'products/{id}/candles', 'products/{id}/stats', 'products/{id}/ticker', 'products/{id}/trades', 'time', 'products/spark-lines', // experimental ], }, 'private': { 'get': [ 'accounts', 'accounts/{id}', 'accounts/{id}/holds', 'accounts/{id}/ledger', 'accounts/{id}/transfers', 'coinbase-accounts', 'fills', 'funding', 'fees', 'margin/profile_information', 'margin/buying_power', 'margin/withdrawal_power', 'margin/withdrawal_power_all', 'margin/exit_plan', 'margin/liquidation_history', 'margin/position_refresh_amounts', 'margin/status', 'oracle', 'orders', 'orders/{id}', 'orders/client:{client_oid}', 'otc/orders', 'payment-methods', 'position', 'profiles', 'profiles/{id}', 'reports/{report_id}', 'transfers', 'transfers/{transfer_id}', 'users/self/exchange-limits', 'users/self/hold-balances', 'users/self/trailing-volume', 'withdrawals/fee-estimate', 'conversions/{conversion_id}', ], 'post': [ 'conversions', 'deposits/coinbase-account', 'deposits/payment-method', 'coinbase-accounts/{id}/addresses', 'funding/repay', 'orders', 'position/close', 'profiles/margin-transfer', 'profiles/transfer', 'reports', 'withdrawals/coinbase', 'withdrawals/coinbase-account', 'withdrawals/crypto', 'withdrawals/payment-method', ], 'delete': [ 'orders', 'orders/client:{client_oid}', 'orders/{id}', ], }, }, 'commonCurrencies': { 'CGLD': 'CELO', }, 'precisionMode': TICK_SIZE, 'fees': { 'trading': { 'tierBased': true, // complicated tier system per coin 'percentage': true, 'maker': 0.4 / 100, // highest fee of all tiers 'taker': 0.6 / 100, // highest fee of all tiers }, 'funding': { 'tierBased': false, 'percentage': false, 'withdraw': { 'BCH': 0, 'BTC': 0, 'LTC': 0, 'ETH': 0, 'EUR': 0.15, 'USD': 25, }, 'deposit': { 'BCH': 0, 'BTC': 0, 'LTC': 0, 'ETH': 0, 'EUR': 0.15, 'USD': 10, }, }, }, 'exceptions': { 'exact': { 'Insufficient funds': InsufficientFunds, 'NotFound': OrderNotFound, 'Invalid API Key': AuthenticationError, 'invalid signature': AuthenticationError, 'Invalid Passphrase': AuthenticationError, 'Invalid order id': InvalidOrder, 'Private rate limit exceeded': RateLimitExceeded, 'Trading pair not available': PermissionDenied, 'Product not found': InvalidOrder, }, 'broad': { 'Order already done': OrderNotFound, 'order not found': OrderNotFound, 'price too small': InvalidOrder, 'price too precise': InvalidOrder, 'under maintenance': OnMaintenance, 'size is too small': InvalidOrder, 'Cancel only mode': OnMaintenance, // https://github.com/ccxt/ccxt/issues/7690 }, }, }); } async fetchCurrencies (params = {}) { /** * @method * @name coinbasepro#fetchCurrencies * @description fetches all available currencies on an exchange * @param {object} params extra parameters specific to the coinbasepro api endpoint * @returns {object} an associative dictionary of currencies */ const response = await this.publicGetCurrencies (params); // // [ // { // id: 'XTZ', // name: 'Tezos', // min_size: '0.000001', // status: 'online', // message: '', // max_precision: '0.000001', // convertible_to: [], // details: { // type: 'crypto', // symbol: 'Τ', // network_confirmations: 60, // sort_order: 53, // crypto_address_link: 'https://tzstats.com/{{address}}', // crypto_transaction_link: 'https://tzstats.com/{{txId}}', // push_payment_methods: [ 'crypto' ], // group_types: [], // display_name: '', // processing_time_seconds: 0, // min_withdrawal_amount: 1 // } // } // ] // const result = {}; for (let i = 0; i < response.length; i++) { const currency = response[i]; const id = this.safeString (currency, 'id'); const name = this.safeString (currency, 'name'); const code = this.safeCurrencyCode (id); const details = this.safeValue (currency, 'details', {}); const status = this.safeString (currency, 'status'); const active = (status === 'online'); result[code] = { 'id': id, 'code': code, 'info': currency, 'type': this.safeString (details, 'type'), 'name': name, 'active': active, 'deposit': undefined, 'withdraw': undefined, 'fee': undefined, 'precision': this.safeNumber (currency, 'max_precision'), 'limits': { 'amount': { 'min': this.safeNumber (details, 'min_size'), 'max': undefined, }, 'withdraw': { 'min': this.safeNumber (details, 'min_withdrawal_amount'), 'max': undefined, }, }, }; } return result; } async fetchMarkets (params = {}) { /** * @method * @name coinbasepro#fetchMarkets * @description retrieves data on all markets for coinbasepro * @param {object} params extra parameters specific to the exchange api endpoint * @returns {[object]} an array of objects representing market data */ const response = await this.publicGetProducts (params); // // [ // { // id: 'BTCAUCTION-USD', // base_currency: 'BTC', // quote_currency: 'USD', // base_min_size: '0.000016', // base_max_size: '1500', // quote_increment: '0.01', // base_increment: '0.00000001', // display_name: 'BTCAUCTION/USD', // min_market_funds: '1', // max_market_funds: '20000000', // margin_enabled: false, // fx_stablecoin: false, // max_slippage_percentage: '0.02000000', // post_only: false, // limit_only: false, // cancel_only: true, // trading_disabled: false, // status: 'online', // status_message: '', // auction_mode: false // }, // { // id: 'BTC-USD', // base_currency: 'BTC', // quote_currency: 'USD', // base_min_size: '0.000016', // base_max_size: '1500', // quote_increment: '0.01', // base_increment: '0.00000001', // display_name: 'BTC/USD', // min_market_funds: '1', // max_market_funds: '20000000', // margin_enabled: false, // fx_stablecoin: false, // max_slippage_percentage: '0.02000000', // post_only: false, // limit_only: false, // cancel_only: false, // trading_disabled: false, // status: 'online', // status_message: '', // auction_mode: false // } // ] // const result = []; for (let i = 0; i < response.length; i++) { const market = response[i]; const id = this.safeString (market, 'id'); const [ baseId, quoteId ] = id.split ('-'); // BTCAUCTION-USD vs BTC-USD conflict workaround, see the output sample above // const baseId = this.safeString (market, 'base_currency'); // const quoteId = this.safeString (market, 'quote_currency'); const base = this.safeCurrencyCode (baseId); const quote = this.safeCurrencyCode (quoteId); const status = this.safeString (market, 'status'); result.push (this.extend (this.fees['trading'], { 'id': id, 'symbol': base + '/' + quote, 'base': base, 'quote': quote, 'settle': undefined, 'baseId': baseId, 'quoteId': quoteId, 'settleId': undefined, 'type': 'spot', 'spot': true, 'margin': this.safeValue (market, 'margin_enabled'), 'swap': false, 'future': false, 'option': false, 'active': (status === 'online'), 'contract': false, 'linear': undefined, 'inverse': undefined, 'contractSize': undefined, 'expiry': undefined, 'expiryDatetime': undefined, 'strike': undefined, 'optionType': undefined, 'precision': { 'amount': this.safeNumber (market, 'base_increment'), 'price': this.safeNumber (market, 'quote_increment'), }, 'limits': { 'leverage': { 'min': undefined, 'max': undefined, }, 'amount': { 'min': undefined, 'max': undefined, }, 'price': { 'min': undefined, 'max': undefined, }, 'cost': { 'min': this.safeNumber (market, 'min_market_funds'), 'max': undefined, }, }, 'info': market, })); } return result; } async fetchAccounts (params = {}) { /** * @method * @name coinbasepro#fetchAccounts * @description fetch all the accounts associated with a profile * @param {object} params extra parameters specific to the coinbasepro api endpoint * @returns {object} a dictionary of [account structures]{@link https://docs.ccxt.com/en/latest/manual.html#account-structure} indexed by the account type */ await this.loadMarkets (); const response = await this.privateGetAccounts (params); // // [ // { // id: '4aac9c60-cbda-4396-9da4-4aa71e95fba0', // currency: 'BTC', // balance: '0.0000000000000000', // available: '0', // hold: '0.0000000000000000', // profile_id: 'b709263e-f42a-4c7d-949a-a95c83d065da' // }, // { // id: 'f75fa69a-1ad1-4a80-bd61-ee7faa6135a3', // currency: 'USDC', // balance: '0.0000000000000000', // available: '0', // hold: '0.0000000000000000', // profile_id: 'b709263e-f42a-4c7d-949a-a95c83d065da' // }, // ] // return this.parseAccounts (response, params); } parseAccount (account) { // // { // id: '4aac9c60-cbda-4396-9da4-4aa71e95fba0', // currency: 'BTC', // balance: '0.0000000000000000', // available: '0', // hold: '0.0000000000000000', // profile_id: 'b709263e-f42a-4c7d-949a-a95c83d065da' // } // const currencyId = this.safeString (account, 'currency'); return { 'id': this.safeString (account, 'id'), 'type': undefined, 'code': this.safeCurrencyCode (currencyId), 'info': account, }; } parseBalance (response) { const result = { 'info': response }; for (let i = 0; i < response.length; i++) { const balance = response[i]; const currencyId = this.safeString (balance, 'currency'); const code = this.safeCurrencyCode (currencyId); const account = this.account (); account['free'] = this.safeString (balance, 'available'); account['used'] = this.safeString (balance, 'hold'); account['total'] = this.safeString (balance, 'balance'); result[code] = account; } return this.safeBalance (result); } async fetchBalance (params = {}) { /** * @method * @name coinbasepro#fetchBalance * @description query for balance and get the amount of funds available for trading or funds locked in orders * @param {object} params extra parameters specific to the coinbasepro api endpoint * @returns {object} a [balance structure]{@link https://docs.ccxt.com/en/latest/manual.html?#balance-structure} */ await this.loadMarkets (); const response = await this.privateGetAccounts (params); return this.parseBalance (response); } async fetchOrderBook (symbol, limit = undefined, params = {}) { /** * @method * @name coinbasepro#fetchOrderBook * @description fetches information on open orders with bid (buy) and ask (sell) prices, volumes and other data * @param {string} symbol unified symbol of the market to fetch the order book for * @param {int|undefined} limit the maximum amount of order book entries to return * @param {object} params extra parameters specific to the coinbasepro api endpoint * @returns {object} A dictionary of [order book structures]{@link https://docs.ccxt.com/en/latest/manual.html#order-book-structure} indexed by market symbols */ await this.loadMarkets (); // level 1 - only the best bid and ask // level 2 - top 50 bids and asks (aggregated) // level 3 - full order book (non aggregated) const request = { 'id': this.marketId (symbol), 'level': 2, // 1 best bidask, 2 aggregated, 3 full }; const response = await this.publicGetProductsIdBook (this.extend (request, params)); // // { // "sequence":1924393896, // "bids":[ // ["0.01825","24.34811287",2], // ["0.01824","72.5463",3], // ["0.01823","424.54298049",6], // ], // "asks":[ // ["0.01826","171.10414904",4], // ["0.01827","22.60427028",1], // ["0.01828","397.46018784",7], // ] // } // const orderbook = this.parseOrderBook (response, symbol); orderbook['nonce'] = this.safeInteger (response, 'sequence'); return orderbook; } parseTicker (ticker, market = undefined) { // // fetchTickers // // [ // 1639472400, // timestamp // 4.26, // low // 4.38, // high // 4.35, // open // 4.27 // close // ] // // fetchTicker // // publicGetProductsIdTicker // // { // "trade_id":843439, // "price":"0.997999", // "size":"80.29769", // "time":"2020-01-28T02:13:33.012523Z", // "bid":"0.997094", // "ask":"0.998", // "volume":"1903188.03750000" // } // // publicGetProductsIdStats // // { // "open": "34.19000000", // "high": "95.70000000", // "low": "7.06000000", // "volume": "2.41000000" // } // let timestamp = undefined; let bid = undefined; let ask = undefined; let last = undefined; let high = undefined; let low = undefined; let open = undefined; let volume = undefined; const symbol = (market === undefined) ? undefined : market['symbol']; if (Array.isArray (ticker)) { last = this.safeString (ticker, 4); timestamp = this.milliseconds (); } else { timestamp = this.parse8601 (this.safeValue (ticker, 'time')); bid = this.safeString (ticker, 'bid'); ask = this.safeString (ticker, 'ask'); high = this.safeString (ticker, 'high'); low = this.safeString (ticker, 'low'); open = this.safeString (ticker, 'open'); last = this.safeString2 (ticker, 'price', 'last'); volume = this.safeString (ticker, 'volume'); } return this.safeTicker ({ 'symbol': symbol, 'timestamp': timestamp, 'datetime': this.iso8601 (timestamp), 'high': high, 'low': low, 'bid': bid, 'bidVolume': undefined, 'ask': ask, 'askVolume': undefined, 'vwap': undefined, 'open': open, 'close': last, 'last': last, 'previousClose': undefined, 'change': undefined, 'percentage': undefined, 'average': undefined, 'baseVolume': volume, 'quoteVolume': undefined, 'info': ticker, }, market); } async fetchTickers (symbols = undefined, params = {}) { /** * @method * @name coinbasepro#fetchTickers * @description fetches price tickers for multiple markets, statistical calculations with the information calculated over the past 24 hours each market * @param {[string]|undefined} symbols unified symbols of the markets to fetch the ticker for, all market tickers are returned if not assigned * @param {object} params extra parameters specific to the coinbasepro api endpoint * @returns {object} an array of [ticker structures]{@link https://docs.ccxt.com/en/latest/manual.html#ticker-structure} */ await this.loadMarkets (); symbols = this.marketSymbols (symbols); const request = {}; const response = await this.publicGetProductsSparkLines (this.extend (request, params)); // // { // YYY-USD: [ // [ // 1639472400, // timestamp // 4.26, // low // 4.38, // high // 4.35, // open // 4.27 // close // ], // [ // 1639468800, // 4.31, // 4.45, // 4.35, // 4.35 // ], // ] // } // const result = {}; const marketIds = Object.keys (response); const delimiter = '-'; for (let i = 0; i < marketIds.length; i++) { const marketId = marketIds[i]; const entry = this.safeValue (response, marketId, []); const first = this.safeValue (entry, 0, []); const market = this.safeMarket (marketId, undefined, delimiter); const symbol = market['symbol']; result[symbol] = this.parseTicker (first, market); } return this.filterByArray (result, 'symbol', symbols); } async fetchTicker (symbol, params = {}) { /** * @method * @name coinbasepro#fetchTicker * @description fetches a price ticker, a statistical calculation with the information calculated over the past 24 hours for a specific market * @param {string} symbol unified symbol of the market to fetch the ticker for * @param {object} params extra parameters specific to the coinbasepro api endpoint * @returns {object} a [ticker structure]{@link https://docs.ccxt.com/en/latest/manual.html#ticker-structure} */ await this.loadMarkets (); const market = this.market (symbol); const request = { 'id': market['id'], }; // publicGetProductsIdTicker or publicGetProductsIdStats const method = this.safeString (this.options, 'fetchTickerMethod', 'publicGetProductsIdTicker'); const response = await this[method] (this.extend (request, params)); // // publicGetProductsIdTicker // // { // "trade_id":843439, // "price":"0.997999", // "size":"80.29769", // "time":"2020-01-28T02:13:33.012523Z", // "bid":"0.997094", // "ask":"0.998", // "volume":"1903188.03750000" // } // // publicGetProductsIdStats // // { // "open": "34.19000000", // "high": "95.70000000", // "low": "7.06000000", // "volume": "2.41000000" // } // return this.parseTicker (response, market); } parseTrade (trade, market = undefined) { // // { // type: 'match', // trade_id: 82047307, // maker_order_id: '0f358725-2134-435e-be11-753912a326e0', // taker_order_id: '252b7002-87a3-425c-ac73-f5b9e23f3caf', // order_id: 'd50ec984-77a8-460a-b958-66f114b0de9b', // side: 'sell', // size: '0.00513192', // price: '9314.78', // product_id: 'BTC-USD', // profile_id: '6244401d-c078-40d9-b305-7ad3551bc3b0', // sequence: 12038915443, // time: '2020-01-31T20:03:41.158814Z' // created_at: '2014-11-07T22:19:28.578544Z', // liquidity: 'T', // fee: '0.00025', // settled: true, // usd_volume: '0.0924556000000000', // user_id: '595eb864313c2b02ddf2937d' // } // const timestamp = this.parse8601 (this.safeString2 (trade, 'time', 'created_at')); const marketId = this.safeString (trade, 'product_id'); market = this.safeMarket (marketId, market, '-'); let feeRate = undefined; let takerOrMaker = undefined; let cost = undefined; const feeCurrencyId = this.safeStringLower (market, 'quoteId'); if (feeCurrencyId !== undefined) { const costField = feeCurrencyId + '_value'; cost = this.safeString (trade, costField); const liquidity = this.safeString (trade, 'liquidity'); if (liquidity !== undefined) { takerOrMaker = (liquidity === 'T') ? 'taker' : 'maker'; feeRate = this.safeString (market, takerOrMaker); } } const feeCost = this.safeString2 (trade, 'fill_fees', 'fee'); const fee = { 'cost': feeCost, 'currency': market['quote'], 'rate': feeRate, }; const id = this.safeString (trade, 'trade_id'); let side = (trade['side'] === 'buy') ? 'sell' : 'buy'; const orderId = this.safeString (trade, 'order_id'); // Coinbase Pro returns inverted side to fetchMyTrades vs fetchTrades const makerOrderId = this.safeString (trade, 'maker_order_id'); const takerOrderId = this.safeString (trade, 'taker_order_id'); if ((orderId !== undefined) || ((makerOrderId !== undefined) && (takerOrderId !== undefined))) { side = (trade['side'] === 'buy') ? 'buy' : 'sell'; } const price = this.safeString (trade, 'price'); const amount = this.safeString (trade, 'size'); return this.safeTrade ({ 'id': id, 'order': orderId, 'info': trade, 'timestamp': timestamp, 'datetime': this.iso8601 (timestamp), 'symbol': market['symbol'], 'type': undefined, 'takerOrMaker': takerOrMaker, 'side': side, 'price': price, 'amount': amount, 'fee': fee, 'cost': cost, }, market); } async fetchMyTrades (symbol = undefined, since = undefined, limit = undefined, params = {}) { /** * @method * @name coinbasepro#fetchMyTrades * @description fetch all trades made by the user * @param {string} symbol unified market symbol * @param {int|undefined} since the earliest time in ms to fetch trades for * @param {int|undefined} limit the maximum number of trades structures to retrieve * @param {object} params extra parameters specific to the coinbasepro api endpoint * @returns {[object]} a list of [trade structures]{@link https://docs.ccxt.com/en/latest/manual.html#trade-structure} */ // as of 2018-08-23 if (symbol === undefined) { throw new ArgumentsRequired (this.id + ' fetchMyTrades() requires a symbol argument'); } await this.loadMarkets (); const market = this.market (symbol); const request = { 'product_id': market['id'], }; if (limit !== undefined) { request['limit'] = limit; } const response = await this.privateGetFills (this.extend (request, params)); return this.parseTrades (response, market, since, limit); } async fetchTrades (symbol, since = undefined, limit = undefined, params = {}) { /** * @method * @name coinbasepro#fetchTrades * @description get the list of most recent trades for a particular symbol * @param {string} symbol unified symbol of the market to fetch trades for * @param {int|undefined} since timestamp in ms of the earliest trade to fetch * @param {int|undefined} limit the maximum amount of trades to fetch * @param {object} params extra parameters specific to the coinbasepro api endpoint * @returns {[object]} a list of [trade structures]{@link https://docs.ccxt.com/en/latest/manual.html?#public-trades} */ await this.loadMarkets (); const market = this.market (symbol); const request = { 'id': market['id'], // fixes issue #2 }; if (limit !== undefined) { request['limit'] = limit; // default 100 } const response = await this.publicGetProductsIdTrades (this.extend (request, params)); return this.parseTrades (response, market, since, limit); } async fetchTradingFees (params = {}) { /** * @method * @name coinbasepro#fetchTradingFees * @description fetch the trading fees for multiple markets * @param {object} params extra parameters specific to the coinbasepro api endpoint * @returns {object} a dictionary of [fee structures]{@link https://docs.ccxt.com/en/latest/manual.html#fee-structure} indexed by market symbols */ await this.loadMarkets (); const response = await this.privateGetFees (params); // // { // "maker_fee_rate": "0.0050", // "taker_fee_rate": "0.0050", // "usd_volume": "43806.92" // } // const maker = this.safeNumber (response, 'maker_fee_rate'); const taker = this.safeNumber (response, 'taker_fee_rate'); const result = {}; for (let i = 0; i < this.symbols.length; i++) { const symbol = this.symbols[i]; result[symbol] = { 'info': response, 'symbol': symbol, 'maker': maker, 'taker': taker, 'percentage': true, 'tierBased': true, }; } return result; } parseOHLCV (ohlcv, market = undefined) { // // [ // 1591514160, // 0.02507, // 0.02507, // 0.02507, // 0.02507, // 0.02816506 // ] // return [ this.safeTimestamp (ohlcv, 0), this.safeNumber (ohlcv, 3), this.safeNumber (ohlcv, 2), this.safeNumber (ohlcv, 1), this.safeNumber (ohlcv, 4), this.safeNumber (ohlcv, 5), ]; } async fetchOHLCV (symbol, timeframe = '1m', since = undefined, limit = undefined, params = {}) { /** * @method * @name coinbasepro#fetchOHLCV * @description fetches historical candlestick data containing the open, high, low, and close price, and the volume of a market * @param {string} symbol unified symbol of the market to fetch OHLCV data for * @param {string} timeframe the length of time each candle represents * @param {int|undefined} since timestamp in ms of the earliest candle to fetch * @param {int|undefined} limit the maximum amount of candles to fetch * @param {object} params extra parameters specific to the coinbasepro api endpoint * @returns {[[int]]} A list of candles ordered as timestamp, open, high, low, close, volume */ await this.loadMarkets (); const market = this.market (symbol); const granularity = this.timeframes[timeframe]; const request = { 'id': market['id'], 'granularity': granularity, }; if (since !== undefined) { request['start'] = this.iso8601 (since); if (limit === undefined) { // https://docs.pro.coinbase.com/#get-historic-rates limit = 300; // max = 300 } else { limit = Math.min (300, limit); } request['end'] = this.iso8601 (this.sum ((limit - 1) * granularity * 1000, since)); } const response = await this.publicGetProductsIdCandles (this.extend (request, params)); // // [ // [1591514160,0.02507,0.02507,0.02507,0.02507,0.02816506], // [1591514100,0.02507,0.02507,0.02507,0.02507,1.63830323], // [1591514040,0.02505,0.02507,0.02505,0.02507,0.19918178] // ] // return this.parseOHLCVs (response, market, timeframe, since, limit); } async fetchTime (params = {}) { /** * @method * @name coinbasepro#fetchTime * @description fetches the current integer timestamp in milliseconds from the exchange server * @param {object} params extra parameters specific to the coinbasepro api endpoint * @returns {int} the current integer timestamp in milliseconds from the exchange server */ const response = await this.publicGetTime (params); // // { // "iso":"2020-05-12T08:00:51.504Z", // "epoch":1589270451.504 // } // return this.safeTimestamp (response, 'epoch'); } parseOrderStatus (status) { const statuses = { 'pending': 'open', 'active': 'open', 'open': 'open', 'done': 'closed', 'canceled': 'canceled', 'canceling': 'open', }; return this.safeString (statuses, status, status); } parseOrder (order, market = undefined) { // // createOrder // // { // "id": "d0c5340b-6d6c-49d9-b567-48c4bfca13d2", // "price": "0.10000000", // "size": "0.01000000", // "product_id": "BTC-USD", // "side": "buy", // "stp": "dc", // "type": "limit", // "time_in_force": "GTC", // "post_only": false, // "created_at": "2016-12-08T20:02:28.53864Z", // "fill_fees": "0.0000000000000000", // "filled_size": "0.00000000", // "executed_value": "0.0000000000000000", // "status": "pending", // "settled": false // } // const timestamp = this.parse8601 (this.safeString (order, 'created_at')); const marketId = this.safeString (order, 'product_id'); market = this.safeMarket (marketId, market, '-'); let status = this.parseOrderStatus (this.safeString (order, 'status')); const doneReason = this.safeString (order, 'done_reason'); if ((status === 'closed') && (doneReason === 'canceled')) { status = 'canceled'; } const price = this.safeString (order, 'price'); const filled = this.safeString (order, 'filled_size'); const amount = this.safeString (order, 'size', filled); const cost = this.safeString (order, 'executed_value'); const feeCost = this.safeNumber (order, 'fill_fees'); let fee = undefined; if (feeCost !== undefined) { fee = { 'cost': feeCost, 'currency': market['quote'], 'rate': undefined, }; } const id = this.safeString (order, 'id'); const type = this.safeString (order, 'type'); const side = this.safeString (order, 'side'); const timeInForce = this.safeString (order, 'time_in_force'); const postOnly = this.safeValue (order, 'post_only'); const stopPrice = this.safeNumber (order, 'stop_price'); const clientOrderId = this.safeString (order, 'client_oid'); return this.safeOrder ({ 'id': id, 'clientOrderId': clientOrderId, 'info': order, 'timestamp': timestamp, 'datetime': this.iso8601 (timestamp), 'lastTradeTimestamp': undefined, 'status': status, 'symbol': market['symbol'], 'type': type, 'timeInForce': timeInForce, 'postOnly': postOnly, 'side': side, 'price': price, 'stopPrice': stopPrice, 'cost': cost, 'amount': amount, 'filled': filled, 'remaining': undefined, 'fee': fee, 'average': undefined, 'trades': undefined, }, market); } async fetchOrder (id, symbol = undefined, params = {}) { /** * @method * @name coinbasepro#fetchOrder * @description fetches information on an order made by the user * @param {string|undefined} symbol not used by coinbasepro fetchOrder * @param {object} params extra parameters specific to the coinbasepro api endpoint * @returns {object} An [order structure]{@link https://docs.ccxt.com/en/latest/manual.html#order-structure} */ await this.loadMarkets (); const request = {}; const clientOrderId = this.safeString2 (params, 'clientOrderId', 'client_oid'); let method = undefined; if (clientOrderId === undefined) { method = 'privateGetOrdersId'; request['id'] = id; } else { method = 'privateGetOrdersClientClientOid'; request['client_oid'] = clientOrderId; params = this.omit (params, [ 'clientOrderId', 'client_oid' ]); } const response = await this[method] (this.extend (request, params)); return this.parseOrder (response); } async fetchOrderTrades (id, symbol = undefined, since = undefined, limit = undefined, params = {}) { /** * @method * @name coinbasepro#fetchOrderTrades * @description fetch all the trades made from a single order * @param {string} id order id * @param {string|undefined} symbol unified market symbol * @param {int|undefined} since the earliest time in ms to fetch trades for * @param {int|undefined} limit the maximum number of trades to retrieve * @param {object} params extra parameters specific to the coinbasepro api endpoint * @returns {[object]} a list of [trade structures]{@link https://docs.ccxt.com/en/latest/manual.html#trade-structure} */ await this.loadMarkets (); let market = undefined; if (symbol !== undefined) { market = this.market (symbol); } const request = { 'order_id': id, }; const response = await this.privateGetFills (this.extend (request, params)); return this.parseTrades (response, market, since, limit); } async fetchOrders (symbol = undefined, since = undefined, limit = undefined, params = {}) { /** * @method * @name coinbasepro#fetchOrders * @description fetches information on multiple orders made by the user * @param {string|undefined} symbol unified market symbol of the market orders were made in * @param {int|undefined} since the earliest time in ms to fetch orders for * @param {int|undefined} limit the maximum number of orde structures to retrieve * @param {object} params extra parameters specific to the coinbasepro api endpoint * @returns {[object]} a list of [order structures]{@link https://docs.ccxt.com/en/latest/manual.html#order-structure} */ const request = { 'status': 'all', }; return await this.fetchOpenOrders (symbol, since, limit, this.extend (request, params)); } async fetchOpenOrders (symbol = undefined, since = undefined, limit = undefined, params = {}) { /** * @method * @name coinbasepro#fetchOpenOrders * @description fetch all unfilled currently open orders * @param {string|undefined} symbol unified market symbol * @param {int|undefined} since the earliest time in ms to fetch open orders for * @param {int|undefined} limit the maximum number of open orders structures to retrieve * @param {object} params extra parameters specific to the coinbasepro api endpoint * @returns {[object]} a list of [order structures]{@link https://docs.ccxt.com/en/latest/manual.html#order-structure} */ await this.loadMarkets (); const request = {}; let market = undefined; if (symbol !== undefined) { market = this.market (symbol); request['product_id'] = market['id']; } if (limit !== undefined) { request['limit'] = limit; // default 100 } const response = await this.privateGetOrders (this.extend (request, params)); return this.parseOrders (response, market, since, limit); } async fetchClosedOrders (symbol = undefined, since = undefined, limit = undefined, params = {}) { /** * @method * @name coinbasepro#fetchClosedOrders * @description fetches information on multiple closed orders made by the user * @param {string|undefined} symbol unified market symbol of the market orders were made in * @param {int|undefined} since the earliest time in ms to fetch orders for * @param {int|undefined} limit the maximum number of orde structures to retrieve * @param {object} params extra parameters specific to the coinbasepro api endpoint * @returns {[object]} a list of [order structures]{@link https://docs.ccxt.com/en/latest/manual.html#order-structure} */ const request = { 'status': 'done', }; return await this.fetchOpenOrders (symbol, since, limit, this.extend (request, params)); } async createOrder (symbol, type, side, amount, price = undefined, params = {}) { /** * @method * @name coinbasepro#createOrder * @description create a trade order * @param {string} symbol unified symbol of the market to create an order in * @param {string} type 'market' or 'limit' * @param {string} side 'buy' or 'sell' * @param {float} amount how much of currency you want to trade in units of base currency * @param {float|undefined} price the price at which the order is to be fullfilled, in units of the quote currency, ignored in market orders * @param {object} params extra parameters specific to the coinbasepro api endpoint * @returns {object} an [order structure]{@link https://docs.ccxt.com/en/latest/manual.html#order-structure} */ await this.loadMarkets (); const market = this.market (symbol); const request = { // common params -------------------------------------------------- // 'client_oid': clientOrderId, 'type': type, 'side': side, 'product_id': market['id'], // 'size': this.amountToPrecision (symbol, amount), // 'stp': 'dc', // self-trade prevention, dc = decrease and cancel, co = cancel oldest, cn = cancel newest, cb = cancel both // 'stop': 'loss', // "loss" = stop loss below price, "entry" = take profit above price // 'stop_price': this.priceToPrecision (symbol, price), // limit order params --------------------------------------------- // 'price': this.priceToPrecision (symbol, price), // 'size': this.amountToPrecision (symbol, amount), // 'time_in_force': 'GTC', // GTC, GTT, IOC, or FOK // 'cancel_after' [optional]* min, hour, day, requires time_in_force to be GTT // 'post_only': false, // invalid when time_in_force is IOC or FOK // market order params -------------------------------------------- // 'size': this.amountToPrecision (symbol, amount), // 'funds': this.costToPrecision (symbol, amount), }; const clientOrderId = this.safeString2 (params, 'clientOrderId', 'client_oid'); if (clientOrderId !== undefined) { request['client_oid'] = clientOrderId; } const stopPrice = this.safeNumber2 (params, 'stopPrice', 'stop_price'); if (stopPrice !== undefined) { request['stop_price'] = this.priceToPrecision (symbol, stopPrice); } const timeInForce = this.safeString2 (params, 'timeInForce', 'time_in_force'); if (timeInForce !== undefined) { request['time_in_force'] = timeInForce; } const postOnly = this.safeValue2