Skip to content

Commit

Permalink
Add redundancy to ARS price node
Browse files Browse the repository at this point in the history
 - Implement ExchangeRateTransformer (@alvasw)
 - Implement BlueLyticsApi (@alvasw)
 - Implement ArsBlueMarketGapProvider (@alvasw)
 - Implement ArsBlueRateTransformer (@alvasw)
 - merge @alvasw improvements
 - fixed bug lossing prices after merge
 - fix tests, all green
 - delete unused code
 - comment most important methods/classes
  • Loading branch information
alvasw authored and rodvar committed Sep 11, 2023
1 parent c70ea6b commit d67f1e2
Show file tree
Hide file tree
Showing 12 changed files with 282 additions and 239 deletions.
61 changes: 61 additions & 0 deletions src/main/java/bisq/price/spot/ArsBlueRateTransformer.java
Original file line number Diff line number Diff line change
@@ -0,0 +1,61 @@
/*
* This file is part of Bisq.
*
* Bisq is free software: you can redistribute it and/or modify it
* under the terms of the GNU Affero General Public License as published by
* the Free Software Foundation, either version 3 of the License, or (at
* your option) any later version.
*
* Bisq is distributed in the hope that it will be useful, but WITHOUT
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU Affero General Public
* License for more details.
*
* You should have received a copy of the GNU Affero General Public License
* along with Bisq. If not, see <http://www.gnu.org/licenses/>.
*/

package bisq.price.spot;

import bisq.price.spot.providers.BlueRateProvider;
import bisq.price.util.bluelytics.ArsBlueMarketGapProvider;
import org.springframework.stereotype.Component;

import java.util.Optional;
import java.util.OptionalDouble;

@Component
public class ArsBlueRateTransformer implements ExchangeRateTransformer {
private final ArsBlueMarketGapProvider blueMarketGapProvider;

public ArsBlueRateTransformer(ArsBlueMarketGapProvider blueMarketGapProvider) {
this.blueMarketGapProvider = blueMarketGapProvider;
}

@Override
public Optional<ExchangeRate> apply(ExchangeRateProvider provider, ExchangeRate originalExchangeRate) {
if (provider instanceof BlueRateProvider) {
return Optional.of(originalExchangeRate);
}

OptionalDouble sellGapMultiplier = blueMarketGapProvider.get();
if (sellGapMultiplier.isEmpty()) {
return Optional.empty();
}

double blueRate = originalExchangeRate.getPrice() * sellGapMultiplier.getAsDouble();

ExchangeRate newExchangeRate = new ExchangeRate(
originalExchangeRate.getCurrency(),
blueRate,
originalExchangeRate.getTimestamp(),
originalExchangeRate.getProvider()
);
return Optional.of(newExchangeRate);
}

@Override
public String supportedCurrency() {
return "ARS";
}
}
27 changes: 0 additions & 27 deletions src/main/java/bisq/price/spot/ExchangeRateProvider.java
Original file line number Diff line number Diff line change
Expand Up @@ -334,31 +334,4 @@ protected long getMarketDataCallDelay() {
protected boolean requiresFilterDuringBulkTickerRetrieval() {
return false;
}

/**
*
* @return true if the implementation of this ExchangeRateProvider already consider currencies
* blue markets if any. Defaults to false.
*/
public boolean alreadyConsidersBlueMarkets() {
return false;
}

/**
* @param originalRate original official rate for a certain currency. E.g. rate for ARS for Argentina PESO
* @return a new exchange rate if the implementation of this price provider already tackles real vs official
* market prices for the given currency.
* e.g. for FIAT ARS official rates are not the ones using in the free trading world.
* Most currencies won't need this, so defaults to null.
*/
public @Nullable ExchangeRate maybeUpdateBlueMarketPriceGapForRate(ExchangeRate originalRate, Double blueMarketGapForCurrency) {
if ("ARS".equalsIgnoreCase(originalRate.getCurrency())) {
double blueRate = originalRate.getPrice() * blueMarketGapForCurrency;
return new ExchangeRate(originalRate.getCurrency(),
BigDecimal.valueOf(blueRate),
new Date(originalRate.getTimestamp()),
originalRate.getProvider());
}
return null;
}
}
94 changes: 34 additions & 60 deletions src/main/java/bisq/price/spot/ExchangeRateService.java
Original file line number Diff line number Diff line change
Expand Up @@ -18,11 +18,7 @@
package bisq.price.spot;

import bisq.common.util.Tuple2;

import bisq.core.util.InlierUtil;

import bisq.price.util.bluelytics.BlueLyticsService;
import com.google.common.collect.Sets;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.core.env.Environment;
Expand All @@ -40,17 +36,22 @@ class ExchangeRateService {
protected final Logger log = LoggerFactory.getLogger(this.getClass());
private final Environment env;
private final List<ExchangeRateProvider> providers;
private final List<ExchangeRateTransformer> transformers;

/**
* Construct an {@link ExchangeRateService} with a list of all
* {@link ExchangeRateProvider} implementations discovered via classpath scanning.
*
* @param providers all {@link ExchangeRateProvider} implementations in ascending
* order of precedence
* @param providers all {@link ExchangeRateProvider} implementations in ascending
* order of precedence
* @param transformers all {@link ExchangeRateTransformer} implementations
*/
public ExchangeRateService(Environment env, List<ExchangeRateProvider> providers) {
public ExchangeRateService(Environment env,
List<ExchangeRateProvider> providers,
List<ExchangeRateTransformer> transformers) {
this.env = env;
this.providers = providers;
this.transformers = transformers;
}

public Map<String, Object> getAllMarketPrices() {
Expand All @@ -76,56 +77,6 @@ public Map<String, Object> getAllMarketPrices() {
return result;
}

/**
* Please do not call provider.get(), use this method instead to consider currencies with blue markets
* @param provider to get the rates from
* @return the exchange rates for the different currencies the provider supports considering bluemarket rates if any
*/
public Set<ExchangeRate> providerCurrentExchangeRates(ExchangeRateProvider provider) {
Map<String, Double> blueMarketGapForCurrency = this.getBlueMarketGapForCurrencies();
Set<ExchangeRate> originalExchangeRates = provider.get();
if (originalExchangeRates == null)
return null;

Set<ExchangeRate> exchangeRates = new HashSet<>();
boolean noOverlapBetweenCurrencies = Sets.intersection(originalExchangeRates.stream().map(ExchangeRate::getCurrency)
.collect(Collectors.toSet()), blueMarketGapForCurrency.keySet())
.isEmpty();

if (provider.alreadyConsidersBlueMarkets() || noOverlapBetweenCurrencies) {
exchangeRates.addAll(originalExchangeRates);
} else {
this.addRatesUpdatingBlueGaps(blueMarketGapForCurrency, provider, exchangeRates, originalExchangeRates);
}
return exchangeRates;
}

private void addRatesUpdatingBlueGaps(Map<String, Double> blueMarketGapForCurrency, ExchangeRateProvider provider, Set<ExchangeRate> exchangeRates, Set<ExchangeRate> originalExchangeRates) {
originalExchangeRates.forEach(er -> {
// default to original rate
ExchangeRate exchangeRateToUse = er;
if (blueMarketGapForCurrency.containsKey(er.getCurrency())) {
ExchangeRate updatedExchangeRate = provider.maybeUpdateBlueMarketPriceGapForRate(er, blueMarketGapForCurrency.get(er.getCurrency()));
if (updatedExchangeRate != null) {
exchangeRateToUse = updatedExchangeRate;
// this.log.info(String.format("Replaced original %s rate of $%s to $%s", er.getCurrency(), BigDecimal.valueOf(er.getPrice()).toEngineeringString(), BigDecimal.valueOf(updatedExchangeRate.getPrice()).toEngineeringString()));
}
}
exchangeRates.add(exchangeRateToUse);
});
}

private Map<String, Double> getBlueMarketGapForCurrencies() {
Map<String, Double> blueMarketGapForCurrencies = new HashMap<>();
// ARS
Double arsBlueMultiplier = BlueLyticsService.getInstance().blueGapMultiplier();
if (!arsBlueMultiplier.isNaN()) {
// this.log.info("Updated ARS/USD multiplier is " + arsBlueMultiplier);
blueMarketGapForCurrencies.put("ARS", arsBlueMultiplier);
}
return blueMarketGapForCurrencies;
}

/**
* For each currency, create an aggregate {@link ExchangeRate} based on the currency's
* rates from all providers. If multiple providers have rates for the currency, then
Expand Down Expand Up @@ -212,20 +163,43 @@ private double getOutlierStdDeviation() {
private Map<String, List<ExchangeRate>> getCurrencyCodeToExchangeRates() {
Map<String, List<ExchangeRate>> currencyCodeToExchangeRates = new HashMap<>();
for (ExchangeRateProvider p : providers) {
Set<ExchangeRate> exchangeRates = providerCurrentExchangeRates(p);
Set<ExchangeRate> exchangeRates = p.get();
if (exchangeRates == null)
continue;
for (ExchangeRate exchangeRate : exchangeRates) {
String currencyCode = exchangeRate.getCurrency();

List<ExchangeRate> transformedExchangeRates = transformers.stream()
.filter(transformer -> transformer.supportedCurrency()
.equalsIgnoreCase(currencyCode)
)
.map(t -> t.apply(p, exchangeRate))
.filter(Optional::isPresent)
.map(Optional::get)
.collect(Collectors.toList());

if (!transformedExchangeRates.isEmpty()) {
log.info(String.format("%s transformed from %s to %s", currencyCode, exchangeRate.getPrice(), transformedExchangeRates.get(0).getPrice()));
}

if (currencyCodeToExchangeRates.containsKey(currencyCode)) {
List<ExchangeRate> l = new ArrayList<>(currencyCodeToExchangeRates.get(currencyCode));
l.add(exchangeRate);
if (transformedExchangeRates.isEmpty()) {
l.add(exchangeRate);
} else {
l.addAll(transformedExchangeRates);
}
currencyCodeToExchangeRates.put(currencyCode, l);
} else {
currencyCodeToExchangeRates.put(currencyCode, List.of(exchangeRate));
if (transformedExchangeRates.isEmpty()) {
currencyCodeToExchangeRates.put(currencyCode, List.of(exchangeRate));
} else {
currencyCodeToExchangeRates.put(currencyCode, transformedExchangeRates);
}
}
}
}

return currencyCodeToExchangeRates;
}

Expand Down
31 changes: 31 additions & 0 deletions src/main/java/bisq/price/spot/ExchangeRateTransformer.java
Original file line number Diff line number Diff line change
@@ -0,0 +1,31 @@
/*
* This file is part of Bisq.
*
* Bisq is free software: you can redistribute it and/or modify it
* under the terms of the GNU Affero General Public License as published by
* the Free Software Foundation, either version 3 of the License, or (at
* your option) any later version.
*
* Bisq is distributed in the hope that it will be useful, but WITHOUT
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU Affero General Public
* License for more details.
*
* You should have received a copy of the GNU Affero General Public License
* along with Bisq. If not, see <http://www.gnu.org/licenses/>.
*/

package bisq.price.spot;

import java.util.Optional;

/**
* An ExchangeRateTransformer allows to apply a transformation on a particular ExchangeRate
* for particular supported currencies. This is useful for countries with currency controls
* that have a "blue" market in place for real/free trades.
*/
public interface ExchangeRateTransformer {
Optional<ExchangeRate> apply(ExchangeRateProvider provider, ExchangeRate exchangeRate);

String supportedCurrency();
}
24 changes: 24 additions & 0 deletions src/main/java/bisq/price/spot/providers/BlueRateProvider.java
Original file line number Diff line number Diff line change
@@ -0,0 +1,24 @@
/*
* This file is part of Bisq.
*
* Bisq is free software: you can redistribute it and/or modify it
* under the terms of the GNU Affero General Public License as published by
* the Free Software Foundation, either version 3 of the License, or (at
* your option) any later version.
*
* Bisq is distributed in the hope that it will be useful, but WITHOUT
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU Affero General Public
* License for more details.
*
* You should have received a copy of the GNU Affero General Public License
* along with Bisq. If not, see <http://www.gnu.org/licenses/>.
*/

package bisq.price.spot.providers;

/**
* Tag for providers that provide a "blue" market exchange rate (unofficial real/free traded)
*/
public interface BlueRateProvider {
}
2 changes: 1 addition & 1 deletion src/main/java/bisq/price/spot/providers/CryptoYa.java
Original file line number Diff line number Diff line change
Expand Up @@ -39,7 +39,7 @@
* This ExchangeRateProvider provides a real market rate (black or "blue") for ARS/BTC
*/
@Component
class CryptoYa extends ExchangeRateProvider {
class CryptoYa extends ExchangeRateProvider implements BlueRateProvider {

private static final String CRYPTO_YA_BTC_ARS_API_URL = "https://criptoya.com/api/btc/ars/0.1";

Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,56 @@
/*
* This file is part of Bisq.
*
* Bisq is free software: you can redistribute it and/or modify it
* under the terms of the GNU Affero General Public License as published by
* the Free Software Foundation, either version 3 of the License, or (at
* your option) any later version.
*
* Bisq is distributed in the hope that it will be useful, but WITHOUT
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU Affero General Public
* License for more details.
*
* You should have received a copy of the GNU Affero General Public License
* along with Bisq. If not, see <http://www.gnu.org/licenses/>.
*/

package bisq.price.util.bluelytics;

import bisq.price.PriceProvider;
import lombok.extern.slf4j.Slf4j;
import org.springframework.stereotype.Component;

import java.time.Duration;
import java.util.Optional;
import java.util.OptionalDouble;

@Slf4j
@Component
public class ArsBlueMarketGapProvider extends PriceProvider<OptionalDouble> {
public interface Listener {
void onUpdate(OptionalDouble sellGapMultiplier);
}

private static final Duration REFRESH_INTERVAL = Duration.ofHours(1);

private final BlueLyticsApi blueLyticsApi = new BlueLyticsApi();
private final Optional<Listener> onUpdateListener;

public ArsBlueMarketGapProvider() {
super(REFRESH_INTERVAL);
this.onUpdateListener = Optional.empty();
}

public ArsBlueMarketGapProvider(Listener onUpdateListener) {
super(REFRESH_INTERVAL);
this.onUpdateListener = Optional.of(onUpdateListener);
}

@Override
protected OptionalDouble doGet() {
OptionalDouble sellGapMultiplier = blueLyticsApi.getSellGapMultiplier();
onUpdateListener.ifPresent(listener -> listener.onUpdate(sellGapMultiplier));
return sellGapMultiplier;
}
}
Loading

0 comments on commit d67f1e2

Please sign in to comment.