ATrade dashboard
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

169 lines
6.5 KiB

9 years ago
from django.http import HttpResponse, HttpResponseRedirect, Http404
9 years ago
from django.template import loader
from django.shortcuts import render, get_object_or_404
from django.urls import reverse
from django.contrib import messages
from django.db import transaction
9 years ago
from .models import RobotInstance, Trade, ClosedTrade
9 years ago
from .forms import NewTradeForm
9 years ago
import redis
import json
import datetime
9 years ago
def overview(request):
r = redis.StrictRedis(unix_socket_path='/var/run/redis/redis')
robot_instances = RobotInstance.objects.order_by('instanceId')
robot_states = []
index = 0
for robot in robot_instances:
entry = {}
9 years ago
raw_state = r.get(robot.instanceId)
if raw_state is not None:
state = json.loads(str(raw_state, 'utf-8'))
try:
entry['positions'] = state['positions']
del state['positions']
9 years ago
except KeyError:
entry['positions'] = dict()
9 years ago
else:
state = dict()
open_pos_counter = 0
pending_pos_counter = 0
for pos in entry['positions']:
if pos['posState']['tag'] == 'PositionOpen':
open_pos_counter += 1
elif pos['posState']['tag'] == 'PositionWaitingOpen':
pending_pos_counter += 1
entry['open_pos_counter'] = open_pos_counter
entry['pending_pos_counter'] = pending_pos_counter
entry['state'] = json.dumps(state, sort_keys=True, indent=2, separators=(',', ': '))
last_store = r.get(robot.instanceId + ":last_store")
if last_store is not None:
entry['last_store'] = datetime.datetime.utcfromtimestamp(float(str(last_store, 'utf-8')[:-1]))
9 years ago
index += 1
entry['index'] = index
entry['instance_id'] = robot.instanceId
robot_states.append(entry)
9 years ago
template = loader.get_template('dashboard/overview.html')
context = {
'robot_instances' : robot_instances,
'robot_states' : robot_states
}
return HttpResponse(template.render(context, request))
def add_instance(request):
instance_id = request.POST['instance_id']
if instance_id == "" or RobotInstance.objects.filter(instanceId=instance_id).count() > 0:
messages.error(request, 'Invalid instance ID specified')
else:
new_instance = RobotInstance(instanceId=instance_id)
new_instance.save()
return HttpResponseRedirect(reverse('overview'))
def delete_instance(request, instance_id):
instance = get_object_or_404(RobotInstance, instanceId=instance_id)
instance.delete()
return HttpResponseRedirect(reverse('overview'))
def trades_index(request):
9 years ago
now = datetime.datetime.utcnow()
new_trade_form = NewTradeForm(initial={'timestamp' : now})
trades = Trade.objects.all()
template = loader.get_template('dashboard/trades.html')
context = {
9 years ago
'trades' : trades,
'new_trade_form' : new_trade_form
}
return HttpResponse(template.render(context, request))
9 years ago
def delete_trade(request, trade_id):
trade = get_object_or_404(Trade, pk=trade_id)
trade.delete()
return HttpResponseRedirect(reverse('trades_index'))
def add_trade(request):
if request.method == 'POST':
form = NewTradeForm(request.POST)
if form.is_valid():
d = form.cleaned_data
quantity_multiplier = 1
if d['operation'] == 'sell':
quantity_multiplier = -1
trade = Trade(account=d['account'], security=d['security'], price=d['price'], quantity=quantity_multiplier * d['quantity'],
volume=d['volume'], volumeCurrency=d['volumeCurrency'], strategyId=d['strategyId'], signalId=d['signalId'], timestamp=d['timestamp'], balanced=False)
trade.save()
return HttpResponseRedirect(reverse('trades_index'))
else:
trades = Trade.objects.all()
template = loader.get_template('dashboard/trades.html')
context = {
'trades' : trades,
'new_trade_form' : form
}
return HttpResponse(template.render(context, request))
raise Http404("Invalid method")
@transaction.atomic
def aggregate_unbalanced_trades():
unbalanced_trades = Trade.objects.filter(balanced=False).order_by('timestamp')
balanced_trades = []
balances = {}
for trade in unbalanced_trades:
balance_key = '/'.join([trade.account, trade.security, trade.strategyId])
try:
balance_entry = balances[balance_key]
except KeyError:
balance_entry = { 'balance' : 0}
print('ts:', trade.timestamp)
if balance_entry['balance'] == 0:
print('new entry: ', balance_key)
balance_entry['balance'] = trade.quantity
direction = ''
if trade.quantity > 0:
direction='long'
else:
direction='short'
balance_entry['trade'] = ClosedTrade(account=trade.account, security=trade.security, entryTime=trade.timestamp, profitCurrency=trade.volumeCurrency,
profit=(-trade.price * trade.quantity), strategyId=trade.strategyId, direction=direction)
balance_entry['ks'] = trade.volume / (trade.price * abs(trade.quantity))
balance_entry['trade_ids'] = [trade.pk]
else:
print('update entry: ', balance_key)
balance_entry['balance'] += trade.quantity
balance_entry['trade'].profit += -trade.price * trade.quantity
balance_entry['ks'] += trade.volume / (trade.price * abs(trade.quantity))
balance_entry['ks'] /= 2
balance_entry['trade_ids'].append(trade.pk)
print('updated: ', balance_entry['balance'])
if balance_entry['balance'] == 0:
balance_entry['trade'].profit *= balance_entry['ks']
balance_entry['trade'].exitTime = trade.timestamp
balanced_trades.append((balance_entry['trade'], balance_entry['trade_ids']))
balances[balance_key] = balance_entry
for trade, trade_ids in balanced_trades:
trade.save()
for trade_id in trade_ids:
tr = Trade.objects.get(pk=trade_id)
tr.balanced = True
tr.save()
def closed_trades_index(request):
aggregate_unbalanced_trades()
closed_trades = ClosedTrade.objects.all()
template = loader.get_template('dashboard/closed_trades.html')
context = {
'closed_trades' : closed_trades
}
return HttpResponse(template.render(context, request))