Skip to content

Commit

Permalink
[Feature] - New CLI option market-data
Browse files Browse the repository at this point in the history
  • Loading branch information
sjanel committed Mar 10, 2024
1 parent f767feb commit a78224a
Show file tree
Hide file tree
Showing 13 changed files with 91 additions and 21 deletions.
6 changes: 3 additions & 3 deletions src/api/interface/src/exchange.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -63,7 +63,7 @@ bool Exchange::canWithdraw(CurrencyCode currencyCode, const CurrencyExchangeFlat
}

bool Exchange::canDeposit(CurrencyCode currencyCode, const CurrencyExchangeFlatSet &currencyExchangeSet) const {
auto lb = currencyExchangeSet.find(currencyCode);
const auto lb = currencyExchangeSet.find(currencyCode);
if (lb == currencyExchangeSet.end()) {
log::trace("{} cannot be deposited on {}", currencyCode, name());
return false;
Expand All @@ -72,7 +72,7 @@ bool Exchange::canDeposit(CurrencyCode currencyCode, const CurrencyExchangeFlatS
}

MarketOrderBook Exchange::queryOrderBook(Market mk, int depth) {
auto marketOrderBook = _exchangePublic.queryOrderBook(mk, depth);
const auto marketOrderBook = _exchangePublic.queryOrderBook(mk, depth);
if (_marketDataSerializerPtr) {
_marketDataSerializerPtr->push(marketOrderBook);
}
Expand All @@ -81,7 +81,7 @@ MarketOrderBook Exchange::queryOrderBook(Market mk, int depth) {

/// Retrieve an ordered vector of recent last trades
TradesVector Exchange::queryLastTrades(Market mk, int nbTrades) {
auto lastTrades = _exchangePublic.queryLastTrades(mk, nbTrades);
const auto lastTrades = _exchangePublic.queryLastTrades(mk, nbTrades);
if (_marketDataSerializerPtr) {
_marketDataSerializerPtr->push(lastTrades);
}
Expand Down
4 changes: 4 additions & 0 deletions src/engine/include/coincenter.hpp
Original file line number Diff line number Diff line change
Expand Up @@ -52,6 +52,10 @@ class Coincenter {
CurrencyCode equiCurrencyCode,
std::optional<int> depth = std::nullopt);

/// Query market data without returning it.
/// This method is especially useful for serialization and metric exports.
void queryMarketDataPerExchange(Market mk, ExchangeNameSpan exchangeNames);

/// Retrieve the last 24h traded volume for exchanges supporting given market.
MonetaryAmountPerExchange getLast24hTradedVolumePerExchange(Market mk, ExchangeNameSpan exchangeNames);

Expand Down
3 changes: 1 addition & 2 deletions src/engine/include/coincentercommand.hpp
Original file line number Diff line number Diff line change
Expand Up @@ -44,7 +44,6 @@ class CoincenterCommand {
CoincenterCommand& setAmount(MonetaryAmount amount);

CoincenterCommand& setDepth(int depth);
CoincenterCommand& setNbLastTrades(int nbTrades) { return setDepth(nbTrades); }

CoincenterCommand& setMarket(Market market);

Expand All @@ -70,7 +69,7 @@ class CoincenterCommand {

MonetaryAmount amount() const { return _amount; }

int nbLastTrades() const { return _n; }
int depth() const { return _n; }
std::optional<int> optDepth() const { return _n == -1 ? std::nullopt : std::optional<int>(_n); }

Market market() const { return _market; }
Expand Down
5 changes: 3 additions & 2 deletions src/engine/include/coincenteroptions.hpp
Original file line number Diff line number Diff line change
Expand Up @@ -97,13 +97,14 @@ class CoincenterCmdLineOptions {

std::string_view lastTrades;

std::string_view marketData;

std::optional<std::string_view> replay;
std::optional<std::string_view> replayMarkets;

CommandLineOptionalInt repeats;
int nbLastTrades = api::ExchangePublic::kNbLastTradesDefault;
int monitoringPort = CoincenterCmdLineOptionsDefinitions::kDefaultMonitoringPort;
int orderbookDepth = 0;
int depth = 0;

bool forceMultiTrade = false;
bool forceSingleTrade = false;
Expand Down
19 changes: 13 additions & 6 deletions src/engine/include/coincenteroptionsdef.hpp
Original file line number Diff line number Diff line change
Expand Up @@ -208,8 +208,7 @@ struct CoincenterAllowedOptions : private CoincenterCmdLineOptionsDefinitions {
"If conversion of cur2 into cur is possible (for each exchange), "
"prints additional column converted to given asset"},
&OptValueType::orderbookCur},
{{{"Public queries", 2300}, "--depth", "<n>", "Override default depth of order book"},
&OptValueType::orderbookDepth},
{{{"Public queries", 2300}, "--depth", "<n>", "Override default depth of order book"}, &OptValueType::depth},
{{{"Public queries", 2400},
"ticker",
"<[exch1,...]>",
Expand Down Expand Up @@ -239,7 +238,7 @@ struct CoincenterAllowedOptions : private CoincenterCmdLineOptionsDefinitions {
"Print last trades for market 'cur1'-'cur2' "
"for all exchanges (or specified one)"},
&OptValueType::lastTrades},
{{{"Public queries", 2800}, "--n", "<n>", kLastTradesN}, &OptValueType::nbLastTrades},
{{{"Public queries", 2800}, "--n", "<n>", kLastTradesN}, &OptValueType::depth},
{{{"Public queries", 2900},
"price",
"<cur1-cur2[,exch1,...]>",
Expand Down Expand Up @@ -406,17 +405,25 @@ struct CoincenterAllowedOptions : private CoincenterCmdLineOptionsDefinitions {
&OptValueType::async},
{{{"Withdraw and deposit", 6010}, "--refresh-time", "<time>", kWithdrawRefreshTime},
&OptValueType::withdrawRefreshTime},
{{{"Withdraw and deposit", 7000},
{{{"Withdraw and deposit", 6020},
"withdraw-apply-all",
"<cur,from-to>",
"Withdraw all available amount instead of a specified amount."},
&OptValueType::withdrawApplyAll},
{{{"Withdraw and deposit", 8000},
{{{"Withdraw and deposit", 6030},
"withdraw-fees",
"<[cur][,exch1,...]>",
"Prints withdraw fees for matching currency and exchanges."
"Prints withdraw fees for matching currency and exchanges.\n"
"Currency and exchanges are optional, if specified, output will be filtered to match them."},
&OptValueType::withdrawFees},
{{{"Automation", 8000},
"market-data",
"<cur1-cur2[,exch1,...]>",
"Query last trades and order books of given market without printing the result on screen, for given exchanges "
"if specified.\n"
"This is the equivalent of calling last-trades and order-book but is useful combined with the repeat "
"command to store market data on disk."},
&OptValueType::marketData},
{{{"Automation", 8000}, "replay-markets", "<duration[,exch1,...]>", "Print markets available for replay."},
&OptValueType::replayMarkets},
{{{"Automation", 8001}, "replay", "<duration[,exch1,...]>", "Replay an algorithm on saved data."},
Expand Down
2 changes: 2 additions & 0 deletions src/engine/include/exchangesorchestrator.hpp
Original file line number Diff line number Diff line change
Expand Up @@ -87,6 +87,8 @@ class ExchangesOrchestrator {

MonetaryAmountPerExchange getLastPricePerExchange(Market mk, ExchangeNameSpan exchangeNames);

MarketDataPerExchange getMarketDataPerExchange(Market mk, ExchangeNameSpan exchangeNames);

MarketsPerExchange pullAvailableMarketsForReplay(TimeWindow timeWindow, ExchangeNameSpan exchangeNames);

void traderConsumeRange(Market market, TimeWindow timeWindow, std::span<MarketTraderEngine> marketTraderEngines,
Expand Down
3 changes: 2 additions & 1 deletion src/engine/include/queryresulttypes.hpp
Original file line number Diff line number Diff line change
Expand Up @@ -40,7 +40,8 @@ using MonetaryAmountByCurrencySetPerExchange =

using TradesPerExchange = FixedCapacityVector<ExchangeWith<TradesVector>, kNbSupportedExchanges>;

using MarketOrderBooksPerExchange = FixedCapacityVector<ExchangeWith<MarketOrderBookVector>, kNbSupportedExchanges>;
using MarketDataPerExchange =
FixedCapacityVector<ExchangeWith<std::pair<MarketOrderBook, TradesVector>>, kNbSupportedExchanges>;

using TradeResultPerExchange = SmallVector<ExchangeWith<TradeResult>, kTypicalNbPrivateAccounts>;

Expand Down
34 changes: 31 additions & 3 deletions src/engine/src/coincenter.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -147,9 +147,9 @@ TransferableCommandResultVector Coincenter::processCommand(
break;
}
case CoincenterCommandType::kLastTrades: {
const auto lastTradesPerExchange =
getLastTradesPerExchange(cmd.market(), cmd.exchangeNames(), cmd.nbLastTrades());
_queryResultPrinter.printLastTrades(cmd.market(), cmd.nbLastTrades(), lastTradesPerExchange);
const int depth = cmd.depth() == 0 ? api::ExchangePublic::kNbLastTradesDefault : cmd.depth();
const auto lastTradesPerExchange = getLastTradesPerExchange(cmd.market(), cmd.exchangeNames(), depth);
_queryResultPrinter.printLastTrades(cmd.market(), depth, lastTradesPerExchange);
break;
}
case CoincenterCommandType::kLast24hTradedVolume: {
Expand Down Expand Up @@ -252,6 +252,11 @@ TransferableCommandResultVector Coincenter::processCommand(
_queryResultPrinter.printDustSweeper(dustSweeper(cmd.exchangeNames(), cmd.cur1()), cmd.cur1());
break;
}
case CoincenterCommandType::kMarketData: {
// No return value here, this command is made only for storing purposes.
queryMarketDataPerExchange(cmd.market(), cmd.exchangeNames());
break;
}
case CoincenterCommandType::kReplay: {
replay(cmd.duration(), cmd.exchangeNames());
break;
Expand Down Expand Up @@ -293,6 +298,29 @@ MarketOrderBookConversionRates Coincenter::getMarketOrderBooks(Market mk, Exchan
return ret;
}

void Coincenter::queryMarketDataPerExchange(Market mk, ExchangeNameSpan exchangeNames) {
const auto marketDataPerExchange = _exchangesOrchestrator.getMarketDataPerExchange(mk, exchangeNames);

// Transform data structures to export metrics input format
MarketOrderBookConversionRates marketOrderBookConversionRates(marketDataPerExchange.size());
TradesPerExchange lastTradesPerExchange(marketDataPerExchange.size());

std::ranges::transform(marketDataPerExchange, marketOrderBookConversionRates.begin(),
[](const auto &exchangeWithPairOrderBooksAndTrades) {
return std::make_tuple(exchangeWithPairOrderBooksAndTrades.first->name(),
exchangeWithPairOrderBooksAndTrades.second.first, std::nullopt);
});

std::ranges::transform(marketDataPerExchange, lastTradesPerExchange.begin(),
[](const auto &exchangeWithPairOrderBooksAndTrades) {
return std::make_pair(exchangeWithPairOrderBooksAndTrades.first,
exchangeWithPairOrderBooksAndTrades.second.second);
});

_metricsExporter.exportOrderbookMetrics(mk, marketOrderBookConversionRates);
_metricsExporter.exportLastTradesMetrics(mk, lastTradesPerExchange);
}

BalancePerExchange Coincenter::getBalance(std::span<const ExchangeName> privateExchangeNames,
const BalanceOptions &balanceOptions) {
CurrencyCode equiCurrency = balanceOptions.equiCurrency();
Expand Down
11 changes: 9 additions & 2 deletions src/engine/src/coincentercommands.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -66,7 +66,7 @@ void CoincenterCommands::addOption(const CoincenterCmdLineOptions &cmdLineOption
_commands.emplace_back(CoincenterCommandType::kOrderbook)
.setMarket(optionParser.parseMarket())
.setExchangeNames(optionParser.parseExchanges())
.setDepth(cmdLineOptions.orderbookDepth)
.setDepth(cmdLineOptions.depth)
.setCur1(cmdLineOptions.orderbookCur);
}

Expand Down Expand Up @@ -184,7 +184,7 @@ void CoincenterCommands::addOption(const CoincenterCmdLineOptions &cmdLineOption
optionParser = StringOptionParser(cmdLineOptions.lastTrades);
_commands.emplace_back(CoincenterCommandType::kLastTrades)
.setMarket(optionParser.parseMarket())
.setNbLastTrades(cmdLineOptions.nbLastTrades)
.setDepth(cmdLineOptions.depth)
.setExchangeNames(optionParser.parseExchanges());
}

Expand All @@ -195,6 +195,13 @@ void CoincenterCommands::addOption(const CoincenterCmdLineOptions &cmdLineOption
.setExchangeNames(optionParser.parseExchanges());
}

if (!cmdLineOptions.marketData.empty()) {
optionParser = StringOptionParser(cmdLineOptions.marketData);
_commands.emplace_back(CoincenterCommandType::kMarketData)
.setMarket(optionParser.parseMarket())
.setExchangeNames(optionParser.parseExchanges());
}

if (cmdLineOptions.replay) {
optionParser = StringOptionParser(*cmdLineOptions.replay);

Expand Down
20 changes: 20 additions & 0 deletions src/engine/src/exchangesorchestrator.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -938,6 +938,26 @@ MonetaryAmountPerExchange ExchangesOrchestrator::getLastPricePerExchange(Market
return lastPricePerExchange;
}

MarketDataPerExchange ExchangesOrchestrator::getMarketDataPerExchange(Market mk, ExchangeNameSpan exchangeNames) {
log::info("Query market data for {} on {}", mk, ConstructAccumulatedExchangeNames(exchangeNames));

UniquePublicSelectedExchanges selectedExchanges = _exchangeRetriever.selectOneAccount(exchangeNames);
std::array<bool, kNbSupportedExchanges> isMarketTradable;

_threadPool.parallelTransform(selectedExchanges.begin(), selectedExchanges.end(), isMarketTradable.begin(),
[mk](Exchange *exchange) { return exchange->queryTradableMarkets().contains(mk); });

FilterVector(selectedExchanges, isMarketTradable);

MarketDataPerExchange ret(selectedExchanges.size());
_threadPool.parallelTransform(
selectedExchanges.begin(), selectedExchanges.end(), ret.begin(), [mk](Exchange *exchange) {
// Call order book and last trades sequentially for this exchange
return std::make_pair(exchange, std::make_pair(exchange->queryOrderBook(mk), exchange->queryLastTrades(mk)));
});
return ret;
}

MarketsPerExchange ExchangesOrchestrator::pullAvailableMarketsForReplay(TimeWindow timeWindow,
ExchangeNameSpan exchangeNames) {
log::info("Query available markets for replay from {} within {}", ConstructAccumulatedExchangeNames(exchangeNames),
Expand Down
2 changes: 1 addition & 1 deletion src/engine/test/coincenteroptions_test.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -59,7 +59,7 @@ TEST_F(CoincenterCmdLineOptionsTest, MergeGlobal) {

CoincenterCmdLineOptions rhs;
rhs.trade = "some value";
rhs.nbLastTrades = 42;
rhs.depth = 42;
rhs.repeatTime = std::chrono::minutes(45);
rhs.monitoringPort = 999;

Expand Down
1 change: 1 addition & 0 deletions src/objects/include/coincentercommandtype.hpp
Original file line number Diff line number Diff line change
Expand Up @@ -30,6 +30,7 @@ enum class CoincenterCommandType : int8_t {
kWithdrawApply,
kDustSweeper,

kMarketData,
kReplay,
kReplayMarkets,

Expand Down
2 changes: 1 addition & 1 deletion src/objects/src/coincentercommandtype.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -16,7 +16,7 @@ constexpr std::string_view kCommandTypeNames[] = {

"Balance", "DepositInfo", "OrdersClosed", "OrdersOpened", "OrdersCancel",
"RecentDeposits", "RecentWithdraws", "Trade", "Buy", "Sell",
"Withdraw", "DustSweeper", "Replay", "ReplayMarkets"};
"Withdraw", "DustSweeper", "MarketData", "Replay", "ReplayMarkets"};

static_assert(std::size(kCommandTypeNames) == static_cast<std::size_t>(CoincenterCommandType::kLast));
} // namespace
Expand Down

0 comments on commit a78224a

Please sign in to comment.