Generated: Mon 2015-06-08 11:27 AST
Source file: /home/kevin/Development/custom apps/valuehorizon-forex/forex/models.py
Stats: 113 executed, 5 missed, 11 excluded, 98 ignored
from django.db import modelsfrom django.db.models import Managerfrom django.core.validators import MinValueValidator, ValidationError# Import misc packagesimport numpy as npfrom datetime import date, timedeltafrom decimal import Decimalfrom pandas import DataFrame, date_rangePRICE_PRECISION = 4DATEFRAME_START_DATE = date(2005,1,1)class Currency(models.Model): """ Represents a currency according to ISO 4217 standards. """ name = models.CharField(max_length=255) symbol = models.CharField(max_length=10, unique=True) ascii_symbol = models.CharField(max_length=20, null=True, blank=True) num_code = models.IntegerField(null=True, blank=True) digits = models.IntegerField(null=True, blank=True) # Digits after decimal (minor unit) description = models.TextField(blank=True) # Cached Data date_modified = models.DateTimeField(null=True, blank=True, editable=False, auto_now=True) date_created = models.DateTimeField(null=True, blank=True, editable=False, auto_now_add=True) class Meta: verbose_name_plural = 'Currencies' verbose_name = 'Currency' ordering = ['name', 'symbol'] def __unicode__(self): return u'%s, %s' % (unicode(self.name), unicode(self.symbol)) def generate_dataframe(self, start_date=None, end_date=None): """ """ first_series_point = CurrencyPrice.objects.filter(currency=self)[0] last_series_point = CurrencyPrice.objects.filter(currency=self).reverse()[0] start_date = first_series_point.date if start_date == None else max(first_series_point.date, start_date) temp_start_date = start_date - timedelta(days=3) # Add lag end_date = last_series_point.date if end_date == None else min(last_series_point.date, end_date) currency_date = CurrencyPrice.objects.filter(currency=self, date__gte=temp_start_date, date__lte=end_date).values_list('date', 'ask_price', 'bid_price') currency_data_array = np.core.records.fromrecords(currency_date, names=['DATE', "ASK", "BID"]) df = DataFrame.from_records(currency_data_array, index='DATE').astype(float) df['MID'] = (df['ASK'] + df['BID']) / 2.0 df['CHANGE'] = df['MID'].pct_change() required_dates = date_range(start_date,end_date) df = df.reindex(required_dates) df = df.fillna(method='ffill') return dfclass CurrencyPriceManager(Manager): """ Adds some added functionality """ def generate_dataframe(self, symbols=None, date_index=None, price_type="mid"): """ Generate a dataframe consisting of the currency prices (specified by symbols) from the start to end date """ # Set defaults if necessary if symbols == None: symbols = list(Currency.objects.all().values_list('symbol', flat=True)) try: start_date = date_index[0] end_date = date_index[-1] except: start_date = DATEFRAME_START_DATE end_date = date.today() date_index = date_range(start_date, end_date) currency_price_data = CurrencyPrice.objects.filter(currency__symbol__in=symbols, date__gte=date_index[0], date__lte=date_index[-1]).values_list('date', 'currency__symbol', 'ask_price', 'bid_price') try: forex_data_array = np.core.records.fromrecords(currency_price_data, names=['date', 'symbol', 'ask_price', 'bid_price']) except IndexError: forex_data_array = np.core.records.fromrecords([(date(1900,1,1) , "", 0, 0)], names=['date', 'symbol', 'ask_price', 'bid_price']) df = DataFrame.from_records(forex_data_array, index='date') df['date'] = df.index if price_type == "mid": df['price'] = (df['ask_price'] + df['bid_price']) / 2 elif price_type == "ask_price": df['price'] = df['ask_price'] elif price_type == "bid_price": df['price'] = df['bid_price'] else: raise ValueError("price_type must be on of 'ask', 'bid' or 'mid'") df = df.pivot(index='date', columns='symbol', values='price') df = df.reindex(date_index) df = df.fillna(method="ffill") unlisted_symbols = list(set(symbols) - set(df.columns)) for unlisted_symbol in unlisted_symbols: df[unlisted_symbol] = np.nan df = df[symbols] return dfclass CurrencyPrice(models.Model): """ Represents a currency price to US """ currency = models.ForeignKey(Currency) date = models.DateField() # Price Data per $1 of US ask_price = models.DecimalField(max_digits=20, decimal_places=PRICE_PRECISION, validators=[MinValueValidator(Decimal('0.00'))]) bid_price = models.DecimalField(max_digits=20, decimal_places=PRICE_PRECISION, validators=[MinValueValidator(Decimal('0.00'))], blank=True, null=True) # Cached Data date_modified = models.DateTimeField(null=True, blank=True, editable=False, auto_now=True) date_created = models.DateTimeField(null=True, blank=True, editable=False, auto_now_add=True) # Add custom managers objects=CurrencyPriceManager() class Meta: verbose_name_plural = 'Currency Prices' verbose_name = 'Currency Price' ordering = ['date', ] unique_together=['date', 'currency'] get_latest_by = "date" def __unicode__(self): return u'%s, %s' % (unicode(self.currency), unicode(self.date),) def save(self, *args, **kwargs): """ Sanitation checks """ if self.ask_price < 0: raise ValidationError("Ask price must be greater than zero") if self.bid_price < 0: raise ValidationError("Bid price must be greater than zero") if self.ask_price < self.bid_price: raise ValidationError("Ask price must be at least Bid price") super(CurrencyPrice, self).save(*args, **kwargs) # Call the "real" save() method. @property def mid_price(self): """ Compute the mid point between the bid and ask prices """ return (self.ask_price + self.bid_price) / Decimal('2.0') @property def spread(self): """ Compute the difference between bid and ask prices """ return (self.ask_price - self.bid_price) @property def ask_price_us(self): """ Calculate the ask_price in USD. This is the inverse of the ask price. """ if self.ask_price != 0: return 1 / Decimal(str(self.ask_price)) else: raise ZeroDivisionError('Ask price is zero') @property def bid_price_us(self): """ Calculate the bid_price in USD. This is the inverse of the bid price. """ if self.bid_price != 0: return 1 / Decimal(str(self.bid_price)) else: raise ZeroDivisionError('Bid price is zero')def convert_currency(from_symbol, to_symbol, value, date): """ """ if from_symbol == to_symbol: return value from_currency = Currency.objects.get(symbol=from_symbol) try: from_currency_price = CurrencyPrice.objects.get(currency=from_currency, date=date).mid_price except CurrencyPrice.DoesNotExist: print "Cannot fetch prices for %s on %s" % (str(from_currency), str(date)) return None to_currency = Currency.objects.get(symbol=to_symbol) try: to_currency_price = CurrencyPrice.objects.get(currency=to_currency, date=date).mid_price except CurrencyPrice.DoesNotExist: print "Cannot fetch prices for %s on %s" % (str(to_currency), str(date)) return None if type(value) == float: output = (value / float(from_currency_price)) * float(to_currency_price) elif type(value) == Decimal: output = Decimal(format((value / from_currency_price) * to_currency_price, '.%sf' % str(PRICE_PRECISION))) else: output = None return output