ccxt-bybit
Version:
A JavaScript / Python / PHP cryptocurrency trading library with support for 130+ exchanges
1,158 lines (1,122 loc) • 48.2 kB
JavaScript
'use strict';
// ---------------------------------------------------------------------------
const bitfinex = require ('./bitfinex.js');
const { ExchangeError, InvalidAddress, ArgumentsRequired, InsufficientFunds, AuthenticationError, OrderNotFound, InvalidOrder, BadRequest, InvalidNonce, BadSymbol, OnMaintenance } = require ('./base/errors');
// ---------------------------------------------------------------------------
module.exports = class bitfinex2 extends bitfinex {
describe () {
return this.deepExtend (super.describe (), {
'id': 'bitfinex2',
'name': 'Bitfinex',
'countries': [ 'VG' ],
'version': 'v2',
'certified': false,
'pro': false,
// new metainfo interface
'has': {
'CORS': false,
'cancelAllOrders': true,
'createDepositAddress': true,
'createLimitOrder': true,
'createMarketOrder': true,
'createOrder': true,
'cancelOrder': true,
'deposit': false,
'editOrder': false,
'fetchDepositAddress': true,
'fetchClosedOrders': false,
'fetchFundingFees': false,
'fetchMyTrades': true,
'fetchOHLCV': true,
'fetchOpenOrders': true,
'fetchOrder': false,
'fetchOpenOrder': true,
'fetchClosedOrder': true,
'fetchOrderTrades': true,
'fetchStatus': true,
'fetchTickers': true,
'fetchTradingFee': false,
'fetchTradingFees': false,
'fetchTransactions': false,
'withdraw': true,
},
'timeframes': {
'1m': '1m',
'5m': '5m',
'15m': '15m',
'30m': '30m',
'1h': '1h',
'3h': '3h',
'6h': '6h',
'12h': '12h',
'1d': '1D',
'1w': '7D',
'2w': '14D',
'1M': '1M',
},
'rateLimit': 1500,
'urls': {
'logo': 'https://user-images.githubusercontent.com/1294454/27766244-e328a50c-5ed2-11e7-947b-041416579bb3.jpg',
'api': {
'v1': 'https://api.bitfinex.com',
'public': 'https://api-pub.bitfinex.com',
'private': 'https://api.bitfinex.com',
},
'www': 'https://www.bitfinex.com',
'doc': [
'https://docs.bitfinex.com/v2/docs/',
'https://github.com/bitfinexcom/bitfinex-api-node',
],
'fees': 'https://www.bitfinex.com/fees',
},
'api': {
'v1': {
'get': [
'symbols',
'symbols_details',
],
},
'public': {
'get': [
'conf/{config}',
'conf/pub:{action}:{object}',
'conf/pub:{action}:{object}:{detail}',
'conf/pub:map:{object}',
'conf/pub:map:{object}:{detail}',
'conf/pub:map:currency:{detail}',
'conf/pub:map:currency:sym', // maps symbols to their API symbols, BAB > BCH
'conf/pub:map:currency:label', // verbose friendly names, BNT > Bancor
'conf/pub:map:currency:unit', // maps symbols to unit of measure where applicable
'conf/pub:map:currency:undl', // maps derivatives symbols to their underlying currency
'conf/pub:map:currency:pool', // maps symbols to underlying network/protocol they operate on
'conf/pub:map:currency:explorer', // maps symbols to their recognised block explorer URLs
'conf/pub:map:tx:method',
'conf/pub:list:{object}',
'conf/pub:list:{object}:{detail}',
'conf/pub:list:currency',
'conf/pub:list:pair:exchange',
'conf/pub:list:pair:margin',
'conf/pub:list:competitions',
'conf/pub:info:{object}',
'conf/pub:info:{object}:{detail}',
'conf/pub:info:pair',
'conf/pub:info:tx:status', // [ deposit, withdrawal ] statuses 1 = active, 0 = maintenance
'conf/pub:fees',
'platform/status',
'tickers',
'ticker/{symbol}',
'trades/{symbol}/hist',
'book/{symbol}/{precision}',
'book/{symbol}/P0',
'book/{symbol}/P1',
'book/{symbol}/P2',
'book/{symbol}/P3',
'book/{symbol}/R0',
'stats1/{key}:{size}:{symbol}:{side}/{section}',
'stats1/{key}:{size}:{symbol}:{side}/last',
'stats1/{key}:{size}:{symbol}:{side}/hist',
'stats1/{key}:{size}:{symbol}/{section}',
'stats1/{key}:{size}:{symbol}/last',
'stats1/{key}:{size}:{symbol}/hist',
'stats1/{key}:{size}:{symbol}:long/last',
'stats1/{key}:{size}:{symbol}:long/hist',
'stats1/{key}:{size}:{symbol}:short/last',
'stats1/{key}:{size}:{symbol}:short/hist',
'candles/trade:{timeframe}:{symbol}/{section}',
'candles/trade:{timeframe}:{symbol}/last',
'candles/trade:{timeframe}:{symbol}/hist',
'status/{type}',
'status/deriv',
'liquidations/hist',
'rankings/{key}:{timeframe}:{symbol}/{section}',
'rankings/{key}:{timeframe}:{symbol}/hist',
],
'post': [
'calc/trade/avg',
'calc/fx',
],
},
'private': {
'post': [
// 'auth/r/orders/{symbol}/new', // outdated
// 'auth/r/stats/perf:{timeframe}/hist', // outdated
'auth/r/wallets',
'auth/r/wallets/hist',
'auth/r/orders',
'auth/r/orders/{symbol}',
'auth/w/order/submit',
'auth/w/order/update',
'auth/w/order/cancel',
'auth/w/order/multi',
'auth/w/order/cancel/multi',
'auth/r/orders/{symbol}/hist',
'auth/r/orders/hist',
'auth/r/order/{symbol}:{id}/trades',
'auth/r/trades/{symbol}/hist',
'auth/r/trades/hist',
'auth/r/ledgers/{currency}/hist',
'auth/r/ledgers/hist',
'auth/r/info/margin/{key}',
'auth/r/info/margin/base',
'auth/r/info/margin/sym_all',
'auth/r/positions',
'auth/w/position/claim',
'auth/r/positions/hist',
'auth/r/positions/audit',
'auth/w/deriv/collateral/set',
'auth/r/funding/offers',
'auth/r/funding/offers/{symbol}',
'auth/w/funding/offer/submit',
'auth/w/funding/offer/cancel',
'auth/w/funding/offer/cancel/all',
'auth/w/funding/close',
'auth/w/funding/auto',
'auth/w/funding/keep',
'auth/r/funding/offers/{symbol}/hist',
'auth/r/funding/offers/hist',
'auth/r/funding/loans',
'auth/r/funding/loans/hist',
'auth/r/funding/loans/{symbol}',
'auth/r/funding/loans/{symbol}/hist',
'auth/r/funding/credits',
'auth/r/funding/credits/hist',
'auth/r/funding/credits/{symbol}',
'auth/r/funding/credits/{symbol}/hist',
'auth/r/funding/trades/{symbol}/hist',
'auth/r/funding/trades/hist',
'auth/r/info/funding/{key}',
'auth/r/info/user',
'auth/r/logins/hist',
'auth/w/transfer',
'auth/w/deposit/address',
'auth/w/deposit/invoice',
'auth/w/withdraw',
'auth/r/movements/{currency}/hist',
'auth/r/movements/hist',
'auth/r/alerts',
'auth/w/alert/set',
'auth/w/alert/price:{symbol}:{price}/del',
'auth/w/alert/{type}:{symbol}:{price}/del',
'auth/calc/order/avail',
'auth/w/settings/set',
'auth/r/settings',
'auth/w/settings/del',
],
},
},
'fees': {
'trading': {
'maker': 0.1 / 100,
'taker': 0.2 / 100,
},
'funding': {
'withdraw': {
'BTC': 0.0004,
'BCH': 0.0001,
'ETH': 0.00135,
'EOS': 0.0,
'LTC': 0.001,
'OMG': 0.15097,
'IOT': 0.0,
'NEO': 0.0,
'ETC': 0.01,
'XRP': 0.02,
'ETP': 0.01,
'ZEC': 0.001,
'BTG': 0.0,
'DASH': 0.01,
'XMR': 0.0001,
'QTM': 0.01,
'EDO': 0.23687,
'DAT': 9.8858,
'AVT': 1.1251,
'SAN': 0.35977,
'USDT': 5.0,
'SPK': 16.971,
'BAT': 1.1209,
'GNT': 2.8789,
'SNT': 9.0848,
'QASH': 1.726,
'YYW': 7.9464,
},
},
},
'options': {
'precision': 'R0', // P0, P1, P2, P3, P4, R0
// convert 'EXCHANGE MARKET' to lowercase 'market'
// convert 'EXCHANGE LIMIT' to lowercase 'limit'
// everything else remains uppercase
'exchangeTypes': {
// 'MARKET': undefined,
'EXCHANGE MARKET': 'market',
// 'LIMIT': undefined,
'EXCHANGE LIMIT': 'limit',
// 'STOP': undefined,
// 'EXCHANGE STOP': undefined,
// 'TRAILING STOP': undefined,
// 'EXCHANGE TRAILING STOP': undefined,
// 'FOK': undefined,
// 'EXCHANGE FOK': undefined,
// 'STOP LIMIT': undefined,
// 'EXCHANGE STOP LIMIT': undefined,
// 'IOC': undefined,
// 'EXCHANGE IOC': undefined,
},
// convert 'market' to 'EXCHANGE MARKET'
// convert 'limit' 'EXCHANGE LIMIT'
// everything else remains as is
'orderTypes': {
'market': 'EXCHANGE MARKET',
'limit': 'EXCHANGE LIMIT',
},
'fiat': {
'USD': 'USD',
'EUR': 'EUR',
'JPY': 'JPY',
'GBP': 'GBP',
},
},
'exceptions': {
'exact': {
'10020': BadRequest,
'10100': AuthenticationError,
'10114': InvalidNonce,
'20060': OnMaintenance,
},
'broad': {
'address': InvalidAddress,
'available balance is only': InsufficientFunds,
'not enough exchange balance': InsufficientFunds,
'Order not found': OrderNotFound,
'symbol: invalid': BadSymbol,
'Invalid order': InvalidOrder,
},
},
});
}
isFiat (code) {
return (code in this.options['fiat']);
}
getCurrencyId (code) {
return 'f' + code;
}
async fetchStatus (params = {}) {
//
// [1] // operative
// [0] // maintenance
//
const response = await this.publicGetPlatformStatus (params);
const status = this.safeValue (response, 0);
const formattedStatus = (status === 1) ? 'ok' : 'maintenance';
this.status = this.extend (this.status, {
'status': formattedStatus,
'updated': this.milliseconds (),
});
return this.status;
}
async fetchMarkets (params = {}) {
const response = await this.v1GetSymbolsDetails (params);
const result = [];
for (let i = 0; i < response.length; i++) {
const market = response[i];
let id = this.safeStringUpper (market, 'pair');
let baseId = undefined;
let quoteId = undefined;
if (id.indexOf (':') >= 0) {
const parts = id.split (':');
baseId = parts[0];
quoteId = parts[1];
} else {
baseId = id.slice (0, 3);
quoteId = id.slice (3, 6);
}
const base = this.safeCurrencyCode (baseId);
const quote = this.safeCurrencyCode (quoteId);
const symbol = base + '/' + quote;
id = 't' + id;
baseId = this.getCurrencyId (baseId);
quoteId = this.getCurrencyId (quoteId);
const precision = {
'price': this.safeInteger (market, 'price_precision'),
'amount': this.safeInteger (market, 'price_precision'),
};
const limits = {
'amount': {
'min': this.safeFloat (market, 'minimum_order_size'),
'max': this.safeFloat (market, 'maximum_order_size'),
},
'price': {
'min': Math.pow (10, -precision['price']),
'max': Math.pow (10, precision['price']),
},
};
limits['cost'] = {
'min': limits['amount']['min'] * limits['price']['min'],
'max': undefined,
};
result.push ({
'id': id,
'symbol': symbol,
'base': base,
'quote': quote,
'baseId': baseId,
'quoteId': quoteId,
'active': true,
'precision': precision,
'limits': limits,
'info': market,
'swap': false,
'spot': false,
'futures': false,
});
}
return result;
}
async fetchBalance (params = {}) {
// this api call does not return the 'used' amount - use the v1 version instead (which also returns zero balances)
await this.loadMarkets ();
const response = await this.privatePostAuthRWallets (params);
const balanceType = this.safeString (params, 'type', 'exchange');
const result = { 'info': response };
for (let b = 0; b < response.length; b++) {
const balance = response[b];
const accountType = balance[0];
let currency = balance[1];
const total = balance[2];
const available = balance[4];
if (accountType === balanceType) {
if (currency[0] === 't') {
currency = currency.slice (1);
}
const code = this.safeCurrencyCode (currency);
const account = this.account ();
// do not fill in zeroes and missing values in the parser
// rewrite and unify the following to use the unified parseBalance
account['total'] = total;
if (!available) {
if (available === 0) {
account['free'] = 0;
account['used'] = total;
} else {
account['free'] = total;
}
} else {
account['free'] = available;
account['used'] = account['total'] - account['free'];
}
result[code] = account;
}
}
return this.parseBalance (result);
}
async fetchOrderBook (symbol, limit = undefined, params = {}) {
await this.loadMarkets ();
const precision = this.safeValue (this.options, 'precision', 'R0');
const request = {
'symbol': this.marketId (symbol),
'precision': precision,
};
if (limit !== undefined) {
request['len'] = limit; // 25 or 100
}
const fullRequest = this.extend (request, params);
const orderbook = await this.publicGetBookSymbolPrecision (fullRequest);
const timestamp = this.milliseconds ();
const result = {
'bids': [],
'asks': [],
'timestamp': timestamp,
'datetime': this.iso8601 (timestamp),
'nonce': undefined,
};
const priceIndex = (fullRequest['precision'] === 'R0') ? 1 : 0;
for (let i = 0; i < orderbook.length; i++) {
const order = orderbook[i];
const price = order[priceIndex];
const amount = Math.abs (order[2]);
const side = (order[2] > 0) ? 'bids' : 'asks';
result[side].push ([ price, amount ]);
}
result['bids'] = this.sortBy (result['bids'], 0, true);
result['asks'] = this.sortBy (result['asks'], 0);
return result;
}
parseTicker (ticker, market = undefined) {
const timestamp = this.milliseconds ();
let symbol = undefined;
if (market !== undefined) {
symbol = market['symbol'];
}
const length = ticker.length;
const last = ticker[length - 4];
return {
'symbol': symbol,
'timestamp': timestamp,
'datetime': this.iso8601 (timestamp),
'high': ticker[length - 2],
'low': ticker[length - 1],
'bid': ticker[length - 10],
'bidVolume': undefined,
'ask': ticker[length - 8],
'askVolume': undefined,
'vwap': undefined,
'open': undefined,
'close': last,
'last': last,
'previousClose': undefined,
'change': ticker[length - 6],
'percentage': ticker[length - 5] * 100,
'average': undefined,
'baseVolume': ticker[length - 3],
'quoteVolume': undefined,
'info': ticker,
};
}
async fetchTickers (symbols = undefined, params = {}) {
await this.loadMarkets ();
const request = {};
if (symbols !== undefined) {
const ids = this.marketIds (symbols);
request['symbols'] = ids.join (',');
} else {
request['symbols'] = 'ALL';
}
const tickers = await this.publicGetTickers (this.extend (request, params));
const result = {};
for (let i = 0; i < tickers.length; i++) {
const ticker = tickers[i];
const id = ticker[0];
if (id in this.markets_by_id) {
const market = this.markets_by_id[id];
const symbol = market['symbol'];
result[symbol] = this.parseTicker (ticker, market);
}
}
return result;
}
async fetchTicker (symbol, params = {}) {
await this.loadMarkets ();
const market = this.market (symbol);
const request = {
'symbol': market['id'],
};
const ticker = await this.publicGetTickerSymbol (this.extend (request, params));
return this.parseTicker (ticker, market);
}
parseTrade (trade, market = undefined) {
//
// fetchTrades (public)
//
// [
// ID,
// MTS, // timestamp
// AMOUNT,
// PRICE
// ]
//
// fetchMyTrades (private)
//
// [
// ID,
// PAIR,
// MTS_CREATE,
// ORDER_ID,
// EXEC_AMOUNT,
// EXEC_PRICE,
// ORDER_TYPE,
// ORDER_PRICE,
// MAKER,
// FEE,
// FEE_CURRENCY,
// ...
// ]
//
const tradeLength = trade.length;
const isPrivate = (tradeLength > 5);
const id = trade[0].toString ();
const amountIndex = isPrivate ? 4 : 2;
let amount = trade[amountIndex];
let cost = undefined;
const priceIndex = isPrivate ? 5 : 3;
const price = trade[priceIndex];
let side = undefined;
let orderId = undefined;
let takerOrMaker = undefined;
let type = undefined;
let fee = undefined;
let symbol = undefined;
const timestampIndex = isPrivate ? 2 : 1;
const timestamp = trade[timestampIndex];
if (isPrivate) {
const marketId = trade[1];
if (marketId !== undefined) {
if (marketId in this.markets_by_id) {
market = this.markets_by_id[marketId];
symbol = market['symbol'];
} else {
symbol = marketId;
}
}
orderId = trade[3].toString ();
takerOrMaker = (trade[8] === 1) ? 'maker' : 'taker';
const feeCost = trade[9];
const feeCurrency = this.safeCurrencyCode (trade[10]);
if (feeCost !== undefined) {
fee = {
'cost': parseFloat (this.feeToPrecision (symbol, Math.abs (feeCost))),
'currency': feeCurrency,
};
}
const orderType = trade[6];
type = this.safeString (this.options['exchangeTypes'], orderType);
}
if (symbol === undefined) {
if (market !== undefined) {
symbol = market['symbol'];
}
}
if (amount !== undefined) {
side = (amount < 0) ? 'sell' : 'buy';
amount = Math.abs (amount);
if (cost === undefined) {
if (price !== undefined) {
cost = amount * price;
}
}
}
return {
'id': id,
'timestamp': timestamp,
'datetime': this.iso8601 (timestamp),
'symbol': symbol,
'order': orderId,
'side': side,
'type': type,
'takerOrMaker': takerOrMaker,
'price': price,
'amount': amount,
'cost': cost,
'fee': fee,
'info': trade,
};
}
async fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {
await this.loadMarkets ();
const market = this.market (symbol);
let sort = '-1';
const request = {
'symbol': market['id'],
};
if (since !== undefined) {
request['start'] = since;
sort = '1';
}
if (limit !== undefined) {
request['limit'] = limit; // default 120, max 5000
}
request['sort'] = sort;
const response = await this.publicGetTradesSymbolHist (this.extend (request, params));
//
// [
// [
// ID,
// MTS, // timestamp
// AMOUNT,
// PRICE
// ]
// ]
//
const trades = this.sortBy (response, 1);
return this.parseTrades (trades, market, undefined, limit);
}
async fetchOHLCV (symbol, timeframe = '1m', since = undefined, limit = 100, params = {}) {
await this.loadMarkets ();
const market = this.market (symbol);
if (limit === undefined) {
limit = 100; // default 100, max 5000
}
if (since === undefined) {
since = this.milliseconds () - this.parseTimeframe (timeframe) * limit * 1000;
}
const request = {
'symbol': market['id'],
'timeframe': this.timeframes[timeframe],
'sort': 1,
'start': since,
'limit': limit,
};
const response = await this.publicGetCandlesTradeTimeframeSymbolHist (this.extend (request, params));
return this.parseOHLCVs (response, market, timeframe, since, limit);
}
parseOrderStatus (status) {
const statuses = {
'ACTIVE': 'open',
'PARTIALLY FILLED': 'open',
'EXECUTED': 'closed',
'CANCELED': 'canceled',
'INSUFFICIENT MARGIN': 'canceled',
'RSN_DUST': 'rejected',
'RSN_PAUSE': 'rejected',
};
return this.safeString (statuses, status, status);
}
parseOrder (order, market = undefined) {
const id = this.safeString (order, 0);
let symbol = undefined;
const marketId = this.safeString (order, 3);
if (marketId in this.markets_by_id) {
market = this.markets_by_id[marketId];
}
if (market !== undefined) {
symbol = market['symbol'];
}
const timestamp = this.safeTimestamp (order, 5);
const remaining = Math.abs (this.safeFloat (order, 6));
const amount = Math.abs (this.safeFloat (order, 7));
const filled = amount - remaining;
const side = (order[7] < 0) ? 'sell' : 'buy';
const orderType = this.safeString (order, 8);
const type = this.safeString (this.safeValue (this.options, 'exchangeTypes'), orderType);
let status = undefined;
const statusString = this.safeString (order, 13);
if (statusString !== undefined) {
const parts = statusString.split (' @ ');
status = this.parseOrderStatus (this.safeString (parts, 0));
}
const price = this.safeFloat (order, 16);
const average = this.safeFloat (order, 17);
const cost = price * filled;
return {
'info': order,
'id': id,
'timestamp': timestamp,
'datetime': this.iso8601 (timestamp),
'lastTradeTimestamp': undefined,
'symbol': symbol,
'type': type,
'side': side,
'price': price,
'amount': amount,
'cost': cost,
'average': average,
'filled': filled,
'remaining': remaining,
'status': status,
'fee': undefined,
'trades': undefined,
};
}
async createOrder (symbol, type, side, amount, price = undefined, params = {}) {
await this.loadMarkets ();
const market = this.market (symbol);
const orderTypes = this.safeValue (this.options, 'orderTypes', {});
const orderType = this.safeString (orderTypes, type, type);
amount = (side === 'sell') ? -amount : amount;
const request = {
'symbol': market['id'],
'type': orderType,
'amount': this.numberToString (amount),
};
if (type !== 'market') {
request['price'] = this.numberToString (price);
}
const response = await this.privatePostAuthWOrderSubmit (this.extend (request, params));
//
// [
// 1578784364.748, // Millisecond Time Stamp of the update
// "on-req", // Purpose of notification ('on-req', 'oc-req', 'uca', 'fon-req', 'foc-req')
// null, // Unique ID of the message
// null, // Ignore
// [
// [
// 37271830598, // Order ID
// null, // Group ID
// 1578784364748, // Client Order ID
// "tBTCUST", // Pair
// 1578784364748, // Millisecond timestamp of creation
// 1578784364748, // Millisecond timestamp of update
// -0.005, // Positive means buy, negative means sell
// -0.005, // Original amount
// "EXCHANGE LIMIT", // Order type (LIMIT, MARKET, STOP, TRAILING STOP, EXCHANGE MARKET, EXCHANGE LIMIT, EXCHANGE STOP, EXCHANGE TRAILING STOP, FOK, EXCHANGE FOK, IOC, EXCHANGE IOC)
// null, // Previous order type
// null, // Millisecond timestamp of Time-In-Force: automatic order cancellation
// null, // Ignore
// 0, // Flags (see https://docs.bitfinex.com/docs/flag-values)
// "ACTIVE", // Order Status
// null, // Ignore
// null, // Ignore
// 20000, // Price
// 0, // Average price
// 0, // The trailing price
// 0, // Auxiliary Limit price (for STOP LIMIT)
// null, // Ignore
// null, // Ignore
// null, // Ignore
// 0, // 1 - hidden order
// null, // If another order caused this order to be placed (OCO) this will be that other order's ID
// null, // Ignore
// null, // Ignore
// null, // Ignore
// "API>BFX", // Origin of action: BFX, ETHFX, API>BFX, API>ETHFX
// null, // Ignore
// null, // Ignore
// null // Meta
// ]
// ],
// null, // Error code
// "SUCCESS", // Status (SUCCESS, ERROR, FAILURE, ...)
// "Submitting 1 orders." // Text of the notification
// ]
//
const status = this.safeString (response, 6);
if (status !== 'SUCCESS') {
const errorCode = response[5];
const errorText = response[7];
throw new ExchangeError (this.id + ' ' + response[6] + ': ' + errorText + ' (#' + errorCode + ')');
}
const orders = this.safeValue (response, 4, []);
const order = this.safeValue (orders, 0);
return this.parseOrder (order, market);
}
async cancelAllOrders (symbol = undefined, params = {}) {
const request = {
'all': 1,
};
const response = await this.privatePostAuthWOrderCancelMulti (this.extend (request, params));
const orders = this.safeValue (response, 4, []);
return this.parseOrders (orders);
}
async cancelOrder (id, symbol = undefined, params = {}) {
const cid = this.safeValue (params, 'cid'); // client order id
let request = undefined;
if (cid !== undefined) {
const cidDate = this.safeValue (params, 'cidDate'); // client order id date
if (cidDate === undefined) {
throw new InvalidOrder (this.id + " canceling an order by client order id ('cid') requires both 'cid' and 'cid_date' ('YYYY-MM-DD')");
}
request = {
'cid': cid,
'cid_date': cidDate,
};
} else {
request = {
'id': parseInt (id),
};
}
const response = await this.privatePostAuthWOrderCancel (this.extend (request, params));
const order = this.safeValue (response, 4);
return this.parseOrder (order);
}
async fetchOpenOrder (id, symbol = undefined, params = {}) {
const request = {
'id': [ parseInt (id) ],
};
const orders = await this.fetchOpenOrders (symbol, undefined, undefined, this.extend (request, params));
const order = this.safeValue (orders, 0);
if (order === undefined) {
throw new OrderNotFound (this.id + ' order ' + id + ' not found');
}
return order;
}
async fetchClosedOrder (id, symbol = undefined, params = {}) {
const request = {
'id': [ parseInt (id) ],
};
const orders = await this.fetchClosedOrders (symbol, undefined, undefined, this.extend (request, params));
const order = this.safeValue (orders, 0);
if (order === undefined) {
throw new OrderNotFound (this.id + ' order ' + id + ' not found');
}
return order;
}
async fetchOpenOrders (symbol = undefined, since = undefined, limit = undefined, params = {}) {
await this.loadMarkets ();
const request = {};
let market = undefined;
let response = undefined;
if (symbol === undefined) {
response = await this.privatePostAuthROrders (this.extend (request, params));
} else {
market = this.market (symbol);
request['symbol'] = market['id'];
response = await this.privatePostAuthROrdersSymbol (this.extend (request, params));
}
return this.parseOrders (response, market, since, limit);
}
async fetchClosedOrders (symbol = undefined, since = undefined, limit = undefined, params = {}) {
// returns the most recent closed or canceled orders up to circa two weeks ago
await this.loadMarkets ();
const request = {};
let market = undefined;
let response = undefined;
if (symbol === undefined) {
response = await this.privatePostAuthROrdersHist (this.extend (request, params));
} else {
market = this.market (symbol);
request['symbol'] = market['id'];
response = await this.privatePostAuthROrdersSymbolHist (this.extend (request, params));
}
if (since !== undefined) {
request['start'] = since;
}
if (limit !== undefined) {
request['limit'] = limit; // default 25, max 2500
}
return this.parseOrders (response, market, since, limit);
}
async fetchOrderTrades (id, symbol = undefined, since = undefined, limit = undefined, params = {}) {
if (symbol === undefined) {
throw new ArgumentsRequired (this.id + ' fetchOrderTrades() requires a symbol argument');
}
await this.loadMarkets ();
const market = this.market (symbol);
const orderId = parseInt (id);
const request = {
'id': orderId,
'symbol': market['id'],
};
// valid for trades upto 10 days old
const response = await this.privatePostAuthROrderSymbolIdTrades (this.extend (request, params));
return this.parseTrades (response, market, since, limit);
}
async fetchMyTrades (symbol = undefined, since = undefined, limit = undefined, params = {}) {
await this.loadMarkets ();
let market = undefined;
const request = {
'end': this.milliseconds (),
};
if (since !== undefined) {
request['start'] = since;
}
if (limit !== undefined) {
request['limit'] = limit; // default 25, max 1000
}
let method = 'privatePostAuthRTradesHist';
if (symbol !== undefined) {
market = this.market (symbol);
request['symbol'] = market['id'];
method = 'privatePostAuthRTradesSymbolHist';
}
const response = await this[method] (this.extend (request, params));
return this.parseTrades (response, market, since, limit);
}
async createDepositAddress (code, params = {}) {
await this.loadMarkets ();
const request = {
'op_renew': 1,
};
const response = await this.fetchDepositAddress (code, this.extend (request, params));
return response;
}
async fetchDepositAddress (code, params = {}) {
await this.loadMarkets ();
// todo rewrite for https://api-pub.bitfinex.com//v2/conf/pub:map:tx:method
const name = this.getCurrencyName (code);
const request = {
'method': name,
'wallet': 'exchange', // 'exchange', 'margin', 'funding' and also old labels 'exchange', 'trading', 'deposit', respectively
'op_renew': 0, // a value of 1 will generate a new address
};
const response = await this.privatePostAuthWDepositAddress (this.extend (request, params));
//
// [
// 1582269616687, // MTS Millisecond Time Stamp of the update
// 'acc_dep', // TYPE Purpose of notification 'acc_dep' for account deposit
// null, // MESSAGE_ID unique ID of the message
// null, // not documented
// [
// null, // PLACEHOLDER
// 'BITCOIN', // METHOD Method of deposit
// 'BTC', // CURRENCY_CODE Currency code of new address
// null, // PLACEHOLDER
// '1BC9PZqpUmjyEB54uggn8TFKj49zSDYzqG', // ADDRESS
// null, // POOL_ADDRESS
// ],
// null, // CODE null or integer work in progress
// 'SUCCESS', // STATUS Status of the notification, SUCCESS, ERROR, FAILURE
// 'success', // TEXT Text of the notification
// ]
//
const result = this.safeValue (response, 4, []);
const poolAddress = this.safeString (result, 5);
const address = (poolAddress === undefined) ? this.safeString (result, 4) : poolAddress;
const tag = (poolAddress === undefined) ? undefined : this.safeString (result, 4);
this.checkAddress (address);
return {
'currency': code,
'address': address,
'tag': tag,
'info': response,
};
}
parseTransaction (transaction, currency = undefined) {
//
// withdraw
//
// [
// 1582271520931, // MTS Millisecond Time Stamp of the update
// "acc_wd-req", // TYPE Purpose of notification 'acc_wd-req' account withdrawal request
// null, // MESSAGE_ID unique ID of the message
// null, // not documented
// [
// 0, // WITHDRAWAL_ID Unique Withdrawal ID
// null, // PLACEHOLDER
// "bitcoin", // METHOD Method of withdrawal
// null, // PAYMENT_ID Payment ID if relevant
// "exchange", // WALLET Sending wallet
// 1, // AMOUNT Amount of Withdrawal less fee
// null, // PLACEHOLDER
// null, // PLACEHOLDER
// 0.0004, // WITHDRAWAL_FEE Fee on withdrawal
// ],
// null, // CODE null or integer Work in progress
// "SUCCESS", // STATUS Status of the notification, it may vary over time SUCCESS, ERROR, FAILURE
// "Invalid bitcoin address (abcdef)", // TEXT Text of the notification
// ]
//
// todo add support for all movements, deposits and withdrawals
//
const data = this.safeValue (transaction, 4, []);
const timestamp = this.safeInteger (transaction, 0);
let code = undefined;
if (currency !== undefined) {
code = currency['code'];
}
let feeCost = this.safeFloat (data, 8);
if (feeCost !== undefined) {
feeCost = Math.abs (feeCost);
}
const amount = this.safeFloat (data, 5);
let id = this.safeValue (data, 0);
let status = 'ok';
if (id === 0) {
id = undefined;
status = 'failed';
}
return {
'info': transaction,
'id': id,
'txid': undefined,
'timestamp': timestamp,
'datetime': this.iso8601 (timestamp),
'address': undefined, // this is actually the tag for XRP transfers (the address is missing)
'tag': this.safeString (data, 3), // refix it properly for the tag from description
'type': 'withdrawal',
'amount': amount,
'currency': code,
'status': status,
'updated': undefined,
'fee': {
'currency': code,
'cost': feeCost,
'rate': undefined,
},
};
}
async withdraw (code, amount, address, tag = undefined, params = {}) {
this.checkAddress (address);
await this.loadMarkets ();
const currency = this.currency (code);
// todo rewrite for https://api-pub.bitfinex.com//v2/conf/pub:map:tx:method
const name = this.getCurrencyName (code);
const request = {
'method': name,
'wallet': 'exchange', // 'exchange', 'margin', 'funding' and also old labels 'exchange', 'trading', 'deposit', respectively
'amount': this.numberToString (amount),
'address': address,
};
if (tag !== undefined) {
request['payment_id'] = tag;
}
const response = await this.privatePostAuthWWithdraw (this.extend (request, params));
//
// [
// 1582271520931, // MTS Millisecond Time Stamp of the update
// "acc_wd-req", // TYPE Purpose of notification 'acc_wd-req' account withdrawal request
// null, // MESSAGE_ID unique ID of the message
// null, // not documented
// [
// 0, // WITHDRAWAL_ID Unique Withdrawal ID
// null, // PLACEHOLDER
// "bitcoin", // METHOD Method of withdrawal
// null, // PAYMENT_ID Payment ID if relevant
// "exchange", // WALLET Sending wallet
// 1, // AMOUNT Amount of Withdrawal less fee
// null, // PLACEHOLDER
// null, // PLACEHOLDER
// 0.0004, // WITHDRAWAL_FEE Fee on withdrawal
// ],
// null, // CODE null or integer Work in progress
// "SUCCESS", // STATUS Status of the notification, it may vary over time SUCCESS, ERROR, FAILURE
// "Invalid bitcoin address (abcdef)", // TEXT Text of the notification
// ]
//
const text = this.safeString (response, 7);
if (text !== 'success') {
this.throwBroadlyMatchedException (this.exceptions['broad'], text, text);
}
const transaction = this.parseTransaction (response, currency);
return this.extend (transaction, {
'address': address,
});
}
nonce () {
return this.milliseconds ();
}
sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {
let request = '/' + this.implodeParams (path, params);
const query = this.omit (params, this.extractParams (path));
if (api === 'v1') {
request = api + request;
} else {
request = this.version + request;
}
let url = this.urls['api'][api] + '/' + request;
if (api === 'public') {
if (Object.keys (query).length) {
url += '?' + this.urlencode (query);
}
}
if (api === 'private') {
this.checkRequiredCredentials ();
const nonce = this.nonce ().toString ();
body = this.json (query);
const auth = '/api/' + request + nonce + body;
const signature = this.hmac (this.encode (auth), this.encode (this.secret), 'sha384');
headers = {
'bfx-nonce': nonce,
'bfx-apikey': this.apiKey,
'bfx-signature': signature,
'Content-Type': 'application/json',
};
}
return { 'url': url, 'method': method, 'body': body, 'headers': headers };
}
async request (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {
const response = await this.fetch2 (path, api, method, params, headers, body);
if (response) {
if ('message' in response) {
if (response['message'].indexOf ('not enough exchange balance') >= 0) {
throw new InsufficientFunds (this.id + ' ' + this.json (response));
}
throw new ExchangeError (this.id + ' ' + this.json (response));
}
return response;
} else if (response === '') {
throw new ExchangeError (this.id + ' returned empty response');
}
return response;
}
handleErrors (statusCode, statusText, url, method, responseHeaders, responseBody, response, requestHeaders, requestBody) {
if (statusCode === 500) {
// See https://docs.bitfinex.com/docs/abbreviations-glossary#section-errorinfo-codes
const errorCode = this.numberToString (response[1]);
const errorText = response[2];
const feedback = this.id + ' ' + errorText;
this.throwExactlyMatchedException (this.exceptions['exact'], errorCode, feedback);
this.throwExactlyMatchedException (this.exceptions['exact'], errorText, feedback);
this.throwBroadlyMatchedException (this.exceptions['broad'], errorText, feedback);
throw new ExchangeError (this.id + ' ' + errorText + ' (#' + errorCode + ')');
}
}
};