diff --git a/config/currency_provider_tracker.go b/config/currency_provider_tracker.go index 221f4ba..1a79716 100644 --- a/config/currency_provider_tracker.go +++ b/config/currency_provider_tracker.go @@ -14,15 +14,13 @@ import ( ) const ( - coinGeckoRestURL = "https://api.coingecko.com/api/v3/coins" - coinGeckoListEndpoint = "list" - coinGeckoTickersEndpoint = "tickers" - osmosisRestURL = "https://api.osmo-api.prod.ojo.network" - osmosisAssetPairsEndpoint = "assetpairs" - crescentRestURL = "https://api.cresc-api.prod.ojo.network" - crescentAssetPairsEndpoint = "assetpairs" - requestTimeout = time.Second * 2 - trackingPeriod = time.Hour * 24 + coinGeckoRestURL = "https://api.coingecko.com/api/v3/coins" + coinGeckoListEndpoint = "list" + coinGeckoTickersEndpoint = "tickers" + osmosisRestURL = "https://api.osmo-api.prod.ojo.network" + osmosisAssetPairsEndpoint = "assetpairs" + requestTimeout = time.Second * 2 + trackingPeriod = time.Hour * 24 ) type ( @@ -90,12 +88,7 @@ func NewCurrencyProviderTracker( return nil, err } - crescentAPIPairs, err := currencyProviderTracker.getCrescentAPIPairs() - if err != nil { - return nil, err - } - - if err := currencyProviderTracker.setCurrencyProviders(osmosisAPIPairs, crescentAPIPairs); err != nil { + if err := currencyProviderTracker.setCurrencyProviders(osmosisAPIPairs); err != nil { return nil, err } @@ -159,36 +152,10 @@ func (t *CurrencyProviderTracker) getOsmosisAPIPairs() (map[string]string, error return osmosisAPIPairs, nil } -// getCrescentAPIPairs queries the crescent-api assetpairs endpoint to get the asset pairs -// supported by it. -func (t *CurrencyProviderTracker) getCrescentAPIPairs() (map[string]string, error) { - client := &http.Client{ - Timeout: requestTimeout, - } - crescentAPIPairs := make(map[string]string) - - crescentResp, err := client.Get(fmt.Sprintf("%s/%s", crescentRestURL, crescentAssetPairsEndpoint)) - if err != nil { - return nil, err - } - defer crescentResp.Body.Close() - var assetPairsResponse []assetPair - if err = json.NewDecoder(crescentResp.Body).Decode(&assetPairsResponse); err != nil { - return nil, err - } - - for _, assetPair := range assetPairsResponse { - crescentAPIPairs[assetPair.Base] = assetPair.Quote - } - - return crescentAPIPairs, nil -} - // setCurrencyProviders queries CoinGecko's tickers endpoint to get all the exchanges // that support each price feeder currency pair and store it in the CurrencyProviders map. func (t *CurrencyProviderTracker) setCurrencyProviders( osmosisAPIPairs map[string]string, - crescentAPIPairs map[string]string, ) error { client := &http.Client{ Timeout: requestTimeout, @@ -199,11 +166,6 @@ func (t *CurrencyProviderTracker) setCurrencyProviders( t.CurrencyProviders[pair.Base] = append(t.CurrencyProviders[pair.Base], "osmosis") } - // check if its a pair supported by the crescent api - if crescentAPIPairs[strings.ToUpper(pair.Base)] == strings.ToUpper(pair.Quote) { - t.CurrencyProviders[pair.Base] = append(t.CurrencyProviders[pair.Base], "crescent") - } - // check if CoinGecko API supports pair pairBaseID := t.coinIDSymbolMap[strings.ToLower(pair.Base)] coinGeckoResp, err := client.Get(fmt.Sprintf("%s/%s/%s", coinGeckoRestURL, pairBaseID, coinGeckoTickersEndpoint)) @@ -259,11 +221,7 @@ func (t *CurrencyProviderTracker) trackCurrencyProviders(ctx context.Context) { if err != nil { t.logger.Error().Err(err).Msg("failed to query osmosis-api for available asset pairs") } - crescentAPIPairs, err := t.getOsmosisAPIPairs() - if err != nil { - t.logger.Error().Err(err).Msg("failed to query crescent-api for available asset pairs") - } - if err := t.setCurrencyProviders(osmosisAPIPairs, crescentAPIPairs); err != nil { + if err := t.setCurrencyProviders(osmosisAPIPairs); err != nil { t.logger.Error().Err(err).Msg("failed to set available providers for currencies") } diff --git a/config/supported_assets.go b/config/supported_assets.go index 69e1eb4..3cff2ca 100644 --- a/config/supported_assets.go +++ b/config/supported_assets.go @@ -14,7 +14,6 @@ var ( provider.ProviderKraken: false, provider.ProviderBinance: false, provider.ProviderBinanceUS: false, - provider.ProviderCrescent: false, provider.ProviderOsmosis: false, provider.ProviderOkx: false, provider.ProviderHuobi: false, diff --git a/oracle/oracle.go b/oracle/oracle.go index 487c81e..24646a9 100644 --- a/oracle/oracle.go +++ b/oracle/oracle.go @@ -498,9 +498,6 @@ func NewProvider( case provider.ProviderPolygon: return provider.NewPolygonProvider(ctx, logger, endpoint, providerPairs...) - case provider.ProviderCrescent: - return provider.NewCrescentProvider(ctx, logger, endpoint, providerPairs...) - case provider.ProviderKujira: return provider.NewKujiraProvider(ctx, logger, endpoint, providerPairs...) diff --git a/oracle/provider/crescent.go b/oracle/provider/crescent.go deleted file mode 100644 index ed70efa..0000000 --- a/oracle/provider/crescent.go +++ /dev/null @@ -1,269 +0,0 @@ -package provider - -import ( - "context" - "encoding/json" - "net/http" - "net/url" - "strings" - "sync" - - "github.com/gorilla/websocket" - "github.com/ojo-network/price-feeder/oracle/types" - "github.com/rs/zerolog" -) - -const ( - crescentV2WSHost = "api.cresc-api.prod.ojo.network" - crescentV2WSPath = "ws" - crescentV2RestHost = "https://api.cresc-api.prod.ojo.network" - crescentV2RestPath = "/assetpairs" - crescentAckMsg = "ack" -) - -var _ Provider = (*CrescentProvider)(nil) - -type ( - // CrescentProvider defines an Oracle provider implemented by OJO's - // Crescent API. - CrescentProvider struct { - wsc *WebsocketController - wsURL url.URL - logger zerolog.Logger - mtx sync.RWMutex - endpoints Endpoint - - priceStore - } - - CrescentTicker struct { - Price string `json:"Price"` - Volume string `json:"Volume"` - } - - CrescentCandle struct { - Close string `json:"Close"` - Volume string `json:"Volume"` - EndTime int64 `json:"EndTime"` - } - - // CrescentPairsSummary defines the response structure for an Crescent pairs - // summary. - CrescentPairsSummary struct { - Data []CrescentPairData `json:"data"` - } - - // CrescentPairData defines the data response structure for an Crescent pair. - CrescentPairData struct { - Base string `json:"base"` - Quote string `json:"quote"` - } -) - -func NewCrescentProvider( - ctx context.Context, - logger zerolog.Logger, - endpoints Endpoint, - pairs ...types.CurrencyPair, -) (*CrescentProvider, error) { - if endpoints.Name != ProviderCrescent { - endpoints = Endpoint{ - Name: ProviderCrescent, - Rest: crescentV2RestHost, - Websocket: crescentV2WSHost, - } - } - - wsURL := url.URL{ - Scheme: "wss", - Host: endpoints.Websocket, - Path: crescentV2WSPath, - } - - crescentV2Logger := logger.With().Str("provider", "crescent").Logger() - - provider := &CrescentProvider{ - wsURL: wsURL, - logger: crescentV2Logger, - endpoints: endpoints, - priceStore: newPriceStore(crescentV2Logger), - } - provider.setCurrencyPairToTickerAndCandlePair(currencyPairToCrescentPair) - - confirmedPairs, err := ConfirmPairAvailability( - provider, - provider.endpoints.Name, - provider.logger, - pairs..., - ) - if err != nil { - return nil, err - } - - provider.setSubscribedPairs(confirmedPairs...) - - provider.wsc = NewWebsocketController( - ctx, - endpoints.Name, - wsURL, - []interface{}{""}, - provider.messageReceived, - defaultPingDuration, - websocket.PingMessage, - crescentV2Logger, - ) - - return provider, nil -} - -func (p *CrescentProvider) StartConnections() { - p.wsc.StartConnections() -} - -// SubscribeCurrencyPairs sends the new subscription messages to the websocket -// and adds them to the providers subscribedPairs array -func (p *CrescentProvider) SubscribeCurrencyPairs(cps ...types.CurrencyPair) { - p.mtx.Lock() - defer p.mtx.Unlock() - - confirmedPairs, err := ConfirmPairAvailability( - p, - p.endpoints.Name, - p.logger, - cps..., - ) - if err != nil { - return - } - - p.setSubscribedPairs(confirmedPairs...) -} - -func (p *CrescentProvider) messageReceived(_ int, _ *WebsocketConnection, bz []byte) { - // check if message is an ack - if string(bz) == crescentAckMsg { - return - } - - var ( - messageResp map[string]interface{} - messageErr error - tickerResp CrescentTicker - tickerErr error - candleResp []CrescentCandle - candleErr error - ) - - messageErr = json.Unmarshal(bz, &messageResp) - if messageErr != nil { - p.logger.Error(). - Int("length", len(bz)). - AnErr("message", messageErr). - Msg("Error on receive message") - } - - // Check the response for currency pairs that the provider is subscribed - // to and determine whether it is a ticker or candle. - for _, pair := range p.subscribedPairs { - crescentPair := currencyPairToCrescentPair(pair) - if msg, ok := messageResp[crescentPair]; ok { - switch v := msg.(type) { - // ticker response - case map[string]interface{}: - tickerString, _ := json.Marshal(v) - tickerErr = json.Unmarshal(tickerString, &tickerResp) - if tickerErr != nil { - p.logger.Error(). - Int("length", len(bz)). - AnErr("ticker", tickerErr). - Msg("Error on receive message") - continue - } - p.setTickerPair( - tickerResp, - crescentPair, - ) - telemetryWebsocketMessage(ProviderCrescent, MessageTypeTicker) - continue - - // candle response - case []interface{}: - // use latest candlestick in list if there is one - if len(v) == 0 { - continue - } - candleString, _ := json.Marshal(v) - candleErr = json.Unmarshal(candleString, &candleResp) - if candleErr != nil { - p.logger.Error(). - Int("length", len(bz)). - AnErr("candle", candleErr). - Msg("Error on receive message") - continue - } - for _, singleCandle := range candleResp { - p.setCandlePair( - singleCandle, - crescentPair, - ) - } - telemetryWebsocketMessage(ProviderCrescent, MessageTypeCandle) - continue - } - } - } -} - -func (ct CrescentTicker) toTickerPrice() (types.TickerPrice, error) { - return types.NewTickerPrice( - ct.Price, - ct.Volume, - ) -} - -func (cc CrescentCandle) toCandlePrice() (types.CandlePrice, error) { - return types.NewCandlePrice( - cc.Close, - cc.Volume, - cc.EndTime, - ) -} - -// setSubscribedPairs sets N currency pairs to the map of subscribed pairs. -func (p *CrescentProvider) setSubscribedPairs(cps ...types.CurrencyPair) { - for _, cp := range cps { - p.subscribedPairs[cp.String()] = cp - } -} - -// GetAvailablePairs returns all pairs to which the provider can subscribe. -// ex.: map["ATOMUSDT" => {}, "OJOUSDC" => {}]. -func (p *CrescentProvider) GetAvailablePairs() (map[string]struct{}, error) { - resp, err := http.Get(p.endpoints.Rest + crescentV2RestPath) - if err != nil { - return nil, err - } - defer resp.Body.Close() - - var pairsSummary []CrescentPairData - if err := json.NewDecoder(resp.Body).Decode(&pairsSummary); err != nil { - return nil, err - } - - availablePairs := make(map[string]struct{}, len(pairsSummary)) - for _, pair := range pairsSummary { - cp := types.CurrencyPair{ - Base: pair.Base, - Quote: pair.Quote, - } - availablePairs[strings.ToUpper(cp.String())] = struct{}{} - } - - return availablePairs, nil -} - -// currencyPairToCrescentPair receives a currency pair and return crescent -// ticker symbol atomusdt@ticker. -func currencyPairToCrescentPair(cp types.CurrencyPair) string { - return cp.Base + "/" + cp.Quote -} diff --git a/oracle/provider/crescent_test.go b/oracle/provider/crescent_test.go deleted file mode 100644 index b87468a..0000000 --- a/oracle/provider/crescent_test.go +++ /dev/null @@ -1,116 +0,0 @@ -package provider - -import ( - "context" - "testing" - - "cosmossdk.io/math" - "github.com/ojo-network/price-feeder/oracle/types" - "github.com/rs/zerolog" - "github.com/stretchr/testify/require" -) - -func TestCrescentProvider_GetTickerPrices(t *testing.T) { - p, err := NewCrescentProvider( - context.TODO(), - zerolog.Nop(), - Endpoint{}, - BCREATOM, - ) - require.NoError(t, err) - - t.Run("valid_request_single_ticker", func(t *testing.T) { - lastPrice := math.LegacyMustNewDecFromStr("34.69000000") - volume := math.LegacyMustNewDecFromStr("2396974.02000000") - - tickerMap := map[string]types.TickerPrice{} - tickerMap["BCRE/ATOM"] = types.TickerPrice{ - Price: lastPrice, - Volume: volume, - } - - p.tickers = tickerMap - - prices, err := p.GetTickerPrices(BCREATOM) - require.NoError(t, err) - require.Len(t, prices, 1) - require.Equal(t, lastPrice, prices[BCREATOM].Price) - require.Equal(t, volume, prices[BCREATOM].Volume) - }) - - t.Run("valid_request_multi_ticker", func(t *testing.T) { - lastPriceAtom := math.LegacyMustNewDecFromStr("34.69000000") - lastPriceLuna := math.LegacyMustNewDecFromStr("41.35000000") - volume := math.LegacyMustNewDecFromStr("2396974.02000000") - - tickerMap := map[string]types.TickerPrice{} - tickerMap["ATOM/USDT"] = types.TickerPrice{ - Price: lastPriceAtom, - Volume: volume, - } - - tickerMap["LUNA/USDT"] = types.TickerPrice{ - Price: lastPriceLuna, - Volume: volume, - } - - p.tickers = tickerMap - prices, err := p.GetTickerPrices( - ATOMUSDT, - LUNAUSDT, - ) - require.NoError(t, err) - require.Len(t, prices, 2) - require.Equal(t, lastPriceAtom, prices[ATOMUSDT].Price) - require.Equal(t, volume, prices[ATOMUSDT].Volume) - require.Equal(t, lastPriceLuna, prices[LUNAUSDT].Price) - require.Equal(t, volume, prices[LUNAUSDT].Volume) - }) - - t.Run("invalid_request_invalid_ticker", func(t *testing.T) { - prices, _ := p.GetTickerPrices(types.CurrencyPair{Base: "FOO", Quote: "BAR"}) - require.Empty(t, prices) - }) -} - -func TestCrescentProvider_GetCandlePrices(t *testing.T) { - p, err := NewCrescentProvider( - context.TODO(), - zerolog.Nop(), - Endpoint{}, - BCREATOM, - ) - require.NoError(t, err) - - t.Run("valid_request_single_candle", func(t *testing.T) { - price := "34.689998626708984000" - volume := "2396974.000000000000000000" - time := int64(1000000) - - candle := CrescentCandle{ - Volume: volume, - Close: price, - EndTime: time, - } - - p.setCandlePair(candle, "BCRE/ATOM") - - prices, err := p.GetCandlePrices(BCREATOM) - require.NoError(t, err) - require.Len(t, prices, 1) - require.Equal(t, math.LegacyMustNewDecFromStr(price), prices[BCREATOM][0].Price) - require.Equal(t, math.LegacyMustNewDecFromStr(volume), prices[BCREATOM][0].Volume) - require.Equal(t, time, prices[BCREATOM][0].TimeStamp) - }) - - t.Run("invalid_request_invalid_candle", func(t *testing.T) { - prices, _ := p.GetCandlePrices(types.CurrencyPair{Base: "FOO", Quote: "BAR"}) - require.Empty(t, prices) - }) -} - -func TestCrescentCurrencyPairToCrescentPair(t *testing.T) { - cp := types.CurrencyPair{Base: "BCRE", Quote: "USDT"} - crescentSymbol := currencyPairToCrescentPair(cp) - require.Equal(t, crescentSymbol, "BCRE/USDT") -} diff --git a/oracle/provider/provider.go b/oracle/provider/provider.go index f7c8b22..5e954c3 100644 --- a/oracle/provider/provider.go +++ b/oracle/provider/provider.go @@ -21,7 +21,6 @@ const ( ProviderMexc types.ProviderName = "mexc" ProviderCrypto types.ProviderName = "crypto" ProviderPolygon types.ProviderName = "polygon" - ProviderCrescent types.ProviderName = "crescent" ProviderEthUniswap types.ProviderName = "eth-uniswap" ProviderKujira types.ProviderName = "kujira" ProviderMock types.ProviderName = "mock"