mirror of
https://github.com/saymrwulf/zipline.git
synced 2026-05-16 21:10:11 +00:00
This data bundle will use the quantopian mirror of the quandl WIKI data instead of downloading from quandl directly. This dramatically improves the speed because we do not pay the rate limiting for quandl and we can send the data in the format zipline expects.
722 lines
25 KiB
ReStructuredText
722 lines
25 KiB
ReStructuredText
Zipline beginner tutorial
|
||
-------------------------
|
||
|
||
Basics
|
||
~~~~~~
|
||
|
||
Zipline is an open-source algorithmic trading simulator written in
|
||
Python.
|
||
|
||
The source can be found at: https://github.com/quantopian/zipline
|
||
|
||
Some benefits include:
|
||
|
||
- Realistic: slippage, transaction costs, order delays.
|
||
- Stream-based: Process each event individually, avoids look-ahead
|
||
bias.
|
||
- Batteries included: Common transforms (moving average) as well as
|
||
common risk calculations (Sharpe).
|
||
- Developed and continuously updated by
|
||
`Quantopian <https://www.quantopian.com>`__ which provides an
|
||
easy-to-use web-interface to Zipline, 10 years of minute-resolution
|
||
historical US stock data, and live-trading capabilities. This
|
||
tutorial is directed at users wishing to use Zipline without using
|
||
Quantopian. If you instead want to get started on Quantopian, see
|
||
`here <https://www.quantopian.com/faq#get-started>`__.
|
||
|
||
This tutorial assumes that you have zipline correctly installed, see the
|
||
`installation
|
||
instructions <https://github.com/quantopian/zipline#installation>`__ if
|
||
you haven't set up zipline yet.
|
||
|
||
Every ``zipline`` algorithm consists of two functions you have to
|
||
define: \* ``initialize(context)`` \* ``handle_data(context, data)``
|
||
|
||
Before the start of the algorithm, ``zipline`` calls the
|
||
``initialize()`` function and passes in a ``context`` variable.
|
||
``context`` is a persistent namespace for you to store variables you
|
||
need to access from one algorithm iteration to the next.
|
||
|
||
After the algorithm has been initialized, ``zipline`` calls the
|
||
``handle_data()`` function once for each event. At every call, it passes
|
||
the same ``context`` variable and an event-frame called ``data``
|
||
containing the current trading bar with open, high, low, and close
|
||
(OHLC) prices as well as volume for each stock in your universe. For
|
||
more information on these functions, see the `relevant part of the
|
||
Quantopian docs <https://www.quantopian.com/help#api-toplevel>`__.
|
||
|
||
My first algorithm
|
||
~~~~~~~~~~~~~~~~~~
|
||
|
||
Lets take a look at a very simple algorithm from the ``examples``
|
||
directory, ``buyapple.py``:
|
||
|
||
.. code-block:: python
|
||
|
||
from zipline.examples import buyapple
|
||
buyapple??
|
||
|
||
|
||
.. code-block:: python
|
||
|
||
from zipline.api import order, record, symbol
|
||
|
||
|
||
def initialize(context):
|
||
pass
|
||
|
||
|
||
def handle_data(context, data):
|
||
order(symbol('AAPL'), 10)
|
||
record(AAPL=data.current(symbol('AAPL'), 'price'))
|
||
|
||
|
||
As you can see, we first have to import some functions we would like to
|
||
use. All functions commonly used in your algorithm can be found in
|
||
``zipline.api``. Here we are using :func:`~zipline.api.order()` which takes two
|
||
arguments: a security object, and a number specifying how many stocks you would
|
||
like to order (if negative, :func:`~zipline.api.order()` will sell/short
|
||
stocks). In this case we want to order 10 shares of Apple at each iteration. For
|
||
more documentation on ``order()``, see the `Quantopian docs
|
||
<https://www.quantopian.com/help#api-order>`__.
|
||
|
||
Finally, the :func:`~zipline.api.record` function allows you to save the value
|
||
of a variable at each iteration. You provide it with a name for the variable
|
||
together with the variable itself: ``varname=var``. After the algorithm
|
||
finished running you will have access to each variable value you tracked
|
||
with :func:`~zipline.api.record` under the name you provided (we will see this
|
||
further below). You also see how we can access the current price data of the
|
||
AAPL stock in the ``data`` event frame (for more information see
|
||
`here <https://www.quantopian.com/help#api-event-properties>`__.
|
||
|
||
Running the algorithm
|
||
~~~~~~~~~~~~~~~~~~~~~
|
||
|
||
To now test this algorithm on financial data, ``zipline`` provides three
|
||
interfaces: A command-line interface, ``IPython Notebook`` magic, and
|
||
:func:`~zipline.run_algorithm`.
|
||
|
||
Command line interface
|
||
^^^^^^^^^^^^^^^^^^^^^^
|
||
|
||
After you installed zipline you should be able to execute the following
|
||
from your command line (e.g. ``cmd.exe`` on Windows, or the Terminal app
|
||
on OSX):
|
||
|
||
.. code-block:: bash
|
||
|
||
$ python -m zipline run --help
|
||
|
||
.. parsed-literal::
|
||
|
||
Usage: __main__.py run [OPTIONS]
|
||
|
||
Run a backtest for the given algorithm.
|
||
|
||
Options:
|
||
-f, --algofile FILENAME The file that contains the algorithm to run.
|
||
-t, --algotext TEXT The algorithm script to run.
|
||
-D, --define TEXT Define a name to be bound in the namespace
|
||
before executing the algotext. For example
|
||
'-Dname=value'. The value may be any python
|
||
expression. These are evaluated in order so
|
||
they may refer to previously defined names.
|
||
--data-frequency [minute|daily]
|
||
The data frequency of the simulation.
|
||
[default: daily]
|
||
--capital-base FLOAT The starting capital for the simulation.
|
||
[default: 10000000.0]
|
||
-b, --bundle BUNDLE-NAME The data bundle to use for the simulation.
|
||
[default: quantopian-quandl]
|
||
--bundle-timestamp TIMESTAMP The date to lookup data on or before.
|
||
[default: <current-time>]
|
||
-s, --start DATE The start date of the simulation.
|
||
-e, --end DATE The end date of the simulation.
|
||
-o, --output FILENAME The location to write the perf data. If this
|
||
is '-' the perf will be written to stdout.
|
||
[default: -]
|
||
--print-algo / --no-print-algo Print the algorithm to stdout.
|
||
--help Show this message and exit.
|
||
|
||
As you can see there are a couple of flags that specify where to find your
|
||
algorithm (``-f``) as well as parameters specifying which data to use,
|
||
defaulting to the :ref:`quantopian-quandl-mirror`. There are also arguments for
|
||
the date range to run the algorithm over (``--start`` and ``--end``). Finally,
|
||
you'll want to save the performance metrics of your algorithm so that you can
|
||
analyze how it performed. This is done via the ``--output`` flag and will cause
|
||
it to write the performance ``DataFrame`` in the pickle Python file format.
|
||
Note that you can also define a configuration file with these parameters that
|
||
you can then conveniently pass to the ``-c`` option so that you don't have to
|
||
supply the command line args all the time (see the .conf files in the examples
|
||
directory).
|
||
|
||
Thus, to execute our algorithm from above and save the results to
|
||
``buyapple_out.pickle`` we would call ``python -m zipline run`` as follows:
|
||
|
||
.. code-block:: python
|
||
|
||
python -m zipline run -f ../../zipline/examples/buyapple.py --start 2000-1-1 --end 2014-1-1 --symbols AAPL -o buyapple_out.pickle
|
||
|
||
|
||
.. parsed-literal::
|
||
|
||
AAPL
|
||
[2015-11-04 22:45:32.820166] INFO: Performance: Simulated 3521 trading days out of 3521.
|
||
[2015-11-04 22:45:32.820314] INFO: Performance: first open: 2000-01-03 14:31:00+00:00
|
||
[2015-11-04 22:45:32.820401] INFO: Performance: last close: 2013-12-31 21:00:00+00:00
|
||
|
||
|
||
``run`` first calls the ``initialize()`` function, and then
|
||
streams the historical stock price day-by-day through ``handle_data()``.
|
||
After each call to ``handle_data()`` we instruct ``zipline`` to order 10
|
||
stocks of AAPL. After the call of the ``order()`` function, ``zipline``
|
||
enters the ordered stock and amount in the order book. After the
|
||
``handle_data()`` function has finished, ``zipline`` looks for any open
|
||
orders and tries to fill them. If the trading volume is high enough for
|
||
this stock, the order is executed after adding the commission and
|
||
applying the slippage model which models the influence of your order on
|
||
the stock price, so your algorithm will be charged more than just the
|
||
stock price \* 10. (Note, that you can also change the commission and
|
||
slippage model that ``zipline`` uses, see the `Quantopian
|
||
docs <https://www.quantopian.com/help#ide-slippage>`__ for more
|
||
information).
|
||
|
||
Lets take a quick look at the performance ``DataFrame``. For this, we
|
||
use ``pandas`` from inside the IPython Notebook and print the first ten
|
||
rows. Note that ``zipline`` makes heavy usage of ``pandas``, especially
|
||
for data input and outputting so it's worth spending some time to learn
|
||
it.
|
||
|
||
.. code-block:: python
|
||
|
||
import pandas as pd
|
||
perf = pd.read_pickle('buyapple_out.pickle') # read in perf DataFrame
|
||
perf.head()
|
||
|
||
.. raw:: html
|
||
|
||
<div style="max-height:1000px;max-width:1500px;overflow:auto;">
|
||
<table border="1" class="dataframe">
|
||
<thead>
|
||
<tr style="text-align: right;">
|
||
<th></th>
|
||
<th>AAPL</th>
|
||
<th>algo_volatility</th>
|
||
<th>algorithm_period_return</th>
|
||
<th>alpha</th>
|
||
<th>benchmark_period_return</th>
|
||
<th>benchmark_volatility</th>
|
||
<th>beta</th>
|
||
<th>capital_used</th>
|
||
<th>ending_cash</th>
|
||
<th>ending_exposure</th>
|
||
<th>...</th>
|
||
<th>short_exposure</th>
|
||
<th>short_value</th>
|
||
<th>shorts_count</th>
|
||
<th>sortino</th>
|
||
<th>starting_cash</th>
|
||
<th>starting_exposure</th>
|
||
<th>starting_value</th>
|
||
<th>trading_days</th>
|
||
<th>transactions</th>
|
||
<th>treasury_period_return</th>
|
||
</tr>
|
||
</thead>
|
||
<tbody>
|
||
<tr>
|
||
<th>2000-01-03 21:00:00</th>
|
||
<td>3.738314</td>
|
||
<td>0.000000e+00</td>
|
||
<td>0.000000e+00</td>
|
||
<td>-0.065800</td>
|
||
<td>-0.009549</td>
|
||
<td>0.000000</td>
|
||
<td>0.000000</td>
|
||
<td>0.00000</td>
|
||
<td>10000000.00000</td>
|
||
<td>0.00000</td>
|
||
<td>...</td>
|
||
<td>0</td>
|
||
<td>0</td>
|
||
<td>0</td>
|
||
<td>0.000000</td>
|
||
<td>10000000.00000</td>
|
||
<td>0.00000</td>
|
||
<td>0.00000</td>
|
||
<td>1</td>
|
||
<td>[]</td>
|
||
<td>0.0658</td>
|
||
</tr>
|
||
<tr>
|
||
<th>2000-01-04 21:00:00</th>
|
||
<td>3.423135</td>
|
||
<td>3.367492e-07</td>
|
||
<td>-3.000000e-08</td>
|
||
<td>-0.064897</td>
|
||
<td>-0.047528</td>
|
||
<td>0.323229</td>
|
||
<td>0.000001</td>
|
||
<td>-34.53135</td>
|
||
<td>9999965.46865</td>
|
||
<td>34.23135</td>
|
||
<td>...</td>
|
||
<td>0</td>
|
||
<td>0</td>
|
||
<td>0</td>
|
||
<td>0.000000</td>
|
||
<td>10000000.00000</td>
|
||
<td>0.00000</td>
|
||
<td>0.00000</td>
|
||
<td>2</td>
|
||
<td>[{u'order_id': u'513357725cb64a539e3dd02b47da7...</td>
|
||
<td>0.0649</td>
|
||
</tr>
|
||
<tr>
|
||
<th>2000-01-05 21:00:00</th>
|
||
<td>3.473229</td>
|
||
<td>4.001918e-07</td>
|
||
<td>-9.906000e-09</td>
|
||
<td>-0.066196</td>
|
||
<td>-0.045697</td>
|
||
<td>0.329321</td>
|
||
<td>0.000001</td>
|
||
<td>-35.03229</td>
|
||
<td>9999930.43636</td>
|
||
<td>69.46458</td>
|
||
<td>...</td>
|
||
<td>0</td>
|
||
<td>0</td>
|
||
<td>0</td>
|
||
<td>0.000000</td>
|
||
<td>9999965.46865</td>
|
||
<td>34.23135</td>
|
||
<td>34.23135</td>
|
||
<td>3</td>
|
||
<td>[{u'order_id': u'd7d4ad03cfec4d578c0d817dc3829...</td>
|
||
<td>0.0662</td>
|
||
</tr>
|
||
<tr>
|
||
<th>2000-01-06 21:00:00</th>
|
||
<td>3.172661</td>
|
||
<td>4.993979e-06</td>
|
||
<td>-6.410420e-07</td>
|
||
<td>-0.065758</td>
|
||
<td>-0.044785</td>
|
||
<td>0.298325</td>
|
||
<td>-0.000006</td>
|
||
<td>-32.02661</td>
|
||
<td>9999898.40975</td>
|
||
<td>95.17983</td>
|
||
<td>...</td>
|
||
<td>0</td>
|
||
<td>0</td>
|
||
<td>0</td>
|
||
<td>-12731.780516</td>
|
||
<td>9999930.43636</td>
|
||
<td>69.46458</td>
|
||
<td>69.46458</td>
|
||
<td>4</td>
|
||
<td>[{u'order_id': u'1fbf5e9bfd7c4d9cb2e8383e1085e...</td>
|
||
<td>0.0657</td>
|
||
</tr>
|
||
<tr>
|
||
<th>2000-01-07 21:00:00</th>
|
||
<td>3.322945</td>
|
||
<td>5.977002e-06</td>
|
||
<td>-2.201900e-07</td>
|
||
<td>-0.065206</td>
|
||
<td>-0.018908</td>
|
||
<td>0.375301</td>
|
||
<td>0.000005</td>
|
||
<td>-33.52945</td>
|
||
<td>9999864.88030</td>
|
||
<td>132.91780</td>
|
||
<td>...</td>
|
||
<td>0</td>
|
||
<td>0</td>
|
||
<td>0</td>
|
||
<td>-12629.274583</td>
|
||
<td>9999898.40975</td>
|
||
<td>95.17983</td>
|
||
<td>95.17983</td>
|
||
<td>5</td>
|
||
<td>[{u'order_id': u'9ea6b142ff09466b9113331a37437...</td>
|
||
<td>0.0652</td>
|
||
</tr>
|
||
</tbody>
|
||
</table>
|
||
<p>5 rows × 39 columns</p>
|
||
</div>
|
||
|
||
|
||
|
||
As you can see, there is a row for each trading day, starting on the
|
||
first business day of 2000. In the columns you can find various
|
||
information about the state of your algorithm. The very first column
|
||
``AAPL`` was placed there by the ``record()`` function mentioned earlier
|
||
and allows us to plot the price of apple. For example, we could easily
|
||
examine now how our portfolio value changed over time compared to the
|
||
AAPL stock price.
|
||
|
||
.. code-block:: python
|
||
|
||
%pylab inline
|
||
figsize(12, 12)
|
||
import matplotlib.pyplot as plt
|
||
|
||
ax1 = plt.subplot(211)
|
||
perf.portfolio_value.plot(ax=ax1)
|
||
ax1.set_ylabel('portfolio value')
|
||
ax2 = plt.subplot(212, sharex=ax1)
|
||
perf.AAPL.plot(ax=ax2)
|
||
ax2.set_ylabel('AAPL stock price')
|
||
|
||
.. parsed-literal::
|
||
|
||
Populating the interactive namespace from numpy and matplotlib
|
||
|
||
.. parsed-literal::
|
||
|
||
<matplotlib.text.Text at 0x7ff5c6147f90>
|
||
|
||
.. image:: tutorial_files/tutorial_11_2.png
|
||
|
||
|
||
As you can see, our algorithm performance as assessed by the
|
||
``portfolio_value`` closely matches that of the AAPL stock price. This
|
||
is not surprising as our algorithm only bought AAPL every chance it got.
|
||
|
||
IPython Notebook
|
||
~~~~~~~~~~~~~~~~
|
||
|
||
The `IPython Notebook <http://ipython.org/notebook.html>`__ is a very
|
||
powerful browser-based interface to a Python interpreter (this tutorial
|
||
was written in it). As it is already the de-facto interface for most
|
||
quantitative researchers ``zipline`` provides an easy way to run your
|
||
algorithm inside the Notebook without requiring you to use the CLI.
|
||
|
||
To use it you have to write your algorithm in a cell and let ``zipline``
|
||
know that it is supposed to run this algorithm. This is done via the
|
||
``%%zipline`` IPython magic command that is available after you
|
||
``import zipline`` from within the IPython Notebook. This magic takes
|
||
the same arguments as the command line interface described above. Thus
|
||
to run the algorithm from above with the same parameters we just have to
|
||
execute the following cell after importing ``zipline`` to register the
|
||
magic.
|
||
|
||
.. code-block:: python
|
||
|
||
%load_ext zipline
|
||
|
||
.. code-block:: python
|
||
|
||
%%zipline --start 2000-1-1 --end 2014-1-1 --symbols AAPL
|
||
from zipline.api import symbol, order, record
|
||
|
||
def initialize(context):
|
||
pass
|
||
|
||
def handle_data(context, data):
|
||
order(symbol('AAPL'), 10)
|
||
record(AAPL=data[symbol('AAPL')].price)
|
||
|
||
Note that we did not have to specify an input file as above since the
|
||
magic will use the contents of the cell and look for your algorithm
|
||
functions there. Also, instead of defining an output file we are
|
||
specifying a variable name with ``-o`` that will be created in the name
|
||
space and contain the performance ``DataFrame`` we looked at above.
|
||
|
||
.. code-block:: python
|
||
|
||
_.head()
|
||
|
||
.. raw:: html
|
||
|
||
<div style="max-height:1000px;max-width:1500px;overflow:auto;">
|
||
<table border="1" class="dataframe">
|
||
<thead>
|
||
<tr style="text-align: right;">
|
||
<th></th>
|
||
<th>AAPL</th>
|
||
<th>algo_volatility</th>
|
||
<th>algorithm_period_return</th>
|
||
<th>alpha</th>
|
||
<th>benchmark_period_return</th>
|
||
<th>benchmark_volatility</th>
|
||
<th>beta</th>
|
||
<th>capital_used</th>
|
||
<th>ending_cash</th>
|
||
<th>ending_exposure</th>
|
||
<th>...</th>
|
||
<th>short_exposure</th>
|
||
<th>short_value</th>
|
||
<th>shorts_count</th>
|
||
<th>sortino</th>
|
||
<th>starting_cash</th>
|
||
<th>starting_exposure</th>
|
||
<th>starting_value</th>
|
||
<th>trading_days</th>
|
||
<th>transactions</th>
|
||
<th>treasury_period_return</th>
|
||
</tr>
|
||
</thead>
|
||
<tbody>
|
||
<tr>
|
||
<th>2000-01-03 21:00:00</th>
|
||
<td>3.738314</td>
|
||
<td>0.000000e+00</td>
|
||
<td>0.000000e+00</td>
|
||
<td>-0.065800</td>
|
||
<td>-0.009549</td>
|
||
<td>0.000000</td>
|
||
<td>0.000000</td>
|
||
<td>0.00000</td>
|
||
<td>10000000.00000</td>
|
||
<td>0.00000</td>
|
||
<td>...</td>
|
||
<td>0</td>
|
||
<td>0</td>
|
||
<td>0</td>
|
||
<td>0.000000</td>
|
||
<td>10000000.00000</td>
|
||
<td>0.00000</td>
|
||
<td>0.00000</td>
|
||
<td>1</td>
|
||
<td>[]</td>
|
||
<td>0.0658</td>
|
||
</tr>
|
||
<tr>
|
||
<th>2000-01-04 21:00:00</th>
|
||
<td>3.423135</td>
|
||
<td>3.367492e-07</td>
|
||
<td>-3.000000e-08</td>
|
||
<td>-0.064897</td>
|
||
<td>-0.047528</td>
|
||
<td>0.323229</td>
|
||
<td>0.000001</td>
|
||
<td>-34.53135</td>
|
||
<td>9999965.46865</td>
|
||
<td>34.23135</td>
|
||
<td>...</td>
|
||
<td>0</td>
|
||
<td>0</td>
|
||
<td>0</td>
|
||
<td>0.000000</td>
|
||
<td>10000000.00000</td>
|
||
<td>0.00000</td>
|
||
<td>0.00000</td>
|
||
<td>2</td>
|
||
<td>[{u'commission': 0.3, u'amount': 10, u'sid': 0...</td>
|
||
<td>0.0649</td>
|
||
</tr>
|
||
<tr>
|
||
<th>2000-01-05 21:00:00</th>
|
||
<td>3.473229</td>
|
||
<td>4.001918e-07</td>
|
||
<td>-9.906000e-09</td>
|
||
<td>-0.066196</td>
|
||
<td>-0.045697</td>
|
||
<td>0.329321</td>
|
||
<td>0.000001</td>
|
||
<td>-35.03229</td>
|
||
<td>9999930.43636</td>
|
||
<td>69.46458</td>
|
||
<td>...</td>
|
||
<td>0</td>
|
||
<td>0</td>
|
||
<td>0</td>
|
||
<td>0.000000</td>
|
||
<td>9999965.46865</td>
|
||
<td>34.23135</td>
|
||
<td>34.23135</td>
|
||
<td>3</td>
|
||
<td>[{u'commission': 0.3, u'amount': 10, u'sid': 0...</td>
|
||
<td>0.0662</td>
|
||
</tr>
|
||
<tr>
|
||
<th>2000-01-06 21:00:00</th>
|
||
<td>3.172661</td>
|
||
<td>4.993979e-06</td>
|
||
<td>-6.410420e-07</td>
|
||
<td>-0.065758</td>
|
||
<td>-0.044785</td>
|
||
<td>0.298325</td>
|
||
<td>-0.000006</td>
|
||
<td>-32.02661</td>
|
||
<td>9999898.40975</td>
|
||
<td>95.17983</td>
|
||
<td>...</td>
|
||
<td>0</td>
|
||
<td>0</td>
|
||
<td>0</td>
|
||
<td>-12731.780516</td>
|
||
<td>9999930.43636</td>
|
||
<td>69.46458</td>
|
||
<td>69.46458</td>
|
||
<td>4</td>
|
||
<td>[{u'commission': 0.3, u'amount': 10, u'sid': 0...</td>
|
||
<td>0.0657</td>
|
||
</tr>
|
||
<tr>
|
||
<th>2000-01-07 21:00:00</th>
|
||
<td>3.322945</td>
|
||
<td>5.977002e-06</td>
|
||
<td>-2.201900e-07</td>
|
||
<td>-0.065206</td>
|
||
<td>-0.018908</td>
|
||
<td>0.375301</td>
|
||
<td>0.000005</td>
|
||
<td>-33.52945</td>
|
||
<td>9999864.88030</td>
|
||
<td>132.91780</td>
|
||
<td>...</td>
|
||
<td>0</td>
|
||
<td>0</td>
|
||
<td>0</td>
|
||
<td>-12629.274583</td>
|
||
<td>9999898.40975</td>
|
||
<td>95.17983</td>
|
||
<td>95.17983</td>
|
||
<td>5</td>
|
||
<td>[{u'commission': 0.3, u'amount': 10, u'sid': 0...</td>
|
||
<td>0.0652</td>
|
||
</tr>
|
||
</tbody>
|
||
</table>
|
||
<p>5 rows × 39 columns</p>
|
||
</div>
|
||
|
||
|
||
Access to previous prices using ``history``
|
||
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
||
|
||
Working example: Dual Moving Average Cross-Over
|
||
^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
|
||
|
||
The Dual Moving Average (DMA) is a classic momentum strategy. It's
|
||
probably not used by any serious trader anymore but is still very
|
||
instructive. The basic idea is that we compute two rolling or moving
|
||
averages (mavg) -- one with a longer window that is supposed to capture
|
||
long-term trends and one shorter window that is supposed to capture
|
||
short-term trends. Once the short-mavg crosses the long-mavg from below
|
||
we assume that the stock price has upwards momentum and long the stock.
|
||
If the short-mavg crosses from above we exit the positions as we assume
|
||
the stock to go down further.
|
||
|
||
As we need to have access to previous prices to implement this strategy
|
||
we need a new concept: History
|
||
|
||
``history()`` is a convenience function that keeps a rolling window of
|
||
data for you. The first argument is the number of bars you want to
|
||
collect, the second argument is the unit (either ``'1d'`` for ``'1m'``
|
||
but note that you need to have minute-level data for using ``1m``). For
|
||
a more detailed description ``history()``'s features, see the
|
||
`Quantopian docs <https://www.quantopian.com/help#ide-history>`__. While
|
||
you can directly use the ``history()`` function on Quantopian, in
|
||
``zipline`` you have to register each history container you want to use
|
||
with ``add_history()`` and pass it the same arguments as the history
|
||
function below. Lets look at the strategy which should make this clear:
|
||
|
||
.. code-block:: python
|
||
|
||
%%zipline --start 2000-1-1 --end 2014-1-1 -o perf_dma
|
||
|
||
|
||
from zipline.api import order_target, record, symbol, history, add_history
|
||
import numpy as np
|
||
|
||
def initialize(context):
|
||
# Register 2 histories that track daily prices,
|
||
# one with a 100 window and one with a 300 day window
|
||
add_history(100, '1d', 'price')
|
||
add_history(300, '1d', 'price')
|
||
|
||
context.i = 0
|
||
|
||
|
||
def handle_data(context, data):
|
||
# Skip first 300 days to get full windows
|
||
context.i += 1
|
||
if context.i < 300:
|
||
return
|
||
|
||
# Compute averages
|
||
# history() has to be called with the same params
|
||
# from above and returns a pandas dataframe.
|
||
short_mavg = history(100, '1d', 'price').mean()
|
||
long_mavg = history(300, '1d', 'price').mean()
|
||
|
||
# Trading logic
|
||
if short_mavg[0] > long_mavg[0]:
|
||
# order_target orders as many shares as needed to
|
||
# achieve the desired number of shares.
|
||
order_target(symbol('AAPL'), 100)
|
||
elif short_mavg[0] < long_mavg[0]:
|
||
order_target(symbol('AAPL'), 0)
|
||
|
||
# Save values for later inspection
|
||
record(AAPL=data[symbol('AAPL')].price,
|
||
short_mavg=short_mavg[0],
|
||
long_mavg=long_mavg[0])
|
||
|
||
|
||
def analyze(context, perf):
|
||
fig = plt.figure()
|
||
ax1 = fig.add_subplot(211)
|
||
perf.portfolio_value.plot(ax=ax1)
|
||
ax1.set_ylabel('portfolio value in $')
|
||
|
||
ax2 = fig.add_subplot(212)
|
||
perf['AAPL'].plot(ax=ax2)
|
||
perf[['short_mavg', 'long_mavg']].plot(ax=ax2)
|
||
|
||
perf_trans = perf.ix[[t != [] for t in perf.transactions]]
|
||
buys = perf_trans.ix[[t[0]['amount'] > 0 for t in perf_trans.transactions]]
|
||
sells = perf_trans.ix[
|
||
[t[0]['amount'] < 0 for t in perf_trans.transactions]]
|
||
ax2.plot(buys.index, perf.short_mavg.ix[buys.index],
|
||
'^', markersize=10, color='m')
|
||
ax2.plot(sells.index, perf.short_mavg.ix[sells.index],
|
||
'v', markersize=10, color='k')
|
||
ax2.set_ylabel('price in $')
|
||
plt.legend(loc=0)
|
||
plt.show()
|
||
|
||
.. image:: tutorial_files/tutorial_22_1.png
|
||
|
||
Here we are explicitly defining an ``analyze()`` function that gets
|
||
automatically called once the backtest is done (this is not possible on
|
||
Quantopian currently).
|
||
|
||
Although it might not be directly apparent, the power of ``history()``
|
||
(pun intended) can not be under-estimated as most algorithms make use of
|
||
prior market developments in one form or another. You could easily
|
||
devise a strategy that trains a classifier with
|
||
`scikit-learn <http://scikit-learn.org/stable/>`__ which tries to
|
||
predict future market movements based on past prices (note, that most of
|
||
the ``scikit-learn`` functions require ``numpy.ndarray``\ s rather than
|
||
``pandas.DataFrame``\ s, so you can simply pass the underlying
|
||
``ndarray`` of a ``DataFrame`` via ``.values``).
|
||
|
||
We also used the ``order_target()`` function above. This and other
|
||
functions like it can make order management and portfolio rebalancing
|
||
much easier. See the `Quantopian documentation on order
|
||
functions <https://www.quantopian.com/help#api-order-methods>`__ fore
|
||
more details.
|
||
|
||
Conclusions
|
||
~~~~~~~~~~~
|
||
|
||
We hope that this tutorial gave you a little insight into the
|
||
architecture, API, and features of ``zipline``. For next steps, check
|
||
out some of the
|
||
`examples <https://github.com/quantopian/zipline/tree/master/zipline/examples>`__.
|
||
|
||
Feel free to ask questions on `our mailing
|
||
list <https://groups.google.com/forum/#!forum/zipline>`__, report
|
||
problems on our `GitHub issue
|
||
tracker <https://github.com/quantopian/zipline/issues?state=open>`__,
|
||
`get
|
||
involved <https://github.com/quantopian/zipline/wiki/Contribution-Requests>`__,
|
||
and `checkout Quantopian <https://quantopian.com>`__.
|