trading_bot.py 3.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110
  1. import random
  2. from datetime import timedelta, datetime
  3. from math import log2, ceil
  4. import model
  5. order_expiry = 43200
  6. def place_order(ownable_id):
  7. """
  8. places a new order according to the algorithm described in `assets/follower.py`
  9. :param ownable_id: on which ownable to place the order
  10. :return: True iff a new order was placed
  11. """
  12. cheapest_buy_order, best_sell_order = model.abs_spread(ownable_id)
  13. if cheapest_buy_order is None or best_sell_order is None:
  14. return False
  15. investors_id = model.bank_id()
  16. orders = model.get_ownable_orders(investors_id, ownable_id)
  17. orders = [random.choice(orders) for _ in range(int(ceil(log2(len(orders)))))]
  18. amounts = [order[3] for order in orders]
  19. amount = sum(amounts) / len(amounts)
  20. expiry = datetime.strptime(model.current_db_time(), '%Y-%m-%d %H:%M:%S') + timedelta(minutes=order_expiry)
  21. model.place_order(buy=bool(random.getrandbits(1)),
  22. ownership_id=model.get_ownership_id(ownable_id, investors_id),
  23. limit=random.uniform(cheapest_buy_order, best_sell_order),
  24. stop_loss=False,
  25. amount=amount,
  26. expiry=expiry)
  27. return True
  28. def main(): # TODO testen
  29. """the initial part of the trading bot algorithm"""
  30. if model.get_user_orders(model.bank_id()):
  31. raise AssertionError('The trading bot already has some orders.')
  32. if input('Are you sure you want to place the initial orders? (type in "yes" or something else):') == 'yes':
  33. for ownable_id in model.ownable_ids():
  34. if ownable_id != model.currency_id():
  35. place_order(ownable_id)
  36. else:
  37. print('Not placing orders.')
  38. model.cleanup()
  39. if __name__ == '__main__':
  40. main()
  41. def notify_expired_orders(orders):
  42. for order in orders:
  43. # order_id = order[0]
  44. ownership_id = order[1]
  45. # check if that was one of the bots orders
  46. bank_ownership_id = model.get_ownership_id(ownership_id, model.bank_id())
  47. if ownership_id != bank_ownership_id:
  48. continue
  49. # create a new order
  50. ownable_id = model.ownable_id_by_ownership_id(ownership_id)
  51. place_order(ownable_id)
  52. def notify_order_traded(ownable_id):
  53. """
  54. Called after a trade has been done and now the auctions are finished.
  55. :param ownable_id: the ownable that was traded
  56. :return: True iff a new order was placed
  57. """
  58. model.connect()
  59. if ownable_id == model.currency_id():
  60. return False
  61. ownership_id = model.get_ownership_id(ownable_id, model.bank_id())
  62. model.cursor.execute('''
  63. SELECT rowid, amount, expiry
  64. FROM orders
  65. WHERE ownership_id = ?
  66. -- no need for ORDER since the bot should have only one order
  67. -- ORDER BY rowid DESC -- equivalent to ordering by time created
  68. LIMIT 1
  69. ''', (ownership_id,))
  70. data = model.cursor.fechtall()
  71. # TODO Neu überdenken, das geht so nicht, wenn die order schon ausgeführt ist, ist sie nicht mehr in der DB
  72. if not data:
  73. return place_order(ownable_id)
  74. my_last_order = model.cursor.fechtone()
  75. last_order_id = my_last_order[0]
  76. last_amount = my_last_order[1]
  77. expiry = my_last_order[2]
  78. dt_order_placed = datetime.strptime(expiry, '%Y-%m-%d %H:%M:%S') - timedelta(minutes=order_expiry)
  79. model.cursor.execute('''
  80. SELECT 2 * SUM(amount) >= ?
  81. FROM transactions
  82. WHERE ownable_id = ?
  83. AND dt >= ?
  84. ''', (last_amount, ownable_id, dt_order_placed))
  85. if model.cursor.fechtone()[0]:
  86. model.delete_order(last_order_id)
  87. return place_order(ownable_id)
  88. return False