{binance} Spot Trading: Liquidity

In previous posts we looked at creating market orders and limit orders with {binance}. We saw a couple of successful trades. However, sometimes trades are not successful and the orders are not filled. Let’s try to understand why.

The setup will be the same as the previous posts and we’ll use the Binance testnet. Let’s take a look at our testnet account balance.

spot_account_balances()
# A tibble: 8 × 3
  asset      free locked
  <chr>     <dbl>  <dbl>
1 BNB      971.       0 
2 BTC        1.41     0 
3 BUSD   37438.     116.
4 ETH       89.3      0 
5 LTC      500        0 
6 TRX   498722.       0 
7 USDT   10111.       0 
8 XRP    50000        0 

Expired Orders

We’ve got plenty of TRX. Let’s try to sell 5000 TRX in exchange for USDT.

spot_new_market_order(symbol = "TRXUSDT", side = "SELL", quantity = 5000)
# A tibble: 1 × 11
  symbol  order_id transact_time       price orig_qty exec_qty status  time_in_force type   side
  <chr>      <int> <dttm>              <dbl>    <dbl>    <dbl> <chr>   <chr>         <chr>  <chr>
1 TRXUSDT      650 2021-11-07 07:11:42     0     5000        0 EXPIRED GTC           MARKET SELL

The order was not successful (status is EXPIRED). What went wrong? It’s not like we don’t have enough TRX. The problem is that there’s nobody who wants to buy our TRX with USDT! A successful trade requires two parties: a seller and a buyer.

Order Book

The order book reflects the orders which are waiting to be executed. Let’s retrieve the order book for all pairs on the testnet.

order_book <- market_price_ticker() %>% pull(symbol) %>% market_order_book()
# A tibble: 20 × 4
   symbol  last_update_id bids              asks             
   <chr>            <int> <list>            <list>           
 1 BNBBUSD              3 <tibble [1 × 2]>  <tibble [0 × 2]> 
 2 BTCBUSD             66 <tibble [1 × 2]>  <tibble [1 × 2]> 
 3 ETHBUSD              4 <tibble [1 × 2]>  <tibble [0 × 2]> 
 4 LTCBUSD              2 <tibble [0 × 2]>  <tibble [0 × 2]> 
 5 TRXBUSD             40 <tibble [1 × 2]>  <tibble [0 × 2]> 
 6 XRPBUSD              2 <tibble [0 × 2]>  <tibble [0 × 2]> 
 7 BNBUSDT         432661 <tibble [15 × 2]> <tibble [3 × 2]> 
 8 BTCUSDT         634251 <tibble [44 × 2]> <tibble [20 × 2]>
 9 ETHUSDT           2352 <tibble [9 × 2]>  <tibble [10 × 2]>
10 LTCUSDT              8 <tibble [0 × 2]>  <tibble [0 × 2]> 
11 TRXUSDT             50 <tibble [0 × 2]>  <tibble [0 × 2]> 
12 XRPUSDT             18 <tibble [0 × 2]>  <tibble [0 × 2]> 
13 BNBBTC               8 <tibble [1 × 2]>  <tibble [0 × 2]> 
14 ETHBTC              22 <tibble [1 × 2]>  <tibble [0 × 2]> 
15 LTCBTC               2 <tibble [0 × 2]>  <tibble [0 × 2]> 
16 TRXBTC               2 <tibble [0 × 2]>  <tibble [0 × 2]> 
17 XRPBTC               3 <tibble [0 × 2]>  <tibble [0 × 2]> 
18 LTCBNB               2 <tibble [0 × 2]>  <tibble [0 × 2]> 
19 TRXBNB               2 <tibble [0 × 2]>  <tibble [0 × 2]> 
20 XRPBNB               2 <tibble [0 × 2]>  <tibble [0 × 2]>

The asks are orders from sellers (these orders specify the price they are asking) and the bids are orders from buyers (these orders specify the price that they are bidding). Interesting! So there are a few pairs (like BNB/USDT, BTC/USDT and ETH/USDT) where there is a lot of interest in trading. But other pairs have only a few orders (or none at all!).

Let’s take a closer look at the order book for TRX/USDT and TRX/BUSD.

order_book %>%
  filter(symbol %in% c("TRXUSDT", "TRXBUSD")) %>%
  select(-last_update_id) %>%
  pivot_longer(bids:asks, names_to = "type", values_to = "orders") %>%
  unnest(cols = c(orders))
# A tibble: 4 × 4
  symbol  type  price      qty
  <chr>   <chr> <dbl>    <dbl>
1 TRXBUSD bids  0.15     773.1
2 TRXBUSD bids  0.14     167  
3 TRXUSDT asks  0.1   336429. 
4 TRXUSDT asks  0.102   2000  

There are just bids for TRX/BUSD and only asks for TRX/USDT. This means that there’s nobody interested in buying our TRX for USDT, but there is interest in buying TRX for BUSD.

Filling Orders

Let’s change our order then, offering to sell 200 TRX for BUSD at the highest bid price.

spot_new_limit_order(symbol = "TRXBUSD", side = "SELL", quantity = 200, price = 0.15)
# A tibble: 1 × 11
  symbol  order_id transact_time       price orig_qty exec_qty status time_in_force type  side
  <chr>      <int> <dttm>              <dbl>    <dbl>    <dbl> <chr>  <chr>         <chr> <chr>
1 TRXBUSD      189 2021-11-07 07:41:13  0.15      200      200 FILLED GTC           LIMIT SELL

Our order was successful (status is FILLED). Checking back on the order book we see that the quantity of the bid has dropped from 773.1 TRX to 573.1 TRX because we sold them 200 TRX.

# A tibble: 2 × 4
  symbol  type  price   qty
  <chr>   <chr> <dbl> <dbl>
1 TRXBUSD bids   0.15 573.1
2 TRXBUSD bids   0.14 167  

Partially Filling Orders

The previous order was filled completely, meaning that we managed to sell the full quantity that we offered. But what happens if the order can’t be completed fully?

Somebody still wants to buy another 573.1 TRX at a price of 0.15 BUSD per TRX. What happens if we offer to sell a larger amount at that price?

spot_new_limit_order(symbol = "TRXBUSD", side = "SELL", quantity = 1000, price = 0.15)
# A tibble: 1 × 11
  symbol  order_id transact_time       price orig_qty exec_qty status           time_in_force type  side
  <chr>      <int> <dttm>              <dbl>    <dbl>    <dbl> <chr>            <chr>         <chr> <chr>
1 TRXBUSD      190 2021-11-07 07:44:36  0.15     1000    573.1 PARTIALLY_FILLED GTC           LIMIT SELL

Aha! So now the status is PARTIALLY_FILLED. The original quantity for the order (orig_qty) was 1000 TRX but only 573.1 TRX of that has been executed (exec_qty).

Time in Force

We can be more specific about the way that we want our order to be handled using the time_in_force option. This specifies how long the order is valid and dictates what happens when there are no matching orders in the order book. Possible values for time_in_force are:

  • "GTC" — (Good Til Canceled) order will remain on the book unless canceled;
  • "IOC" — (Immediate Or Cancel) try to fill as much as possible before it expires; or
  • "FOK" — (Fill Or Kill) Order will expire if full order cannot be filled.

Let’s look at the updated order book for TRX/BUSD.

# A tibble: 3 × 4
  symbol  type  price   qty
  <chr>   <chr> <dbl> <dbl>
1 TRXBUSD bids   0.13  1000
2 TRXBUSD asks   0.14  3000
3 TRXBUSD asks   0.15  5000

Somebody is offering to buy 1000 TRX at 0.13 BUSD per TRX. Let’s offer more TRX than they are requesting but specify "FOK" for time_in_force.

spot_new_limit_order(symbol = "TRXBUSD", side = "SELL", quantity = 5000, price = 0.13, time_in_force = "FOK")
# A tibble: 1 × 11
  symbol  order_id transact_time       price orig_qty exec_qty status  time_in_force type  side
  <chr>      <int> <dttm>              <dbl>    <dbl>    <dbl> <chr>   <chr>         <chr> <chr>
1 TRXBUSD      208 2021-11-07 08:10:47  0.13     5000        0 EXPIRED FOK           LIMIT SELL

The full order could not be filled (we offered 5000 TRX but there was only 1000 TRX requested), so it expired. The executed quantity (exec_qty) is zero: no trading happened.

Let’s try again with time_in_force set to "IOC".

spot_new_limit_order(symbol = "TRXBUSD", side = "SELL", quantity = 5000, price = 0.13, time_in_force = "IOC")
# A tibble: 1 × 11
  symbol  order_id transact_time       price orig_qty exec_qty status  time_in_force type  side
  <chr>      <int> <dttm>              <dbl>    <dbl>    <dbl> <chr>   <chr>         <chr> <chr>
1 TRXBUSD      209 2021-11-07 08:10:52  0.13     5000     1000 EXPIRED IOC           LIMIT SELL

That order has also expired but now the executed quantity is 1000 TRX. So, 1000 TRX was requested and we offered 5000 TRX. Of that, 1000 TRX was sold and the balace of our order (the remaining 4000 TRX) was cancelled.

Some time later there’s another 1000 TRX requested at the same price. Let’s try with time_in_force set to "GTC".

spot_new_limit_order(symbol = "TRXBUSD", side = "SELL", quantity = 5000, price = 0.13, time_in_force = "GTC")
# A tibble: 1 × 11
  symbol  order_id transact_time       price orig_qty exec_qty status           time_in_force type  side
  <chr>      <int> <dttm>              <dbl>    <dbl>    <dbl> <chr>            <chr>         <chr> <chr>
1 TRXBUSD      211 2021-11-07 08:11:05  0.13     5000     1000 PARTIALLY_FILLED GTC           LIMIT SELL

Aha! So now the status of the order is PARTIALLY_FILLED. Again the executed quantity is 1000 TRX, meaning that we’ve sold the full quantity requested. However, now, because we specified "GTC", the balance of the order (another 4000 TRX) is still pending. The order persists on the books until either we cancel it or it is filled.