Execution Algorithms 101

Problem Definition

  • Two variables: Time and Price
  • Liquidity
  • Bid-Ask Spread

A simple algorithm

def buy_sell_logic(self, update):
    # Check if the time limit has exceeded
    if self.watch.ts >= self.exec_request.end_time:
        self._time_exceeded(update)
        return

    if self.om.num_orders() == 0:
        # Send limit order
        order_price = self.book.passive_price(self.buysell, num_ticks=0)
        order = self.om.create_order('L', self.buysell, self.size, order_price)
        self.om.send_order(order)

How to check if we did a good job?

Metrics

  • Arrival Mid Price
  • Arrival Aggressive Price
  • VWAP 
  • TWAP 
  • Implementation Shortfall

Execution Algorithm Improvements

def divide_orders(self):
    self._sub_orders = []
    for i in range(self.num_orders):
        order_size = max(self.size // self._num_orders, 100)
        self._sub_orders.append(order_size) 

def buy_sell_logic(self, update):
    # Check if the time limit has exceeded
    if self.watch.ts >= self.exec_request.end_time:
        self._time_exceeded(update)
            return
    
     # Get the current bucket number
     bucket = self._get_bucket()

     if self.om.num_orders() == 0 and bucket > self._current_bucket:
        # Send limit order
        order_size = self._sub_orders[bucket]
        order_price = self.book.passive_price(self.buysell, num_ticks=0)
        order = self.om.create_order('L', self.buysell, order_size, order_price)
        self.om.send_order(order)

More Improvements

Passive TWAP algorithm

def divide_orders(self):
    self._sub_orders = []
    for i in range(self.num_orders):
        order_size = max(self.size // self._num_orders, 100)
        self._sub_orders.append(order_size)
    
def buy_sell_logic(self, update):
    # Check if the time limit has exceeded
    if self.watch.ts >= self.exec_request.end_time:
        self._time_exceeded(update)
        return

    # Get the latest top price
    order_price = self.book.passive_price(self.buysell, 0)

    if self.om.num_orders() > 0:
        if time_diff >= self._interval:
            # Modify to improve/aggressive price
            self.om.modify_price(self.buysell, order_price)

    # Get the current bucket number
    bucket = self._get_bucket()

    if self.om.num_orders() == 0 and bucket > self._current_bucket:
        # Send limit order
        order_size = self._sub_orders[bucket]
        order_price = self.book.passive_price(self.buysell, num_ticks=0)
        order = self.om.create_order('L', self.buysell, order_size, order_price)
        self.om.send_order(order)

Production setup

  • API server receives the orders and passes them to the run_execs.py process through RabbitMQ.
  • API server maintains the status of the order including the benchmarks.
  • Benchmarks as well as the order status updates are sent from run_execs.py process.

Production setup

Issues with prices

  • Nanex process fetches the live prices. We push the "important" updates in binary format to kafka. (We have not benchmarked this operation.)
  • For NASDAQ, the prices are fetched from Google/Yahoo/Robinhood. This affects both the trading and the benchmarking.
  • There are often issues with the price APIs - creating more issues in reliably measuring the performance.
  • Backtests are likely to have inferior results on average - this is due to the presence of hidden orders in market.

Issues with IB

  • BUY orders have to wait till SELL orders get completed.

APEX results by securities

APEX results by securities

Security Average PNL (bps)
FSTA 6.11
FMAT 5.86
VOX 5.49
VBR 4.61
FIDU 4.27

APEX results by securities

Security Average PNL (bps)
IVV -2.68
VUG -1.45
IVE -1.27
XLE -1.00
REM -0.69

APEX results by date

APEX results by date

Date Total PNL ($)
2017-07-27 4435
2017-08-15 2642
2017-08-23 1725
2017-03-15 1066
​2017-08-18 932

APEX results by date

Date Total PNL ($)
2017-08-28 -2353
2017-06-14 -1964
2017-06-09 -685
2017-06-22 -293
2017-09-06 -267

APEX results by size

Size Average PNL (bps)
<500 1.20
>=500 1.05

Next steps

Execution Algorithms 101

By Hardik Patel

Execution Algorithms 101

  • 861