@proton/ccxt
Version:
A JavaScript / TypeScript / Python / C# / PHP cryptocurrency trading library with support for 130+ exchanges
1,168 lines (1,166 loc) • 65.3 kB
JavaScript
// ----------------------------------------------------------------------------
// 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
// EDIT THE CORRESPONDENT .ts FILE INSTEAD
// ---------------------------------------------------------------------------
import Exchange from './abstract/novadax.js';
import { AuthenticationError, ExchangeError, PermissionDenied, BadRequest, CancelPending, OrderNotFound, InsufficientFunds, RateLimitExceeded, InvalidOrder, AccountSuspended, BadSymbol, OnMaintenance, ArgumentsRequired, AccountNotEnabled } from './base/errors.js';
import { TICK_SIZE } from './base/functions/number.js';
import { sha256 } from './static_dependencies/noble-hashes/sha256.js';
import { md5 } from './static_dependencies/noble-hashes/md5.js';
// ---------------------------------------------------------------------------
export default class novadax extends Exchange {
describe() {
return this.deepExtend(super.describe(), {
'id': 'novadax',
'name': 'NovaDAX',
'countries': ['BR'],
// 60 requests per second = 1000ms / 60 = 16.6667ms between requests (public endpoints, limited by IP address)
// 20 requests per second => cost = 60 / 20 = 3 (private endpoints, limited by API Key)
'rateLimit': 16.6667,
'version': 'v1',
// new metainfo interface
'has': {
'CORS': undefined,
'spot': true,
'margin': false,
'swap': false,
'future': false,
'option': false,
'addMargin': false,
'cancelOrder': true,
'createOrder': true,
'createReduceOnlyOrder': false,
'createStopLimitOrder': true,
'createStopMarketOrder': true,
'createStopOrder': true,
'fetchAccounts': true,
'fetchBalance': true,
'fetchBorrowRate': false,
'fetchBorrowRateHistories': false,
'fetchBorrowRateHistory': false,
'fetchBorrowRates': false,
'fetchBorrowRatesPerSymbol': false,
'fetchClosedOrders': true,
'fetchDepositAddress': false,
'fetchDepositAddresses': false,
'fetchDepositAddressesByNetwork': false,
'fetchDeposits': true,
'fetchDepositsWithdrawals': true,
'fetchFundingHistory': false,
'fetchFundingRate': false,
'fetchFundingRateHistory': false,
'fetchFundingRates': false,
'fetchIndexOHLCV': false,
'fetchLeverage': false,
'fetchLeverageTiers': false,
'fetchMarkets': true,
'fetchMarkOHLCV': false,
'fetchMyTrades': true,
'fetchOHLCV': true,
'fetchOpenInterestHistory': false,
'fetchOpenOrders': true,
'fetchOrder': true,
'fetchOrderBook': true,
'fetchOrders': true,
'fetchOrderTrades': true,
'fetchPosition': false,
'fetchPositions': false,
'fetchPositionsRisk': false,
'fetchPremiumIndexOHLCV': false,
'fetchTicker': true,
'fetchTickers': true,
'fetchTime': true,
'fetchTrades': true,
'fetchTradingFee': false,
'fetchTradingFees': false,
'fetchTransactions': true,
'fetchWithdrawals': true,
'reduceMargin': false,
'setLeverage': false,
'setMarginMode': false,
'setPositionMode': false,
'transfer': true,
'withdraw': true,
},
'timeframes': {
'1m': 'ONE_MIN',
'5m': 'FIVE_MIN',
'15m': 'FIFTEEN_MIN',
'30m': 'HALF_HOU',
'1h': 'ONE_HOU',
'1d': 'ONE_DAY',
'1w': 'ONE_WEE',
'1M': 'ONE_MON',
},
'urls': {
'logo': 'https://user-images.githubusercontent.com/1294454/92337550-2b085500-f0b3-11ea-98e7-5794fb07dd3b.jpg',
'api': {
'public': 'https://api.novadax.com',
'private': 'https://api.novadax.com',
},
'www': 'https://www.novadax.com.br',
'doc': [
'https://doc.novadax.com/pt-BR/',
],
'fees': 'https://www.novadax.com.br/fees-and-limits',
'referral': 'https://www.novadax.com.br/?s=ccxt',
},
'api': {
'public': {
'get': {
'common/symbol': 1.2,
'common/symbols': 1.2,
'common/timestamp': 1.2,
'market/tickers': 1.2,
'market/ticker': 1.2,
'market/depth': 1.2,
'market/trades': 1.2,
'market/kline/history': 1.2,
},
},
'private': {
'get': {
'orders/get': 3,
'orders/list': 3,
'orders/fill': 3,
'orders/fills': 3,
'account/getBalance': 3,
'account/subs': 3,
'account/subs/balance': 3,
'account/subs/transfer/record': 3,
'wallet/query/deposit-withdraw': 3,
},
'post': {
'orders/create': 3,
'orders/cancel': 3,
'account/withdraw/coin': 3,
'account/subs/transfer': 3,
},
},
},
'fees': {
'trading': {
'tierBased': false,
'percentage': true,
'taker': this.parseNumber('0.005'),
'maker': this.parseNumber('0.0025'),
},
},
'requiredCredentials': {
'apiKey': true,
'secret': true,
},
'precisionMode': TICK_SIZE,
'exceptions': {
'exact': {
'A99999': ExchangeError,
// 'A10000': ExchangeError, // 200 Success Successful request
'A10001': BadRequest,
'A10002': ExchangeError,
'A10003': AuthenticationError,
'A10004': RateLimitExceeded,
'A10005': PermissionDenied,
'A10006': AccountSuspended,
'A10007': AccountNotEnabled,
'A10011': BadSymbol,
'A10012': BadSymbol,
'A10013': OnMaintenance,
'A30001': OrderNotFound,
'A30002': InvalidOrder,
'A30003': InvalidOrder,
'A30004': InvalidOrder,
'A30005': InvalidOrder,
'A30006': InvalidOrder,
'A30007': InsufficientFunds,
'A30008': InvalidOrder,
'A30009': InvalidOrder,
'A30010': CancelPending,
'A30011': InvalidOrder,
'A30012': InvalidOrder,
'A40004': InsufficientFunds, // {"code":"A40004","data":[],"message":"sub account balance Insufficient"}
},
'broad': {},
},
'options': {
'fetchOHLCV': {
'volume': 'amount', // 'amount' for base volume or 'vol' for quote volume
},
'transfer': {
'fillResponseFromRequest': true,
},
},
});
}
async fetchTime(params = {}) {
/**
* @method
* @name novadax#fetchTime
* @description fetches the current integer timestamp in milliseconds from the exchange server
* @param {object} params extra parameters specific to the novadax api endpoint
* @returns {int} the current integer timestamp in milliseconds from the exchange server
*/
const response = await this.publicGetCommonTimestamp(params);
//
// {
// "code":"A10000",
// "data":1599090512080,
// "message":"Success"
// }
//
return this.safeInteger(response, 'data');
}
async fetchMarkets(params = {}) {
/**
* @method
* @name novadax#fetchMarkets
* @description retrieves data on all markets for novadax
* @param {object} params extra parameters specific to the exchange api endpoint
* @returns {[object]} an array of objects representing market data
*/
const response = await this.publicGetCommonSymbols(params);
//
// {
// "code":"A10000",
// "data":[
// {
// "amountPrecision":8,
// "baseCurrency":"BTC",
// "minOrderAmount":"0.001",
// "minOrderValue":"25",
// "pricePrecision":2,
// "quoteCurrency":"BRL",
// "status":"ONLINE",
// "symbol":"BTC_BRL",
// "valuePrecision":2
// },
// ],
// "message":"Success"
// }
//
const result = [];
const data = this.safeValue(response, 'data', []);
for (let i = 0; i < data.length; i++) {
const market = data[i];
const baseId = this.safeString(market, 'baseCurrency');
const quoteId = this.safeString(market, 'quoteCurrency');
const id = this.safeString(market, 'symbol');
const base = this.safeCurrencyCode(baseId);
const quote = this.safeCurrencyCode(quoteId);
const status = this.safeString(market, 'status');
result.push({
'id': id,
'symbol': base + '/' + quote,
'base': base,
'quote': quote,
'settle': undefined,
'baseId': baseId,
'quoteId': quoteId,
'settleId': undefined,
'type': 'spot',
'spot': true,
'margin': false,
'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.parseNumber(this.parsePrecision(this.safeString(market, 'amountPrecision'))),
'price': this.parseNumber(this.parsePrecision(this.safeString(market, 'pricePrecision'))),
'cost': this.parseNumber(this.parsePrecision(this.safeString(market, 'valuePrecision'))),
},
'limits': {
'leverage': {
'min': undefined,
'max': undefined,
},
'amount': {
'min': this.safeNumber(market, 'minOrderAmount'),
'max': undefined,
},
'price': {
'min': undefined,
'max': undefined,
},
'cost': {
'min': this.safeNumber(market, 'minOrderValue'),
'max': undefined,
},
},
'info': market,
});
}
return result;
}
parseTicker(ticker, market = undefined) {
//
// fetchTicker, fetchTickers
//
// {
// "ask":"61946.1",
// "baseVolume24h":"164.41930186",
// "bid":"61815",
// "high24h":"64930.72",
// "lastPrice":"61928.41",
// "low24h":"61156.32",
// "open24h":"64512.46",
// "quoteVolume24h":"10308157.95",
// "symbol":"BTC_BRL",
// "timestamp":1599091115090
// }
//
const timestamp = this.safeInteger(ticker, 'timestamp');
const marketId = this.safeString(ticker, 'symbol');
const symbol = this.safeSymbol(marketId, market, '_');
const open = this.safeString(ticker, 'open24h');
const last = this.safeString(ticker, 'lastPrice');
const baseVolume = this.safeString(ticker, 'baseVolume24h');
const quoteVolume = this.safeString(ticker, 'quoteVolume24h');
return this.safeTicker({
'symbol': symbol,
'timestamp': timestamp,
'datetime': this.iso8601(timestamp),
'high': this.safeString(ticker, 'high24h'),
'low': this.safeString(ticker, 'low24h'),
'bid': this.safeString(ticker, 'bid'),
'bidVolume': undefined,
'ask': this.safeString(ticker, 'ask'),
'askVolume': undefined,
'vwap': undefined,
'open': open,
'close': last,
'last': last,
'previousClose': undefined,
'change': undefined,
'percentage': undefined,
'average': undefined,
'baseVolume': baseVolume,
'quoteVolume': quoteVolume,
'info': ticker,
}, market);
}
async fetchTicker(symbol, params = {}) {
/**
* @method
* @name novadax#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 novadax api endpoint
* @returns {object} a [ticker structure]{@link https://docs.ccxt.com/#/?id=ticker-structure}
*/
await this.loadMarkets();
const market = this.market(symbol);
const request = {
'symbol': market['id'],
};
const response = await this.publicGetMarketTicker(this.extend(request, params));
//
// {
// "code":"A10000",
// "data":{
// "ask":"61946.1",
// "baseVolume24h":"164.41930186",
// "bid":"61815",
// "high24h":"64930.72",
// "lastPrice":"61928.41",
// "low24h":"61156.32",
// "open24h":"64512.46",
// "quoteVolume24h":"10308157.95",
// "symbol":"BTC_BRL",
// "timestamp":1599091115090
// },
// "message":"Success"
// }
//
const data = this.safeValue(response, 'data', {});
return this.parseTicker(data, market);
}
async fetchTickers(symbols = undefined, params = {}) {
/**
* @method
* @name novadax#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 novadax api endpoint
* @returns {object} a dictionary of [ticker structures]{@link https://docs.ccxt.com/#/?id=ticker-structure}
*/
await this.loadMarkets();
symbols = this.marketSymbols(symbols);
const response = await this.publicGetMarketTickers(params);
//
// {
// "code":"A10000",
// "data":[
// {
// "ask":"61879.36",
// "baseVolume24h":"164.40955092",
// "bid":"61815",
// "high24h":"64930.72",
// "lastPrice":"61820.04",
// "low24h":"61156.32",
// "open24h":"64624.19",
// "quoteVolume24h":"10307493.92",
// "symbol":"BTC_BRL",
// "timestamp":1599091291083
// },
// ],
// "message":"Success"
// }
//
const data = this.safeValue(response, 'data', []);
const result = {};
for (let i = 0; i < data.length; i++) {
const ticker = this.parseTicker(data[i]);
const symbol = ticker['symbol'];
result[symbol] = ticker;
}
return this.filterByArray(result, 'symbol', symbols);
}
async fetchOrderBook(symbol, limit = undefined, params = {}) {
/**
* @method
* @name novadax#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 novadax api endpoint
* @returns {object} A dictionary of [order book structures]{@link https://docs.ccxt.com/#/?id=order-book-structure} indexed by market symbols
*/
await this.loadMarkets();
const market = this.market(symbol);
const request = {
'symbol': market['id'],
};
if (limit !== undefined) {
request['limit'] = limit; // default 10, max 20
}
const response = await this.publicGetMarketDepth(this.extend(request, params));
//
// {
// "code":"A10000",
// "data":{
// "asks":[
// ["0.037159","0.3741"],
// ["0.037215","0.2706"],
// ["0.037222","1.8459"],
// ],
// "bids":[
// ["0.037053","0.3857"],
// ["0.036969","0.8101"],
// ["0.036953","1.5226"],
// ],
// "timestamp":1599280414448
// },
// "message":"Success"
// }
//
const data = this.safeValue(response, 'data', {});
const timestamp = this.safeInteger(data, 'timestamp');
return this.parseOrderBook(data, market['symbol'], timestamp, 'bids', 'asks');
}
parseTrade(trade, market = undefined) {
//
// public fetchTrades
//
// {
// "amount":"0.0632",
// "price":"0.037288",
// "side":"BUY",
// "timestamp":1599279694576
// }
//
// private fetchOrderTrades
//
// {
// "id": "608717046691139584",
// "orderId": "608716957545402368",
// "symbol": "BTC_BRL",
// "side": "BUY",
// "amount": "0.0988",
// "price": "45514.76",
// "fee": "0.0000988 BTC",
// "feeAmount": "0.0000988",
// "feeCurrency": "BTC",
// "role": "MAKER",
// "timestamp": 1565171053345
// }
//
// private fetchMyTrades (same endpoint as fetchOrderTrades)
//
// {
// "id": "608717046691139584",
// "orderId": "608716957545402368",
// "symbol": "BTC_BRL",
// "side": "BUY",
// "amount": "0.0988",
// "price": "45514.76",
// "fee": "0.0000988 BTC",
// "feeAmount": "0.0000988",
// "feeCurrency": "BTC",
// "role": "MAKER",
// "timestamp": 1565171053345
// }
//
const id = this.safeString(trade, 'id');
const orderId = this.safeString(trade, 'orderId');
const timestamp = this.safeInteger(trade, 'timestamp');
const side = this.safeStringLower(trade, 'side');
const priceString = this.safeString(trade, 'price');
const amountString = this.safeString(trade, 'amount');
const marketId = this.safeString(trade, 'symbol');
const symbol = this.safeSymbol(marketId, market, '_');
const takerOrMaker = this.safeStringLower(trade, 'role');
const feeString = this.safeString(trade, 'fee');
let fee = undefined;
if (feeString !== undefined) {
const feeCurrencyId = this.safeString(trade, 'feeCurrency');
const feeCurrencyCode = this.safeCurrencyCode(feeCurrencyId);
fee = {
'cost': this.safeString(trade, 'feeAmount'),
'currency': feeCurrencyCode,
};
}
return this.safeTrade({
'id': id,
'order': orderId,
'timestamp': timestamp,
'datetime': this.iso8601(timestamp),
'symbol': symbol,
'type': undefined,
'side': side,
'price': priceString,
'amount': amountString,
'cost': undefined,
'takerOrMaker': takerOrMaker,
'fee': fee,
'info': trade,
}, market);
}
async fetchTrades(symbol, since = undefined, limit = undefined, params = {}) {
/**
* @method
* @name novadax#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 novadax 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 = {
'symbol': market['id'],
};
if (limit !== undefined) {
request['limit'] = limit; // default 100
}
const response = await this.publicGetMarketTrades(this.extend(request, params));
//
// {
// "code":"A10000",
// "data":[
// {"amount":"0.0632","price":"0.037288","side":"BUY","timestamp":1599279694576},
// {"amount":"0.0052","price":"0.03715","side":"SELL","timestamp":1599276606852},
// {"amount":"0.0058","price":"0.037188","side":"SELL","timestamp":1599275187812},
// ],
// "message":"Success"
// }
//
const data = this.safeValue(response, 'data', []);
return this.parseTrades(data, market, since, limit);
}
async fetchOHLCV(symbol, timeframe = '1m', since = undefined, limit = undefined, params = {}) {
/**
* @method
* @name novadax#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 novadax 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 request = {
'symbol': market['id'],
'unit': this.safeString(this.timeframes, timeframe, timeframe),
};
const duration = this.parseTimeframe(timeframe);
const now = this.seconds();
if (limit === undefined) {
limit = 3000; // max
}
if (since === undefined) {
request['from'] = now - limit * duration;
request['to'] = now;
}
else {
const startFrom = this.parseToInt(since / 1000);
request['from'] = startFrom;
request['to'] = this.sum(startFrom, limit * duration);
}
const response = await this.publicGetMarketKlineHistory(this.extend(request, params));
//
// {
// "code": "A10000",
// "data": [
// {
// "amount": 8.25709100,
// "closePrice": 62553.20,
// "count": 29,
// "highPrice": 62592.87,
// "lowPrice": 62553.20,
// "openPrice": 62554.23,
// "score": 1602501480,
// "symbol": "BTC_BRL",
// "vol": 516784.2504067500
// }
// ],
// "message": "Success"
// }
//
const data = this.safeValue(response, 'data', []);
return this.parseOHLCVs(data, market, timeframe, since, limit);
}
parseOHLCV(ohlcv, market = undefined) {
//
// {
// "amount": 8.25709100,
// "closePrice": 62553.20,
// "count": 29,
// "highPrice": 62592.87,
// "lowPrice": 62553.20,
// "openPrice": 62554.23,
// "score": 1602501480,
// "symbol": "BTC_BRL",
// "vol": 516784.2504067500
// }
//
const options = this.safeValue(this.options, 'fetchOHLCV', {});
const volumeField = this.safeString(options, 'volume', 'amount'); // or vol
return [
this.safeTimestamp(ohlcv, 'score'),
this.safeNumber(ohlcv, 'openPrice'),
this.safeNumber(ohlcv, 'highPrice'),
this.safeNumber(ohlcv, 'lowPrice'),
this.safeNumber(ohlcv, 'closePrice'),
this.safeNumber(ohlcv, volumeField),
];
}
parseBalance(response) {
const data = this.safeValue(response, 'data', []);
const result = {
'info': response,
'timestamp': undefined,
'datetime': undefined,
};
for (let i = 0; i < data.length; i++) {
const balance = data[i];
const currencyId = this.safeString(balance, 'currency');
const code = this.safeCurrencyCode(currencyId);
const account = this.account();
account['total'] = this.safeString(balance, 'balance');
account['free'] = this.safeString(balance, 'available');
account['used'] = this.safeString(balance, 'hold');
result[code] = account;
}
return this.safeBalance(result);
}
async fetchBalance(params = {}) {
/**
* @method
* @name novadax#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 novadax 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.privateGetAccountGetBalance(params);
//
// {
// "code": "A10000",
// "data": [
// {
// "available": "1.23",
// "balance": "0.23",
// "currency": "BTC",
// "hold": "1"
// }
// ],
// "message": "Success"
// }
//
return this.parseBalance(response);
}
async createOrder(symbol, type, side, amount, price = undefined, params = {}) {
/**
* @method
* @name novadax#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 novadax api endpoint
* @returns {object} an [order structure]{@link https://docs.ccxt.com/#/?id=order-structure}
*/
await this.loadMarkets();
const market = this.market(symbol);
let uppercaseType = type.toUpperCase();
const uppercaseSide = side.toUpperCase();
const request = {
'symbol': market['id'],
'side': uppercaseSide, // or SELL
// 'amount': this.amountToPrecision (symbol, amount),
// "price": "1234.5678", // required for LIMIT and STOP orders
// 'operator': '' // for stop orders, can be found in order introduction
// 'stopPrice': this.priceToPrecision (symbol, stopPrice),
// 'accountId': '...', // subaccount id, optional
};
const stopPrice = this.safeValue2(params, 'triggerPrice', 'stopPrice');
if (stopPrice === undefined) {
if ((uppercaseType === 'STOP_LIMIT') || (uppercaseType === 'STOP_MARKET')) {
throw new ArgumentsRequired(this.id + ' createOrder() requires a stopPrice parameter for ' + uppercaseType + ' orders');
}
}
else {
if (uppercaseType === 'LIMIT') {
uppercaseType = 'STOP_LIMIT';
}
else if (uppercaseType === 'MARKET') {
uppercaseType = 'STOP_MARKET';
}
const defaultOperator = (uppercaseSide === 'BUY') ? 'LTE' : 'GTE';
request['operator'] = this.safeString(params, 'operator', defaultOperator);
request['stopPrice'] = this.priceToPrecision(symbol, stopPrice);
params = this.omit(params, ['triggerPrice', 'stopPrice']);
}
if ((uppercaseType === 'LIMIT') || (uppercaseType === 'STOP_LIMIT')) {
request['price'] = this.priceToPrecision(symbol, price);
request['amount'] = this.amountToPrecision(symbol, amount);
}
else if ((uppercaseType === 'MARKET') || (uppercaseType === 'STOP_MARKET')) {
if (uppercaseSide === 'SELL') {
request['amount'] = this.amountToPrecision(symbol, amount);
}
else if (uppercaseSide === 'BUY') {
let value = this.safeNumber(params, 'value');
const createMarketBuyOrderRequiresPrice = this.safeValue(this.options, 'createMarketBuyOrderRequiresPrice', true);
if (createMarketBuyOrderRequiresPrice) {
if (price !== undefined) {
if (value === undefined) {
value = amount * price;
}
}
else if (value === undefined) {
throw new InvalidOrder(this.id + " createOrder() requires the price argument with market buy orders to calculate total order cost (amount to spend), where cost = amount * price. Supply a price argument to createOrder() call if you want the cost to be calculated for you from price and amount, or, alternatively, add .options['createMarketBuyOrderRequiresPrice'] = false and supply the total cost value in the 'amount' argument or in the 'value' extra parameter (the exchange-specific behaviour)");
}
}
else {
value = (value === undefined) ? amount : value;
}
request['value'] = this.costToPrecision(symbol, value);
}
}
request['type'] = uppercaseType;
const response = await this.privatePostOrdersCreate(this.extend(request, params));
//
// {
// "code": "A10000",
// "data": {
// "amount": "0.001",
// "averagePrice": null,
// "filledAmount": "0",
// "filledFee": "0",
// "filledValue": "0",
// "id": "870613508008464384",
// "operator": "GTE",
// "price": "210000",
// "side": "BUY",
// "status": "SUBMITTED",
// "stopPrice": "211000",
// "symbol": "BTC_BRL",
// "timestamp": 1627612035528,
// "type": "STOP_LIMIT",
// "value": "210"
// },
// "message": "Success"
// }
//
const data = this.safeValue(response, 'data', {});
return this.parseOrder(data, market);
}
async cancelOrder(id, symbol = undefined, params = {}) {
/**
* @method
* @name novadax#cancelOrder
* @description cancels an open order
* @param {string} id order id
* @param {string|undefined} symbol not used by novadax cancelOrder ()
* @param {object} params extra parameters specific to the novadax api endpoint
* @returns {object} An [order structure]{@link https://docs.ccxt.com/#/?id=order-structure}
*/
await this.loadMarkets();
const request = {
'id': id,
};
const response = await this.privatePostOrdersCancel(this.extend(request, params));
//
// {
// "code": "A10000",
// "data": {
// "result": true
// },
// "message": "Success"
// }
//
const data = this.safeValue(response, 'data', {});
return this.parseOrder(data);
}
async fetchOrder(id, symbol = undefined, params = {}) {
/**
* @method
* @name novadax#fetchOrder
* @description fetches information on an order made by the user
* @param {string|undefined} symbol not used by novadax fetchOrder
* @param {object} params extra parameters specific to the novadax api endpoint
* @returns {object} An [order structure]{@link https://docs.ccxt.com/#/?id=order-structure}
*/
await this.loadMarkets();
const request = {
'id': id,
};
const response = await this.privateGetOrdersGet(this.extend(request, params));
//
// {
// "code": "A10000",
// "data": {
// "id": "608695623247466496",
// "symbol": "BTC_BRL",
// "type": "MARKET",
// "side": "SELL",
// "price": null,
// "averagePrice": "0",
// "amount": "0.123",
// "filledAmount": "0",
// "value": null,
// "filledValue": "0",
// "filledFee": "0",
// "status": "REJECTED",
// "timestamp": 1565165945588
// },
// "message": "Success"
// }
//
const data = this.safeValue(response, 'data', {});
return this.parseOrder(data);
}
async fetchOrders(symbol = undefined, since = undefined, limit = undefined, params = {}) {
/**
* @method
* @name novadax#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 novadax api endpoint
* @returns {[object]} a list of [order structures]{@link https://docs.ccxt.com/#/?id=order-structure}
*/
await this.loadMarkets();
const request = {
// 'symbol': market['id'],
// 'status': 'SUBMITTED,PROCESSING', // SUBMITTED, PROCESSING, PARTIAL_FILLED, CANCELING, FILLED, CANCELED, REJECTED
// 'fromId': '...', // order id to begin with
// 'toId': '...', // order id to end up with
// 'fromTimestamp': since,
// 'toTimestamp': this.milliseconds (),
// 'limit': limit, // default 100, max 100
};
let market = undefined;
if (symbol !== undefined) {
market = this.market(symbol);
request['symbol'] = market['id'];
}
if (limit !== undefined) {
request['limit'] = limit; // default 100, max 100
}
if (since !== undefined) {
request['fromTimestamp'] = since;
}
const response = await this.privateGetOrdersList(this.extend(request, params));
//
// {
// "code": "A10000",
// "data": [
// {
// "id": "608695678650028032",
// "symbol": "BTC_BRL",
// "type": "MARKET",
// "side": "SELL",
// "price": null,
// "averagePrice": "0",
// "amount": "0.123",
// "filledAmount": "0",
// "value": null,
// "filledValue": "0",
// "filledFee": "0",
// "status": "REJECTED",
// "timestamp": 1565165958796
// },
// ],
// "message": "Success"
// }
//
const data = this.safeValue(response, 'data', []);
return this.parseOrders(data, market, since, limit);
}
async fetchOpenOrders(symbol = undefined, since = undefined, limit = undefined, params = {}) {
/**
* @method
* @name novadax#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 novadax api endpoint
* @returns {[object]} a list of [order structures]{@link https://docs.ccxt.com/#/?id=order-structure}
*/
const request = {
'status': 'SUBMITTED,PROCESSING,PARTIAL_FILLED,CANCELING',
};
return await this.fetchOrders(symbol, since, limit, this.extend(request, params));
}
async fetchClosedOrders(symbol = undefined, since = undefined, limit = undefined, params = {}) {
/**
* @method
* @name novadax#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 novadax api endpoint
* @returns {[object]} a list of [order structures]{@link https://docs.ccxt.com/#/?id=order-structure}
*/
const request = {
'status': 'FILLED,CANCELED,REJECTED',
};
return await this.fetchOrders(symbol, since, limit, this.extend(request, params));
}
async fetchOrderTrades(id, symbol = undefined, since = undefined, limit = undefined, params = {}) {
/**
* @method
* @name novadax#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 novadax api endpoint
* @returns {[object]} a list of [trade structures]{@link https://docs.ccxt.com/#/?id=trade-structure}
*/
await this.loadMarkets();
const request = {
'id': id,
};
const response = await this.privateGetOrdersFill(this.extend(request, params));
let market = undefined;
if (symbol !== undefined) {
market = this.market(symbol);
}
const data = this.safeValue(response, 'data', []);
//
// {
// "code": "A10000",
// "data": [
// {
// "id": "608717046691139584",
// "orderId": "608716957545402368",
// "symbol": "BTC_BRL",
// "side": "BUY",
// "amount": "0.0988",
// "price": "45514.76",
// "fee": "0.0000988 BTC",
// "feeAmount": "0.0000988",
// "feeCurrency": "BTC",
// "role": "MAKER",
// "timestamp": 1565171053345
// },
// ],
// "message": "Success"
// }
//
return this.parseTrades(data, market, since, limit);
}
parseOrderStatus(status) {
const statuses = {
'SUBMITTED': 'open',
'PROCESSING': 'open',
'PARTIAL_FILLED': 'open',
'CANCELING': 'open',
'FILLED': 'closed',
'CANCELED': 'canceled',
'REJECTED': 'rejected',
};
return this.safeString(statuses, status, status);
}
parseOrder(order, market = undefined) {
//
// createOrder, fetchOrders, fetchOrder
//
// {
// "amount": "0.001",
// "averagePrice": null,
// "filledAmount": "0",
// "filledFee": "0",
// "filledValue": "0",
// "id": "870613508008464384",
// "operator": "GTE",
// "price": "210000",
// "side": "BUY",
// "status": "SUBMITTED",
// "stopPrice": "211000",
// "symbol": "BTC_BRL",
// "timestamp": 1627612035528,
// "type": "STOP_LIMIT",
// "value": "210"
// }
//
// cancelOrder
//
// {
// "result": true
// }
//
const id = this.safeString(order, 'id');
const amount = this.safeString(order, 'amount');
const price = this.safeString(order, 'price');
const cost = this.safeString2(order, 'filledValue', 'value');
const type = this.safeStringLower(order, 'type');
const side = this.safeStringLower(order, 'side');
const status = this.parseOrderStatus(this.safeString(order, 'status'));
const timestamp = this.safeInteger(order, 'timestamp');
const average = this.safeString(order, 'averagePrice');
const filled = this.safeString(order, 'filledAmount');
let fee = undefined;
const feeCost = this.safeNumber(order, 'filledFee');
if (feeCost !== undefined) {
fee = {
'cost': feeCost,
'currency': undefined,
};
}
const marketId = this.safeString(order, 'symbol');
const symbol = this.safeSymbol(marketId, market, '_');
const stopPrice = this.safeNumber(order, 'stopPrice');
return this.safeOrder({
'id': id,
'clientOrderId': undefined,
'info': order,
'timestamp': timestamp,
'datetime': this.iso8601(timestamp),
'lastTradeTimestamp': undefined,
'symbol': symbol,
'type': type,
'timeInForce': undefined,
'postOnly': undefined,
'side': side,
'price': price,
'stopPrice': stopPrice,
'triggerPrice': stopPrice,
'amount': amount,
'cost': cost,
'average': average,
'filled': filled,
'remaining': undefined,
'status': status,
'fee': fee,
'trades': undefined,
}, market);
}
async transfer(code, amount, fromAccount, toAccount, params = {}) {
/**
* @method
* @name novadax#transfer
* @description transfer currency internally between wallets on the same account
* @param {string} code unified currency code
* @param {float} amount amount to transfer
* @param {string} fromAccount account to transfer from
* @param {string} toAccount account to transfer to
* @param {object} params extra parameters specific to the novadax api endpoint
* @returns {object} a [transfer structure]{@link https://docs.ccxt.com/#/?id=transfer-structure}
*/
await this.loadMarkets();
const currency = this.currency(code);
if (fromAccount !== 'main' && toAccount !== 'main') {
throw new ExchangeError(this.id + ' transfer() supports transfers between main account and subaccounts only');
}
// master-transfer-in = from master account to subaccount
// master-transfer-out = from subaccount to master account
const type = (fromAccount === 'main') ? 'master-transfer-in' : 'master-transfer-out';
const request = {
'transferAmount': this.currencyToPrecision(code, amount),
'currency': currency['id'],
'subId': (type === 'master-transfer-in') ? toAccount : fromAccount,
'transferType': type,
};
const response = await this.privatePostAccountSubsTransfer(this.extend(request, params));
//
// {
// "code":"A10000",
// "message":"Success",
// "data":40
// }
//
const transfer = this.parseTransfer(response, currency);
const transferOptions = this.safeValue(this.options, 'transfer', {});
const fillResponseFromRequest = this.safeValue(transferOptions, 'fillResponseFromRequest', true);
if (fillResponseFromRequest) {
transfer['fromAccount'] = fromAccount;
transfer['toAccount'] = toAccount;
transfer['amount'] = amount;
}
return transfer;
}
parseTransfer(transfer, currency = undefined) {
//
// {
// "code":"A10000",
// "message":"Success",
// "data":40
// }
//
const id = this.safeString(transfer, 'data');
const status = this.safeString(transfer, 'message');
const currencyCode = this.safeCurrencyCode(undefined, currency);
return {
'info': transfer,
'id': id,
'amount': undefined,
'code': currencyCode,
'currency': currencyCode,
'fromAccount': undefined,
'toAccount': undefined,
'timestamp': undefined,
'datetime': undefined,
'status': status,
};
}
parseTransferStatus(status) {
const statuses = {
'SUCCESS': 'pending',
};
return this.safeString(statuses, status, 'failed');
}
async withdraw(code, amount, address, tag = undefined,