forked from lausser/piebot
-
Notifications
You must be signed in to change notification settings - Fork 0
/
PieBot.py
248 lines (190 loc) · 8.71 KB
/
PieBot.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
from functions import *
import argparse
import gc
import schedule
import signal
from prometheus_client import start_http_server, Gauge, Enum
pre_flight_checks()
balance = Gauge('coin_balance', 'The amount of coins', ['coin', 'account'])
price = Gauge('coin_price', 'The price of a coin in USDT', ['coin', 'account'])
state = Enum('coin_state', 'The management state of a coin', ['coin', 'account'], states=['managed', 'unmanaged']) # managed, unmanaged
# Hard codes the minimum order value
min_order_value = 0.25
# Checks whether the Rebalance threshold has been defined, and allows the bot to run if it hasn't
try:
rebalance_threshold
except NameError:
rebalance_threshold = None
if rebalance_threshold is None:
uses_threshold = False
else:
if rebalance_threshold > 0:
uses_threshold = True
else:
uses_threshold = False
# Buy more coins at a regular interval
def buy(pairs):
# Let users know the bot has been called and is running
print()
current_time(True)
print(colored("Buy", "yellow"))
print(colored("Placing orders...", "cyan"))
total_portfolio_value = get_portfolio_value(pairs, True)
total_usdt_reserve = (total_portfolio_value / 100) * (usdt_reserve * 100)
total_usdt_value = get_coin_balance("USDT")
total_usdt_available = total_usdt_value - total_usdt_reserve
required_usdt = buy_order_value * len(pairs)
if required_usdt <= total_usdt_available:
for pair in pairs:
order_value = buy_order_value
if environment == "production":
order_confirmed = False
order = order_buy(pair[1], order_value)
time.sleep(0.1)
if order.status_code == 200:
order_confirmed = True
print_value = round(order_value, 2)
current_time(True)
print(str(print_value) + " USDT - " + pair[0], end=" ")
print(colored("[BUY]", "green"))
if not order_confirmed:
print(order.status_code, order.reason)
print(order.content)
else:
print_value = round(order_value, 2)
current_time(True)
print(str(print_value) + " USDT - " + pair[0], end=" ")
print(colored("[BUY]", "green"))
else:
print(colored("Not enough USDT available (have {}, need {})".format(total_usdt_available, required_usdt), "yellow"))
total_portfolio_value = get_portfolio_value(pairs, True)
print("Total portfolio value is {:.4f}USDT ({:.4f}stable)".format(total_portfolio_value, total_usdt_value))
gc.collect()
print(colored("Waiting to be called...", "cyan"))
# Rebalance all coins so they are on target
def rebalance(pairs):
# Let users know the bot has been called and is running
print()
current_time(True)
print(colored("Rebalance", "yellow"))
print(colored("Placing orders...", "cyan"))
order_data = []
total_portfolio_value = 0
for pair in pairs:
coin_balance = get_coin_balance(pair[0])
coin_price = get_coin_price(pair[1])
pair_value = coin_balance * coin_price
order_data.append([pair[0], pair[1], coin_price, pair_value])
total_portfolio_value += pair_value
# Equally divide the balance by the number of coins, so we know the target value each coin should aim for
target_per_coin = total_portfolio_value / len(pairs)
print("target_per_coin is {}".format(target_per_coin))
buy_orders_data = []
sell_orders_data = []
for pair in order_data:
coin_price = pair[2]
pair_value = pair[3]
# The coin value is over target
if pair_value > target_per_coin:
difference = pair_value - target_per_coin
if difference >= min_order_value:
if uses_threshold:
difference_percentage = (((pair_value - target_per_coin) / target_per_coin) * 100)
print("difference+ of {} is {:.2f} ({:.2f}%)".format(pair[0], difference, difference_percentage))
if difference_percentage >= (rebalance_threshold * 100):
order_value = difference / coin_price
sell_orders_data.append([pair[0], pair[1], order_value, difference])
else:
order_value = difference / coin_price
sell_orders_data.append([pair[0], pair[1], order_value, difference])
# The coin value is under target
elif pair_value < target_per_coin:
difference = target_per_coin - pair_value
if difference >= min_order_value:
if uses_threshold:
difference_percentage = 100 if pair_value == 0 else (((target_per_coin - pair_value) / pair_value) * 100)
print("difference- of {} is {:.2f} ({:.2f}%)".format(pair[0], difference, difference_percentage))
if difference_percentage >= (rebalance_threshold * 100):
order_value = difference
buy_orders_data.append([pair[0], pair[1], order_value, difference])
else:
order_value = difference
buy_orders_data.append([pair[0], pair[1], order_value, difference])
if len(sell_orders_data) >= 1:
for order in sell_orders_data:
if environment == "production":
order_confirmed = False
order_request = order_sell(order[1], order[2])
time.sleep(0.1)
if order_request.status_code == 200:
order_confirmed = True
print_value = round(order[3], 2)
current_time(True)
print(str(print_value) + " USDT - " + order[0], end=" ")
print(colored("[SELL]", "magenta"))
if not order_confirmed:
print(order_request.status_code, order_request.reason)
print(order_request.content)
else:
print_value = round(order[3], 2)
current_time(True)
print(str(print_value) + " USDT - " + order[0], end=" ")
print(colored("[SELL]", "magenta"))
if len(buy_orders_data) >= 1:
for order in buy_orders_data:
if environment == "production":
order_confirmed = False
order_request = order_buy(order[1], order[2])
time.sleep(0.1)
if order_request.status_code == 200:
order_confirmed = True
print_value = round(order[3], 2)
current_time(True)
print(str(print_value) + " USDT - " + order[0], end=" ")
print(colored("[BUY]", "green"))
if not order_confirmed:
print(order_request.status_code, order_request.reason)
print(order_request.content)
else:
print_value = round(order[3], 2)
current_time(True)
print(str(print_value) + " USDT - " + order[0], end=" ")
print(colored("[BUY]", "green"))
total_orders = len(sell_orders_data) + len(buy_orders_data)
if total_orders == 0:
current_time(True)
print(colored("No coins were eligible to be rebalanced", "yellow"))
del order_data
del buy_orders_data
del sell_orders_data
total_portfolio_value = get_portfolio_value(pairs, True)
print("Total portfolio value is {:.4f}USDT".format(total_portfolio_value))
gc.collect()
print(colored("Waiting to be called...", "cyan"))
def update_exporter(pairs):
pairs = get_account_details()
for pair in pairs:
balance.labels(pair["coin"], pair["account"]).set(pair["balance"])
price.labels(pair["coin"], pair["account"]).set(pair["price"])
state.labels(pair["coin"], pair["account"]).state(pair["state"])
if environment == "production":
print(colored("Waiting to be called...", "cyan"))
if rebalance_frequency > 0:
schedule.every(rebalance_frequency).hours.at(":00").do(rebalance, pairs=pair_list)
schedule.every(buy_frequency).hours.at(":30").do(buy, pairs=pair_list)
schedule.every(5).minutes.do(update_exporter, pairs=pair_list)
start_http_server(19000)
stop = StopSignal()
while not stop.stop_now:
schedule.run_pending()
time.sleep(1)
else:
parser = argparse.ArgumentParser()
parser.add_argument("task")
args = parser.parse_args()
if (args.task == "buy") or (args.task == "Buy"):
buy(pairs=pair_list)
elif (args.task == "rebalance") or (args.task == "Rebalance"):
rebalance(pairs=pair_list)
else:
print(colored("Please specify which task you want to run", "red"))