In [5]:
%matplotlib inline

In [6]:
from __future__ import (absolute_import, division, print_function,
                        unicode_literals)


import argparse
import collections
import datetime


import backtrader as bt


class St(bt.SignalStrategy):
    params = (
        ('pfast', 13),
        ('pslow', 50),
        ('printdata', False),
        ('stake', 1000),
        ('short', False),
    )

    def __init__(self):
        self.sfast = bt.indicators.SMA(period=self.p.pfast)
        self.sslow = bt.indicators.SMA(period=self.p.pslow)
        self.cover = bt.indicators.CrossOver(self.sfast, self.sslow)
        if self.p.short:
            self.signal_add(bt.SIGNAL_LONGSHORT, self.cover)
        else:
            self.signal_add(bt.SIGNAL_LONG, self.cover)

    def start(self):
        super(self.__class__, self).start()
        if self.p.printdata:
            txtfields = list()
            txtfields.append('Len')
            txtfields.append('Datetime')
            txtfields.append('Open')
            txtfields.append('High')
            txtfields.append('Low')
            txtfields.append('Close')
            txtfields.append('Volume')
            txtfields.append('OpenInterest')
            print(','.join(txtfields))

    def next(self):
        super(self.__class__, self).next()
        if self.p.printdata:
            # Print only 1st data ... is just a check that things are running
            txtfields = list()
            txtfields.append('%04d' % len(self))
            txtfields.append(self.data.datetime.datetime(0).isoformat())
            txtfields.append('%.2f' % self.data0.open[0])
            txtfields.append('%.2f' % self.data0.high[0])
            txtfields.append('%.2f' % self.data0.low[0])
            txtfields.append('%.2f' % self.data0.close[0])
            txtfields.append('%.2f' % self.data0.volume[0])
            txtfields.append('%.2f' % self.data0.openinterest[0])
            print(','.join(txtfields))


_TFRAMES = collections.OrderedDict(
    (
        ('minutes', bt.TimeFrame.Minutes),
        ('days', bt.TimeFrame.Days),
        ('weeks', bt.TimeFrame.Weeks),
        ('months', bt.TimeFrame.Months),
        ('years', bt.TimeFrame.Years),
    )
)

_TFS = _TFRAMES.keys()


def runstrat(args=None):
    args = parse_args(args)

    cerebro = bt.Cerebro()
    cerebro.broker.set_cash(args.cash)

    dkwargs = dict()
    if args.fromdate:
        fromdate = datetime.datetime.strptime(args.fromdate, '%Y-%m-%d')
        dkwargs['fromdate'] = fromdate

    if args.todate:
        todate = datetime.datetime.strptime(args.todate, '%Y-%m-%d')
        dkwargs['todate'] = todate

    if args.timeframe:
        dkwargs['timeframe'] = _TFRAMES[args.timeframe]

    if args.compression:
        dkwargs['compression'] = args.compression

    # data0 = bt.feeds.BacktraderCSVData(dataname=args.data0, **dkwargs)
    data0 = bt.feeds.VCData(dataname=args.data0, historical=True, **dkwargs)
    cerebro.adddata(data0, name='Data0')

    cerebro.addstrategy(St, short=args.short, printdata=args.printdata)
    cerebro.addsizer(bt.sizers.FixedSize, stake=args.stake)

    # Own analyzerset
    cerebro.addanalyzer(bt.analyzers.TimeReturn, timeframe=bt.TimeFrame.Years)
    cerebro.addanalyzer(bt.analyzers.SharpeRatio, timeframe=bt.TimeFrame.Years)
    cerebro.addanalyzer(bt.analyzers.SQN,)

    if args.pyfolio:
        cerebro.addanalyzer(bt.analyzers.PyFolio, _name='pyfolio',
                            timeframe=_TFRAMES[args.pftimeframe])

    if args.printout:
        print('Start run')
    results = cerebro.run()
    if args.printout:
        print('End Run')
    strat = results[0]

    # Results of own analyzers
    al = strat.analyzers.timereturn
    print('-- Time Return:')
    for k, v in al.get_analysis().items():
        print('{}: {}'.format(k, v))

    al = strat.analyzers.sharperatio
    print('-- Sharpe Ratio:')
    for k, v in al.get_analysis().items():
        print('{}: {}'.format(k, v))

    al = strat.analyzers.sqn
    print('-- SQN:')
    for k, v in al.get_analysis().items():
        print('{}: {}'.format(k, v))

    if args.pyfolio:
        pyfoliozer = strat.analyzers.getbyname('pyfolio',)

        returns, positions, transactions, gross_lev = pyfoliozer.get_pf_items()
        if args.printout:
            print('-- RETURNS')
            print(returns)
            print('-- POSITIONS')
            print(positions)
            print('-- TRANSACTIONS')
            print(transactions)
            print('-- GROSS LEVERAGE')
            print(gross_lev)

        if True:
            import pyfolio as pf
            pf.create_full_tear_sheet(
                returns,
                positions=positions,
                transactions=transactions,
                gross_lev=gross_lev,
                round_trips=True)

    if args.plot:
        pkwargs = dict(style='bar')
        if args.plot is not True:  # evals to True but is not True
            pkwargs = eval('dict(' + args.plot + ')')  # args were passed

        cerebro.plot(**pkwargs)


def parse_args(pargs=None):

    parser = argparse.ArgumentParser(
        formatter_class=argparse.ArgumentDefaultsHelpFormatter,
        description='Sample for pivot point and cross plotting')

    parser.add_argument('--data0', required=True,
                        # default='../../datas/yhoo-1996-2015.txt',
                        help='Data to be read in')

    parser.add_argument('--timeframe', required=False,
                        default=next(iter(_TFS)), choices=_TFS,
                        help='Starting date in YYYY-MM-DD format')

    parser.add_argument('--compression', required=False,
                        default=1, type=int,
                        help='Starting date in YYYY-MM-DD format')

    if False:
        parser.add_argument('--data1', required=False,
                            default='../../datas/orcl-1995-2014.txt',
                            help='Data to be read in')

    parser.add_argument('--fromdate', required=False,
                        default='2013-01-01',
                        help='Starting date in YYYY-MM-DD format')

    parser.add_argument('--todate', required=False,
                        default='2015-12-31',
                        help='Ending date in YYYY-MM-DD format')

    parser.add_argument('--stake', required=False, action='store',
                        default=10, type=int,
                        help=('Stake size'))

    parser.add_argument('--short', required=False, action='store_true',
                        help=('Go short too'))

    parser.add_argument('--cash', required=False, action='store',
                        type=float, default=50000,
                        help=('Cash to start with'))

    parser.add_argument('--pyfolio', required=False, action='store_true',
                        help=('Do pyfolio things'))

    parser.add_argument('--pftimeframe', required=False,
                        default='days', choices=_TFS,
                        help='Starting date in YYYY-MM-DD format')

    parser.add_argument('--printout', required=False, action='store_true',
                        help=('Print infos'))

    parser.add_argument('--printdata', required=False, action='store_true',
                        help=('Print data lines'))

    # Plot options
    parser.add_argument('--plot', '-p', nargs='?', required=False,
                        metavar='kwargs', const=True,
                        help=('Plot the read data applying any kwargs passed\n'
                              '\n'
                              'For example:\n'
                              '\n'
                              '  --plot style="candle" (to plot candles)\n'))

    if pargs is not None:
        return parser.parse_args(pargs)

    return parser.parse_args()

In [7]:
runstrat('--data0 015ES --timeframe days --compression 1 --pyfolio --printout --cash 200000 --short'.split())


---------------------------------------------------------------------------
UnboundLocalError                         Traceback (most recent call last)
<ipython-input-7-3298d240ea06> in <module>()
----> 1 runstrat('--data0 015ES --timeframe days --compression 1 --pyfolio --printout --cash 200000 --short'.split())

<ipython-input-6-c5b3d4ae126d> in runstrat(args)
     94 
     95     # data0 = bt.feeds.BacktraderCSVData(dataname=args.data0, **dkwargs)
---> 96     data0 = bt.feeds.VCData(dataname=args.data0, historical=True, **dkwargs)
     97     cerebro.adddata(data0, name='Data0')
     98 

d:\dro\01-docs\01-home\src\backtrader\backtrader\metabase.py in __call__(cls, *args, **kwargs)
     85         _obj, args, kwargs = cls.donew(*args, **kwargs)
     86         _obj, args, kwargs = cls.dopreinit(_obj, *args, **kwargs)
---> 87         _obj, args, kwargs = cls.doinit(_obj, *args, **kwargs)
     88         _obj, args, kwargs = cls.dopostinit(_obj, *args, **kwargs)
     89         return _obj

d:\dro\01-docs\01-home\src\backtrader\backtrader\metabase.py in doinit(cls, _obj, *args, **kwargs)
     75 
     76     def doinit(cls, _obj, *args, **kwargs):
---> 77         _obj.__init__(*args, **kwargs)
     78         return _obj, args, kwargs
     79 

d:\dro\01-docs\01-home\src\backtrader\backtrader\feeds\vcdata.py in __init__(self, **kwargs)
    245 
    246     def __init__(self, **kwargs):
--> 247         self.store = vcstore.VCStore(**kwargs)
    248 
    249         # Correct a copy past directly from VisualChart

d:\dro\01-docs\01-home\src\backtrader\backtrader\stores\vcstore.py in __call__(cls, *args, **kwargs)
    180         if cls._singleton is None:
    181             cls._singleton = (
--> 182                 super(MetaSingleton, cls).__call__(*args, **kwargs))
    183 
    184         return cls._singleton

d:\dro\01-docs\01-home\src\backtrader\backtrader\metabase.py in __call__(cls, *args, **kwargs)
     85         _obj, args, kwargs = cls.donew(*args, **kwargs)
     86         _obj, args, kwargs = cls.dopreinit(_obj, *args, **kwargs)
---> 87         _obj, args, kwargs = cls.doinit(_obj, *args, **kwargs)
     88         _obj, args, kwargs = cls.dopostinit(_obj, *args, **kwargs)
     89         return _obj

d:\dro\01-docs\01-home\src\backtrader\backtrader\metabase.py in doinit(cls, _obj, *args, **kwargs)
     75 
     76     def doinit(cls, _obj, *args, **kwargs):
---> 77         _obj.__init__(*args, **kwargs)
     78         return _obj, args, kwargs
     79 

d:\dro\01-docs\01-home\src\backtrader\backtrader\stores\vcstore.py in __init__(self)
    305 
    306         if not self._load_comtypes():
--> 307             msg = self._RT_TYPELIB, txt
    308             txt = 'Failed to import comtypes'
    309             msg = self._RT_COMTYPES, txt

UnboundLocalError: local variable 'txt' referenced before assignment