mirror of
https://github.com/haveno-dex/haveno.git
synced 2025-06-28 08:37:16 -04:00
Introduce OfferRepository interface and TomP2P impl
This change begins the process of breaking up the monolithic MessageFacade abstraction into smaller, cohesive Repositories (in the Domain-Driven Design sense of the word) that abstract callers away from networking details. It also begins the process of restructuring the msg.listeners package, such that individual *Listener interfaces are co-located with their respective Repositories
This commit is contained in:
parent
ef68e08b50
commit
40807b07f8
10 changed files with 395 additions and 312 deletions
|
@ -23,6 +23,7 @@ import io.bitsquare.crypto.CryptoModule;
|
||||||
import io.bitsquare.gui.GuiModule;
|
import io.bitsquare.gui.GuiModule;
|
||||||
import io.bitsquare.msg.DefaultMessageModule;
|
import io.bitsquare.msg.DefaultMessageModule;
|
||||||
import io.bitsquare.msg.MessageModule;
|
import io.bitsquare.msg.MessageModule;
|
||||||
|
import io.bitsquare.offer.OfferModule;
|
||||||
import io.bitsquare.persistence.Persistence;
|
import io.bitsquare.persistence.Persistence;
|
||||||
import io.bitsquare.settings.Settings;
|
import io.bitsquare.settings.Settings;
|
||||||
import io.bitsquare.trade.TradeModule;
|
import io.bitsquare.trade.TradeModule;
|
||||||
|
@ -70,6 +71,7 @@ public class BitsquareModule extends AbstractBitsquareModule {
|
||||||
install(bitcoinModule());
|
install(bitcoinModule());
|
||||||
install(cryptoModule());
|
install(cryptoModule());
|
||||||
install(tradeModule());
|
install(tradeModule());
|
||||||
|
install(offerModule());
|
||||||
install(guiModule());
|
install(guiModule());
|
||||||
|
|
||||||
bindConstant().annotatedWith(Names.named("appName")).to(appName);
|
bindConstant().annotatedWith(Names.named("appName")).to(appName);
|
||||||
|
@ -95,6 +97,10 @@ public class BitsquareModule extends AbstractBitsquareModule {
|
||||||
return new TradeModule(properties);
|
return new TradeModule(properties);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
protected OfferModule offerModule() {
|
||||||
|
return new OfferModule(properties);
|
||||||
|
}
|
||||||
|
|
||||||
protected GuiModule guiModule() {
|
protected GuiModule guiModule() {
|
||||||
return new GuiModule(properties, primaryStage);
|
return new GuiModule(properties, primaryStage);
|
||||||
}
|
}
|
||||||
|
|
|
@ -20,9 +20,8 @@ package io.bitsquare.gui.main.trade.offerbook;
|
||||||
import io.bitsquare.bank.BankAccount;
|
import io.bitsquare.bank.BankAccount;
|
||||||
import io.bitsquare.locale.Country;
|
import io.bitsquare.locale.Country;
|
||||||
import io.bitsquare.locale.CurrencyUtil;
|
import io.bitsquare.locale.CurrencyUtil;
|
||||||
import io.bitsquare.msg.MessageFacade;
|
|
||||||
import io.bitsquare.msg.listeners.OfferBookListener;
|
|
||||||
import io.bitsquare.offer.Offer;
|
import io.bitsquare.offer.Offer;
|
||||||
|
import io.bitsquare.offer.OfferRepository;
|
||||||
import io.bitsquare.user.User;
|
import io.bitsquare.user.User;
|
||||||
import io.bitsquare.util.Utilities;
|
import io.bitsquare.util.Utilities;
|
||||||
|
|
||||||
|
@ -44,18 +43,18 @@ import static com.google.common.base.Preconditions.checkArgument;
|
||||||
* Holds and manages the unsorted and unfiltered offerbook list of both buy and sell offers.
|
* Holds and manages the unsorted and unfiltered offerbook list of both buy and sell offers.
|
||||||
* It is handled as singleton by Guice and is used by 2 instances of OfferBookModel (one for Buy one for Sell).
|
* It is handled as singleton by Guice and is used by 2 instances of OfferBookModel (one for Buy one for Sell).
|
||||||
* As it is used only by the Buy and Sell UIs we treat it as local UI model.
|
* As it is used only by the Buy and Sell UIs we treat it as local UI model.
|
||||||
* It also use OfferBookListener as the lists items class and we don't want to get any dependency out of the package
|
* It also use OfferRepository.Listener as the lists items class and we don't want to get any dependency out of the
|
||||||
* for that.
|
* package for that.
|
||||||
*/
|
*/
|
||||||
public class OfferBook {
|
public class OfferBook {
|
||||||
|
|
||||||
private static final Logger log = LoggerFactory.getLogger(OfferBook.class);
|
private static final Logger log = LoggerFactory.getLogger(OfferBook.class);
|
||||||
|
|
||||||
private final MessageFacade messageFacade;
|
private final OfferRepository offerRepository;
|
||||||
private final User user;
|
private final User user;
|
||||||
|
|
||||||
private final ObservableList<OfferBookListItem> offerBookListItems = FXCollections.observableArrayList();
|
private final ObservableList<OfferBookListItem> offerBookListItems = FXCollections.observableArrayList();
|
||||||
private final OfferBookListener offerBookListener;
|
private final OfferRepository.Listener offerRepositoryListener;
|
||||||
private final ChangeListener<BankAccount> bankAccountChangeListener;
|
private final ChangeListener<BankAccount> bankAccountChangeListener;
|
||||||
private final ChangeListener<Number> invalidationListener;
|
private final ChangeListener<Number> invalidationListener;
|
||||||
private String fiatCode;
|
private String fiatCode;
|
||||||
|
@ -69,14 +68,14 @@ public class OfferBook {
|
||||||
///////////////////////////////////////////////////////////////////////////////////////////
|
///////////////////////////////////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
@Inject
|
@Inject
|
||||||
OfferBook(MessageFacade messageFacade, User user) {
|
OfferBook(OfferRepository offerRepository, User user) {
|
||||||
this.messageFacade = messageFacade;
|
this.offerRepository = offerRepository;
|
||||||
this.user = user;
|
this.user = user;
|
||||||
|
|
||||||
bankAccountChangeListener = (observableValue, oldValue, newValue) -> setBankAccount(newValue);
|
bankAccountChangeListener = (observableValue, oldValue, newValue) -> setBankAccount(newValue);
|
||||||
invalidationListener = (ov, oldValue, newValue) -> requestOffers();
|
invalidationListener = (ov, oldValue, newValue) -> requestOffers();
|
||||||
|
|
||||||
offerBookListener = new OfferBookListener() {
|
offerRepositoryListener = new OfferRepository.Listener() {
|
||||||
@Override
|
@Override
|
||||||
public void onOfferAdded(Offer offer) {
|
public void onOfferAdded(Offer offer) {
|
||||||
addOfferToOfferBookListItems(offer);
|
addOfferToOfferBookListItems(offer);
|
||||||
|
@ -143,15 +142,15 @@ public class OfferBook {
|
||||||
private void addListeners() {
|
private void addListeners() {
|
||||||
log.debug("addListeners ");
|
log.debug("addListeners ");
|
||||||
user.currentBankAccountProperty().addListener(bankAccountChangeListener);
|
user.currentBankAccountProperty().addListener(bankAccountChangeListener);
|
||||||
messageFacade.addOfferBookListener(offerBookListener);
|
offerRepository.addListener(offerRepositoryListener);
|
||||||
messageFacade.invalidationTimestampProperty().addListener(invalidationListener);
|
offerRepository.invalidationTimestampProperty().addListener(invalidationListener);
|
||||||
}
|
}
|
||||||
|
|
||||||
private void removeListeners() {
|
private void removeListeners() {
|
||||||
log.debug("removeListeners ");
|
log.debug("removeListeners ");
|
||||||
user.currentBankAccountProperty().removeListener(bankAccountChangeListener);
|
user.currentBankAccountProperty().removeListener(bankAccountChangeListener);
|
||||||
messageFacade.removeOfferBookListener(offerBookListener);
|
offerRepository.removeListener(offerRepositoryListener);
|
||||||
messageFacade.invalidationTimestampProperty().removeListener(invalidationListener);
|
offerRepository.invalidationTimestampProperty().removeListener(invalidationListener);
|
||||||
}
|
}
|
||||||
|
|
||||||
private void addOfferToOfferBookListItems(Offer offer) {
|
private void addOfferToOfferBookListItems(Offer offer) {
|
||||||
|
@ -161,7 +160,7 @@ public class OfferBook {
|
||||||
}
|
}
|
||||||
|
|
||||||
private void requestOffers() {
|
private void requestOffers() {
|
||||||
messageFacade.getOffers(fiatCode);
|
offerRepository.getOffers(fiatCode);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@ -174,11 +173,11 @@ public class OfferBook {
|
||||||
addListeners();
|
addListeners();
|
||||||
setBankAccount(user.getCurrentBankAccount());
|
setBankAccount(user.getCurrentBankAccount());
|
||||||
pollingTimer = Utilities.setInterval(1000, (animationTimer) -> {
|
pollingTimer = Utilities.setInterval(1000, (animationTimer) -> {
|
||||||
messageFacade.requestInvalidationTimeStampFromDHT(fiatCode);
|
offerRepository.requestInvalidationTimeStampFromDHT(fiatCode);
|
||||||
return null;
|
return null;
|
||||||
});
|
});
|
||||||
|
|
||||||
messageFacade.getOffers(fiatCode);
|
offerRepository.getOffers(fiatCode);
|
||||||
}
|
}
|
||||||
|
|
||||||
private void stopPolling() {
|
private void stopPolling() {
|
||||||
|
|
|
@ -18,22 +18,17 @@
|
||||||
package io.bitsquare.msg;
|
package io.bitsquare.msg;
|
||||||
|
|
||||||
import io.bitsquare.arbitrator.Arbitrator;
|
import io.bitsquare.arbitrator.Arbitrator;
|
||||||
import io.bitsquare.msg.listeners.AddOfferListener;
|
|
||||||
import io.bitsquare.msg.listeners.ArbitratorListener;
|
import io.bitsquare.msg.listeners.ArbitratorListener;
|
||||||
import io.bitsquare.msg.listeners.BootstrapListener;
|
import io.bitsquare.msg.listeners.BootstrapListener;
|
||||||
import io.bitsquare.msg.listeners.GetPeerAddressListener;
|
import io.bitsquare.msg.listeners.GetPeerAddressListener;
|
||||||
import io.bitsquare.msg.listeners.IncomingMessageListener;
|
import io.bitsquare.msg.listeners.IncomingMessageListener;
|
||||||
import io.bitsquare.msg.listeners.OfferBookListener;
|
|
||||||
import io.bitsquare.msg.listeners.OutgoingMessageListener;
|
import io.bitsquare.msg.listeners.OutgoingMessageListener;
|
||||||
import io.bitsquare.network.Peer;
|
import io.bitsquare.network.Peer;
|
||||||
import io.bitsquare.offer.Offer;
|
|
||||||
|
|
||||||
import java.security.PublicKey;
|
import java.security.PublicKey;
|
||||||
|
|
||||||
import java.util.Locale;
|
import java.util.Locale;
|
||||||
|
|
||||||
import javafx.beans.property.LongProperty;
|
|
||||||
|
|
||||||
public interface MessageFacade extends MessageBroker {
|
public interface MessageFacade extends MessageBroker {
|
||||||
|
|
||||||
void sendMessage(Peer peer, Message message, OutgoingMessageListener listener);
|
void sendMessage(Peer peer, Message message, OutgoingMessageListener listener);
|
||||||
|
@ -46,25 +41,11 @@ public interface MessageFacade extends MessageBroker {
|
||||||
|
|
||||||
void removeIncomingMessageListener(IncomingMessageListener listener);
|
void removeIncomingMessageListener(IncomingMessageListener listener);
|
||||||
|
|
||||||
void addOffer(Offer offer, AddOfferListener addOfferListener);
|
|
||||||
|
|
||||||
void addArbitratorListener(ArbitratorListener listener);
|
void addArbitratorListener(ArbitratorListener listener);
|
||||||
|
|
||||||
void getArbitrators(Locale defaultLanguageLocale);
|
void getArbitrators(Locale defaultLanguageLocale);
|
||||||
|
|
||||||
LongProperty invalidationTimestampProperty();
|
|
||||||
|
|
||||||
void addOfferBookListener(OfferBookListener offerBookListener);
|
|
||||||
|
|
||||||
void requestInvalidationTimeStampFromDHT(String fiatCode);
|
|
||||||
|
|
||||||
void getOffers(String fiatCode);
|
|
||||||
|
|
||||||
void removeOffer(Offer offer);
|
|
||||||
|
|
||||||
void init(int clientPort, BootstrapListener bootstrapListener);
|
void init(int clientPort, BootstrapListener bootstrapListener);
|
||||||
|
|
||||||
void getPeerAddress(PublicKey messagePublicKey, GetPeerAddressListener getPeerAddressListener);
|
void getPeerAddress(PublicKey messagePublicKey, GetPeerAddressListener getPeerAddressListener);
|
||||||
|
|
||||||
void removeOfferBookListener(OfferBookListener offerBookListener);
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -18,16 +18,13 @@
|
||||||
package io.bitsquare.msg;
|
package io.bitsquare.msg;
|
||||||
|
|
||||||
import io.bitsquare.arbitrator.Arbitrator;
|
import io.bitsquare.arbitrator.Arbitrator;
|
||||||
import io.bitsquare.msg.listeners.AddOfferListener;
|
|
||||||
import io.bitsquare.msg.listeners.ArbitratorListener;
|
import io.bitsquare.msg.listeners.ArbitratorListener;
|
||||||
import io.bitsquare.msg.listeners.BootstrapListener;
|
import io.bitsquare.msg.listeners.BootstrapListener;
|
||||||
import io.bitsquare.msg.listeners.GetPeerAddressListener;
|
import io.bitsquare.msg.listeners.GetPeerAddressListener;
|
||||||
import io.bitsquare.msg.listeners.IncomingMessageListener;
|
import io.bitsquare.msg.listeners.IncomingMessageListener;
|
||||||
import io.bitsquare.msg.listeners.OfferBookListener;
|
|
||||||
import io.bitsquare.msg.listeners.OutgoingMessageListener;
|
import io.bitsquare.msg.listeners.OutgoingMessageListener;
|
||||||
import io.bitsquare.network.Peer;
|
import io.bitsquare.network.Peer;
|
||||||
import io.bitsquare.network.tomp2p.TomP2PPeer;
|
import io.bitsquare.network.tomp2p.TomP2PPeer;
|
||||||
import io.bitsquare.offer.Offer;
|
|
||||||
import io.bitsquare.user.User;
|
import io.bitsquare.user.User;
|
||||||
|
|
||||||
import com.google.common.util.concurrent.FutureCallback;
|
import com.google.common.util.concurrent.FutureCallback;
|
||||||
|
@ -39,15 +36,12 @@ import java.security.PublicKey;
|
||||||
import java.util.ArrayList;
|
import java.util.ArrayList;
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
import java.util.Locale;
|
import java.util.Locale;
|
||||||
import java.util.Map;
|
|
||||||
|
|
||||||
import javax.annotation.Nullable;
|
import javax.annotation.Nullable;
|
||||||
|
|
||||||
import javax.inject.Inject;
|
import javax.inject.Inject;
|
||||||
|
|
||||||
import javafx.application.Platform;
|
import javafx.application.Platform;
|
||||||
import javafx.beans.property.LongProperty;
|
|
||||||
import javafx.beans.property.SimpleLongProperty;
|
|
||||||
|
|
||||||
import net.tomp2p.dht.FutureGet;
|
import net.tomp2p.dht.FutureGet;
|
||||||
import net.tomp2p.dht.FuturePut;
|
import net.tomp2p.dht.FuturePut;
|
||||||
|
@ -58,7 +52,6 @@ import net.tomp2p.futures.BaseFutureAdapter;
|
||||||
import net.tomp2p.futures.BaseFutureListener;
|
import net.tomp2p.futures.BaseFutureListener;
|
||||||
import net.tomp2p.futures.FutureDirect;
|
import net.tomp2p.futures.FutureDirect;
|
||||||
import net.tomp2p.peers.Number160;
|
import net.tomp2p.peers.Number160;
|
||||||
import net.tomp2p.peers.Number640;
|
|
||||||
import net.tomp2p.storage.Data;
|
import net.tomp2p.storage.Data;
|
||||||
import net.tomp2p.utils.Utils;
|
import net.tomp2p.utils.Utils;
|
||||||
|
|
||||||
|
@ -82,10 +75,8 @@ class TomP2PMessageFacade implements MessageFacade {
|
||||||
private final P2PNode p2pNode;
|
private final P2PNode p2pNode;
|
||||||
private final User user;
|
private final User user;
|
||||||
|
|
||||||
private final List<OfferBookListener> offerBookListeners = new ArrayList<>();
|
|
||||||
private final List<ArbitratorListener> arbitratorListeners = new ArrayList<>();
|
private final List<ArbitratorListener> arbitratorListeners = new ArrayList<>();
|
||||||
private final List<IncomingMessageListener> incomingMessageListeners = new ArrayList<>();
|
private final List<IncomingMessageListener> incomingMessageListeners = new ArrayList<>();
|
||||||
private final LongProperty invalidationTimestamp = new SimpleLongProperty(0);
|
|
||||||
|
|
||||||
|
|
||||||
///////////////////////////////////////////////////////////////////////////////////////////
|
///////////////////////////////////////////////////////////////////////////////////////////
|
||||||
|
@ -152,166 +143,6 @@ class TomP2PMessageFacade implements MessageFacade {
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
///////////////////////////////////////////////////////////////////////////////////////////
|
|
||||||
// Offer
|
|
||||||
///////////////////////////////////////////////////////////////////////////////////////////
|
|
||||||
|
|
||||||
public void addOffer(Offer offer, AddOfferListener addOfferListener) {
|
|
||||||
Number160 locationKey = Number160.createHash(offer.getCurrency().getCurrencyCode());
|
|
||||||
try {
|
|
||||||
final Data offerData = new Data(offer);
|
|
||||||
|
|
||||||
// the offer is default 30 days valid
|
|
||||||
int defaultOfferTTL = 30 * 24 * 60 * 60;
|
|
||||||
offerData.ttlSeconds(defaultOfferTTL);
|
|
||||||
log.trace("Add offer to DHT requested. Added data: [locationKey: " + locationKey +
|
|
||||||
", hash: " + offerData.hash().toString() + "]");
|
|
||||||
FuturePut futurePut = p2pNode.addProtectedData(locationKey, offerData);
|
|
||||||
futurePut.addListener(new BaseFutureListener<BaseFuture>() {
|
|
||||||
@Override
|
|
||||||
public void operationComplete(BaseFuture future) throws Exception {
|
|
||||||
// deactivate it for the moment until the port forwarding bug is fixed
|
|
||||||
// if (future.isSuccess()) {
|
|
||||||
Platform.runLater(() -> {
|
|
||||||
addOfferListener.onComplete();
|
|
||||||
offerBookListeners.stream().forEach(listener -> {
|
|
||||||
try {
|
|
||||||
Object offerDataObject = offerData.object();
|
|
||||||
if (offerDataObject instanceof Offer) {
|
|
||||||
log.error("Added offer to DHT with ID: " + ((Offer) offerDataObject).getId());
|
|
||||||
listener.onOfferAdded((Offer) offerDataObject);
|
|
||||||
}
|
|
||||||
} catch (ClassNotFoundException | IOException e) {
|
|
||||||
e.printStackTrace();
|
|
||||||
log.error("Add offer to DHT failed: " + e.getMessage());
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
// TODO will be removed when we don't use polling anymore
|
|
||||||
writeInvalidationTimestampToDHT(locationKey);
|
|
||||||
log.trace("Add offer to DHT was successful. Added data: [locationKey: " + locationKey +
|
|
||||||
", value: " + offerData + "]");
|
|
||||||
});
|
|
||||||
/* }
|
|
||||||
else {
|
|
||||||
Platform.runLater(() -> {
|
|
||||||
addOfferListener.onFailed("Add offer to DHT failed.",
|
|
||||||
new Exception("Add offer to DHT failed. Reason: " + future.failedReason()));
|
|
||||||
log.error("Add offer to DHT failed. Reason: " + future.failedReason());
|
|
||||||
});
|
|
||||||
}*/
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void exceptionCaught(Throwable t) throws Exception {
|
|
||||||
Platform.runLater(() -> {
|
|
||||||
addOfferListener.onFailed("Add offer to DHT failed with an exception.", t);
|
|
||||||
log.error("Add offer to DHT failed with an exception: " + t.getMessage());
|
|
||||||
});
|
|
||||||
}
|
|
||||||
});
|
|
||||||
} catch (IOException e) {
|
|
||||||
Platform.runLater(() -> {
|
|
||||||
addOfferListener.onFailed("Add offer to DHT failed with an exception.", e);
|
|
||||||
log.error("Add offer to DHT failed with an exception: " + e.getMessage());
|
|
||||||
});
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
//TODO remove is failing, probably due Coin or Fiat class (was working before)
|
|
||||||
// objects are identical but returned object form network might have some problem with serialisation?
|
|
||||||
public void removeOffer(Offer offer) {
|
|
||||||
Number160 locationKey = Number160.createHash(offer.getCurrency().getCurrencyCode());
|
|
||||||
try {
|
|
||||||
final Data offerData = new Data(offer);
|
|
||||||
log.trace("Remove offer from DHT requested. Removed data: [locationKey: " + locationKey +
|
|
||||||
", hash: " + offerData.hash().toString() + "]");
|
|
||||||
FutureRemove futureRemove = p2pNode.removeFromDataMap(locationKey, offerData);
|
|
||||||
futureRemove.addListener(new BaseFutureListener<BaseFuture>() {
|
|
||||||
@Override
|
|
||||||
public void operationComplete(BaseFuture future) throws Exception {
|
|
||||||
// deactivate it for the moment until the port forwarding bug is fixed
|
|
||||||
// if (future.isSuccess()) {
|
|
||||||
Platform.runLater(() -> {
|
|
||||||
offerBookListeners.stream().forEach(offerBookListener -> {
|
|
||||||
try {
|
|
||||||
Object offerDataObject = offerData.object();
|
|
||||||
if (offerDataObject instanceof Offer) {
|
|
||||||
log.trace("Remove offer from DHT was successful. Removed data: [key: " +
|
|
||||||
locationKey + ", " +
|
|
||||||
"offer: " + (Offer) offerDataObject + "]");
|
|
||||||
offerBookListener.onOfferRemoved((Offer) offerDataObject);
|
|
||||||
}
|
|
||||||
} catch (ClassNotFoundException | IOException e) {
|
|
||||||
e.printStackTrace();
|
|
||||||
log.error("Remove offer from DHT failed. Error: " + e.getMessage());
|
|
||||||
}
|
|
||||||
});
|
|
||||||
writeInvalidationTimestampToDHT(locationKey);
|
|
||||||
});
|
|
||||||
/* }
|
|
||||||
else {
|
|
||||||
log.error("Remove offer from DHT failed. Cause: future.isSuccess() = false, locationKey: " +
|
|
||||||
locationKey + ", Reason: " + future.failedReason());
|
|
||||||
}*/
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void exceptionCaught(Throwable t) throws Exception {
|
|
||||||
log.error("Remove offer from DHT failed. Error: " + t.getMessage());
|
|
||||||
}
|
|
||||||
});
|
|
||||||
} catch (IOException e) {
|
|
||||||
e.printStackTrace();
|
|
||||||
log.error("Remove offer from DHT failed. Error: " + e.getMessage());
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public void getOffers(String currencyCode) {
|
|
||||||
Number160 locationKey = Number160.createHash(currencyCode);
|
|
||||||
log.trace("Get offers from DHT requested for locationKey: " + locationKey);
|
|
||||||
FutureGet futureGet = p2pNode.getDataMap(locationKey);
|
|
||||||
futureGet.addListener(new BaseFutureAdapter<BaseFuture>() {
|
|
||||||
@Override
|
|
||||||
public void operationComplete(BaseFuture baseFuture) throws Exception {
|
|
||||||
if (baseFuture.isSuccess()) {
|
|
||||||
final Map<Number640, Data> dataMap = futureGet.dataMap();
|
|
||||||
final List<Offer> offers = new ArrayList<>();
|
|
||||||
if (dataMap != null) {
|
|
||||||
for (Data offerData : dataMap.values()) {
|
|
||||||
try {
|
|
||||||
Object offerDataObject = offerData.object();
|
|
||||||
if (offerDataObject instanceof Offer) {
|
|
||||||
offers.add((Offer) offerDataObject);
|
|
||||||
}
|
|
||||||
} catch (ClassNotFoundException | IOException e) {
|
|
||||||
e.printStackTrace();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
Platform.runLater(() -> offerBookListeners.stream().forEach(listener ->
|
|
||||||
listener.onOffersReceived(offers)));
|
|
||||||
}
|
|
||||||
|
|
||||||
log.trace("Get offers from DHT was successful. Stored data: [key: " + locationKey
|
|
||||||
+ ", values: " + futureGet.dataMap() + "]");
|
|
||||||
}
|
|
||||||
else {
|
|
||||||
final Map<Number640, Data> dataMap = futureGet.dataMap();
|
|
||||||
if (dataMap == null || dataMap.size() == 0) {
|
|
||||||
log.trace("Get offers from DHT delivered empty dataMap.");
|
|
||||||
Platform.runLater(() -> offerBookListeners.stream().forEach(listener ->
|
|
||||||
listener.onOffersReceived(new ArrayList<>())));
|
|
||||||
}
|
|
||||||
else {
|
|
||||||
log.error("Get offers from DHT was not successful with reason:" + baseFuture.failedReason());
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
///////////////////////////////////////////////////////////////////////////////////////////
|
///////////////////////////////////////////////////////////////////////////////////////////
|
||||||
// Trade process
|
// Trade process
|
||||||
///////////////////////////////////////////////////////////////////////////////////////////
|
///////////////////////////////////////////////////////////////////////////////////////////
|
||||||
|
@ -452,14 +283,6 @@ class TomP2PMessageFacade implements MessageFacade {
|
||||||
// Event Listeners
|
// Event Listeners
|
||||||
///////////////////////////////////////////////////////////////////////////////////////////
|
///////////////////////////////////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
public void addOfferBookListener(OfferBookListener listener) {
|
|
||||||
offerBookListeners.add(listener);
|
|
||||||
}
|
|
||||||
|
|
||||||
public void removeOfferBookListener(OfferBookListener listener) {
|
|
||||||
offerBookListeners.remove(listener);
|
|
||||||
}
|
|
||||||
|
|
||||||
public void addArbitratorListener(ArbitratorListener listener) {
|
public void addArbitratorListener(ArbitratorListener listener) {
|
||||||
arbitratorListeners.add(listener);
|
arbitratorListeners.add(listener);
|
||||||
}
|
}
|
||||||
|
@ -477,87 +300,6 @@ class TomP2PMessageFacade implements MessageFacade {
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
/*
|
|
||||||
* We store the timestamp of any change of the offer list (add, remove offer) and we poll in intervals for changes.
|
|
||||||
* If we detect a change we request the offer list from the DHT.
|
|
||||||
* Polling should be replaced by a push based solution later.
|
|
||||||
*/
|
|
||||||
|
|
||||||
///////////////////////////////////////////////////////////////////////////////////////////
|
|
||||||
// Polling
|
|
||||||
///////////////////////////////////////////////////////////////////////////////////////////
|
|
||||||
|
|
||||||
private void writeInvalidationTimestampToDHT(Number160 locationKey) {
|
|
||||||
invalidationTimestamp.set(System.currentTimeMillis());
|
|
||||||
try {
|
|
||||||
FuturePut putFuture = p2pNode.putData(getInvalidatedLocationKey(locationKey),
|
|
||||||
new Data(invalidationTimestamp.get()));
|
|
||||||
putFuture.addListener(new BaseFutureListener<BaseFuture>() {
|
|
||||||
@Override
|
|
||||||
public void operationComplete(BaseFuture future) throws Exception {
|
|
||||||
if (putFuture.isSuccess())
|
|
||||||
log.trace("Update invalidationTimestamp to DHT was successful. TimeStamp=" +
|
|
||||||
invalidationTimestamp.get());
|
|
||||||
else
|
|
||||||
log.error("Update invalidationTimestamp to DHT failed with reason:" + putFuture.failedReason());
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void exceptionCaught(Throwable t) throws Exception {
|
|
||||||
log.error("Update invalidationTimestamp to DHT failed with exception:" + t.getMessage());
|
|
||||||
}
|
|
||||||
});
|
|
||||||
} catch (IOException e) {
|
|
||||||
log.error("Update invalidationTimestamp to DHT failed with exception:" + e.getMessage());
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public LongProperty invalidationTimestampProperty() {
|
|
||||||
return invalidationTimestamp;
|
|
||||||
}
|
|
||||||
|
|
||||||
public void requestInvalidationTimeStampFromDHT(String currencyCode) {
|
|
||||||
Number160 locationKey = Number160.createHash(currencyCode);
|
|
||||||
FutureGet getFuture = p2pNode.getData(getInvalidatedLocationKey(locationKey));
|
|
||||||
getFuture.addListener(new BaseFutureListener<BaseFuture>() {
|
|
||||||
@Override
|
|
||||||
public void operationComplete(BaseFuture future) throws Exception {
|
|
||||||
if (getFuture.isSuccess()) {
|
|
||||||
Data data = getFuture.data();
|
|
||||||
if (data != null && data.object() instanceof Long) {
|
|
||||||
final Object object = data.object();
|
|
||||||
Platform.runLater(() -> {
|
|
||||||
Long timeStamp = (Long) object;
|
|
||||||
//log.trace("Get invalidationTimestamp from DHT was successful. TimeStamp=" + timeStamp);
|
|
||||||
invalidationTimestamp.set(timeStamp);
|
|
||||||
});
|
|
||||||
}
|
|
||||||
else {
|
|
||||||
//log.error("Get invalidationTimestamp from DHT failed. Data = " + data);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
else if (getFuture.data() == null) {
|
|
||||||
// OK as nothing is set at the moment
|
|
||||||
// log.trace("Get invalidationTimestamp from DHT returns null. That is ok for the startup.");
|
|
||||||
}
|
|
||||||
else {
|
|
||||||
log.error("Get invalidationTimestamp from DHT failed with reason:" + getFuture.failedReason());
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void exceptionCaught(Throwable t) throws Exception {
|
|
||||||
log.error("Get invalidationTimestamp from DHT failed with exception:" + t.getMessage());
|
|
||||||
t.printStackTrace();
|
|
||||||
}
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
private Number160 getInvalidatedLocationKey(Number160 locationKey) {
|
|
||||||
return Number160.createHash(locationKey + "invalidated");
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
///////////////////////////////////////////////////////////////////////////////////////////
|
///////////////////////////////////////////////////////////////////////////////////////////
|
||||||
// Incoming message handler
|
// Incoming message handler
|
||||||
///////////////////////////////////////////////////////////////////////////////////////////
|
///////////////////////////////////////////////////////////////////////////////////////////
|
||||||
|
|
|
@ -15,16 +15,20 @@
|
||||||
* along with Bitsquare. If not, see <http://www.gnu.org/licenses/>.
|
* along with Bitsquare. If not, see <http://www.gnu.org/licenses/>.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
package io.bitsquare.msg.listeners;
|
package io.bitsquare.offer;
|
||||||
|
|
||||||
import io.bitsquare.offer.Offer;
|
import io.bitsquare.AbstractBitsquareModule;
|
||||||
|
|
||||||
import java.util.List;
|
import java.util.Properties;
|
||||||
|
|
||||||
public interface OfferBookListener {
|
public class OfferModule extends AbstractBitsquareModule {
|
||||||
void onOfferAdded(Offer offer);
|
|
||||||
|
|
||||||
void onOffersReceived(List<Offer> offers);
|
public OfferModule(Properties properties) {
|
||||||
|
super(properties);
|
||||||
|
}
|
||||||
|
|
||||||
void onOfferRemoved(Offer offer);
|
@Override
|
||||||
}
|
protected void configure() {
|
||||||
|
bind(OfferRepository.class).to(TomP2POfferRepository.class).asEagerSingleton();
|
||||||
|
}
|
||||||
|
}
|
49
src/main/java/io/bitsquare/offer/OfferRepository.java
Normal file
49
src/main/java/io/bitsquare/offer/OfferRepository.java
Normal file
|
@ -0,0 +1,49 @@
|
||||||
|
/*
|
||||||
|
* This file is part of Bitsquare.
|
||||||
|
*
|
||||||
|
* Bitsquare 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.
|
||||||
|
*
|
||||||
|
* Bitsquare 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 Bitsquare. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
*/
|
||||||
|
|
||||||
|
package io.bitsquare.offer;
|
||||||
|
|
||||||
|
import io.bitsquare.msg.listeners.AddOfferListener;
|
||||||
|
|
||||||
|
import java.util.List;
|
||||||
|
|
||||||
|
import javafx.beans.property.LongProperty;
|
||||||
|
|
||||||
|
public interface OfferRepository {
|
||||||
|
|
||||||
|
void getOffers(String fiatCode);
|
||||||
|
|
||||||
|
void addOffer(Offer offer, AddOfferListener addOfferListener);
|
||||||
|
|
||||||
|
void removeOffer(Offer offer);
|
||||||
|
|
||||||
|
void addListener(Listener listener);
|
||||||
|
|
||||||
|
void removeListener(Listener listener);
|
||||||
|
|
||||||
|
LongProperty invalidationTimestampProperty();
|
||||||
|
|
||||||
|
void requestInvalidationTimeStampFromDHT(String fiatCode);
|
||||||
|
|
||||||
|
interface Listener {
|
||||||
|
void onOfferAdded(Offer offer);
|
||||||
|
|
||||||
|
void onOffersReceived(List<Offer> offers);
|
||||||
|
|
||||||
|
void onOfferRemoved(Offer offer);
|
||||||
|
}
|
||||||
|
}
|
294
src/main/java/io/bitsquare/offer/TomP2POfferRepository.java
Normal file
294
src/main/java/io/bitsquare/offer/TomP2POfferRepository.java
Normal file
|
@ -0,0 +1,294 @@
|
||||||
|
/*
|
||||||
|
* This file is part of Bitsquare.
|
||||||
|
*
|
||||||
|
* Bitsquare 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.
|
||||||
|
*
|
||||||
|
* Bitsquare 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 Bitsquare. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
*/
|
||||||
|
|
||||||
|
package io.bitsquare.offer;
|
||||||
|
|
||||||
|
import io.bitsquare.msg.P2PNode;
|
||||||
|
import io.bitsquare.msg.listeners.AddOfferListener;
|
||||||
|
|
||||||
|
import java.io.IOException;
|
||||||
|
|
||||||
|
import java.util.ArrayList;
|
||||||
|
import java.util.List;
|
||||||
|
import java.util.Map;
|
||||||
|
|
||||||
|
import javax.inject.Inject;
|
||||||
|
|
||||||
|
import javafx.application.Platform;
|
||||||
|
import javafx.beans.property.LongProperty;
|
||||||
|
import javafx.beans.property.SimpleLongProperty;
|
||||||
|
|
||||||
|
import net.tomp2p.dht.FutureGet;
|
||||||
|
import net.tomp2p.dht.FuturePut;
|
||||||
|
import net.tomp2p.dht.FutureRemove;
|
||||||
|
import net.tomp2p.futures.BaseFuture;
|
||||||
|
import net.tomp2p.futures.BaseFutureAdapter;
|
||||||
|
import net.tomp2p.futures.BaseFutureListener;
|
||||||
|
import net.tomp2p.peers.Number160;
|
||||||
|
import net.tomp2p.peers.Number640;
|
||||||
|
import net.tomp2p.storage.Data;
|
||||||
|
|
||||||
|
import org.slf4j.Logger;
|
||||||
|
import org.slf4j.LoggerFactory;
|
||||||
|
|
||||||
|
class TomP2POfferRepository implements OfferRepository {
|
||||||
|
|
||||||
|
private static final Logger log = LoggerFactory.getLogger(TomP2POfferRepository.class);
|
||||||
|
|
||||||
|
private final List<Listener> offerRepositoryListeners = new ArrayList<>();
|
||||||
|
private final LongProperty invalidationTimestamp = new SimpleLongProperty(0);
|
||||||
|
|
||||||
|
private final P2PNode p2pNode;
|
||||||
|
|
||||||
|
@Inject
|
||||||
|
public TomP2POfferRepository(P2PNode p2pNode) {
|
||||||
|
this.p2pNode = p2pNode;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void addOffer(Offer offer, AddOfferListener addOfferListener) {
|
||||||
|
Number160 locationKey = Number160.createHash(offer.getCurrency().getCurrencyCode());
|
||||||
|
try {
|
||||||
|
final Data offerData = new Data(offer);
|
||||||
|
|
||||||
|
// the offer is default 30 days valid
|
||||||
|
int defaultOfferTTL = 30 * 24 * 60 * 60;
|
||||||
|
offerData.ttlSeconds(defaultOfferTTL);
|
||||||
|
log.trace("Add offer to DHT requested. Added data: [locationKey: " + locationKey +
|
||||||
|
", hash: " + offerData.hash().toString() + "]");
|
||||||
|
FuturePut futurePut = p2pNode.addProtectedData(locationKey, offerData);
|
||||||
|
futurePut.addListener(new BaseFutureListener<BaseFuture>() {
|
||||||
|
@Override
|
||||||
|
public void operationComplete(BaseFuture future) throws Exception {
|
||||||
|
// deactivate it for the moment until the port forwarding bug is fixed
|
||||||
|
// if (future.isSuccess()) {
|
||||||
|
Platform.runLater(() -> {
|
||||||
|
addOfferListener.onComplete();
|
||||||
|
offerRepositoryListeners.stream().forEach(listener -> {
|
||||||
|
try {
|
||||||
|
Object offerDataObject = offerData.object();
|
||||||
|
if (offerDataObject instanceof Offer) {
|
||||||
|
log.error("Added offer to DHT with ID: " + ((Offer) offerDataObject).getId());
|
||||||
|
listener.onOfferAdded((Offer) offerDataObject);
|
||||||
|
}
|
||||||
|
} catch (ClassNotFoundException | IOException e) {
|
||||||
|
e.printStackTrace();
|
||||||
|
log.error("Add offer to DHT failed: " + e.getMessage());
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
writeInvalidationTimestampToDHT(locationKey);
|
||||||
|
log.trace("Add offer to DHT was successful. Added data: [locationKey: " + locationKey +
|
||||||
|
", value: " + offerData + "]");
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void exceptionCaught(Throwable t) throws Exception {
|
||||||
|
Platform.runLater(() -> {
|
||||||
|
addOfferListener.onFailed("Add offer to DHT failed with an exception.", t);
|
||||||
|
log.error("Add offer to DHT failed with an exception: " + t.getMessage());
|
||||||
|
});
|
||||||
|
}
|
||||||
|
});
|
||||||
|
} catch (IOException e) {
|
||||||
|
Platform.runLater(() -> {
|
||||||
|
addOfferListener.onFailed("Add offer to DHT failed with an exception.", e);
|
||||||
|
log.error("Add offer to DHT failed with an exception: " + e.getMessage());
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
//TODO remove is failing, probably due Coin or Fiat class (was working before)
|
||||||
|
// objects are identical but returned object form network might have some problem with serialisation?
|
||||||
|
public void removeOffer(Offer offer) {
|
||||||
|
Number160 locationKey = Number160.createHash(offer.getCurrency().getCurrencyCode());
|
||||||
|
try {
|
||||||
|
final Data offerData = new Data(offer);
|
||||||
|
log.trace("Remove offer from DHT requested. Removed data: [locationKey: " + locationKey +
|
||||||
|
", hash: " + offerData.hash().toString() + "]");
|
||||||
|
FutureRemove futureRemove = p2pNode.removeFromDataMap(locationKey, offerData);
|
||||||
|
futureRemove.addListener(new BaseFutureListener<BaseFuture>() {
|
||||||
|
@Override
|
||||||
|
public void operationComplete(BaseFuture future) throws Exception {
|
||||||
|
// deactivate it for the moment until the port forwarding bug is fixed
|
||||||
|
// if (future.isSuccess()) {
|
||||||
|
Platform.runLater(() -> {
|
||||||
|
offerRepositoryListeners.stream().forEach(listener -> {
|
||||||
|
try {
|
||||||
|
Object offerDataObject = offerData.object();
|
||||||
|
if (offerDataObject instanceof Offer) {
|
||||||
|
log.trace("Remove offer from DHT was successful. Removed data: [key: " +
|
||||||
|
locationKey + ", " +
|
||||||
|
"offer: " + (Offer) offerDataObject + "]");
|
||||||
|
listener.onOfferRemoved((Offer) offerDataObject);
|
||||||
|
}
|
||||||
|
} catch (ClassNotFoundException | IOException e) {
|
||||||
|
e.printStackTrace();
|
||||||
|
log.error("Remove offer from DHT failed. Error: " + e.getMessage());
|
||||||
|
}
|
||||||
|
});
|
||||||
|
writeInvalidationTimestampToDHT(locationKey);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void exceptionCaught(Throwable t) throws Exception {
|
||||||
|
log.error("Remove offer from DHT failed. Error: " + t.getMessage());
|
||||||
|
}
|
||||||
|
});
|
||||||
|
} catch (IOException e) {
|
||||||
|
e.printStackTrace();
|
||||||
|
log.error("Remove offer from DHT failed. Error: " + e.getMessage());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public void getOffers(String currencyCode) {
|
||||||
|
Number160 locationKey = Number160.createHash(currencyCode);
|
||||||
|
log.trace("Get offers from DHT requested for locationKey: " + locationKey);
|
||||||
|
FutureGet futureGet = p2pNode.getDataMap(locationKey);
|
||||||
|
futureGet.addListener(new BaseFutureAdapter<BaseFuture>() {
|
||||||
|
@Override
|
||||||
|
public void operationComplete(BaseFuture baseFuture) throws Exception {
|
||||||
|
if (baseFuture.isSuccess()) {
|
||||||
|
final Map<Number640, Data> dataMap = futureGet.dataMap();
|
||||||
|
final List<Offer> offers = new ArrayList<>();
|
||||||
|
if (dataMap != null) {
|
||||||
|
for (Data offerData : dataMap.values()) {
|
||||||
|
try {
|
||||||
|
Object offerDataObject = offerData.object();
|
||||||
|
if (offerDataObject instanceof Offer) {
|
||||||
|
offers.add((Offer) offerDataObject);
|
||||||
|
}
|
||||||
|
} catch (ClassNotFoundException | IOException e) {
|
||||||
|
e.printStackTrace();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
Platform.runLater(() -> offerRepositoryListeners.stream().forEach(listener ->
|
||||||
|
listener.onOffersReceived(offers)));
|
||||||
|
}
|
||||||
|
|
||||||
|
log.trace("Get offers from DHT was successful. Stored data: [key: " + locationKey
|
||||||
|
+ ", values: " + futureGet.dataMap() + "]");
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
final Map<Number640, Data> dataMap = futureGet.dataMap();
|
||||||
|
if (dataMap == null || dataMap.size() == 0) {
|
||||||
|
log.trace("Get offers from DHT delivered empty dataMap.");
|
||||||
|
Platform.runLater(() -> offerRepositoryListeners.stream().forEach(listener ->
|
||||||
|
listener.onOffersReceived(new ArrayList<>())));
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
log.error("Get offers from DHT was not successful with reason:" + baseFuture.failedReason());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void addListener(Listener listener) {
|
||||||
|
offerRepositoryListeners.add(listener);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void removeListener(Listener listener) {
|
||||||
|
offerRepositoryListeners.remove(listener);
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* We store the timestamp of any change of the offer list (add, remove offer) and we poll
|
||||||
|
* in intervals for changes. If we detect a change we request the offer list from the DHT.
|
||||||
|
* Polling should be replaced by a push based solution later.
|
||||||
|
*/
|
||||||
|
|
||||||
|
///////////////////////////////////////////////////////////////////////////////////////////
|
||||||
|
// Polling
|
||||||
|
///////////////////////////////////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
|
private void writeInvalidationTimestampToDHT(Number160 locationKey) {
|
||||||
|
invalidationTimestamp.set(System.currentTimeMillis());
|
||||||
|
try {
|
||||||
|
FuturePut putFuture = p2pNode.putData(getInvalidatedLocationKey(locationKey),
|
||||||
|
new Data(invalidationTimestamp.get()));
|
||||||
|
putFuture.addListener(new BaseFutureListener<BaseFuture>() {
|
||||||
|
@Override
|
||||||
|
public void operationComplete(BaseFuture future) throws Exception {
|
||||||
|
if (putFuture.isSuccess())
|
||||||
|
log.trace("Update invalidationTimestamp to DHT was successful. TimeStamp=" +
|
||||||
|
invalidationTimestamp.get());
|
||||||
|
else
|
||||||
|
log.error("Update invalidationTimestamp to DHT failed with reason:" + putFuture.failedReason());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void exceptionCaught(Throwable t) throws Exception {
|
||||||
|
log.error("Update invalidationTimestamp to DHT failed with exception:" + t.getMessage());
|
||||||
|
}
|
||||||
|
});
|
||||||
|
} catch (IOException e) {
|
||||||
|
log.error("Update invalidationTimestamp to DHT failed with exception:" + e.getMessage());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public LongProperty invalidationTimestampProperty() {
|
||||||
|
return invalidationTimestamp;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void requestInvalidationTimeStampFromDHT(String currencyCode) {
|
||||||
|
Number160 locationKey = Number160.createHash(currencyCode);
|
||||||
|
FutureGet getFuture = p2pNode.getData(getInvalidatedLocationKey(locationKey));
|
||||||
|
getFuture.addListener(new BaseFutureListener<BaseFuture>() {
|
||||||
|
@Override
|
||||||
|
public void operationComplete(BaseFuture future) throws Exception {
|
||||||
|
if (getFuture.isSuccess()) {
|
||||||
|
Data data = getFuture.data();
|
||||||
|
if (data != null && data.object() instanceof Long) {
|
||||||
|
final Object object = data.object();
|
||||||
|
Platform.runLater(() -> {
|
||||||
|
Long timeStamp = (Long) object;
|
||||||
|
//log.trace("Get invalidationTimestamp from DHT was successful. TimeStamp=" + timeStamp);
|
||||||
|
invalidationTimestamp.set(timeStamp);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
//log.error("Get invalidationTimestamp from DHT failed. Data = " + data);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else if (getFuture.data() == null) {
|
||||||
|
// OK as nothing is set at the moment
|
||||||
|
// log.trace("Get invalidationTimestamp from DHT returns null. That is ok for the startup.");
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
log.error("Get invalidationTimestamp from DHT failed with reason:" + getFuture.failedReason());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void exceptionCaught(Throwable t) throws Exception {
|
||||||
|
log.error("Get invalidationTimestamp from DHT failed with exception:" + t.getMessage());
|
||||||
|
t.printStackTrace();
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
private Number160 getInvalidatedLocationKey(Number160 locationKey) {
|
||||||
|
return Number160.createHash(locationKey + "invalidated");
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
|
@ -25,6 +25,7 @@ import io.bitsquare.msg.MessageFacade;
|
||||||
import io.bitsquare.network.Peer;
|
import io.bitsquare.network.Peer;
|
||||||
import io.bitsquare.offer.Direction;
|
import io.bitsquare.offer.Direction;
|
||||||
import io.bitsquare.offer.Offer;
|
import io.bitsquare.offer.Offer;
|
||||||
|
import io.bitsquare.offer.OfferRepository;
|
||||||
import io.bitsquare.persistence.Persistence;
|
import io.bitsquare.persistence.Persistence;
|
||||||
import io.bitsquare.settings.Settings;
|
import io.bitsquare.settings.Settings;
|
||||||
import io.bitsquare.trade.handlers.ErrorMessageHandler;
|
import io.bitsquare.trade.handlers.ErrorMessageHandler;
|
||||||
|
@ -78,6 +79,7 @@ public class TradeManager {
|
||||||
private final BlockChainFacade blockChainFacade;
|
private final BlockChainFacade blockChainFacade;
|
||||||
private final WalletFacade walletFacade;
|
private final WalletFacade walletFacade;
|
||||||
private final CryptoFacade cryptoFacade;
|
private final CryptoFacade cryptoFacade;
|
||||||
|
private final OfferRepository offerRepository;
|
||||||
|
|
||||||
//TODO store TakerAsSellerProtocol in trade
|
//TODO store TakerAsSellerProtocol in trade
|
||||||
private final Map<String, SellerTakesOfferProtocol> takerAsSellerProtocolMap = new HashMap<>();
|
private final Map<String, SellerTakesOfferProtocol> takerAsSellerProtocolMap = new HashMap<>();
|
||||||
|
@ -99,7 +101,8 @@ public class TradeManager {
|
||||||
|
|
||||||
@Inject
|
@Inject
|
||||||
public TradeManager(User user, Settings settings, Persistence persistence, MessageFacade messageFacade,
|
public TradeManager(User user, Settings settings, Persistence persistence, MessageFacade messageFacade,
|
||||||
BlockChainFacade blockChainFacade, WalletFacade walletFacade, CryptoFacade cryptoFacade) {
|
BlockChainFacade blockChainFacade, WalletFacade walletFacade, CryptoFacade cryptoFacade,
|
||||||
|
OfferRepository offerRepository) {
|
||||||
this.user = user;
|
this.user = user;
|
||||||
this.settings = settings;
|
this.settings = settings;
|
||||||
this.persistence = persistence;
|
this.persistence = persistence;
|
||||||
|
@ -107,6 +110,7 @@ public class TradeManager {
|
||||||
this.blockChainFacade = blockChainFacade;
|
this.blockChainFacade = blockChainFacade;
|
||||||
this.walletFacade = walletFacade;
|
this.walletFacade = walletFacade;
|
||||||
this.cryptoFacade = cryptoFacade;
|
this.cryptoFacade = cryptoFacade;
|
||||||
|
this.offerRepository = offerRepository;
|
||||||
|
|
||||||
Object offersObject = persistence.read(this, "offers");
|
Object offersObject = persistence.read(this, "offers");
|
||||||
if (offersObject instanceof Map) {
|
if (offersObject instanceof Map) {
|
||||||
|
@ -189,7 +193,7 @@ public class TradeManager {
|
||||||
(message, throwable) -> {
|
(message, throwable) -> {
|
||||||
errorMessageHandler.onFault(message);
|
errorMessageHandler.onFault(message);
|
||||||
createOfferCoordinatorMap.remove(offer.getId());
|
createOfferCoordinatorMap.remove(offer.getId());
|
||||||
});
|
}, offerRepository);
|
||||||
createOfferCoordinatorMap.put(offer.getId(), createOfferCoordinator);
|
createOfferCoordinatorMap.put(offer.getId(), createOfferCoordinator);
|
||||||
createOfferCoordinator.start();
|
createOfferCoordinator.start();
|
||||||
}
|
}
|
||||||
|
@ -210,7 +214,7 @@ public class TradeManager {
|
||||||
offers.remove(offer.getId());
|
offers.remove(offer.getId());
|
||||||
persistOffers();
|
persistOffers();
|
||||||
|
|
||||||
messageFacade.removeOffer(offer);
|
offerRepository.removeOffer(offer);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
|
@ -20,6 +20,7 @@ package io.bitsquare.trade.protocol.createoffer;
|
||||||
import io.bitsquare.btc.WalletFacade;
|
import io.bitsquare.btc.WalletFacade;
|
||||||
import io.bitsquare.msg.MessageFacade;
|
import io.bitsquare.msg.MessageFacade;
|
||||||
import io.bitsquare.offer.Offer;
|
import io.bitsquare.offer.Offer;
|
||||||
|
import io.bitsquare.offer.OfferRepository;
|
||||||
import io.bitsquare.persistence.Persistence;
|
import io.bitsquare.persistence.Persistence;
|
||||||
import io.bitsquare.trade.handlers.FaultHandler;
|
import io.bitsquare.trade.handlers.FaultHandler;
|
||||||
import io.bitsquare.trade.handlers.TransactionResultHandler;
|
import io.bitsquare.trade.handlers.TransactionResultHandler;
|
||||||
|
@ -90,22 +91,25 @@ public class CreateOfferCoordinator {
|
||||||
private final TransactionResultHandler resultHandler;
|
private final TransactionResultHandler resultHandler;
|
||||||
private final FaultHandler faultHandler;
|
private final FaultHandler faultHandler;
|
||||||
private final Model model;
|
private final Model model;
|
||||||
|
private final OfferRepository offerRepository;
|
||||||
|
|
||||||
public CreateOfferCoordinator(Persistence persistence, Offer offer, WalletFacade walletFacade,
|
public CreateOfferCoordinator(Persistence persistence, Offer offer, WalletFacade walletFacade,
|
||||||
MessageFacade messageFacade, TransactionResultHandler resultHandler,
|
MessageFacade messageFacade, TransactionResultHandler resultHandler,
|
||||||
FaultHandler faultHandler) {
|
FaultHandler faultHandler, OfferRepository offerRepository) {
|
||||||
this(offer, walletFacade, messageFacade, resultHandler, faultHandler, new Model(persistence));
|
this(offer, walletFacade, messageFacade, resultHandler, faultHandler, new Model(persistence), offerRepository);
|
||||||
}
|
}
|
||||||
|
|
||||||
// for recovery from model
|
// for recovery from model
|
||||||
public CreateOfferCoordinator(Offer offer, WalletFacade walletFacade, MessageFacade messageFacade,
|
public CreateOfferCoordinator(Offer offer, WalletFacade walletFacade, MessageFacade messageFacade,
|
||||||
TransactionResultHandler resultHandler, FaultHandler faultHandler, Model model) {
|
TransactionResultHandler resultHandler, FaultHandler faultHandler, Model model,
|
||||||
|
OfferRepository offerRepository) {
|
||||||
this.offer = offer;
|
this.offer = offer;
|
||||||
this.walletFacade = walletFacade;
|
this.walletFacade = walletFacade;
|
||||||
this.messageFacade = messageFacade;
|
this.messageFacade = messageFacade;
|
||||||
this.resultHandler = resultHandler;
|
this.resultHandler = resultHandler;
|
||||||
this.faultHandler = faultHandler;
|
this.faultHandler = faultHandler;
|
||||||
this.model = model;
|
this.model = model;
|
||||||
|
this.offerRepository = offerRepository;
|
||||||
|
|
||||||
model.setState(State.INITED);
|
model.setState(State.INITED);
|
||||||
}
|
}
|
||||||
|
@ -129,7 +133,7 @@ public class CreateOfferCoordinator {
|
||||||
|
|
||||||
private void onOfferFeeTxBroadCasted() {
|
private void onOfferFeeTxBroadCasted() {
|
||||||
model.setState(State.OFFER_FEE_BROAD_CASTED);
|
model.setState(State.OFFER_FEE_BROAD_CASTED);
|
||||||
PublishOfferToDHT.run(this::onOfferPublishedToDHT, this::onFailed, messageFacade, offer);
|
PublishOfferToDHT.run(this::onOfferPublishedToDHT, this::onFailed, offerRepository, offer);
|
||||||
}
|
}
|
||||||
|
|
||||||
private void onOfferPublishedToDHT() {
|
private void onOfferPublishedToDHT() {
|
||||||
|
@ -159,7 +163,7 @@ public class CreateOfferCoordinator {
|
||||||
case OFFER_FEE_BROAD_CASTED:
|
case OFFER_FEE_BROAD_CASTED:
|
||||||
// actually the only replay case here, tx publish was successful but storage to dht failed.
|
// actually the only replay case here, tx publish was successful but storage to dht failed.
|
||||||
// Republish the offer to DHT
|
// Republish the offer to DHT
|
||||||
PublishOfferToDHT.run(this::onOfferPublishedToDHT, this::onFailed, messageFacade, offer);
|
PublishOfferToDHT.run(this::onOfferPublishedToDHT, this::onFailed, offerRepository, offer);
|
||||||
break;
|
break;
|
||||||
case OFFER_PUBLISHED_TO_DHT:
|
case OFFER_PUBLISHED_TO_DHT:
|
||||||
// should be impossible
|
// should be impossible
|
||||||
|
|
|
@ -17,9 +17,9 @@
|
||||||
|
|
||||||
package io.bitsquare.trade.protocol.createoffer.tasks;
|
package io.bitsquare.trade.protocol.createoffer.tasks;
|
||||||
|
|
||||||
import io.bitsquare.msg.MessageFacade;
|
|
||||||
import io.bitsquare.msg.listeners.AddOfferListener;
|
import io.bitsquare.msg.listeners.AddOfferListener;
|
||||||
import io.bitsquare.offer.Offer;
|
import io.bitsquare.offer.Offer;
|
||||||
|
import io.bitsquare.offer.OfferRepository;
|
||||||
import io.bitsquare.trade.handlers.FaultHandler;
|
import io.bitsquare.trade.handlers.FaultHandler;
|
||||||
import io.bitsquare.trade.handlers.ResultHandler;
|
import io.bitsquare.trade.handlers.ResultHandler;
|
||||||
|
|
||||||
|
@ -29,9 +29,9 @@ import org.slf4j.LoggerFactory;
|
||||||
public class PublishOfferToDHT {
|
public class PublishOfferToDHT {
|
||||||
private static final Logger log = LoggerFactory.getLogger(PublishOfferToDHT.class);
|
private static final Logger log = LoggerFactory.getLogger(PublishOfferToDHT.class);
|
||||||
|
|
||||||
public static void run(ResultHandler resultHandler, FaultHandler faultHandler, MessageFacade messageFacade,
|
public static void run(ResultHandler resultHandler, FaultHandler faultHandler, OfferRepository offerRepository,
|
||||||
Offer offer) {
|
Offer offer) {
|
||||||
messageFacade.addOffer(offer, new AddOfferListener() {
|
offerRepository.addOffer(offer, new AddOfferListener() {
|
||||||
@Override
|
@Override
|
||||||
public void onComplete() {
|
public void onComplete() {
|
||||||
resultHandler.onResult();
|
resultHandler.onResult();
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue