All Downloads are FREE. Search and download functionalities are using the official Maven repository.

info.bitrich.xchangestream.poloniex2.PoloniexStreamingMarketDataService Maven / Gradle / Ivy

The newest version!
package info.bitrich.xchangestream.poloniex2;

import static org.knowm.xchange.poloniex.PoloniexAdapters.adaptPoloniexDepth;
import static org.knowm.xchange.poloniex.PoloniexAdapters.adaptPoloniexTicker;

import com.fasterxml.jackson.databind.ObjectMapper;
import com.google.common.base.Supplier;
import com.google.common.base.Suppliers;
import info.bitrich.xchangestream.core.StreamingMarketDataService;
import info.bitrich.xchangestream.poloniex2.dto.*;
import info.bitrich.xchangestream.service.netty.StreamingObjectMapperHelper;
import io.reactivex.Observable;
import java.math.BigDecimal;
import java.util.List;
import java.util.Map;
import java.util.Optional;
import java.util.SortedMap;
import org.knowm.xchange.currency.CurrencyPair;
import org.knowm.xchange.dto.marketdata.OrderBook;
import org.knowm.xchange.dto.marketdata.Ticker;
import org.knowm.xchange.dto.marketdata.Trade;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

/** Created by Lukas Zaoralek on 10.11.17. */
public class PoloniexStreamingMarketDataService implements StreamingMarketDataService {
  private static final Logger LOG =
      LoggerFactory.getLogger(PoloniexStreamingMarketDataService.class);
  private static final String TICKER_CHANNEL_ID = "1002";

  private final PoloniexStreamingService service;
  private final Supplier> streamingTickers;

  public PoloniexStreamingMarketDataService(
      PoloniexStreamingService service, Map currencyIdMap) {
    this.service = service;
    final ObjectMapper mapper = StreamingObjectMapperHelper.getObjectMapper();

    streamingTickers =
        Suppliers.memoize(
            () ->
                service
                    .subscribeChannel(TICKER_CHANNEL_ID)
                    .map(
                        s -> {
                          PoloniexWebSocketTickerTransaction ticker =
                              mapper.readValue(
                                  s.toString(), PoloniexWebSocketTickerTransaction.class);
                          CurrencyPair currencyPair = currencyIdMap.get(ticker.getPairId());
                          return adaptPoloniexTicker(
                              ticker.toPoloniexTicker(currencyPair), currencyPair);
                        })
                    .share());
  }

  @Override
  public Observable getOrderBook(CurrencyPair currencyPair, Object... args) {
    Observable subscribedOrderbook =
        service
            .subscribeCurrencyPairChannel(currencyPair)
            .scan(
                Optional.empty(),
                (Optional orderbook,
                    List poloniexWebSocketEvents) ->
                    poloniexWebSocketEvents.stream()
                        .filter(
                            s ->
                                s instanceof PoloniexWebSocketOrderbookInsertEvent
                                    || s instanceof PoloniexWebSocketOrderbookModifiedEvent)
                        .reduce(
                            orderbook,
                            (poloniexOrderbook, s) -> getPoloniexOrderbook(orderbook, s),
                            (o1, o2) -> {
                              throw new UnsupportedOperationException("No parallel execution");
                            }))
            .filter(Optional::isPresent)
            .map(Optional::get);

    return subscribedOrderbook.map(s -> adaptPoloniexDepth(s.toPoloniexDepth(), currencyPair));
  }

  @Override
  public Observable getTicker(CurrencyPair currencyPair, Object... args) {
    return streamingTickers.get().filter(ticker -> ticker.getCurrencyPair().equals(currencyPair));
  }

  @Override
  public Observable getTrades(CurrencyPair currencyPair, Object... args) {
    Observable subscribedTrades =
        service
            .subscribeCurrencyPairChannel(currencyPair)
            .flatMapIterable(poloniexWebSocketEvents -> poloniexWebSocketEvents)
            .filter(PoloniexWebSocketTradeEvent.class::isInstance)
            .map(PoloniexWebSocketTradeEvent.class::cast)
            .share();

    return subscribedTrades.map(
        s -> PoloniexWebSocketAdapter.convertPoloniexWebSocketTradeEventToTrade(s, currencyPair));
  }

  private Optional getPoloniexOrderbook(
      final Optional orderbook, final PoloniexWebSocketEvent s) {
    if (s.getEventType().equals("i")) {
      OrderbookInsertEvent insertEvent = ((PoloniexWebSocketOrderbookInsertEvent) s).getInsert();
      SortedMap asks =
          insertEvent.toDepthLevels(OrderbookInsertEvent.ASK_SIDE);
      SortedMap bids =
          insertEvent.toDepthLevels(OrderbookInsertEvent.BID_SIDE);
      return Optional.of(new PoloniexOrderbook(asks, bids));
    } else {
      OrderbookModifiedEvent modifiedEvent =
          ((PoloniexWebSocketOrderbookModifiedEvent) s).getModifiedEvent();
      orderbook
          .orElseThrow(
              () -> new IllegalStateException("Orderbook update received before initial snapshot"))
          .modify(modifiedEvent);
      return orderbook;
    }
  }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy