Skip to content

Commit c1b4de0

Browse files
committed
feat: change hubi websocket endpoint
1 parent b43e44f commit c1b4de0

File tree

7 files changed

+184
-86
lines changed

7 files changed

+184
-86
lines changed

reactive-crypto-hubi/src/main/kotlin/com/njkim/reactivecrypto/hubi/HubiJsonObjectMapper.kt

+12-5
Original file line numberDiff line numberDiff line change
@@ -23,19 +23,26 @@ import com.fasterxml.jackson.databind.JsonDeserializer
2323
import com.njkim.reactivecrypto.core.ExchangeJsonObjectMapper
2424
import com.njkim.reactivecrypto.core.common.model.currency.CurrencyPair
2525
import com.njkim.reactivecrypto.core.common.model.order.TradeSideType
26+
import com.njkim.reactivecrypto.core.common.util.CurrencyPairUtil
2627
import java.io.IOException
2728
import java.math.BigDecimal
28-
import java.time.Instant
29-
import java.time.ZoneId
29+
import java.time.ZoneOffset
3030
import java.time.ZonedDateTime
31+
import java.time.format.DateTimeFormatterBuilder
32+
import java.util.*
3133

3234
class HubiJsonObjectMapper : ExchangeJsonObjectMapper {
3335

3436
override fun zonedDateTimeDeserializer(): JsonDeserializer<ZonedDateTime>? {
37+
val dateTimeFormatter = DateTimeFormatterBuilder()
38+
.parseCaseInsensitive()
39+
.appendPattern("MMM dd, yyyy hh:mm:ss a")
40+
.toFormatter(Locale.ENGLISH)
41+
3542
return object : JsonDeserializer<ZonedDateTime>() {
3643
@Throws(IOException::class, JsonProcessingException::class)
3744
override fun deserialize(p: JsonParser, ctxt: DeserializationContext): ZonedDateTime {
38-
return Instant.ofEpochMilli(p.valueAsLong).atZone(ZoneId.systemDefault())
45+
return ZonedDateTime.parse(p.valueAsString, dateTimeFormatter.withZone(ZoneOffset.UTC))
3946
}
4047
}
4148
}
@@ -44,8 +51,8 @@ class HubiJsonObjectMapper : ExchangeJsonObjectMapper {
4451
return object : JsonDeserializer<CurrencyPair>() {
4552
@Throws(IOException::class, JsonProcessingException::class)
4653
override fun deserialize(p: JsonParser, ctxt: DeserializationContext): CurrencyPair {
47-
val splitCurrencyPair = p.valueAsString.split("_")
48-
return CurrencyPair.parse(splitCurrencyPair[0], splitCurrencyPair[1])
54+
val rawCurrencyPair = p.valueAsString
55+
return CurrencyPairUtil.parse(rawCurrencyPair) ?: error("can't find currencyPair : $rawCurrencyPair")
4956
}
5057
}
5158
}

reactive-crypto-hubi/src/main/kotlin/com/njkim/reactivecrypto/hubi/HubiWebsocketClient.kt

+83-31
Original file line numberDiff line numberDiff line change
@@ -27,21 +27,20 @@ import com.njkim.reactivecrypto.core.common.model.order.TickData
2727
import com.njkim.reactivecrypto.core.common.model.order.TradeSideType
2828
import com.njkim.reactivecrypto.core.common.util.toEpochMilli
2929
import com.njkim.reactivecrypto.core.websocket.AbstractExchangeWebsocketClient
30-
import com.njkim.reactivecrypto.hubi.model.HubiMessageFrame
31-
import com.njkim.reactivecrypto.hubi.model.HubiOrderBook
32-
import com.njkim.reactivecrypto.hubi.model.HubiTickDataWrapper
30+
import com.njkim.reactivecrypto.hubi.model.HubiDepthResponse
3331
import mu.KotlinLogging
3432
import reactor.core.publisher.Flux
3533
import reactor.kotlin.core.publisher.toFlux
3634
import reactor.netty.http.client.HttpClient
35+
import java.math.BigDecimal
3736
import java.time.ZonedDateTime
3837
import java.util.concurrent.ConcurrentHashMap
3938
import java.util.concurrent.atomic.AtomicLong
4039

4140
class HubiWebsocketClient : AbstractExchangeWebsocketClient() {
4241
private val log = KotlinLogging.logger {}
4342

44-
private val baseUri = "wss://api.hubi.com/ws/connect/v1"
43+
private val baseUri = "wss://api.hubi.com/ws/futures/public/market"
4544

4645
private val objectMapper: ObjectMapper = createJsonObjectMapper().objectMapper()
4746

@@ -50,12 +49,12 @@ class HubiWebsocketClient : AbstractExchangeWebsocketClient() {
5049
}
5150

5251
override fun createDepthSnapshot(subscribeTargets: List<CurrencyPair>): Flux<OrderBook> {
53-
val subscribeRequests = subscribeTargets
54-
.map { "${it.baseCurrency.symbol}${it.quoteCurrency.symbol}".toLowerCase() }
52+
val currentOrderBookMap: MutableMap<CurrencyPair, OrderBook> = ConcurrentHashMap()
53+
54+
val subscribeRequests = subscribeTargets.asSequence()
55+
.map { "${it.baseCurrency.symbol}${it.quoteCurrency.symbol}".toUpperCase() }
5556
.map { symbol ->
56-
"""
57-
{"channel":"depth_all","symbol":"$symbol"}
58-
""".trimIndent()
57+
"""{"op":"subscribe", "channel":"/api/depth/depth", "key":"$symbol"}"""
5958
}
6059
.toFlux()
6160

@@ -68,66 +67,119 @@ class HubiWebsocketClient : AbstractExchangeWebsocketClient() {
6867
.then()
6968
.thenMany(inbound.aggregateFrames().receive().asString())
7069
}
71-
.filter { it.contains("\"dataType\":\"depth_all\"") }
72-
.map { objectMapper.readValue<HubiMessageFrame<HubiOrderBook>>(it) }
70+
.filter { it.contains(""""event":"/api/depth/depth"""") }
71+
.map { objectMapper.readValue<HubiDepthResponse>(it) }
7372
.map { messageFrame ->
7473
val eventTime = ZonedDateTime.now()
7574
OrderBook(
76-
"${messageFrame.symbol}${eventTime.toEpochMilli()}",
77-
messageFrame.symbol,
75+
"${messageFrame.key}${eventTime.toEpochMilli()}",
76+
messageFrame.key,
7877
eventTime,
7978
ExchangeVendor.HUBI,
80-
messageFrame.data.bids.map { OrderBookUnit(it.price, it.amount, TradeSideType.BUY, null) },
81-
messageFrame.data.asks.map { OrderBookUnit(it.price, it.amount, TradeSideType.SELL, null) }.sortedBy { it.price }
79+
messageFrame.buyDepth.map { OrderBookUnit(it.price, it.qty, TradeSideType.BUY, it.count) },
80+
messageFrame.sellDepth.map { OrderBookUnit(it.price, it.qty, TradeSideType.SELL, it.count) }
81+
.sortedBy { it.price }
8282
)
8383
}
84+
.map { orderBook ->
85+
if (!currentOrderBookMap.containsKey(orderBook.currencyPair)) {
86+
val filteredOrderBook = orderBook.copy(
87+
bids = orderBook.bids.filter { it.quantity > BigDecimal.ZERO },
88+
asks = orderBook.asks.filter { it.quantity > BigDecimal.ZERO }
89+
)
90+
currentOrderBookMap[orderBook.currencyPair] = filteredOrderBook
91+
return@map filteredOrderBook
92+
}
93+
94+
val prevOrderBook = currentOrderBookMap[orderBook.currencyPair]!!
95+
96+
val askMap: MutableMap<BigDecimal, OrderBookUnit> = prevOrderBook.asks
97+
.associateBy { it.price.stripTrailingZeros() }
98+
.toMutableMap()
99+
100+
orderBook.asks.forEach { updatedAsk ->
101+
askMap.compute(updatedAsk.price.stripTrailingZeros()) { _, oldValue ->
102+
when {
103+
updatedAsk.quantity <= BigDecimal.ZERO -> null
104+
oldValue == null -> updatedAsk
105+
else -> oldValue.copy(
106+
quantity = updatedAsk.quantity,
107+
orderNumbers = updatedAsk.orderNumbers
108+
)
109+
}
110+
}
111+
}
112+
113+
val bidMap: MutableMap<BigDecimal, OrderBookUnit> = prevOrderBook.bids
114+
.associateBy { it.price.stripTrailingZeros() }
115+
.toMutableMap()
116+
117+
orderBook.bids.forEach { updatedBid ->
118+
bidMap.compute(updatedBid.price.stripTrailingZeros()) { _, oldValue ->
119+
when {
120+
updatedBid.quantity <= BigDecimal.ZERO -> null
121+
oldValue == null -> updatedBid
122+
else -> oldValue.copy(
123+
quantity = updatedBid.quantity,
124+
orderNumbers = updatedBid.orderNumbers
125+
)
126+
}
127+
}
128+
}
129+
130+
val currentOrderBook = prevOrderBook.copy(
131+
eventTime = orderBook.eventTime,
132+
asks = askMap.values.sortedBy { orderBookUnit -> orderBookUnit.price },
133+
bids = bidMap.values.sortedByDescending { orderBookUnit -> orderBookUnit.price }
134+
)
135+
currentOrderBookMap[currentOrderBook.currencyPair] = currentOrderBook
136+
currentOrderBook
137+
}
138+
.doFinally { currentOrderBookMap.clear() } // cleanup memory limit orderBook when disconnected
84139
}
85140

86141
override fun createTradeWebsocket(subscribeTargets: List<CurrencyPair>): Flux<TickData> {
87142
val lastPublishedTimestamp: MutableMap<CurrencyPair, AtomicLong> = ConcurrentHashMap()
88143

89-
val subscribeRequests = subscribeTargets
90-
.map { "${it.baseCurrency.symbol}${it.quoteCurrency.symbol}".toLowerCase() }
144+
val subscribeRequests = subscribeTargets.asSequence()
145+
.map { "${it.baseCurrency.symbol}${it.quoteCurrency.symbol}".toUpperCase() }
91146
.map { symbol ->
92-
"""
93-
{"channel":"trade_history","symbol":"$symbol"}
94-
""".trimIndent()
147+
"""{"op":"subscribe", "channel":"/api/depth/depth", "key":"$symbol"}"""
95148
}
96149
.toFlux()
97150

98151
return HttpClient.create()
99152
.wiretap(log.isDebugEnabled)
100-
.websocket(65536)
153+
.websocket(262144)
101154
.uri(baseUri)
102155
.handle { inbound, outbound ->
103156
outbound.sendString(subscribeRequests)
104157
.then()
105-
.thenMany(inbound.aggregateFrames(65536).receive().asString())
158+
.thenMany(inbound.aggregateFrames().receive().asString())
106159
}
107-
.filter { it.contains("\"dataType\":\"trade_history\"") }
108-
.map { objectMapper.readValue<HubiMessageFrame<HubiTickDataWrapper>>(it) }
109-
.map { it.data }
160+
.filter { it.contains(""""event":"/api/depth/depth"""") }
161+
.map { objectMapper.readValue<HubiDepthResponse>(it) }
110162
.flatMapIterable {
111163
it.trades
112164
.takeWhile { hubiTickData ->
113165
// hubi trade history response contain history data
114166
val lastTradeEpochMilli =
115167
lastPublishedTimestamp.computeIfAbsent(hubiTickData.symbol) { AtomicLong() }
116-
val isNew = hubiTickData.time.toEpochMilli() > lastTradeEpochMilli.toLong()
168+
val isNew = hubiTickData.timestamp.toEpochMilli() > lastTradeEpochMilli.toLong()
117169
if (isNew) {
118-
lastTradeEpochMilli.set(hubiTickData.time.toEpochMilli())
170+
lastTradeEpochMilli.set(hubiTickData.timestamp.toEpochMilli())
119171
}
120172
isNew
121173
}
122174
.map { hubiTickData ->
123175
TickData(
124-
"${hubiTickData.symbol}${hubiTickData.time}",
125-
hubiTickData.time,
176+
"${hubiTickData.symbol}${hubiTickData.timestamp}",
177+
hubiTickData.timestamp,
126178
hubiTickData.price,
127-
hubiTickData.amount,
179+
hubiTickData.qty,
128180
hubiTickData.symbol,
129181
ExchangeVendor.HUBI,
130-
hubiTickData.type
182+
if (hubiTickData.buyActive) TradeSideType.BUY else TradeSideType.SELL
131183
)
132184
}
133185
.reversed()
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,73 @@
1+
/*
2+
* Copyright 2019 namjug-kim
3+
*
4+
* LINE Corporation licenses this file to you under the Apache License,
5+
* version 2.0 (the "License"); you may not use this file except in compliance
6+
* with the License. You may obtain a copy of the License at:
7+
*
8+
* https://www.apache.org/licenses/LICENSE-2.0
9+
*
10+
* Unless required by applicable law or agreed to in writing, software
11+
* distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
12+
* WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
13+
* License for the specific language governing permissions and limitations
14+
* under the License.
15+
*/
16+
17+
package com.njkim.reactivecrypto.hubi.model
18+
19+
import com.fasterxml.jackson.annotation.JsonProperty
20+
import com.njkim.reactivecrypto.core.common.model.currency.CurrencyPair
21+
import java.math.BigDecimal
22+
import java.time.ZonedDateTime
23+
24+
data class HubiDepthResponse(
25+
@get:JsonProperty("buyDepth")
26+
val buyDepth: List<HubiDepthUnit>,
27+
28+
@get:JsonProperty("sellDepth")
29+
val sellDepth: List<HubiDepthUnit>,
30+
31+
@get:JsonProperty("trades")
32+
val trades: List<HubiTradeUnit>,
33+
34+
@get:JsonProperty("key")
35+
val key: CurrencyPair,
36+
37+
@get:JsonProperty("event")
38+
val event: String
39+
)
40+
41+
data class HubiDepthUnit(
42+
@get:JsonProperty("price")
43+
val price: BigDecimal,
44+
45+
@get:JsonProperty("qty")
46+
val qty: BigDecimal,
47+
48+
@get:JsonProperty("count")
49+
val count: Int,
50+
51+
@get:JsonProperty("iceCount")
52+
val iceCount: Int
53+
)
54+
55+
data class HubiTradeUnit(
56+
@get:JsonProperty("id")
57+
val id: String,
58+
59+
@get:JsonProperty("symbol")
60+
val symbol: CurrencyPair,
61+
62+
@get:JsonProperty("price")
63+
val price: BigDecimal,
64+
65+
@get:JsonProperty("qty")
66+
val qty: BigDecimal,
67+
68+
@get:JsonProperty("buyActive")
69+
val buyActive: Boolean,
70+
71+
@get:JsonProperty("timestamp")
72+
val timestamp: ZonedDateTime
73+
)

reactive-crypto-hubi/src/main/kotlin/com/njkim/reactivecrypto/hubi/model/HubiMessageFrame.kt

+5-5
Original file line numberDiff line numberDiff line change
@@ -24,12 +24,12 @@ data class HubiMessageFrame<T>(
2424
@get:JsonProperty("timestamp")
2525
val timestamp: ZonedDateTime?,
2626

27-
@get:JsonProperty("symbol")
28-
val symbol: CurrencyPair,
27+
@get:JsonProperty("key")
28+
val key: CurrencyPair,
2929

30-
@get:JsonProperty("dataType")
31-
val dataType: String,
30+
@get:JsonProperty("event")
31+
val event: String,
3232

3333
@get:JsonProperty("data")
3434
val data: T
35-
)
35+
)

reactive-crypto-hubi/src/main/kotlin/com/njkim/reactivecrypto/hubi/model/HubiOrderBook.kt

-36
This file was deleted.

reactive-crypto-hubi/src/test/java/com/njkim/reactivecrypto/hubi/HubiWebsocketClientJavaTest.java

+6-4
Original file line numberDiff line numberDiff line change
@@ -17,7 +17,7 @@ public class HubiWebsocketClientJavaTest {
1717
@Test
1818
public void hubi_tick_data_subscribe() {
1919
// given
20-
CurrencyPair targetCurrencyPair = CurrencyPair.parse("BTC", "USDT");
20+
CurrencyPair targetCurrencyPair = CurrencyPair.parse("XBTC", "USD");
2121
Flux<TickData> tickDataFlux = new HubiWebsocketClient()
2222
.createTradeWebsocket(Collections.singletonList(targetCurrencyPair));
2323

@@ -52,12 +52,14 @@ public void hubi_tick_data_subscribe() {
5252
@Test
5353
public void hubi_orderBook_subscribe() {
5454
// given
55-
CurrencyPair targetCurrencyPair = CurrencyPair.parse("BTC", "USDT");
55+
CurrencyPair targetCurrencyPair = CurrencyPair.parse("XBTC", "USD");
5656
Flux<OrderBook> orderBookFlux = new HubiWebsocketClient()
5757
.createDepthSnapshot(Collections.singletonList(targetCurrencyPair));
5858

5959
// when
60-
StepVerifier.create(orderBookFlux.limitRequest(2))
60+
StepVerifier.create(orderBookFlux.limitRequest(5))
61+
// skip first 3 request
62+
.expectNextCount(3)
6163
// then
6264
.assertNext(orderBook -> {
6365
assertThat(orderBook).isNotNull();
@@ -115,4 +117,4 @@ public void hubi_orderBook_subscribe() {
115117
})
116118
.verifyComplete();
117119
}
118-
}
120+
}

0 commit comments

Comments
 (0)