Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Remove Cryptowatch API (it has been switch off) #454

Merged
merged 1 commit into from
Oct 9, 2023
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
6 changes: 3 additions & 3 deletions src/api/common/CMakeLists.txt
Original file line number Diff line number Diff line change
Expand Up @@ -19,7 +19,7 @@ function(add_common_test name)
add_unit_test(
${name}
${MY_UNPARSED_ARGUMENTS}
src/cryptowatchapi.cpp
src/commonapi.cpp
LIBRARIES
coincenter_api-objects
coincenter_http-request
Expand All @@ -28,8 +28,8 @@ function(add_common_test name)
endfunction()

add_common_test(
cryptowatchapi_test
test/cryptowatchapi_test.cpp
commonapi_test
test/commonapi_test.cpp
)

add_common_test(
Expand Down
55 changes: 55 additions & 0 deletions src/api/common/include/commonapi.hpp
Original file line number Diff line number Diff line change
@@ -0,0 +1,55 @@
#pragma once

#include <chrono>
#include <mutex>

#include "cachedresult.hpp"
#include "cct_flatset.hpp"
#include "curlhandle.hpp"
#include "currencycode.hpp"
#include "exchangebase.hpp"
#include "timedef.hpp"

namespace cct {
class CoincenterInfo;
namespace api {
/// Public API connected to different exchanges, providing fast methods to retrieve huge amount of data.
class CommonAPI : public ExchangeBase {
public:
using Fiats = FlatSet<CurrencyCode>;

CommonAPI(const CoincenterInfo &config, Duration fiatsUpdateFrequency = std::chrono::hours(96),
bool loadFromFileCacheAtInit = true);

/// Returns a new set of fiat currencies.
Fiats queryFiats() {
std::lock_guard<std::mutex> guard(_fiatsMutex);
return _fiatsCache.get();
}

/// Tells whether given currency code is a fiat currency or not.
/// Fiat currencies are traditional currencies, such as EUR, USD, GBP, KRW, etc.
/// Information here: https://en.wikipedia.org/wiki/Fiat_money
bool queryIsCurrencyCodeFiat(CurrencyCode currencyCode) {
std::lock_guard<std::mutex> guard(_fiatsMutex);
return _fiatsCache.get().contains(currencyCode);
}

void updateCacheFile() const override;

private:
struct FiatsFunc {
FiatsFunc();

Fiats operator()();

CurlHandle _curlHandle;
};

CachedResultVault _cachedResultVault;
const CoincenterInfo &_coincenterInfo;
std::mutex _fiatsMutex;
CachedResult<FiatsFunc> _fiatsCache;
};
} // namespace api
} // namespace cct
99 changes: 0 additions & 99 deletions src/api/common/include/cryptowatchapi.hpp

This file was deleted.

4 changes: 2 additions & 2 deletions src/api/common/include/exchangebase.hpp
Original file line number Diff line number Diff line change
Expand Up @@ -30,7 +30,7 @@ class UniqueQueryHandle {
}

~UniqueQueryHandle() {
if (_pCachedResultVault) {
if (_pCachedResultVault != nullptr) {
_pCachedResultVault->unfreezeAll();
}
}
Expand All @@ -43,7 +43,7 @@ class ExchangeBase {
public:
virtual void updateCacheFile() const {}

virtual ~ExchangeBase() {}
virtual ~ExchangeBase() = default;

protected:
ExchangeBase() = default;
Expand Down
16 changes: 8 additions & 8 deletions src/api/common/include/exchangepublicapi.hpp
Original file line number Diff line number Diff line change
Expand Up @@ -4,7 +4,7 @@
#include <string_view>

#include "cct_string.hpp"
#include "cryptowatchapi.hpp"
#include "commonapi.hpp"
#include "currencycode.hpp"
#include "currencyexchangeflatset.hpp"
#include "exchangebase.hpp"
Expand All @@ -28,7 +28,7 @@ class ExchangePublic : public ExchangeBase {
static constexpr int kDefaultDepth = MarketOrderBook::kDefaultDepth;
static constexpr int kNbLastTradesDefault = 100;

using Fiats = CryptowatchAPI::Fiats;
using Fiats = CommonAPI::Fiats;

virtual ~ExchangePublic() = default;

Expand Down Expand Up @@ -57,15 +57,15 @@ class ExchangePublic : public ExchangeBase {
Fiats fiats = queryFiats();
MarketSet markets;
MarketsPath conversionPath = findMarketsPath(a.currencyCode(), toCurrency, markets, fiats, true);
return convert(a, toCurrency, conversionPath, fiats, marketOrderBookMap, true, priceOptions);
return convert(a, toCurrency, conversionPath, fiats, marketOrderBookMap, priceOptions);
}

/// Attempts to convert amount into a target currency.
/// Conversion is made according to given price options, which uses the 'Maker' prices by default.
/// No external calls is made with this version, it has all what it needs
std::optional<MonetaryAmount> convert(MonetaryAmount a, CurrencyCode toCurrency, const MarketsPath &conversionPath,
const Fiats &fiats, MarketOrderBookMap &marketOrderBookMap,
bool canUseCryptowatchAPI, const PriceOptions &priceOptions = PriceOptions());
const PriceOptions &priceOptions = PriceOptions());

/// Retrieve the fixed withdrawal fees per currency.
/// Depending on the exchange, this could be retrieved dynamically,
Expand Down Expand Up @@ -97,7 +97,7 @@ class ExchangePublic : public ExchangeBase {
/// Retrieve the last price of given market.
virtual MonetaryAmount queryLastPrice(Market mk) = 0;

Fiats queryFiats() { return _cryptowatchApi.queryFiats(); }
Fiats queryFiats() { return _commonApi.queryFiats(); }

/// Get the name of the exchange in lower case.
std::string_view name() const { return _name; }
Expand Down Expand Up @@ -160,18 +160,18 @@ class ExchangePublic : public ExchangeBase {

const ExchangeInfo &exchangeInfo() const { return _exchangeInfo; }

CryptowatchAPI &cryptowatchAPI() { return _cryptowatchApi; }
CommonAPI &commonAPI() { return _commonApi; }

protected:
friend class ExchangePrivate;

ExchangePublic(std::string_view name, FiatConverter &fiatConverter, CryptowatchAPI &cryptowatchApi,
ExchangePublic(std::string_view name, FiatConverter &fiatConverter, CommonAPI &commonApi,
const CoincenterInfo &coincenterInfo);

string _name;
CachedResultVault _cachedResultVault;
FiatConverter &_fiatConverter;
CryptowatchAPI &_cryptowatchApi;
CommonAPI &_commonApi;
const CoincenterInfo &_coincenterInfo;
const ExchangeInfo &_exchangeInfo;
};
Expand Down
6 changes: 3 additions & 3 deletions src/api/common/include/exchangepublicapi_mock.hpp
Original file line number Diff line number Diff line change
Expand Up @@ -2,17 +2,17 @@

#include <gmock/gmock.h>

#include "cryptowatchapi.hpp"
#include "commonapi.hpp"
#include "exchangepublicapi.hpp"
#include "exchangepublicapitypes.hpp"
#include "fiatconverter.hpp"

namespace cct::api {
class MockExchangePublic : public ExchangePublic {
public:
MockExchangePublic(std::string_view name, FiatConverter &fiatConverter, CryptowatchAPI &cryptowatchApi,
MockExchangePublic(std::string_view name, FiatConverter &fiatConverter, CommonAPI &commonApi,
const CoincenterInfo &config)
: ExchangePublic(name, fiatConverter, cryptowatchApi, config) {}
: ExchangePublic(name, fiatConverter, commonApi, config) {}

MOCK_METHOD(bool, healthCheck, (), (override));
MOCK_METHOD(CurrencyExchangeFlatSet, queryTradableCurrencies, (), (override));
Expand Down
92 changes: 92 additions & 0 deletions src/api/common/src/commonapi.cpp
Original file line number Diff line number Diff line change
@@ -0,0 +1,92 @@
#include "commonapi.hpp"

#include <cctype>
#include <string_view>

#include "cct_exception.hpp"
#include "cct_json.hpp"
#include "cct_log.hpp"
#include "coincenterinfo.hpp"
#include "curloptions.hpp"
#include "file.hpp"
#include "permanentcurloptions.hpp"
#include "timedef.hpp"

namespace cct::api {
namespace {

File GetFiatCacheFile(std::string_view dataDir) {
return {dataDir, File::Type::kCache, "fiatcache.json", File::IfError::kNoThrow};
}

} // namespace

CommonAPI::CommonAPI(const CoincenterInfo& config, Duration fiatsUpdateFrequency, bool loadFromFileCacheAtInit)
: _coincenterInfo(config), _fiatsCache(CachedResultOptions(fiatsUpdateFrequency, _cachedResultVault)) {
if (loadFromFileCacheAtInit) {
json data = GetFiatCacheFile(_coincenterInfo.dataDir()).readAllJson();
if (!data.empty()) {
int64_t timeEpoch = data["timeepoch"].get<int64_t>();
auto& fiatsFile = data["fiats"];
Fiats fiats;
fiats.reserve(static_cast<Fiats::size_type>(fiatsFile.size()));
for (json& val : fiatsFile) {
log::trace("Reading fiat {} from cache file", val.get<std::string_view>());
fiats.emplace_hint(fiats.end(), std::move(val.get_ref<string&>()));
}
log::debug("Loaded {} fiats from cache file", fiats.size());
_fiatsCache.set(std::move(fiats), TimePoint(std::chrono::seconds(timeEpoch)));
}
}
}

namespace {
constexpr std::string_view kFiatsUrl = "https://datahub.io/core/currency-codes/r/codes-all.json";
}

CommonAPI::FiatsFunc::FiatsFunc()
: _curlHandle(kFiatsUrl, nullptr, PermanentCurlOptions::Builder().setFollowLocation().build()) {}

CommonAPI::Fiats CommonAPI::FiatsFunc::operator()() {
json dataCSV = json::parse(_curlHandle.query("", CurlOptions(HttpRequestType::kGet)));
Fiats fiats;
for (const json& fiatData : dataCSV) {
static constexpr std::string_view kCodeKey = "AlphabeticCode";
static constexpr std::string_view kWithdrawalDateKey = "WithdrawalDate";
auto codeIt = fiatData.find(kCodeKey);
auto withdrawalDateIt = fiatData.find(kWithdrawalDateKey);
if (codeIt != fiatData.end() && !codeIt->is_null() && withdrawalDateIt != fiatData.end() &&
withdrawalDateIt->is_null()) {
fiats.insert(CurrencyCode(codeIt->get<std::string_view>()));
log::debug("Stored {} fiat", codeIt->get<std::string_view>());
}
}
if (fiats.empty()) {
throw exception("Error parsing currency codes, no fiats found in {}", dataCSV.dump());
}

log::info("Stored {} fiats", fiats.size());
return fiats;
}

void CommonAPI::updateCacheFile() const {
File fiatsCacheFile = GetFiatCacheFile(_coincenterInfo.dataDir());
json data = fiatsCacheFile.readAllJson();
auto fiatsPtrLastUpdatedTimePair = _fiatsCache.retrieve();
auto timeEpochIt = data.find("timeepoch");
if (timeEpochIt != data.end()) {
int64_t lastTimeFileUpdated = timeEpochIt->get<int64_t>();
if (TimePoint(std::chrono::seconds(lastTimeFileUpdated)) >= fiatsPtrLastUpdatedTimePair.second) {
return; // No update
}
}
data.clear();
if (fiatsPtrLastUpdatedTimePair.first != nullptr) {
for (CurrencyCode fiatCode : *fiatsPtrLastUpdatedTimePair.first) {
data["fiats"].emplace_back(fiatCode.str());
}
data["timeepoch"] = TimestampToS(fiatsPtrLastUpdatedTimePair.second);
fiatsCacheFile.write(data);
}
}
} // namespace cct::api
Loading