Backtesting of Algorithmic Cryptocurrenc
Backtesting of Algorithmic Cryptocurrenc
Trading Strategies
Abstract
  1
      Master in Applied Data Science student at Frankfurt School of Finance & Management | Address:
Hermann-Löns-Straße 5, 57250 Netphen, Germany | Student ID: 8405672 | Email: jan@spoerer.me |
Cell number: +49 171 5395666 | linkedin.com/in/janspoerer
BACKTESTING ALGOR. CRYPTOC. TRADING STRATEGIES                                                    II
1 Acknowledgment
      First, I would like to thank my first thesis advisor Prof. Dr. Philipp Sandner2 ,
the head of the Frankfurt School Blockchain Center. Naturally, he gave me domain-
specific guidance; but even more importantly, he connected me with blockchain
practitioners that used the software presented here. The momentum that these
projects caused was vital for me to stay focused and develop quantbacktest quickly.
      Third, I would like to thank Jian Guo4 , a fellow student of mine from the
Master in Applied Data Science at Frankfurt School, for his practical contributions
to the source code. He supported me with urgently needed analyses for Immutable
Insight GmbH. Also, he set up the time complexity monitoring for this software and
presented it at a code review event for feedback from other software developers. In
addition to that, he implemented various financial metrics, notably the maximum
drawdown function, capitalizing on his background in the hedge fund industry.
  2
    frankfurt-school.de/en/home/research/staff/Philipp-Sandner, linkedin.com/in/philippsandner.
  3
    andonians.com, frankfurt-school.de/en/home/research/staff/Vahe-Andonians.
  4
    linkedin.com/in/jian-guo-57390981.
  5
    blockchainfonds.com.
  6
    linkedin.com/in/katharinagehra.
  7
    linkedin.com/in/volker-henning-winterer-17b93ab.
BACKTESTING ALGOR. CRYPTOC. TRADING STRATEGIES                                      III
      Lastly, I would like to thank everyone who was involved with proofreading
and technical advice: Leon Berghoff8 , Tobias Burggraf9 , Frederic Herold10 , Nicholas
Herold11 , and Sascha Wildgrube12 .
  8
    linkedin.com/in/leon-berghoff-753b52128.
  9
    linkedin.com/in/tobias-burggraf-966001138.
 10
    linkedin.com/in/frederic-herold-182034148.
 11
    linkedin.com/in/nicholas-h-7518aa128.
 12
    wildgrube.com, linkedin.com/in/sascha-wildgrube-4135b64.
BACKTESTING ALGOR. CRYPTOC. TRADING STRATEGIES                                     IV
2 List of Figures
4 Contents
1 Acknowledgment II
2 List of Figures IV
4 Contents VIII
5 Introduction 1
5.1 Motivation . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 1
5.1.3.2 Excel . . . . . . . . . . . . . . . . . . . . . . . . . . 2
5.1.3.6 QuantConnect . . . . . . . . . . . . . . . . . . . . . 4
5.1.3.7 Python . . . . . . . . . . . . . . . . . . . . . . . . . 4
7 Data 12
8.1.1 Scenario . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 14
8.1.3 df_prices . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 14
8.1.4 df_trading_journal . . . . . . . . . . . . . . . . . . . . . . . 15
8.1.5 df_performance_metrics . . . . . . . . . . . . . . . . . . . . 15
8.3.2.2 Usage . . . . . . . . . . . . . . . . . . . . . . . . . . 34
9 Results 38
10 Conclusion 44
10.1 Summary . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 44
10.4 Limitations . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 46
11 References 53
12 Appendix 58
5 Introduction
5.1 Motivation
5.1.3.2       Excel Excel is a good tool for managing tabular data. In this case, how-
ever, the price data is three-dimensional: It has two identifying properties (datetime
and identifier) and one value (price). Multi-dimensional data is generally hard to
manage in Excel directly as Excel consists of tables; tables are two-dimensional. Ex-
cel’s index-match() and vlookup() functions are also able to handle multi-criteria
queries, but they are inefficient and likely to crash. Using Excel’s native language
VBA and its array data type would be a possible solution to this, but then the
advantages that Excel has (i.e., a user-friendly interface that most people in finance
know how to use) would vanish. In summary, Excel is not a feasible solution for
this problem, and Excel VBA is a downgrade in comparison to Python, which offers
more functionality for handling multi-dimensional data than VBA does.
  13
       Some   backtesting   libraries   that   are   not    reviewed   in   detail   here   are:   PyAlgoTrade
(gbeced.github.io/pyalgotrade/docs/v0.17/html/),              pybacktest      github.com/ematvey/pybacktest,
bt      github.com/pmorissette/bt,        backtrader         github.com/mementum/backtrader,         ProfitPy
code.google.com/archive/p/profitpy/,            pinkfish        github.com/fja05680/pinkfish,        QSTrader
github.com/mhallsmoore/qstrader,               pysystemtrade           github.com/robcarver17/pysystemtrade,
QTPyLib https://github.com/ranaroussi/qtpylib,             Prophet github.com/Emsu/prophet,        Backtesting
pypi.org/project/Backtesting/.
BACKTESTING ALGOR. CRYPTOC. TRADING STRATEGIES                                                         3
5.1.3.4       Quantopian and Zipline Quantopian is one of the first platforms that
come to mind when thinking about state-of-the-art backtesting of algorithmic trad-
ing strategies. It is a hosted backtesting and deployment platform with a large
proportion of closed-source codebase. The community consists of individuals that
compete for the best out-of-sample portfolio compositions15 ; Quantopian awards
real funding to the best strategies. Zipline is one open-source library that Quan-
topian uses for backtesting. I have considered Quantopian when deciding about a
technical platform and thought about ways to achieve the goals of this study within
the technical framework that Quantopian provides.
        I have decided against using Quantopian for the following reasons: Quantopian
runs backtests on its servers. As a consequence, they offer a relatively low-effort,
reliable, sufficiently documented, and actively maintained platform for backtesting
and deploying quantitative trading strategies. Similar to other proprietary solu-
tions, one issue is that Quantopian’s asset universe consists of U.S. stocks that are
tradeable on the platform. So cryptocurrency data is not yet available on Quan-
topian, at least not natively. There are workarounds to get cryptocurrency data into
 14
      Equilla is Tradesignal’s language.
 15
      See the website for detailed algorithm requirements of Quantopian: www.quantopian.com/contest.
BACKTESTING ALGOR. CRYPTOC. TRADING STRATEGIES                                                                4
5.1.3.5        Crypto Wizards Crypto Wizards17 aim to solve all of the problems
mentioned above. So far, the focus of the project is promising for the use case
presented. At the time of writing this thesis, the project is still in an early phase,
with the project’s start dating only to 2019. When the preparatory work for the
thesis started, I did not even know that this project existed. In the explanation
videos of Crypto Wizards, the developer hints that using the project is free now,
but he implied that it might not be anymore in the future. What also makes us
concerned is that Crypto Wizards’ website is still in a nascent stage and contains
suspicious customer reviews, raising some doubts about the project’s integrity18 .
These doubts about the project’s integrity, combined with the for-profit nature of
the project, make this tool unacceptable for most users. Time will tell if the project
develops into something worth considering for the use case presented here.
5.1.3.7        Python The Python libraries pandas, NumPy, and TA-Lib are commonly
used in research. They are particularly good for cash-neutral strategies that are
tensor-based. One can perform tensor-based calculations with moderate effort and
  16
       See quantopian.com/posts/cryptocurrency-data, quantopian.com/posts/crypto-currencies-backtest,
quantopian.com/posts/how-to-trade-cryptocurrencies-with-zipline.
  17
     The video gives a short introduction to the features                  of   the     Crypto      Wizards
backtesting     tool   youtube.com/watch?v=shvRlciJlZ0;     this   video   goes       more   into     depth
youtube.com/watch?v=TkbwdPq3uWQ; this is their website: cryptowizards.net/.
  18
     Here is a printout of the website cryptowizards.net/ from February                       21,     2020:
drive.google.com/open?id=19QeMAbAiMCkrFy3LVc3e3eqTxM0Y9DVn.
  19
     See quantconnect.com/docs/data-library/crypto and algo-trading.capitalmarketsciooutlook.com for an
initial overview of QuantConnect.
BACKTESTING ALGOR. CRYPTOC. TRADING STRATEGIES                                                 5
coding knowledge. As pointed out in sections 5.2 Scope and Approach (page 6)
and 10.3 Implications for Research (page 45), tensor-based calculations as used by
Jegadeesh and Titman, 2001 will probably continue to dominate research in finan-
cial markets and continue to exhibit unique advantages over the strictly realistic
simulation approach that is used by quantbacktest. Many researchers will stick
with the mentioned Python libraries instead of using quantbacktest. Practition-
ers and researchers with a more applied focus, however, will most likely not want
to make the considerable effort that is involved in simulating trades by just using
common mathematical Python libraries. I can tell from my work on this study and
professional experience that it is not a trivial task to develop realistic backtesting
software. For use cases that require a close resemblance of reality (e.g., cash balance
modeling), I expect many people to prefer using the tool presented here instead of
starting from zero. Starting from zero will likely compromise quality, budget, and
time.
  20
       See     spectrum.ieee.org/computing/software/the-top-programming-languages-2019   and
tiobe.com/tiobe-index/ for popularity rankings of programming languages.
BACKTESTING ALGOR. CRYPTOC. TRADING STRATEGIES                                            6
2. process flexible data inputs even from the researchers’ self-sourced data;
  5. provide users with commonly-known input and output formats that can be
      inspected quickly (CSV and Excel).
      The objective is to create a reliable and easy-to-use tool that speeds up the
process of backtesting algorithmic trading strategies for cryptocurrencies. The main
contribution of this thesis is not its empirical results – the focus is on the software
that produced those results. Ideally, this thesis paves the way for even more fruitful
future research on cryptocurrency markets.
BACKTESTING ALGOR. CRYPTOC. TRADING STRATEGIES                                           7
      As this thesis contains many technical terms, I would like to introduce the
notation that I will use throughout the text. A minimum of formatting conventions
makes it easier for the reader to distinguish purely technical terms from conceptual
terms.
      First, Python objects and file paths are formatted in monospace font, as
this is the font that closely resembles the fonts of development environments (text
editors). Many people associate code and software with this type of font. For
example, the table that contains the price data is denoted as df_price_data.
      Second, any callable Python function has a suffix consisting of one left and one
right bracket, like so: name_of_the_function(). The usage of open-close-bracket
suffixes is a standard convention in software documentations. For example, the order
execution function is denoted as execute_order().
      Third, internal references to other sections are written in italics and contain
the section number. For example, a reference to the data section looks like this:
“Please refer to section 7 Data”.
      All uncommon terms are introduced according to APA style rules. For exam-
ple, the abbreviation “NVT ratio” is introduced as: “network value to transactions
ratio (NVT ratio)”. After introducing a term once, the normal font will be used.
Further reference can be found in Publication manual of the American Psychological
BACKTESTING ALGOR. CRYPTOC. TRADING STRATEGIES                                            8
     The early work of Markowitz, 1952 and Lintner, 1965a, 1965b; Mossin, 1966 is
concerned with the role of variance and market risk in financial asset markets. The
arbitrage pricing theory by Ross, 1976 is a controversial generalization of established
methods; it opened the door for more flexible approaches to asset pricing and more
accommodating factor models beyond market risk considerations. Asness et al.,
2018; Banz, 1981; Fama and French, 1992, 1993, 1996 study the size factor (based
on market capitalization); Carhart, 1997; Jegadeesh and Titman, 1993 study the
momentum factor; Ang et al., 2009 study the volatility factor. Sentiment, albeit
being a broad concept and not a strictly defined risk factor, also plays a vital role
in equity markets, as Jiang et al., 2019 exemplify.
     The equity market risk factors presented in the previous paragraph were al-
ready covered in cryptocurrency markets, as Sovbetov, 2018, Shen et al., 2019, and
Li and Yi, 2019 summarize. Despite the short time of operation of cryptocurrency
BACKTESTING ALGOR. CRYPTOC. TRADING STRATEGIES                                                                  9
Figure 1 .   Overview of research coverage of different asset pricing factors for equity markets and cryp-
tocurrency markets. The overview aims to compare research coverage in equity markets and cryptocurrency
markets in the three different research categories statistical characteristics, inherent characteristics, and
non-statistical, non-inherent approaches. Own visualization created with Google Slides.
       Risk factors that stem from the inherent properties of the securitized/tok-
enized asset (such as investment and profitability, Fama and French, 2015, 2017)
are harder to establish in cryptocurrency asset pricing research as direct analogies
are more challenging to draw. Those factors are called fundamental accounting fac-
tors. Profitability was presented by Basu, 1977 and later adjusted in other models
such as the Fama-French five-factor model to reflect not current, but reported fu-
ture profitability. As stock markets are concerned with trading equity capital of
cash-generating businesses, the profitability factor is an inherent factor of the un-
derlying asset. The profitability factor, along with the investment factor, cannot be
observed in cryptocurrency markets as it simply does not exist there. Other fun-
damental factors face researchers and investors with similar applicability problems
for cryptocurrency markets. Asness et al., 2013; Fama and French, 1998 study the
value factor, which is based on companies’ accounting information and, therefore,
not applicable to cryptocurrencies. Bhandari, 1988 studies the debt factor, which is
again part of financial accounting. Direct analyses of the blockchain networks are a
probable path forward to replace factors that are present in the stock market, but
BACKTESTING ALGOR. CRYPTOC. TRADING STRATEGIES                                        10
      Hubrich, 2017 draws an analogy between the book value of equity and the
on-chain transaction value and thereby introduces a cryptocurrency-specific value
factor. Similarly, Woo, 2017 utilizes on-chain transaction volume by creating a
growth factor that assumes that a cryptocurrency’s value arises from the transaction
activity on the network. The growth factor is thereby expressed as the Network Value
to Transactions Ratio (NVT) and is coined “a PE ratio for Bitcoin” (p. 1) by the
author. The NVT is the fraction of market capitalization over on-chain transactions.
The NVT calculation can be smoothed by using a moving average of the last n days,
like so:
         Methodologies for finding fair values for cryptocurrencies are less well under-
stood than the same methodologies in traditional asset classes. In traditional asset
classes, discounted future cash flows are usually the underlying driver of asset prices,
making future cash flows and their riskiness the two key variables of consideration.
Risk factors are often a reflection of risk and return. A similarly easy answer does not
exist in the cryptocurrency domain, making direct modeling of supply and demand
of cryptocurrencies (as opposed to price modeling) an interesting field of study. The
supply of Bitcoin, for example, can be modeled by using the fraction of its cost to
mine over the block reward. Hayes, 2015 propose three supply-based metrics for
predicting the Bitcoin price and achieved an R2 of 0.84. The idea behind Hayes’
model is that the Bitcoin price should have a fundamental connection to the cost
of electricity per unit of physical work (W ) performed, the efficiency of mining as
measured by W per unit of mining effort, the market price of Bitcoin (as measured
by USD per Bitcoin), and the difficulty of mining (as measured by the complexity
of the hashing problem). For the demand of Bitcoin, Athey et al., 2016 propose an
approach to measure user adaption and provide a link between user adaption and
Bitcoin price.
  21
       One needs to keep in mind that Thaler’s article is not up-to-date anymore and the mentioned monthly
returns probably changed.
BACKTESTING ALGOR. CRYPTOC. TRADING STRATEGIES                                                              12
teristics. Some investors try to determine gold’s fair value using the stock-to-flow
ratio, which is a metric that expresses the fraction of available stock/inventory of
gold in the world over the yearly amount of mined gold:
                                         Inventory (stock) of already mined (and not in industrial use) gold
Annual stock-to-flow ratio =
                                                        Annual volume (flow) of mined gold
The stock-to-flow ratio is useful for commodities that have little industrial applica-
tions. Therefore, the stock-to-flow ratio should be used for assets whose primary
economic function is a store of value, such as gold. The prices of other rare metals
such as silver are too heavily influenced by industrial demand to be modeled reliably
with the stock-to-flow ratio. One can argue that the analogy to gold in the cryp-
tocurrency space is Bitcoin: Bitcoin is not the most efficient means of payment, but
Bitcoin is inherently secure, making it useful as a store of value. The stock-to-flow
ratio became so popular in the Bitcoin community that there are already several
websites that offer live monitoring of Bitcoin’s stock-to-flow ratio22 .
7 Data
          The choice for the optimal price data frequency is mostly dependent on the
requirements of the strategies that one wants to backtest. Day traders have different
  22
       These websites show the historical and the current stock-to-flow ratios for Bitcoin: digitalik.net/btc/
and lookintobitcoin.com/charts/stock-to-flow-model/.
BACKTESTING ALGOR. CRYPTOC. TRADING STRATEGIES                                      13
requirements than longer-term investors. For people that are undecided about the
frequency to use, I list some trade-offs between using higher and lower frequency
data here.
        Higher frequencies are associated with more storage requirements, higher mem-
ory usage, more computational load, and are often harder to acquire. Lower frequen-
cies may necessitate longer time horizons to be tested for reliable backtest results.
        For this thesis, I think that a relatively low frequency (daily) is the right
choice. Strategies that work with daily frequencies can be configured to be easily
understandable while still being economically sensible. Also, many research articles
use daily data, and experience shows that daily rebalancings are often the maxi-
mum sensible rebalancing frequency that does not incur prohibitive trading costs.
quantbacktest is not designed for arbitrage trading and high-frequency approaches,
as discussed in section 10.4.3 Arbitrage Trading.
        quantbacktest can also handle any other frequency, but I will consistently
use daily data throughout this study.
        In all test runs, the ITSA TOKENBASE price data was used. ITSA e.V.23
(ITSA) is a non-profit organization that provides the distributed-ledger technol-
ogy (DLT) community with standardized identifiers and with a unified classifica-
tion framework for cryptocurrencies. ITSA gathered their data from CoinGecko24 .
The columns from the ITSA TOKENBASE CSV file need to be renamed as
quantbacktest expects specific column names for the price data input: token_date
is renamed to datetime, token_itin is renamed to id, and token_price is renamed
to price. The price column contains mid prices; bid and ask prices are not listed
separately.
 23
      itsa.global.
 24
      coingecko.com.
BACKTESTING ALGOR. CRYPTOC. TRADING STRATEGIES                                        14
      8.1.2   Batch Processing. When using the term batch processing, I mean
that the user tests multiple backtests with different scenarios in a single exe-
cution of the program.      Scenarios can differ in many possible ways; notably,
they can assume varying minimum cash constraints, time intervals, rebalanc-
ing frequencies, or variations in other parameters. The advantage of this batch
processing functionality is that the user gets summarized results for all sce-
narios in a single df_performance_metrics table (explained in section 8.1.5
df_performance_metrics). As a visualization of this summary, the user gets out-of-
the-box heatmaps for time-varying robustness of the results (as an example, figure 9
is given in section 12.1 Additional Visualizations). Similarly, the user gets a heatmap
that shows the strategy’s robustness to variations in hyperparameters (please refer
to figure 10 in section 12.1 Additional Visualizations for an example).
      Thus, batch processing allows the user to configure a complete set of scenar-
ios. The user does not need to keep track of which scenarios were already tested
and receives immediate visual robustness results. The risk and return metrics are
conveniently stored in a single CSV file along with the respective input parameters.
Batch processing helps to minimize errors and saves time.
             The input price data can be in CSV or Excel format. The table should at
     minimum have an asset identifier (usually a string), a date or datetime (usually a
     string or an integer that represents a Unix epoch), and a price (usually a float).
     The asset identifiers are assumed to appear in a column called id, the timestamps
     are assumed to be in a column called datetime, and the prices are assumed to be
     in a column called price.
     Sharpe ratio. Some of these metrics are used in section 9 Results, starting at page
     38 to evaluate the exemplary strategies. The metrics are:
 1   ’ S t r a t e g y metadata −−−> ’ ,
 2   ’ S t r a t e g y ID ’ ,
 3   ’ Strategy label ’ ,
 4   ’ Trading i n f o −−−> ’ ,
 5   ’ Begin time o f t e s t e d i n t e r v a l ’ ,
 6   ’ End time o f t e s t e d i n t e r v a l ’ ,
 7   ’ D uration o f t h e t e s t e d i n t e r v a l ’ ,
 8   ’ D uration o f t h e t e s t e d i n t e r v a l ( i n days ) ’ ,
 9   ’ Average c a s h ’ ,
10   ’ Average t i c k e t s i z e ’ ,
11   ’ Number o f t r a d e s ’ ,
12   ’ Number o f un iqu e a s s e t s t r a d e d ’ ,
13   ’ Total t r a n s a c t i o n c o s t ’ ,
14   ’ Return m e t r i c s −−−> ’ ,
15   ’USD a n n u a l i z e d ROI ( from f i r s t t o l a s t t r a d e ) ’ ,
16   ’ C r y p t o c u r r e n c y a n n u a l i z e d ROI d e l t a ( from f i r s t t o l a s t t r a d e ) ’ ,
17   ’ Ending benchmark v a l u e ( f i r s t t o l a s t t r a d e ) ’ ,
18   ’ I n i t i a l budget ’ ,
19   ’ Ending p o r t f o l i o v a l u e ’ ,
20   ’ Risk m e t r i c s −−−> ’ ,
21   ’ H ol d in g p e r i o d v o l a t i l i t y ’ ,
22   ’ Annual v o l a t i l i t y ’ ,
23   ’ Monthly v o l a t i l i t y ’ ,
24   ’ Weekly v o l a t i l i t y ’ ,
25   ’ Beta r e l a t i v e t o benchmark ’ ,
26   ’Maximum drawdown ’ ,
27   ’Maximum drawdown d u r a t i o n ’ ,
28   ’Maximum drawdown peak d a t e ’ ,
29   ’Maximum drawdown t r o u g h d a t e ’ ,
30   ’ Other m e t r i c s −−−> ’ ,
31   ’ Alpha ’ ,
32   ’ Sharpe r a t i o ( h o l d i n g p e r i o d ) ’ ,
33   ’ Sharpe r a t i o ( y e a r l y ) ’ ,
34   ’ B e g i n n i n g benchmark v a l u e ( f i r s t t o l a s t t r a d e ) ’ ,
35   ’ Other i n f o −−−> ’ ,
36   ’ S t a r t time ’ ,
37   ’ End time ’ ,
38   ’ Parameter 1 ’ ,
39   ’ Parameter 2 ’ ,
40   ’ Comments ’ ,
41   ’ Benchmark r e t u r n m e t r i c s −−−> ’ ,
42   ’ Benchmark USD a n n u a l i z e d ROI ( from f i r s t t o l a s t t r a d e ) ’ ,
43   ’ Benchmark c r y p t o c u r r e n c y a n n u a l i z e d ROI d e l t a ( from f i r s t t o l a s t t r a d e ) ’ ,
44   ’ Benchmark e n d i n g benchmark v a l u e ( f i r s t t o l a s t t r a d e ) ’ ,
45   ’ Benchmark i n i t i a l budget ’ ,
46   ’ Benchmark e n d i n g p o r t f o l i o v a l u e ’ ,
     BACKTESTING ALGOR. CRYPTOC. TRADING STRATEGIES                                            17
47   ’ Benchmark r i s k m e t r i c s −−−> ’ ,
48   ’ Benchmark h o l d i n g p e r i o d v o l a t i l i t y ’ ,
49   ’ Benchmark annual v o l a t i l i t y ’ ,
50   ’ Benchmark monthly v o l a t i l i t y ’ ,
51   ’ Benchmark weekly v o l a t i l i t y ’ ,
52   ’ Benchmark Beta r e l a t i v e t o benchmark ’ ,
53   ’ Benchmark maximum drawdown ’ ,
54   ’ Benchmark maximum drawdown d u r a t i o n ’ ,
55   ’ Benchmark maximum drawdown peak d a t e ’ ,
56   ’ Benchmark maximum drawdown t r o u g h d a t e ’ ,
57   ’ Benchmark o t h e r m e t r i c s −−−> ’ ,
58   ’ Benchmark Sharpe r a t i o ( h o l d i n g p e r i o d ) ’ ,
59   ’ Benchmark Sharpe r a t i o ( y e a r l y ) ’
            8.2.1       General Program Flow. The backtest starts with a data loading
     process. Only CSV format and Excel format are allowed. The CSV file with price
     data will be loaded into memory as a pandas DataFrame, df_price_data.
            The steps are described in more detail in the figures 2 and 3 below.
BACKTESTING ALGOR. CRYPTOC. TRADING STRATEGIES                                                             18
Figure 2 .   The high-level backtesting workflow in six steps. This overview is relevant to all users. First,
the user determines which hyperparameters and settings to use (column 1). Then, the specified scenario(s)
will be executed separately (column 2). The run of a single scenario (column 3) is explained in more detail
in figure 3. All results are stored to disk in CSV format and in image format (column 4). The user can
open the results and choose to use additional software such as Tableau and Excel for further analyses
(column 5). The generated insights can be used to decide to take a strategy online (i.e., trade real money)
or to change the strategy (column 6). Own visualization created with Google Slides.
BACKTESTING ALGOR. CRYPTOC. TRADING STRATEGIES                                                            19
Figure 3 .    The low-level backtesting workflow in three steps. First, the user chooses strategy hyper-
parameters, such as optimization constraints and machine learning hyperparameters, and other settings,
such as the number of rounding decimal places, warnings handling, and price data gap tolerance settings
(see “Hyperparameters/settings” in column 1). Then, quantbacktest will run the scenario(s) specified by
the user (repeating process between columns 1 and 2). Trades change the account balance and exposure
states in df_trading_journal, and the trades create a feedback loop to the strategy. The strategy has
access to the current portfolio state through df_trading_journal. For example, the current cash posi-
tion is recorded there. The strategy may behave differently under certain circumstances (e.g., if certain
exposure limits are already crossed, the strategy may decide to divest certain assets), making the feed-
back loop essential. After every trade has been recorded in df_trading_journal, it is converted to a
standardized, fixed-step format. This format is called df_returns and is shown in column 3. It contains
returns and portfolio values for fixed, user-defined intervals. An interval can be hourly, daily, weekly, or
any arbitrary frequency. From this standardized format, common risk and return metrics can be calculated
(df_results_metrics). The process depicted here is also part of figure 2, column 3 on page 18. The user
can decide to run another backtest with adjusted settings after inspecting the results. The user should be
aware of the risk of overfitting when running many scenarios and repeating this process. Own visualization
created with Google Slides.
8.2.2.1      General Order Simulation The order simulation logic is at the heart
of quantbacktest. The function execute_order() handles this step. All strate-
gies use the execute_order() function as an interface for simulated order routing.
execute_order() may or may not execute the requests of a strategy, depending on
the cash balance, exposure constraints, and other checks. It also performs a sanity
check whether the request is consistent in itself by comparing the signal type (buy or
sell) with the order amount (positive or negative). Orders can also be filled partially
     BACKTESTING ALGOR. CRYPTOC. TRADING STRATEGIES                                       20
     if the settings allow for partial order filling. As execute_order() contains exten-
     sive logic and tests, it aims to mitigate and raise errors that occur in the strategy
     definition and the settings. The extensive logic within the order execution simula-
     tion aims to resemble real-world order execution systems as closely as possible and
     to reduce the required amount of logic that needs to be included in the strategy
     functions.
     One can see that this function takes many arguments. The function is thus further
     divided into inner functions that handle the most complex sub-tasks.
              Slippage25 is the time- and the volume-induced difference between the initially
    observed price and the real execution price and is the result of high-volume trades,
    slow internet speeds on the client’s side, delayed order routing by the broker, and
    other factors (Chan, 2009, p. 23). I define market impact to be one subset of
    slippage (volume-induced slippage). Slippage is taken care of by the tool; users
    specify slippage as a setting. There is no volume-dependent slippage model yet.
              For simplicity, slippage also accounts for the bid-ask spread26 , so it is highly
    recommended to specify a slippage higher than 0%. The spread is the difference
    between the bid price and the ask price and reflects the loss that one would make
    when buying and immediately selling an asset at an individual exchange.
              Commissions are fees that brokers and exchanges charge for their services.
    They may be flat (per order) or relative to the nominal amount traded, both of
    which options can be simulated with this tool.
    of the total daily trading volume of any asset in equity trading (Chan, 2009, pp.
    87–88).
          This function calculates the mark-to-market returns for one trade to the next
    at a given frequency.
1   previous_trading_journal_row ,
2   current_trading_journal_row ,
3   df_prices ,
4   strategy_hyperparameters ,
5   display_options ,
6   general_settings ,
7   constraints
          The calculate returns function takes one adjacent pair of trading journal entries
    and calculates all returns (at a given frequency) that fall between these two trades.
    For example, suppose that df_trading_journal contains a trade on day one and
    another trade on day four. The calculate_returns_single() function will get the
    row from day one (row one) and the row from day four (row two) as an input. It
    will calculate the portfolio value on day one, day two, day three, and day four. It
    finally returns a list of dictionaries:
1   list_of_dict_returns
    The dictionaries within this list not only contain returns, but also other helpful
    fields:
1   datetime
     BACKTESTING ALGOR. CRYPTOC. TRADING STRATEGIES                                                                        23
 2   portfolio_value portfolio_return
 3   relative_portfolio_return
 4   dict_of_assets_in_portfolio
 5   benchmark_portfolio_value
 6   benchmark_portfolio_value_normalized
 7   portfolio_value_normalized
 8   benchmark_return
 9   benchmark_relative_return
10   benchmark_dict_of_assets_in_portfolio
             The function returns a list that contains returns at a specified frequency for
     the timeframe between two given trades with the size of the list given by:
             Returns in themselves are not a performance metric; they are just a means for
     making calculations of performance metrics possible. For completeness and as the
     daily returns are so central, the method of generating daily returns from the trading
     journal is outlined here along with the metrics. Expressing the calculation of the
     daily return calculation in purely mathematical terms is not possible.
             As a first step, the method calculates the portfolio value for each day, given the
     cash holdings and asset holdings. If there is no portfolio information in the trading
     journal for a given day, it is assumed that there were no cash or asset changes since
     the last trade. This is not a problematic assumption. For example, if at day one,
     there were $100 and 1 Bitcoin in the portfolio and one wanted to calculate the
     portfolio value of day two without having a trading journal entry for that day, one
     would assume that the cash value is still $10028 and that the number of Bitcoin is
     still 1. Bitcoin will be re-evaluated with current market prices on day two, and the
     result will be added to the cash value to yield the portfolio value for day two.
      28
           Daily costs for margin positions (interest payments) will be included as soon as margin trading is
     supported by quantbacktest.
     BACKTESTING ALGOR. CRYPTOC. TRADING STRATEGIES                                                                                              24
 3   Dict f i e l d s :      ’ timestamp ’ ( d a t e t i m e . d a t e t i m e ) , ’ p o r t f o l i o _ v a l u e ’ ( f l o a t ) ,
 4                           ’ return ’ ( f l o a t ) , ’ relative_return ’ ( f l o a t ) ,
 5                           ’ d i c t _ o f _ a s s e t s _ i n _ p o r t f o l i o ’ ( d i c t with i t i n s a s k e y s and
 6                           i n t e g e r with t h e number o f p i e c e s h e l d o f t h i s a s s e t a s a s
 7                           values )
 8
15   Trades                                              Frequency i n c r e m e n t s
16
17   −No t r a d e −                                    Minute 1
18   Trade 1                                            Minute 2
19   −No t r a d e −                                    Minute 3
20   −No t r a d e −                                    Minute 4
21   Trade 2                                            Minute 5
22
33   d a t e t i m e . d a t e t i m e o b j e c t s assume ’ t h e r e a r e e x a c t l y 3600∗24 s e c o n d s i n e v e r y
34   day ’ . h t t p s : / / d o c s . python . o r g /2/ l i b r a r y / d a t e t i m e . html#datetime −o b j e c t s
35   """
              Arguments:
 1   df_trading_journal ,
 2   df_prices ,
 3   strategy_hyperparameters ,
 4   display_options ,
 5   general_settings ,
    BACKTESTING ALGOR. CRYPTOC. TRADING STRATEGIES                                     25
6 constraints
            Return:
1   df_returns
            Arguments:
1   annualized_portfolio_return ,
2   risk_free_rate ,
3   beta_exposure ,
4   an nualized_market_return
            Return:
1   alpha
market risk. Large alphas are generally desirable. Jensen’s alpha is defined as:
           Arguments:
1   df_daily_returns ,
2   df_daily_benchmark_returns
           Return:
1   beta
           β (beta) reflects the sensitivity of a portfolio to the systematic risk of the mar-
    ket. The market is defined here as the user-defined benchmark. If β is higher than
    one, the portfolio is expected to react overproportionately to market movements.
    If β is smaller than one, the portfolio is expected to react underproportionately to
    market movements. β can be smaller than zero. Assets and portfolios with betas
    smaller than zero can serve as a hedge against market risk. β is calculated as
with ri being the portfolio returns and rM being the market returns.
           Arguments:
    BACKTESTING ALGOR. CRYPTOC. TRADING STRATEGIES                                         27
1 df_daily_returns
          Return:
1   drawdown ,
2   maximum_drawdown_duration . days ,
3   peak_date ,
4   trough_date
          The maximum drawdown does not necessarily take the highest portfolio value
    as an input for the high point. It may also use an earlier high point. The maximum
    drawdown represents the maximum relative decrease in portfolio value. If, for ex-
    ample, the portfolio would be at $100 at day 1, $50 at day 2, $200 at day 3, and
    $120 at day 4, the maximum drawdown would be 50%, the high date would be day
    1, and the low date would be day 2. The maximum drawdown duration would be
    one day.
          The maximum drawdown is a risk measure and can give investors an indication
    of what they should be ready to lose if they invest in an individual portfolio (or follow
    a specific strategy). Future losses can be even higher than the backtested maximum
    drawdown.
            Arguments:
 1   df_trading_journal ,
 2   float_budget_in_usd ,
 3   timeframe_whole_or_first_to_last_trade ,
 4   d f=None ,
 5   df_benchmark=None ,
 6   df_price_column_name= ’ token_price_usd ’ ,
 7   df_time_column_name= ’ time ’ ,
 8   df_benchmark_price_column_name= ’ token_price_usd ’ ,
 9   df_benchmark_time_column_name= ’ time ’ ,
10   df_trading_journal_price_column_name= ’ P o r t f o l i o v a l u e ’ ,
11   df_trading_journal_time_column_name= ’ Date o f e x e c u t i o n ’
            Return:
 1   return_on_investment
with n being the number of years and Ki being the portfolio value in year i.
     Interpolating short time intervals to one year in this way can lead to substantial bias
     because shorter time intervals usually exhibit significant return variance. Scaling
     larger time intervals down to one year, however, is usually not problematic.
            Arguments:
    BACKTESTING ALGOR. CRYPTOC. TRADING STRATEGIES                                                              29
1   df_daily_returns ,
2   portfolio_roi_usd ,
3   days=None ,
4   r i s k _ f r e e _ r a t e=None
              Return:
1   sharpe_ratio
    The risk-free rate should be deducted to make the numerator and the denominator
    fit, as originally intended by William Sharpe, 1966.29
              Arguments:
1   df_trading_journal
      29
           There is some controversy about whether to subtract the risk-free rate from the portfolio return. For
    that reason, both ways of calculating the Sharpe ratio are implemented here, and it is left to the user to
    decide which one to use. I recommend using the Sharpe ratio that includes the risk-free rate. Only excess
    returns should have an associated risk in the denominator of the formula. As a supporting example, if one
    omits the risk-free rate when calculating the Sharpe ratio, a risk-free asset would have a Sharpe ratio of
    infinity. Assuming a positive risk-free rate, omitting the risk-free rate would yield inflated results. Chan,
    2009, pp. 43–44, one of the most renowned practitioners in algorithmic trading, argues against adjusting
    for the risk-free rate. He argues that one can earn “a credit interest close to the risk-free rate,” giving an
    excess return of R + rf − rf = R. Arguing against this, and to the best of my knowledge, I do not think
    that it is realistic to assume that cash holdings can reliably be invested at the risk-free rate; there is not
    a single cryptocurrency exchange that offers risk-free interest rates on unused funds today.
    BACKTESTING ALGOR. CRYPTOC. TRADING STRATEGIES                                             30
           Return:
1   total_transaction_cost
           The sum of transaction costs is helpful when making trade-offs between trans-
    action costs and rebalancing intervals. It can also help to contribute cost impact
    assessments for trading with different exchanges and brokers.
           Arguments:
1   df_daily_returns ,
2   time_adjustment_in_days=None
           Return:
1   volatility
           Investors usually have a preference for specific volatility (σ) observation pe-
    riods. Long-term investors may be familiar with yearly volatility, while investors
    with shorter holding periods may be more familiar with daily and weekly volatility.
    Therefore, the backtesting allows users to adjust volatility measures to any time
    interval. Adjusted volatilities are also used in the Sharpe ratio calculation. The
    time-adjusted volatility (Chan, 2009, p. 44) is defined as
                                                          √
                                                            adjusted duration
       Volatility σ = σof all daily returns ∗ q                                            .
                                                original duration (trading period in days)
    The assumption behind this formula is that returns are serially uncorrelated. This
    approach can be highly problematic as volatility may change significantly over time.
    BACKTESTING ALGOR. CRYPTOC. TRADING STRATEGIES                                                                         31
    8.3.2.1        Code Access, System Requirements, and Setup All code is avail-
    able in the section 12 Appendix from page 58. Nevertheless, I recommend to refer
    to the code from the GitLab repository or to install the module by typing pip
    install quantbacktest in a shell. After a successful installation, please define
    the arguments that are presented in the following paragraphs. With all arguments
    defined, start the backtesting by calling backtest_visualizer():
1   from q u a n t b a c k t e s t import b a c k t e s t _ v i s u a l i z e r
2
3   # For managing d a t e s
4   from d a t e t i m e import d a t e t i m e
5
7   backtest_visualizer (
8         f i l e _ p a t h _ w i t h _ p r i c e _ d a t a= ’ /home/ j a n s p o e r e r / code / j a n s p o e r e r /
          q u a n t b a c k t e s t / q u a n t b a c k t e s t / a s s e t s / raw_itsa_data /20190717
     BACKTESTING ALGOR. CRYPTOC. TRADING STRATEGIES                                                                                           32
       _itsa_tokenbase_top600_wtd302_token_daily . c s v ’ ,
 9     # ONLY LEAVE THIS LINE UNCOMMENTED IF YOU WANT TO USE ETH−ADDRESSES AS
       ASSET IDENTIFIERS !
10     # file_path_with_token_data =’ raw_itsa_data /20190717
       _itsa_tokenbase_top600_wtd301_token . c s v ’ ,                                     # Only f o r multi −a s s e t
       strategies .
11      name_of_foreign_key_in_price_data_table= ’ t o k e n _ i t i n ’ ,
12      name_of_foreign_key_in_token_metadata_table= ’ t o k e n _ i t i n ’ ,
13     # 1: execute_strategy_white_noise ()
14     # 2 : Not used anymore , can be r e a s s i g n e d
15     # 3 : e x e c u t e _ s t r a t e g y _ m u l t i _ a s s e t ( ) −> Uses s t r a t e g y t a b l e
16     # 4 : execute_strategy_ma_crossover ( )
17      i n t _ c h o s e n _ s t r a t e g y =4,
18      d i c t _ c r y p t o _ o p t i o n s={
19              ’ general ’ : {
20                      ’ percentage_buying_fees_and_spread ’ : 0 . 0 0 5 ,                                    # 0.26% i s t h e
       t a k e r f e e f o r low−volume c l i e n t s a t kraken . com h t t p s : / /www. kraken . com/
       f e a t u r e s / f e e −s c h e d u l e
21                      ’ percentage_selling_fees_and_spread ’ : 0.005 ,                                            # 0.26% i s t h e
       t a k e r f e e f o r low−volume c l i e n t s a t kraken . com h t t p s : / /www. kraken . com/
       f e a t u r e s / f e e −s c h e d u l e
22                    # A d d i t i o n a l f e e s may apply f o r d e p o s i t i n g money .
23                      ’ absolute_fee_buy_order ’ : 0 . 0 ,
24                      ’ absolute_fee_sell_order ’ : 0.0 ,
25             }
26      },
27      float_budget_in_usd =10 000 00. 00 ,
28      f i l e _ p a t h _ w i t h _ s i g n a l _ d a t a=f i l e _ p a t h _ w i t h _ s i g n a l _ d a t a ,
29      s t r a t e g y _ h y p e r p a r a m e t e r s=s t r a t e g y _ h y p e r p a r a m e t e r s ,
30      margin_loan_rate =0.05 ,
31      l i s t _ t i m e s _ o f _ s p l i t _ f o r _ r o b u s t n e s s _ t e s t =[
32              [ datetime (2014 , 1 , 1) , datetime (2019 , 5 , 30) ]
33      ],
34      b e n c h m a r k _ d a t a _ s p e c i f i c a t i o n s={
35              ’ name_of_column_with_benchmark_primary_key ’ :                                         ’ id ’ ,      # W i l l be i d
       a f t e r p r o c e s s i n g . Columns w i l l be renamed .
36              ’ benchmark_key ’ :               ’TP3B−248N−Q ’ ,             # Ether : T22F−QJGB−N, B i t c o i n : TP3B
       −248N−Q
37              ’ file_path_with_benchmark_data ’ :                             ’ /home/ j a n s p o e r e r / code / j a n s p o e r e r /
       q u a n t b a c k t e s t / q u a n t b a c k t e s t / a s s e t s / raw_itsa_data /20190717
       _itsa_tokenbase_top600_wtd302_token_daily . c s v ’ ,
38              ’ risk_free_rate ’ : 0.02
39      },
40      d i s p l a y _ o p t i o n s=d i s p l a y _ o p t i o n s ,
41      c o n s t r a i n t s=c o n s t r a i n t s ,
42      g e n e r a l _ s e t t i n g s=g e n e r a l _ s e t t i n g s ,
43      comments=comments ,
     BACKTESTING ALGOR. CRYPTOC. TRADING STRATEGIES                                                                  33
44 )
               The repository30 is stored within the Frankfurt School Blockchain Center Git-
     Lab Group. There is one public repository with the newest version and one private
     repository that contains history before April 21, 2020. The reason for this split is
     that the older history contains confidential data that had to be removed before any
     public release. The repository is structured as follows:
 1   quantbacktest /
 2   |
 3   | −− q u a n t b a c k t e s t /
 4   |        | −− a s s e t s /
 5   |        | −− components /
 6   |              | −− _0_wrappers . py
 7   |              | −− _1_data_preparation . py
 8   |              | −− _ 2 _ s t r a t e g y _ e x e c u t i o n . py
 9   |              | −− _ 3 _ i n d i v i d u a l _ m e t r i c s . py
10   |              | −− _3_performance_evaluation . py
11   |              | −− _ h e l p e r _ f u n c t i o n s . py
12   |        | −− __init__ . py
13   |
14   | −− t e s t s /
15   |        | −− __init__ . py
16   |
17   | −− LICENSE
18   | −− MANIFEST. i n
19   | −− README. md
20   | −− s e t u p . py
               For using the tool, using a Unix-like operating system such as Apple macOS
     or a Linux distribution (e.g., Ubuntu 18.04) is recommended as the tool is designed
     to be accessed from a Unix shell. Alternatively, shells in Windows 10, such as those
     provided by Anaconda31 and git for windows 32 , may be used. The author used
     Anaconda (Python package manager), zsh (shell), and Ubuntu 18.04 LTS (Linux
     distribution, operating system).
         30
            gitlab.com/fsbc/theses/quantbacktest.
         31
            anaconda.com.
         32
            gitforwindows.org.
     BACKTESTING ALGOR. CRYPTOC. TRADING STRATEGIES                                                                      34
           Installing the necessary dependencies should not pose a problem as most li-
     braries are commonly used, and many users probably have them pre-installed. No
     specialized backtesting libraries or other financial libraries were used. Ubuntu, ma-
     cOS, and Windows 10 users can use Anaconda to install libraries. Please refer to
     the read.me that you can find on GitLab, it contains additional information that is
     required to set up the project.
     8.3.2.2      Usage Before starting the program, the user defines settings. Possible
     settings are described in the following paragraphs.
4 }
 4   strategy_hyperparameters = {
 5          ’ maximum_deviation_in_days ’ : 3 0 0 ,
 6          ’ prices_table_id_column_name ’ :                         ’ token_itin ’ ,
 7          ’ excel_worksheet_name ’ : excel_worksheet_name ,                                        # S e t t h i s t o None i f CSV
             i s used !
 8          ’ buy_parameter_space ’ : [ 9 . 8 ] ,                    # [ 1 1 , 2 0 ] # Times 1 0 ! W i l l be d i v i d e d by
             10.
 9          ’ sell_parameter_space ’ : [ 9 . 7 ] ,                     # [ 5 , 9 ] # Times 1 0 ! W i l l be d i v i d e d by
           10.
10          ’ maximum_relative_exposure_per_buy ’ : 0 . 3 4 ,
11          ’ f r e q u e n c y ’ : Timedelta ( days =1) ,
12          ’ moving_average_window_in_days ’ : 1 4 ,
13          ’ id ’ :   ’TP3B−248N−Q ’ ,
14          ’ b o o l e a n _ a l l o w _ p a r t i a l l y _ f i l l e d _ o r d e r s ’ : True ,
15   }
               The optional comments dictionary does not impact the logic of the back-
    testing at all; it is only a place for users and developers to record useful informa-
    tion. The comments that the user defines are saved in the df_results_metrics
    DataFrame and stored to disk as a CSV file when the backtesting finishes. De-
    velopers can add additional information to the comments in their strategies as
    the program runs. Some exemplary comments are the display_options and the
    strategy_hyperparameters and are listed here:
1   comments = {
2          ’ display_options ’ : repr ( display_options ) ,
3          ’ strategy_hyperparameters ’ : repr ( strategy_hyperparameters ) ,
4   }
               The user can specify the benchmark and the file path to the benchmark data
    within the benchmark_data_specifications dictionary. Furthermore, this set-
    ting contains the risk-free rate.
1   b e n c h m a r k _ d a t a _ s p e c i f i c a t i o n s={
    BACKTESTING ALGOR. CRYPTOC. TRADING STRATEGIES                                                           37
2        ’ name_of_column_with_benchmark_primary_key ’ :             ’ id ’ ,   # W i l l be i d a f t e r
        p r o c e s s i n g . Columns w i l l be renamed .
3        ’ benchmark_key ’ :    ’TP3B−248N−Q ’ ,      # Ether : T22F−QJGB−N, B i t c o i n : TP3B−248
        N−Q
4        ’ file_path_with_benchmark_data ’ :           ’ raw_itsa_data /20190717
        _itsa_tokenbase_top600_wtd302_token_daily . c s v ’ ,
5        ’ risk_free_rate ’ : 0.02 ,
6   }
9 Results
         The white noise strategy is based on a simple random number generator. Ev-
ery day, a random number generator produces a buy or a sell signal with equal
probability. In the presence of a buy signal, 34% of the remaining cash is converted
into Bitcoin; in the presence of a sell signal, all Bitcoin in the portfolio is sold. This
simple strategy tests if the backtesting works as expected.
         From a strategy like this, one can expect two results: 1) lower market exposure
(as measured by the equity beta) due to the cash-heaviness of the portfolio, and 2)
a drag to performance as the frequent rebalancings cause high transaction costs (a
typical run will produce approximately 2000 trades).
         The two expected properties that I just mentioned can also be observed from
the results of the test run. The results are visualized in the equity curve from figure
4 and summarized in the output results/backtesting_result_metrics.csv. In
  33
       pandas.pydata.org/pandas-docs/stable/user_guide/advanced.html.
BACKTESTING ALGOR. CRYPTOC. TRADING STRATEGIES                                         39
this run, the beta is 0.46, indicating that the market returns only moderately influ-
ence the portfolio returns. This is consistent with expectation 1). Also, the portfolio
value drifts downward over time. This is consistent with the high transaction costs
of 726,361.07 USD in total and with expectation 2). The results are a back-of-the-
envelope indication that quantbacktest works correctly. In addition to tests like
this, I made many manual checks on individual transactions that also verified the
correctness of the method.
Figure 4 .   Standardized and benchmarked (portfolio value = 1 and Bitcoin price = 1 at t0 ) equity curve
for a white noise strategy. This portfolio was created using random signals and is only a proof-of-concept
for the backtesting method, and not an economically sensible strategy. Buy and sell signals are generated
every day with equal-weighted probability. The minimum cash requirement for this strategy is 100 USD. To
avoid rounding problems, I set the budget to 1m USD. Of course, this may lead to a higher market impact
than shown here. The percentage spread and slippage are 0.5% in total; no fixed fees were considered.
Own visualization created with Tableau Desktop based on results/df_result_metrics.csv.
reflect in prices.
Figure 5 .    Standardized and benchmarked (portfolio value = 1 and Bitcoin price = 1 at t0 ) for a
momentum strategy. This portfolio was created using a 14-day moving average strategy. If the Bitcoin
price is above its 14-day moving average, 34% of the remaining cash in the portfolio will be spent to
enter a long position in Bitcoin. If the Bitcoin price is below its 14-day moving average, all exposure will
be liquidated. Rebalancing happens every day. The minimum cash requirement for this strategy is 100
USD. To avoid rounding problems, I set the budget to 1m USD. Of course, this may lead to a higher
market impact than shown here. The percentage spread and slippage are in total 0.5%, no fixed fees were
considered. Own visualization created with Tableau Desktop based on results/df_result_metrics.csv.
here. In this case, only Bitcoin is traded, but a signal table can contain signals for
any number of assets as long as they are covered by the given price data.
      The strategy, at least at first sight, seems to have favorable risk management
properties. The diagram shows that the sentiment data helps to stay out of the mar-
ket before (or at the beginning of) some bad market phases. The performance of the
sentiment strategy is only slightly impaired, with an annualized underperformance
of -5.52% in comparison to a simple investment in Bitcoin (costs already considered).
The maximum drawdown was mitigated from 83.64% (Bitcoin, peak at 2017-12-16,
trough at 2018-12-15) to 66.23% (portfolio, peak at 2017-12-16, through at 2018-04-
07) because the position was exited before the bottom of the cryptocurrency crash
of early 2018.
      Despite the attractive risk-return trade-off that this strategy can achieve rel-
ative to its benchmark, there is a chance of overfitting in how the black-box signal
table was created and how the hyperparameters were chosen that determine the
sentiment threshold for entry and exit. The strategy only performed a few trades,
and those trades were (possibly luckily) favorable. Also, the strong performance of
Bitcoin in the past is the major driver for this long-only, Bitcoin-only strategy. Live-
trading or paper trading will show if the risk-mitigating properties of this strategy
reflect a true edge.
BACKTESTING ALGOR. CRYPTOC. TRADING STRATEGIES                                                             44
Figure 6 .   Standardized and benchmarked (portfolio value = 1 and Bitcoin price = 1 at t0 ) equity curve
for a sentiment strategy. This portfolio was created using sentiment data as buy and sell signals. If the
sentiment is above the buying threshold, a Bitcoin position will be increased each day (34% of remaining
cash is used each day) until cash is depleted. All Bitcoin is sold if the sentiment value is smaller than the
selling threshold. Rebalancing happens very day. The minimum cash requirement for this strategy is 100
USD. To avoid rounding problems, I set the budget to 1m USD. Of course, this may, in turn, lead to a higher
market impact than shown here. The percentage spread and slippage are in total 0.5%, no fixed fees were
considered. Own visualization created with Tableau Desktop based on results/df_result_metrics.csv.
10 Conclusion
10.1 Summary
7 Data, I described the price data that is used in the backtesting and presented
trade-offs between higher-frequency and lower-frequency data. Section 8 Method
(Trading Strategy and Backtesting) contains the technical setup of the backtesting
and its functions. I also presented assumptions and made internal calculations of
the tool transparent. Lastly, in 9 Results, I tested some commonly known and
easy-to-understand strategies with the backtesting and analyzed if the results meet
expectations. All three strategies, 1) white noise, 2) momentum, and 3) sentiment
yielded the expected results. From these findings, I concluded that the backtesting
works well for realistic applications.
       The tool presented allows investors to build and test algorithmic trading strate-
gies for all major cryptocurrencies. Thanks to the ITSA TOKENBASE price data
integration, investors can access price data of approximately 600 different cryp-
tocurrencies and build strategies around these cryptocurrencies without the need
for advanced technical expertise. Consequently, the tool paves the way for broader
investor sophistication in the cryptocurrency space and facilitates sound investment
decisions.
       There are strategy parameters that are hard to test without a certain level
of backtesting sophistication. Transaction costs are a major concern in algorithmic
trading, and they have a significant impact on the required certainty that a trader
needs to have about the magnitude of market mispricing (Chan, 2009, p. 22). This
is because low gross returns may be dominated by transaction costs. Only a back-
testing tool that is capable of realistically simulating transaction costs can aid in
setting the minimum expected gross return threshold accurately.
       This tool makes complex backtesting tasks easier for researchers and can serve
to calculate portfolio returns and risk-relevant metrics realistically. Backtests can
BACKTESTING ALGOR. CRYPTOC. TRADING STRATEGIES                                       46
reflect trading costs, can simulate portfolio constraints, and can be separated into
training and test periods.
       I see most use of this backtesting for later steps in the quantitative trading
process and less in the academic idea generation phase. Initial research and hy-
pothesis testing can also be performed without a realistic backtesting engine. The
tool is most relevant for researchers that have to take into account fees and risk
management constraints as realistically as possible.
10.4 Limitations
         Second, there are already market participants that use sophisticated, low-
latency software (possibly even hardware, in the future34 ) to address arbitrage. I
believe that a backtesting tool for arbitrage trading should be managed in a separate,
dedicated arbitrage backtesting project.
  34
       In traditional finance, high-frequency traders use hardware-accelerated network stacks, network pro-
cessing units, field-programmable gate arrays, and relocation to gain an edge. These technological im-
provements lead to situations where it is not enough anymore to compete with software; one also needs to
employ sophisticated hardware to stay competitive.
BACKTESTING ALGOR. CRYPTOC. TRADING STRATEGIES                                                              48
potential is that all steps in one scenario run of the backtesting depend on each
other. For example, the cash balance after the first trade is dependent on the full
simulation of the first trade. The second trade cannot be simulated without know-
ing the cash balance after the first trade, so the backtest waits for the first trade
to finish before simulating the second trade. It is, however, possible to use multiple
CPU cores by running several scenarios in parallel.
          10.4.6      Improved Benchmarks. The results from this thesis all stem
from Bitcoin-only trading strategies with Bitcoin as a benchmark. The benchmark
influences relative performance metrics, such as market risk exposure (beta). Even
though Bitcoin has a consistently dominant market position in terms of market
capitalization, a more fine-grained benchmarking system, similar to those bench-
marks used in equity markets, would add value. Sophisticated benchmarking is an
especially pressing issue for multi-asset strategies that also trade altcoins. The back-
testing needs to provide a way to calculate a benchmark from a given price series.
A custom benchmark could be a market capitalization-weighted benchmark or an-
other approach. Incorporating the index calculation method proposed by Trimborn
and Härdle, 2018 would be an example of how this backtesting could be further
improved. Alternatively, one could use already available indices such as the market
capitalization-weighted Bloomberg Galaxy Crypto Index35 .
  35
       bloomberg.com/professional/product/indices/bloomberg-galaxy-crypto-index.
  36
       The date of publication is not stated on the white paper, so here is a trusted source that confirms the
date of publication: “Liechtenstein Blockchain Act,” 2018, pp. 9–10.
  37
     The corona crisis of 2020 happened at the same time that this thesis was written and could not be
considered in the presented results.
BACKTESTING ALGOR. CRYPTOC. TRADING STRATEGIES                                      49
     The backtested performance may not hold in a more mature and in a more
regulated cryptocurrency market. Retail investors may experience more regula-
tory scrutiny (e.g., tighter know your customer regulations for cryptocurrency ex-
changes), limiting market access. Already in 2014, the U.S. Federal Reserve showed
interest in Bitcoin (Badev & Chen, 2014), and the Liechtenstein regulation for dis-
tributed ledger technology sets an example for actual regulatory action (“Liechten-
stein Blockchain Act,” 2018, pp. 11, 28).
     The potential of blockchain, and in particular the potential for Bitcoin, is not
fully used today, and many of the applications described by Iansiti and Lakhani,
2017, p. 7 and “Liechtenstein Blockchain Act,” 2018, p. 18 are still under devel-
opment. Changes in the purpose of Bitcoin may change price movement patterns
in the future. Bitcoin could also lose importance as other DLT concepts such as
Ethereum, Hyperledger Fabric, and Corda offer a wide field of applications that is
not available for Bitcoin (Valenta & Sandner, 2017).
     Analogies from traditional finance show that inefficiencies in asset pricing tend
to vanish as the understanding of an asset’s characteristics increases (McLean & Pon-
tiff, 2016). Brauneis and Mestel, 2018 and Khuntia and Pattanayak, 2018 observe a
similar trend of diminishing inefficiencies in cryptocurrency markets. Another effect
is that previously well-established risk factors may lose relevance. Fama and French,
2019 recently described how even one of their cornerstone findings in finance, the
study of the value factor, seems to lose relevance as a risk factor over time. Fama
and French do not provide an intuitive explanation for this phenomenon, but one can
BACKTESTING ALGOR. CRYPTOC. TRADING STRATEGIES                                                             50
suggest that the rise of technology companies and a general shift in how economic
value is created (value of expensed intangibles) plays a role here. In the future,
similar market shifts are thinkable in cryptocurrency markets.
       Time will help to solve the presented maturity-related issues. Expanding the
use of the tool to intraday data could help to mitigate the importance of systematic
events and make strategies more robust despite the limited timeframe of available
data. Users of this tool need to be aware of the constant change that is present in
cryptocurrency markets and consider this when re-training and re-configuring their
models.
Figure 7 .    A visual example of selection bias. The figure shows that failed cryptocurrencies may fall
through the cracks today, leading to inflated backtesting results. In this example, the hindsight performance
of those cryptocurrencies that are still on the market is 10% on average. But the overall performance of all
five cryptocurrencies is −34%. The early days of cryptocurrencies may not be adequately reflected in the
composition of today’s cryptocurrency price databases. If this is true, backtests will often lead to biased
results. Own visualization created with Google Slides.
BACKTESTING ALGOR. CRYPTOC. TRADING STRATEGIES                                       51
11 References
Ang, A., Hodrick, R., Xing, Y., & Zhang, X. (2009). High idiosyncratic volatility
       and low returns: International and further U.S. evidence. Journal of Financial
       Economics, 91 (1), 1–23.
Asness, C. S., Frazzini, A., Israel, R., Moskowitz, T. J., & Pedersen, L. H. (2018). Size
       matters, if you control your junk. Journal of Financial Economics, 129 (3),
       479–509.
Asness, C. S., Moskowitz, T. J., & Pedersen, L. H. (2013). Value and momentum
       everywhere. The Journal of Finance, 68 (3), 929–985.
Athey, S., Parashkevov, I., Sarukkai, V., & Xia, J. (2016). Bitcoin pricing, adoption,
       and usage: Theory and evidence. Stanford University Graduate School of
       Business Research Paper.
Avramov, D., Kaplanski, G., & Subrahmanyam, A. (2018). The predictability of
       equity returns from past returns: A new moving average-based perspective.
       SSRN.
Badev, A. I., & Chen, M. (2014). Bitcoin: Technical background and data analysis.
       Federal Reserve Board working paper.
Banz, R. W. (1981). The relationship between return and market value of common
       stocks. Journal of Financial Economics, 9 (1), 3–18.
Basu, S. (1977). Investment performance of common stocks in relation to their price-
       earnings ratios: A test of the efficient market hypothesis. The Journal of
       Finance, 32 (3), 663–682.
Berghoff, L. (2020). Applying asset pricing factors in quantitative cryptoasset trading
       strategies (Bachelor’s Thesis). Frankfurt School of Finance and Management.
Bhandari, L. C. (1988). Debt/equity ratio and expected common stock returns:
       Empirical evidence. The Journal of Finance, 43 (2), 507–528.
Brauneis, A., & Mestel, R. (2018). Price discovery of cryptocurrencies: Bitcoin and
       beyond. Economics Letters, 165, 58–61.
Carhart, M. M. (1997). On persistence in mutual fund performance. The Journal of
       Finance, 52 (1), 57–82.
BACKTESTING ALGOR. CRYPTOC. TRADING STRATEGIES                                       54
Chan, E. (2009). Quantitative trading: How to build your own algorithmic trading
       business. John Wiley & Sons, Inc.
Daniel, K., & Moskowitz, T. J. (2016). Momentum crashes. Journal of Financial
       Economics, 122 (2), 221–247.
Desai, H., & Jain, P. C. (1997). Long-run common stock returns following stock
       splits and reverse splits. The Journal of Business, 70 (3), 409–433.
Fama, E. F., & French, K. R. (1992). The cross-section of expected stock returns.
       The Journal of Finance, 47 (2), 427–465.
Fama, E. F., & French, K. R. (1993). Common risk factors in the returns on stocks
       and bonds. Journal of Financial Economics, 33 (1), 3–56.
Fama, E. F., & French, K. R. (1996). Multifactor explanations of asset pricing
       anomalies. The Journal of Finance, 51 (1), 55–84.
Fama, E. F., & French, K. R. (1998). Value versus growth: The international evi-
       dence. The Journal of Finance, 53 (6), 1975–1999.
Fama, E. F., & French, K. R. (2015). A five-factor asset pricing model. Journal of
       Financial Economics, 116 (1), 1–22.
Fama, E. F., & French, K. R. (2017). International tests of a five-factor asset pricing
       model. Journal of Financial Economics, 123 (3), 441–463.
Fama, E. F., & French, K. R. (2019). The value premium. Fama-Miller Working
       Paper, (20-01).
Feng, G., Giglio, S., & Xiu, D. (2017). Taming the factor zoo. Chicago Booth research
       paper, (17-04).
Frino, A., & Oetomo, T. (2005). Slippage in futures markets: Evidence from the
       Sydney Futures Exchange. Journal of Futures Markets: Futures, Options,
       and Other Derivative Products, 25 (12), 1129–1146.
Fürst, J. C. F. (2019). Crypto markets: A quantitative empirical analysis of Bitcoin
       and Ethereum (Bachelor’s Thesis).
Grobys, K., & Sapkota, N. (2019). Cryptocurrencies and momentum. Economics
       Letters, 180, 6–10.
Harvey, C. R., & Liu, Y. (2019). A census of the factor zoo. SSRN.
Hayes, A. (2015). A cost of production model for Bitcoin. SSRN.
BACKTESTING ALGOR. CRYPTOC. TRADING STRATEGIES                                      55
Hong, H., Lim, T., & Stein, J. C. (2000). Bad news travels slowly: Size, analyst cov-
       erage, and the profitability of momentum strategies. The Journal of Finance,
       55 (1), 265–295.
Hubrich, S. (2017). “Know When to Hodl ‘Em, Know When to Fodl ‘Em”: An
       investigation of factor based investing in the cryptocurrency space. SSRN.
Iansiti, M., & Lakhani, K. R. (2017). The truth about blockchain. Harvard Business
       Review.
Jegadeesh, N., & Titman, S. (1993). Returns to buying winners and selling losers:
       Implications for stock market efficiency. The Journal of Finance, 48 (1), 65–
       91.
Jegadeesh, N., & Titman, S. (2001). Profitability of momentum strategies: An eval-
       uation of alternative explanations. The Journal of Finance, 56 (2), 699–720.
Jensen, M. C. (1968). The performance of mutual funds in the period 1945–1964.
       The Journal of Finance, 23 (2), 389–416.
Jiang, F., Lee, J., Martin, X., & Zhou, G. (2019). Manager sentiment and stock
       returns. Journal of Financial Economics, 132 (1), 126–149.
Kakushadze, Z. (2018). Cryptoasset factor models. Algorithmic Finance, 7 (3–4),
       87–104.
Khuntia, S., & Pattanayak, J. (2018). Adaptive market hypothesis and evolving
       predictability of Bitcoin. Economics Letters, 167, 26–28.
Li, J., & Yi, G. (2019). Toward a factor structure in crypto asset returns. The
       Journal of Alternative Investments, 21 (4), 56–66.
Liechtenstein Blockchain Act [Full title: Unofficial translation of the government
       consultation report and the draft-law on transaction systems based on trust-
       worthy technologies (Blockchain Act)]. (2018).
Lintner, J. (1965a). Security prices, risk, and maximal gains from diversification.
       The Journal of Finance, 20 (4), 587–615.
Lintner, J. (1965b). The valuation of risk assets and the selection of risky invest-
       ments in stock portfolios and capital budgets. The Review of Economics and
       Statistics, 47 (1), 13–37.
Markowitz, H. (1952). Portfolio selection. The Journal of Finance, 7 (1), 77–91.
BACKTESTING ALGOR. CRYPTOC. TRADING STRATEGIES                                       56
Woo, W. (2017). Is Bitcoin in a bubble? Check the NVT ratio. Forbes. https://
      www.forbes.com/sites/wwoo/2017/09/29/is-bitcoin-in-a-bubble-check-the-
      nvt-ratio/
BACKTESTING ALGOR. CRYPTOC. TRADING STRATEGIES                                      58
12 Appendix
       The three diagrams in this section give additional insights into the outputs
that quantbacktest provides. As an overview, figure 8 shows the built-in equity
curve plot. It is created without the need for additional software and is automati-
cally saved as a PNG file. Figure 9 shows a visual analysis of the results from the
results/df_result_metrics.csv output and aims to portrait quantbacktest’s
ability to handle multiple periods in a single run of the backtesting. Figure 10
uses the same data source (results/df_result_metrics.csv), but uses two sets
of strategy hyperparameters that were analyzed in a single run of the backtest. The
strategy that was used in the displayed backtest will not be discussed in detail here
as the strategy is not the primary objective of showing these diagrams.
BACKTESTING ALGOR. CRYPTOC. TRADING STRATEGIES                                                           59
Figure 8 .    Standardized and benchmarked (portfolio value = 1 and Bitcoin price = 1 at t0 ) equity
curve. This visualization shows an automatically generated output from one multi-asset run (scenario) of
quantbacktest. While the visual appeal of this diagram is lower than those from Tableau or Google Data
Studio, it provides the user with a zero-effort way of analyzing results without using additional software.
The user can choose the benchmark. The visualization was created directly by quantbacktest using the
open-source matplotlib visualization library.
Figure 9 .    Heatmap for visual robustness comparisons (1/2): Time sensitivity. quantbacktest also
automatically creates a heatmap (using matplotlib), but I decided to show the results of a Tableau
visualization. The heatmap from matplotlib does not look nearly as good as this one. Own visualization
created using data from the results/df_result_metrics.csv file and Tableau.
BACKTESTING ALGOR. CRYPTOC. TRADING STRATEGIES                                                           60
Figure 10 .    Heatmap for visual robustness comparisons (2/2): Parameter sensitivity. The heatmap
shows the profitability (annualized return on investment) in percent (and highlighted by color) for differ-
ent thresholds of two arbitrary dimensions that could be used to execute cryptocurrency trading strate-
gies. Unlike the other heatmap (figure 9), this heatmap shows robustness for different parameter set-
tings, not time intervals. This visualization is not directly available from quantbacktest at the mo-
ment but needs to be created with another software tool. Own visualization created using data from the
results/df_result_metrics.csv file and the commercial Tableau visualization software.
     BACKTESTING ALGOR. CRYPTOC. TRADING STRATEGIES                                                                                   61
 1   from q u a n t b a c k t e s t import b a c k t e s t _ v i s u a l i z e r
 2
 3   # I m p o r t i n g modules from t h i s r e p o s i t o r y
 4   import s y s
 5
 6   # For managing d a t e s
 7   from d a t e t i m e import d a t e t i m e
 8
 9   # For a l l o w i n g f o r f l e x i b l e time d i f f e r e n c e s ( f r e q u e n c i e s )
10   from pandas . t s e r i e s . o f f s e t s import Timedelta
11
12
13   display_options = {
14           ’ boolean_plot_heatmap ’ : F a l s e ,
15           ’ boolean_test ’ : False ,                  # I f multi −a s s e t s t r a t e g y i s used , t h i s w i l l
           c a u s e s a m p l i n g o f t h e s i g n a l s t o s p e e d up t h e run f o r t e s t i n g d u r i n g
           development .
16           ’ warning_no_price_for_last_day ’ : F a l s e ,
17           ’ warning_no_price_during_execution ’ : F a l s e ,
18           ’ w a r n i n g _ n o _ p r i c e _ f o r _ i n t e r m e d i a t e _ v a l u a t i o n ’ : True ,
19           ’ warning_alternative_date ’ : False ,
20           ’ warning_calculate_daily_returns_alternative_date ’ : False ,
21           ’ warning_no_price_for_calculate_daily_returns ’ : False ,
22           ’ warning_buy_order_could_not_be_filled ’ : True ,
23           ’ w a r n i n g _ s e l l _ o r d e r _ c o u l d _ n o t _ b e _ f i l l e d ’ : True ,
24           ’ errors_on_benchmark_gap ’ : True ,
25           ’ boolean_plot_equity_curve ’ : False ,
26           ’ boolean_save_equity_curve_to_disk ’ : True ,
27           ’ string_results_directory ’ :                       ’ /home/ j a n s p o e r e r / code / j a n s p o e r e r /tmp/
           results ’
28   }
29
     BACKTESTING ALGOR. CRYPTOC. TRADING STRATEGIES                                                                                                62
30   general_settings = {
31          ’ rounding_decimal_places ’ : 4 ,
32          ’ rounding_decimal_places_for_security_quantities ’ : 0 ,
33   }
34
35   f i l e _ p a t h _ w i t h _ s i g n a l _ d a t a = ’ /home/ j a n s p o e r e r / code / j a n s p o e r e r / q u a n t b a c k t e s t
           / quantbacktest / a s s e t s / strategy_tables / t e s t . csv ’
36   excel_worksheet_name = ’ w e i g h t s ’
37
38   strategy_hyperparameters = {
39          ’ maximum_deviation_in_days ’ : 3 0 0 ,
40          ’ prices_table_id_column_name ’ :                         ’ token_itin ’ ,
41          ’ excel_worksheet_name ’ : excel_worksheet_name ,                                        # S e t t h i s t o None i f CSV
             i s used !
42         # For OpenMetrics : 9 . 8
43          ’ buy_parameter_space ’ : [ 9 . 8 ] ,                    # [ 1 1 , 2 0 ] # Times 1 0 ! W i l l be d i v i d e d by
             10.
44         # For OpenMetrics : 9 . 7
45          ’ sell_parameter_space ’ : [ 9 . 7 ] ,                     # [ 5 , 9 ] # Times 1 0 ! W i l l be d i v i d e d by
           10.
46          ’ maximum_relative_exposure_per_buy ’ : 0 . 3 4 ,
47          ’ f r e q u e n c y ’ : Timedelta ( days =1) ,
48          ’ moving_average_window_in_days ’ : 1 4 ,
49          ’ id ’ :   ’TP3B−248N−Q ’ ,
50          ’ b o o l e a n _ a l l o w _ p a r t i a l l y _ f i l l e d _ o r d e r s ’ : True ,
51   }
52
53   constraints = {
54          ’ maximum_individual_asset_exposure_all ’ : 1 . 0 ,                                      # Not y e t implemented
55          ’ m aximum_individual_a sset_exposure_individua l ’ : { } ,                                      # Not y e t
           implemented
56          ’ maximum_gross_exposure ’ : 1 . 0 ,                       # Already implemented
57          ’ boolean_allow_shortselling ’ : False ,                              # S h o r t s e l l i n g not y e t implemented
58          ’ minimum_cash ’ : 1 0 0 ,
59   }
60
61   comments = {
62          ’ display_options ’ : repr ( display_options ) ,
63          ’ strategy_hyperparameters ’ : repr ( strategy_hyperparameters )
64   }
65
66   backtest_visualizer (
67          f i l e _ p a t h _ w i t h _ p r i c e _ d a t a= ’ /home/ j a n s p o e r e r / code / j a n s p o e r e r /
           q u a n t b a c k t e s t / q u a n t b a c k t e s t / a s s e t s / raw_itsa_data /20190717
           _itsa_tokenbase_top600_wtd302_token_daily . c s v ’ ,
68         # ONLY LEAVE THIS LINE UNCOMMENTED IF YOU WANT TO USE ETH−ADDRESSES AS
           ASSET IDENTIFIERS !
69         # file_path_with_token_data =’ raw_itsa_data /20190717
           _itsa_tokenbase_top600_wtd301_token . c s v ’ ,                                 # Only f o r multi −a s s e t
      BACKTESTING ALGOR. CRYPTOC. TRADING STRATEGIES                                                                                            63
          strategies .
 70       name_of_foreign_key_in_price_data_table= ’ t o k e n _ i t i n ’ ,
 71       name_of_foreign_key_in_token_metadata_table= ’ t o k e n _ i t i n ’ ,
 72       # 1: execute_strategy_white_noise ()
 73       # 2 : Not used anymore , can be r e a s s i g n e d
 74       # 3 : e x e c u t e _ s t r a t e g y _ m u l t i _ a s s e t ( ) −> Uses s t r a t e g y t a b l e
 75       # 4 : execute_strategy_ma_crossover ( )
 76       i n t _ c h o s e n _ s t r a t e g y =4,
 77       d i c t _ c r y p t o _ o p t i o n s={
 78               ’ general ’ : {
 79                       ’ percentage_buying_fees_and_spread ’ : 0 . 0 0 5 ,                                    # 0.26% i s t h e
          t a k e r f e e f o r low−volume c l i e n t s a t kraken . com h t t p s : / /www. kraken . com/
          f e a t u r e s / f e e −s c h e d u l e
 80                       ’ percentage_selling_fees_and_spread ’ : 0.005 ,                                            # 0.26% i s t h e
          t a k e r f e e f o r low−volume c l i e n t s a t kraken . com h t t p s : / /www. kraken . com/
          f e a t u r e s / f e e −s c h e d u l e
 81                      # A d d i t i o n a l f e e s may apply f o r d e p o s i t i n g money .
 82                       ’ absolute_fee_buy_order ’ : 0 . 0 ,
 83                       ’ absolute_fee_sell_order ’ : 0.0 ,
 84               }
 85       },
 86       float_budget_in_usd =10 000 00. 00 ,
 87       f i l e _ p a t h _ w i t h _ s i g n a l _ d a t a=f i l e _ p a t h _ w i t h _ s i g n a l _ d a t a ,
 88       s t r a t e g y _ h y p e r p a r a m e t e r s=s t r a t e g y _ h y p e r p a r a m e t e r s ,
 89       margin_loan_rate =0.05 ,
 90       l i s t _ t i m e s _ o f _ s p l i t _ f o r _ r o b u s t n e s s _ t e s t =[
 91               [ datetime (2014 , 1 , 1) , datetime (2019 , 5 , 30) ]
 92       ],
 93       b e n c h m a r k _ d a t a _ s p e c i f i c a t i o n s={
 94               ’ name_of_column_with_benchmark_primary_key ’ :                                         ’ id ’ ,      # W i l l be i d
          a f t e r p r o c e s s i n g . Columns w i l l be renamed .
 95               ’ benchmark_key ’ :                ’TP3B−248N−Q ’ ,            # Ether : T22F−QJGB−N, B i t c o i n : TP3B
          −248N−Q
 96               ’ file_path_with_benchmark_data ’ :                             ’ /home/ j a n s p o e r e r / code / j a n s p o e r e r /
          q u a n t b a c k t e s t / q u a n t b a c k t e s t / a s s e t s / raw_itsa_data /20190717
          _itsa_tokenbase_top600_wtd302_token_daily . c s v ’ ,
 97               ’ risk_free_rate ’ : 0.02
 98       },
 99       d i s p l a y _ o p t i o n s=d i s p l a y _ o p t i o n s ,
100       c o n s t r a i n t s=c o n s t r a i n t s ,
101       g e n e r a l _ s e t t i n g s=g e n e r a l _ s e t t i n g s ,
102       comments=comments ,
103   )
      Code Snippet 34: This example triggers a backtest and is used to change the settings
      – tests/__init__.py.
     BACKTESTING ALGOR. CRYPTOC. TRADING STRATEGIES                                                                               64
 8   # For managing d a t e s
 9   from d a t e t i m e import d a t e t i m e
10
11   # For v e c t o r o p e r a t i o n s
12   from numpy import a r r a y
13
14   # For p l o t s ( e s p e c i a l l y t h e heatmap )
15   import m a t p l o t l i b . p y p l o t a s p l t
16
20
21   def backtest (
22                file_path_with_price_data ,
23                int_chosen_strategy ,
24                dict_crypto_options ,
25                float_budget_in_usd ,
26                margin_loan_rate ,
27                display_options ,
28                general_settings ,
29                constraints ,
30                start_time ,
31                comments ,
32                f i l e _ p a t h _ w i t h _ s i g n a l _ d a t a=None ,
33                file_path_with_token_data=None ,# Only f o r multi −a s s e t s t r a t e g i e s .
34                name_of_foreign_key_in_price_data_table=None ,
35                name_of_foreign_key_in_token_metadata_table=None ,
36                b o o l e a n _ a l l o w _ s h o r t i n g=F a l s e ,
37                l i s t _ t r a d i n g _ e x e c u t i o n _ d e l a y _ a f t e r _ s i g n a l _ i n _ h o u r s={
38                        ’ delay_before_buying ’ : 0 ,
39                        ’ delay_before_selling ’ : 0
40                },
41                minimum_expected_mispricing_trigger_in_percent={
42                        ’ mispricing_when_buying ’ : 0 . 0 ,
43                        ’ mispricing_when_selling ’ : 0.0
44                },
45                s t r a t e g y _ h y p e r p a r a m e t e r s=None ,
     BACKTESTING ALGOR. CRYPTOC. TRADING STRATEGIES                                                                                 65
46               sell_at_the_end=True ,
47               l i s t _ t i m e s _ o f _ s p l i t _ f o r _ r o b u s t n e s s _ t e s t=None ,
48               b e n c h m a r k _ d a t a _ s p e c i f i c a t i o n s={# B i t c o i n a s d e f a u l t benchmark
49                       ’ name_of_column_with_benchmark_primary_key ’ :                                         ’ id ’ ,
50                       ’ benchmark_key ’ :              ’TP3B−248N−Q ’ ,
51                       ’ file_path_with_benchmark_data ’ :                             ’ raw_itsa_data /20190717
       _itsa_tokenbase_top600_wtd302_token_daily . c s v ’
52               }
53      ):
54      " " " B a c k t e s t s a u s e r −d e f i n e d s t r a t e g y .
55
70      d f _ p r i c e s = prepare_data (
71               f i l e _ p a t h _ w i t h _ p r i c e _ d a t a=f i l e _ p a t h _ w i t h _ p r i c e _ d a t a ,
72               file_path_with_token_data=file_path_with_token_data ,
73               s t r a t e g y _ h y p e r p a r a m e t e r s=s t r a t e g y _ h y p e r p a r a m e t e r s ,
74               name_of_foreign_key_in_price_data_table=
       name_of_foreign_key_in_price_data_table ,
75               name_of_foreign_key_in_token_metadata_table=
       name_of_foreign_key_in_token_metadata_table ,
76      )
77
78      i f l i s t _ t i m e s _ o f _ s p l i t _ f o r _ r o b u s t n e s s _ t e s t i s None :
79               dfs_results = test_strategies (
80                      d f _ p r i c e s=d f _ p r i c e s ,
81                      i n t _ c h o s e n _ s t r a t e g y=i n t _ c h o s e n _ s t r a t e g y ,
82                      float_budget_in_usd=float_budget_in_usd ,
83                      f i l e _ p a t h _ w i t h _ s i g n a l _ d a t a=f i l e _ p a t h _ w i t h _ s i g n a l _ d a t a ,
84                      margin_loan_rate=margin_loan_rate ,
85                      b o o l e a n _ a l l o w _ s h o r t i n g=b o o l e a n _ a l l o w _ s h o r t i n g ,
      BACKTESTING ALGOR. CRYPTOC. TRADING STRATEGIES                                                                                             66
 86                l i s t _ t r a d i n g _ e x e c u t i o n _ d e l a y _ a f t e r _ s i g n a l _ i n _ h o u r s=
        list_trading_execution_delay_after_signal_in_hours ,
 87                d i c t _ c r y p t o _ o p t i o n s=d i c t _ c r y p t o _ o p t i o n s ,
 88                minimum_expected_mispricing_trigger_in_percent=
        minimum_expected_mispricing_trigger_in_percent ,
 89                s t r a t e g y _ h y p e r p a r a m e t e r s=s t r a t e g y _ h y p e r p a r a m e t e r s ,
 90                sell_at_the_end=sell_at_the_end ,
 91                b e n c h m a r k _ d a t a _ s p e c i f i c a t i o n s=b e n c h m a r k _ d a t a _ s p e c i f i c a t i o n s ,
 92                d i s p l a y _ o p t i o n s=d i s p l a y _ o p t i o n s ,
 93                g e n e r a l _ s e t t i n g s=g e n e r a l _ s e t t i n g s ,
 94                s t a r t _ t i m e=s t a r t _ t i m e ,
 95                comments=comments
 96            )
 97
 98      else :
 99            f o r sell_parameter in strategy_hyperparameters [ ’ sell_parameter_space
        ’ ]:
100                f o r buy_parameter i n s t r a t e g y _ h y p e r p a r a m e t e r s [ ’
        buy_parameter_space ’ ] :
101                       strategy_hyperparameters [ ’ sell_parameter ’ ] = sell_parameter
        /10
102                       s t r a t e g y _ h y p e r p a r a m e t e r s [ ’ buy_parameter ’ ] = buy_parameter /10
103
125                                           comments=comments
126                                   )
127
128                                   try :
129                                           d f s _ r e s u l t s [ 0 ] = d f s _ r e s u l t s [ 0 ] . append (
130                                                   dfs_intermediate_results [ 0 ]
131                                           )
132                                           d f s _ r e s u l t s [ 1 ] = d f s _ r e s u l t s [ 1 ] . append (
133                                                   dfs_intermediate_results [ 1 ]
134                                           )
135                                   except :
136                                           dfs_results = dfs_intermediate_results
137
138       save_dataframe_to_csv (
139            dfs_results [0] ,
140           string_name= ’ b a c k t e s t i n g _ r e s u l t _ m e t r i c s ’ ,
141            s t r i n g _ d i r e c t o r y=d i s p l a y _ o p t i o n s [ ’ s t r i n g _ r e s u l t s _ d i r e c t o r y ’ ] ,
142       )
143
144       comments = {
145           ∗∗{
146                    ’ file_path_with_price_data ’ : file_path_with_price_data ,
147                    ’ file_path_with_token_data ’ : file_path_with_token_data ,
148                    ’ name_of_foreign_key_in_price_dat_table ’ :
         name_of_foreign_key_in_price_data_table ,
149                    ’ name_of_foreign_key_in_token_metadata_table ’ :
         name_of_foreign_key_in_token_metadata_table ,
150                    ’ float_budget_in_usd ’ : float_budget_in_usd ,
151                    ’ margin_loan_rate ’ : margin_loan_rate ,
152                    ’ boolean_allow_shorting ’ : boolean_allow_shorting ,
153                    ’ list_trading_execution_delay_after_signal_in_hours ’ :
          list_trading_execution_delay_after_signal_in_hours ,
154                    ’ dict_crypto_options ’ : dict_crypto_options ,
155                    ’ minimum_expected_mispricing_trigger_in_percent ’ :
         minimum_expected_mispricing_trigger_in_percent ,
156                    ’ strategy_hyperparameters ’ : strategy_hyperparameters ,
157                    ’ sell_at_the_end ’ : sell_at_the_end ,
158           },
159           ∗∗ comments
160       }
161
162
163       return {
164            ’ df_performance ’ : d f s _ r e s u l t s [ 0 ] ,
165            ’ df_trading_journal ’ : dfs_results [ 1 ] ,
166            ’ comments ’ : comments
167       }
168
169   d e f plot_robustness_heatmap (
      BACKTESTING ALGOR. CRYPTOC. TRADING STRATEGIES                                                                  68
170             df_performance ,
171             display_options ,
172             boolean_save=True ,
173             boolean_show=F a l s e
174      ):
175      " " " P l o t s each row t h a t i s c o n t a i n e d i n t h e p e r f o r m a n c e t a b l e .
176
180      dict_heatmap = {
181              ’ Begin time o f t e s t e d i n t e r v a l ’ : [ ] ,
182              ’ Duration o f t h e t e s t e d i n t e r v a l ’ : [ ] ,
183              ’USD a n n u a l i z e d ROI ( from f i r s t t o l a s t t r a d e ) ’ :     []
184      }
185
213       )
214       figure . colorbar ( points )
215
228       i f boolean_save :
229              p l t . s a v e f i g ( s t r i n g _ d i r e c t o r y + ’ / robustness_heatmap_ ’ + s t r (
         n u m b e r _ o f _ r e s u l t _ f i l e s _ p l u s _ 1 ) + ’ . png ’ )
230
231       i f boolean_show :
232              p l t . show ( )
233
261      ):
262      " " " P r i n t s and p l o t s r e s u l t s from t h e p e r f o r m a n c e t a b l e . " " "
263
264      s t a r t _ t i m e = d a t e t i m e . now ( )
265      dict_backtesting = backtest (
266             f i l e _ p a t h _ w i t h _ p r i c e _ d a t a=f i l e _ p a t h _ w i t h _ p r i c e _ d a t a ,
267             file_path_with_token_data=file_path_with_token_data ,
268             name_of_foreign_key_in_price_data_table=
        name_of_foreign_key_in_price_data_table ,
269             name_of_foreign_key_in_token_metadata_table=
        name_of_foreign_key_in_token_metadata_table ,
270             float_budget_in_usd=float_budget_in_usd ,
271             margin_loan_rate=margin_loan_rate ,
272             b o o l e a n _ a l l o w _ s h o r t i n g=b o o l e a n _ a l l o w _ s h o r t i n g ,
273             l i s t _ t r a d i n g _ e x e c u t i o n _ d e l a y _ a f t e r _ s i g n a l _ i n _ h o u r s=
        list_trading_execution_delay_after_signal_in_hours ,
274             d i c t _ c r y p t o _ o p t i o n s=d i c t _ c r y p t o _ o p t i o n s ,
275             minimum_expected_mispricing_trigger_in_percent=
        minimum_expected_mispricing_trigger_in_percent ,
276             s t r a t e g y _ h y p e r p a r a m e t e r s=s t r a t e g y _ h y p e r p a r a m e t e r s ,
277             sell_at_the_end=sell_at_the_end ,
278             l i s t _ t i m e s _ o f _ s p l i t _ f o r _ r o b u s t n e s s _ t e s t=
        list_times_of_split_for_robustness_test ,
279             b e n c h m a r k _ d a t a _ s p e c i f i c a t i o n s=b e n c h m a r k _ d a t a _ s p e c i f i c a t i o n s ,
280             i n t _ c h o s e n _ s t r a t e g y=i n t _ c h o s e n _ s t r a t e g y ,
281             f i l e _ p a t h _ w i t h _ s i g n a l _ d a t a=f i l e _ p a t h _ w i t h _ s i g n a l _ d a t a ,
282             d i s p l a y _ o p t i o n s=d i s p l a y _ o p t i o n s ,
283             c o n s t r a i n t s=c o n s t r a i n t s ,
284             g e n e r a l _ s e t t i n g s=g e n e r a l _ s e t t i n g s ,
285             s t a r t _ t i m e=s t a r t _ t i m e ,
286             comments=comments
287      )
288      end_time = d a t e t i m e . now ( )
289      e l a p s e d _ t i m e = end_time − s t a r t _ t i m e
290      asset_name = l i s t ( d i c t _ c r y p t o _ o p t i o n s . k e y s ( ) ) [ 0 ]
291      p r i n t ( " \n " )
292      p r i n t ( " Execution s t a r t e d at : " + s t r ( start_time ) + " , f i n i s h e d at : " +
        s t r ( end_time ) + " , e l a p s e d time : " , s t r ( e l a p s e d _ t i m e . t o t a l _ s e c o n d s ( ) ) + " s
        ")
293      p r i n t ( " ∗∗∗∗      Performance o v e r v i e w −>" , asset_name , "<− ∗∗∗∗ " )
294      p r i n t ( " Key m e t r i c s " )
295      p r i n t ( " ∗USD a n n u a l i z e d ROI ( from f i r s t t o l a s t t r a d e ) : " , " { 0 : . 2 % } " .
        format ( d i c t _ b a c k t e s t i n g [ ’ df_performance ’ ] [ ’USD a n n u a l i z e d ROI ( from f i r s t
          to l a s t trade ) ’ ] . i l o c [ −1]) )
296      i f type ( d i c t _ b a c k t e s t i n g [ ’ df_performance ’ ] [ ’ C r y p t o c u r r e n c y a n n u a l i z e d
        ROI d e l t a ( from f i r s t t o l a s t t r a d e ) ’ ] . i l o c [ − 1 ] ) i s s t r :
297             p r i n t ( " ∗ C r y p t o c u r r e n c y a n n u a l i z e d ROI d e l t a ( from f i r s t t o l a s t
        t r a d e ) : " , d i c t _ b a c k t e s t i n g [ ’ df_performance ’ ] [ ’ C r y p t o c u r r e n c y a n n u a l i z e d
        ROI d e l t a ( from f i r s t t o l a s t t r a d e ) ’ ] . i l o c [ − 1 ] )
      BACKTESTING ALGOR. CRYPTOC. TRADING STRATEGIES                                                                                              71
298          else :
299                 p r i n t ( " ∗ C r y p t o c u r r e n c y a n n u a l i z e d ROI d e l t a ( from f i r s t t o l a s t
            t r a d e ) : " , " { 0 : . 2 % } " . format ( d i c t _ b a c k t e s t i n g [ ’ df_performance ’ ] [ ’
            C r y p t o c u r r e n c y a n n u a l i z e d ROI d e l t a ( from f i r s t t o l a s t t r a d e ) ’ ] . i l o c
            [ −1]) )
300          p r i n t ( " Number o f t r a d e s : " , l e n ( d i c t _ b a c k t e s t i n g [ ’ d f _ t r a d i n g _ j o u r n a l ’ ] ) )
301          p r i n t ( " Other m e t r i c s " )
302          p r i n t ( " ∗∗∗∗ Assumptions ∗∗∗∗ " )
303          p r i n t ( " ∗ Budget : " , d i c t _ b a c k t e s t i n g [ ’ comments ’ ] [ ’ float_budget_in_usd ’ ] )
304          p r i n t ( " ∗ Check arguments and parameter d e f a u l t s f o r a f u l l                              l i s t of
            assumptions . ∗ " )
305
306          plot_robustness_heatmap (
307                 d i c t _ b a c k t e s t i n g [ ’ df_performance ’ ] ,
308                 d i s p l a y _ o p t i o n s=d i s p l a y _ o p t i o n s ,
309                 boolean_show=d i s p l a y _ o p t i o n s [ ’ boolean_plot_heatmap ’ ]
310          )
311
  1   " " " This module c o n t a i n s f u n c t i o n s t h a t load , save , and m a n i p u l a t e data . " " "
  2
  7   # For managing t a b l e s p r o p e r l y
  8   from pandas import read_csv , merge
  9   import pandas
 10
11
 12   d e f load_data (
 13                 file_path_with_price_data ,
 14                 strategy_hyperparameters ,
 15                 file_path_with_token_data=None ,
 16                 name_of_foreign_key_in_price_data_table=None ,
 17                 name_of_foreign_key_in_token_metadata_table=None
 18          ):
 19          " " " Loads p r i c e data and metadata .
 20
58              d f _ p r i c e s = merge_data (
59                      d f _ p r i c e s=d f _ p r i c e s ,
60                      df_token_metadata=df_token_metadata ,
61                      name_of_foreign_key_in_price_data_table=
       name_of_foreign_key_in_price_data_table ,
62                      name_of_foreign_key_in_token_metadata_table=
       name_of_foreign_key_in_token_metadata_table
63              )
64
65      d f _ p r i c e s . rename (
66              columns={
67                      s t r a t e g y _ h y p e r p a r a m e t e r s [ ’ prices_table_id_column_name ’ ] :   ’ id ’
68              },
69              i n p l a c e=True
70      )
      BACKTESTING ALGOR. CRYPTOC. TRADING STRATEGIES                                                                                             73
71
 72       d f _ p r i c e s . s e t _ i n d e x ( [ ’ d a t e t i m e ’ , ’ i d ’ ] , i n p l a c e=True )
 73
 74       d f _ p r i c e s . s o r t _ i n d e x ( l e v e l =[ ’ d a t e t i m e ’ , ’ i d ’ ] , a s c e n d i n g =[1 , 1 ] , i n p l a c e
          =True )
 75
 76        i f not d f _ p r i c e s . i n d e x . i s _ l e x s o r t e d ( ) :
 77               r a i s e V a l u e E r r o r ( ’ d f _ p r i c e s i s not l e x s o r t e d . ’ )
 78
 79       return df_prices
 80
 81   d e f merge_data (
 82               df_prices ,
 83              df_token_metadata ,
 84               name_of_foreign_key_in_price_data_table ,
 85              name_of_foreign_key_in_token_metadata_table
 86       ):
 87        " " " E n r i c h e s p r i c e data with metadata about t h e t o k e n s . " " "
 88
 89       df_prices_enriched_with_metadata = merge (
 90               df_prices ,
 91              df_token_metadata ,
 92               l e f t _ o n=name_of_foreign_key_in_price_data_table ,
 93               r ig h t_ on=name_of_foreign_key_in_token_metadata_table
 94       )
 95
 96       r e t u r n df_prices_enriched_with_metadata
 97
 98   d e f prepare_data (
 99               file_path_with_price_data ,
100               strategy_hyperparameters ,
101               file_path_with_token_data=None ,
102               name_of_foreign_key_in_price_data_table=None ,
103              name_of_foreign_key_in_token_metadata_table=None
104       ):
105        " " " Loads p r i c e s and token data and j o i n s t h o s e i n t o a pandas DataFrame .
          """
106
107       d f _ p r i c e s = load_data (
108               f i l e _ p a t h _ w i t h _ p r i c e _ d a t a=f i l e _ p a t h _ w i t h _ p r i c e _ d a t a ,
109               file_path_with_token_data=file_path_with_token_data ,
110               s t r a t e g y _ h y p e r p a r a m e t e r s=s t r a t e g y _ h y p e r p a r a m e t e r s ,
111               name_of_foreign_key_in_price_data_table=
          name_of_foreign_key_in_price_data_table ,
112              name_of_foreign_key_in_token_metadata_table=
          name_of_foreign_key_in_token_metadata_table
113       )
114
116
117   d e f save_dataframe_to_csv (
118                 df_prices ,
119                 string_name ,
120                 string_directory ,
121          ):
122          " " " Saves a pandas Dataframe t o c s v w i t h o u t o v e r w r i t i n g e x i s t i n g              files .
            """
123          result_no = len (
124                 [ name f o r name i n l i s t d i r ( s t r i n g _ d i r e c t o r y ) i f path . i s f i l e (
125                        path . j o i n (
126                               string_directory ,
127                               name
128                        )
129                 )]
130          ) / 2
131
132
  3   # For deep−c o p i e d d i c t i o n a r i e s
  4   from copy import deepcopy
  5
  6   # For managing d a t e s
  7   from d a t e t i m e import datetime , t i m e d e l t a
  8
  9   # For managing t a b l e s p r o p e r l y
 10   from numpy import where
 11   from pandas import DataFrame , read_csv , r e a d _ e x c e l , date_range , Timedelta ,
            Timestamp
 12
 13   # For mathematical o p e r a t i o n s
 14   from math import i s n a n
 15
 16   # For n i c e −l o o k i n g p r o g r e s s b a r s
 17   from tqdm import tqdm
 18
 19   # For w h i t e n o i s e s t r a t e g y
 20   from random import c h o i c e
     BACKTESTING ALGOR. CRYPTOC. TRADING STRATEGIES                                                                75
21
27
28   def initialize_trading_journal () :
29        " " " I n i t i a l i z e s a pandas DataFrame t h a t s e r v e s a s a t r a d i n g j o u r n a l .
30
31        I n i t i a l i z a t i o n i s i m p o r t a n t f o r d e t e r m i n i n g t h e column o r d e r .
32        """
33        d f _ t r a d i n g _ j o u r n a l = DataFrame ( columns =[
34              ’ datetime ’ ,
35              ’ Cash ’ ,
36              ’ Cash b e f o r e ’ ,
37              ’ Asset ’ ,
38              ’ Buy o r s e l l ’ ,
39              ’ Number bought ’ ,
40              ’ P r i c e ( quote w i t h o u t any f e e s ) ’ ,
41              ’ Value bought ’ ,
42              ’ P o r t f o l i o value ’ ,
43              ’ Dict of a s s e t s in p o r t f o l i o ’ ,
44              ’ Absolute f e e s ( as absolute ) ’ ,
45              ’ Current e q u i t y margin ’ ,
46              ’ Exposure ( i n c u r r e n c y ) ’ ,
47              ’ Exposure ( number ) ’ ,
48              ’ Gross e x p o s u r e ’ ,
49              ’ I n t e r e s t paid ’ ,
50              ’ Money s p e n t ’ ,
51              ’ R e l a t i v e f e e s ( as absolute ) ’ ,
52              ’ R e l a t i v e f e e s ( as r e l a t i v e ) ’ ,
53              ’ S t r a t e g y ID ’ ,
54              ’ Total exposure ’ ,
55              ’ Total f e e s ( as ab solute ) ’ ,
56              ’ Total f e e s ( as r e l a t i v e ) ’
57        ])
58
59        return df_trading_journal
60
61   def execute_order (
62              boolean_buy ,
63              index ,
64              date ,
65              crypto_key ,
66              number_to_be_bought ,
67              strategy_id ,
68              df_prices ,
69              df_trading_journal ,
      BACKTESTING ALGOR. CRYPTOC. TRADING STRATEGIES                                                                                        76
 70            margin_loan_rate ,
 71            fees ,
 72            float_budget_in_usd ,
 73            price ,
 74            display_options ,
 75            constraints ,
 76            general_settings ,
 77            boolean_allow_partially_filled_orders
 78      ):
 79      " " " Executes a l l kinds of or der s .
 80
 87      order = {
 88            ’ d a t e t i m e ’ : date ,
 89            ’ S t r a t e g y ID ’ : s t r a t e g y _ i d ,
 90            ’ A s s e t ’ : crypto_key ,
 91            ’ Buy o r s e l l ’ : boolean_buy
 92      }
 93
103      d e f reduce_quantity_until_max_gross_exposure_is_met (
104                  number_to_be_bought ,
105                   df_prices ,
106                   dict_of_assets_in_portfolio ,
107                   time ,
108                   display_options ,
109                   constraints ,
110                   rounding_accuracy ,
111                   cash_value ,
112                   general_settings ,
113                   crypto_key
114            ):
      BACKTESTING ALGOR. CRYPTOC. TRADING STRATEGIES                                                                                         77
140            r e t u r n initial_number_to_be_bought
141
142      d e f quantity_that_can_be_bought_given_budget ( o r d e r ,
        dict_of_assets_in_portfolio , boolean_allow_partially_filled_orders ,
        number_to_be_bought , a v a i l a b l e _ f u n d s , g e n e r a l _ s e t t i n g s , d i s p l a y _ o p t i o n s ,
        date ) :
143            initial_number_to_be_bought = number_to_be_bought
144
150           # Todo : I n d i v i d u a l a s s e t c o n s t r a i n t s .
151           # c o n s t r a i n t s [ ’ maximum_individual_asset_exposure_all ’ ]
152            d i c t _ o f _ a s s e t s _ i n _ p o r t f o l i o = deepcopy ( d i c t _ o f _ a s s e t s _ i n _ p o r t f o l i o )
153            number_to_be_bought =
        reduce_quantity_until_max_gross_exposure_is_met (
154                   number_to_be_bought=number_to_be_bought ,
      BACKTESTING ALGOR. CRYPTOC. TRADING STRATEGIES                                                                                 78
155                  d f _ p r i c e s=d f _ p r i c e s ,
156                  d i c t _ o f _ a s s e t s _ i n _ p o r t f o l i o=d i c t _ o f _ a s s e t s _ i n _ p o r t f o l i o ,
157                  time=date ,
158                  d i s p l a y _ o p t i o n s=d i s p l a y _ o p t i o n s ,
159                  c o n s t r a i n t s=c o n s t r a i n t s ,
160                  rounding_accuracy=g e n e r a l _ s e t t i n g s [ ’ r o u n d i n g _ d e c i m a l _ p l a c e s ’ ] ,
161                  cash_value=o r d e r [ ’ Cash b e f o r e ’ ] − ( number_to_be_bought ∗ p r i c e )
        − f e e s [ ’ absolute_fee_buy_order ’ ] − round (
162                         f e e s [ ’ percentage_buying_fees_and_spread ’ ] ∗ p r i c e ∗
        number_to_be_bought ,
163                         g e n e r a l _ s e t t i n g s [ ’ rounding_decimal_places ’ ]
164                  ),
165                  g e n e r a l _ s e t t i n g s=g e n e r a l _ s e t t i n g s ,
166                  crypto_key=crypto_key
167            )
168
169            i f boolean_allow_partially_filled_orders :
170                  i f number_to_be_bought == 0 and initial_number_to_be_bought !=
        0:
171                          i f d i s p l a y _ o p t i o n s [ ’ warning_buy_order_could_not_be_filled ’ ] :
172                                 p r i n t ( f ’ Order e x e c u t i o n warning : Buy o r d e r f o r {
        initial_number_to_be_bought } u n i t s o f { crypto_key } c o u l d not be f i l l e d . ’ )
173                   e l i f number_to_be_bought < initial_number_to_be_bought :
174                          i f d i s p l a y _ o p t i o n s [ ’ warning_buy_order_could_not_be_filled ’ ] :
175                                 p r i n t ( f ’ Order e x e c u t i o n warning : Buy o r d e r f o r {
        initial_number_to_be_bought } u n i t s o f { crypto_key } c o u l d o n l y p a r t i a l l y
        be f i l l e d : {number_to_be_bought} u n i t s bought . ’ )
176                   e l i f number_to_be_bought > initial_number_to_be_bought :
177                         r a i s e V a l u e E r r o r ( f ’ I t i s not p o s s i b l e t h a t t h e s i g n a l
        q u a n t i t y { initial_number_to_be_bought } i s l o w e r than t h e f i n a l q u a n t i t y {
        number_to_be_bought} f o r { o r d e r [ " A s s e t " ] } . ’ )
178                  r e t u r n number_to_be_bought
179            else :
180                  i f initial_number_to_be_bought != number_to_be_bought :
181                         r a i s e I n p u t E r r o r ( ’ Quantity {number_to_be_bought} f o r {
        crypto_key } cannot be c o v e r e d with t h e g i v e n budget o r c o n s t r a i n t s .
        Maximum { max_possible_quantity } u n i t s can be bought . ’ )
182                  else :
183                         r e t u r n number_to_be_bought
184
188           # Todo : I n d i v i d u a l a s s e t c o n s t r a i n t s .
189           # c o n s t r a i n t s [ ’ maximum_individual_asset_exposure_all ’ ]
190
      BACKTESTING ALGOR. CRYPTOC. TRADING STRATEGIES                                                                                  79
200           d i c t _ o f _ a s s e t s _ i n _ p o r t f o l i o = deepcopy ( o r d e r [ ’ D i c t o f a s s e t s i n
        portfolio ’ ])
201           number_to_be_bought =
        reduce_quantity_until_max_gross_exposure_is_met (
202                   number_to_be_bought=number_to_be_bought ,
203                   d f _ p r i c e s=d f _ p r i c e s ,
204                   d i c t _ o f _ a s s e t s _ i n _ p o r t f o l i o=d i c t _ o f _ a s s e t s _ i n _ p o r t f o l i o ,
205                   time=date ,
206                   d i s p l a y _ o p t i o n s=d i s p l a y _ o p t i o n s ,
207                   c o n s t r a i n t s=c o n s t r a i n t s ,
208                   rounding_accuracy=g e n e r a l _ s e t t i n g s [ ’ r o u n d i n g _ d e c i m a l _ p l a c e s ’ ] ,
209                   cash_value=o r d e r [ ’ Cash b e f o r e ’ ] − ( number_to_be_bought ∗ p r i c e )
        − f e e s [ ’ absolute_fee_buy_order ’ ] − round (
210                          f e e s [ ’ percentage_buying_fees_and_spread ’ ] ∗ p r i c e ∗
        number_to_be_bought ,
211                          g e n e r a l _ s e t t i n g s [ ’ rounding_decimal_places ’ ]
212                   ),
213                   g e n e r a l _ s e t t i n g s=g e n e r a l _ s e t t i n g s ,
214                   crypto_key=crypto_key
215           )
216
217           i f boolean_allow_partially_filled_orders :
218                   i f number_to_be_bought == 0 and initial_number_to_be_bought !=
        0:
219                           i f d i s p l a y _ o p t i o n s [ ’ warning_buy_order_could_not_be_filled ’ ] :
220                                  p r i n t ( f ’ Order e x e c u t i o n warning : Buy o r d e r f o r {
        number_to_be_bought} u n i t s o f { crypto_key } c o u l d not be f i l l e d . ’ )
221                   e l i f number_to_be_bought > initial_number_to_be_bought :
222                           i f d i s p l a y _ o p t i o n s [ ’ warning_buy_order_could_not_be_filled ’ ] :
223                                  p r i n t ( f ’ Order e x e c u t i o n warning : Buy o r d e r f o r {
        number_to_be_bought} u n i t s o f { crypto_key } c o u l d not be f i l l e d . ’ )
224                   e l i f number_to_be_bought < initial_number_to_be_bought :
225                          r a i s e V a l u e E r r o r ( ’ I t i s not p o s s i b l e t h a t t h e s i g n a l
        q u a n t i t y { initial_number_to_be_bought } i s h i g h e r ( l e s s a s s e t s s o l d ) than
        t h e f i n a l q u a n t i t y {number_to_be_bought} f o r { o r d e r [ " A s s e t " ] } . ’ )
226                   r e t u r n number_to_be_bought
227           else :
228                   i f number_to_be_bought != initial_number_to_be_bought :
229                          r a i s e I n p u t E r r o r ( ’ Quantity {number_to_be_bought} f o r {
      BACKTESTING ALGOR. CRYPTOC. TRADING STRATEGIES                                                                                        80
233      i f boolean_buy :
234            number_to_be_bought = quantity_that_can_be_bought_given_budget (
235                   o r d e r=o r d e r ,
236                   d i c t _ o f _ a s s e t s _ i n _ p o r t f o l i o=o r d e r [ ’ D i c t o f a s s e t s i n p o r t f o l i o ’
        ],
237                   b o o l e a n _ a l l o w _ p a r t i a l l y _ f i l l e d _ o r d e r s=
        boolean_allow_partially_filled_orders ,
238                  number_to_be_bought=number_to_be_bought ,
239                   a v a i l a b l e _ f u n d s=a v a i l a b l e _ f u n d s ,
240                   g e n e r a l _ s e t t i n g s=g e n e r a l _ s e t t i n g s ,
241                   d i s p l a y _ o p t i o n s=d i s p l a y _ o p t i o n s ,
242                   d a t e=d a t e
243            )
244      else :
245            number_to_be_bought = q u a n t i t y _ t h a t _ c a n _ b e _ s o l d _ g i v e n _ p o r t f o l i o (
246                   o r d e r=o r d e r ,
247                   d i c t _ o f _ a s s e t s _ i n _ p o r t f o l i o=o r d e r [ ’ D i c t o f a s s e t s i n p o r t f o l i o ’
        ],
248                   b o o l e a n _ a l l o w _ p a r t i a l l y _ f i l l e d _ o r d e r s=
        boolean_allow_partially_filled_orders ,
249                  number_to_be_bought=number_to_be_bought ,
250                   d f _ t r a d i n g _ j o u r n a l=d f _ t r a d i n g _ j o u r n a l ,
251                   g e n e r a l _ s e t t i n g s=g e n e r a l _ s e t t i n g s ,
252                   d i s p l a y _ o p t i o n s=d i s p l a y _ o p t i o n s ,
253                   d a t e=d a t e
254            )
255
260      o r d e r [ ’ D i c t o f a s s e t s i n p o r t f o l i o ’ ] [ crypto_key ] = o r d e r [ ’ D i c t o f a s s e t s
          i n p o r t f o l i o ’ ] [ crypto_key ] + number_to_be_bought
261
262      i f number_to_be_bought != 0 :
263            i f boolean_buy :
264                   order [ ’ Absolute f e e s ( as absolute ) ’ ] = f e e s [ ’
        absolute_fee_buy_order ’ ]
265                   o r d e r [ ’ R e l a t i v e f e e s ( a s a b s o l u t e ) ’ ] = round (
266                          f e e s [ ’ percentage_buying_fees_and_spread ’ ] ∗ p r i c e ∗
        number_to_be_bought ,
267                          g e n e r a l _ s e t t i n g s [ ’ rounding_decimal_places ’ ]
268                   )
269                   order [ ’ R e l a t i v e f e e s ( as r e l a t i v e ) ’ ] = f e e s [ ’
      BACKTESTING ALGOR. CRYPTOC. TRADING STRATEGIES                                                                         81
        percentage_buying_fees_and_spread ’ ]
270                   o r d e r [ ’ T o t a l f e e s ( a s a b s o l u t e ) ’ ] = f e e s [ ’ absolute_fee_buy_order
        ’ ] + order [ ’ R e l a t i v e f e e s ( as absolute ) ’ ]
271                   o r d e r [ ’ T o t a l f e e s ( a s r e l a t i v e ) ’ ] = round (
272                          o r d e r [ ’ T o t a l f e e s ( a s a b s o l u t e ) ’ ] / ( number_to_be_bought ∗
        price ) ,
273                          g e n e r a l _ s e t t i n g s [ ’ rounding_decimal_places ’ ]
274                  )
275            else :
276                   order [ ’ Absolute f e e s ( as absolute ) ’ ] = f e e s [ ’
        absolute_fee_sell_order ’ ]
277                   o r d e r [ ’ R e l a t i v e f e e s ( a s a b s o l u t e ) ’ ] = round (
278                          f e e s [ ’ percentage_selling_fees_and_spread ’ ] ∗ p r i c e ∗
        number_to_be_bought ∗ ( −1) ,
279                          g e n e r a l _ s e t t i n g s [ ’ rounding_decimal_places ’ ]
280                  )
281                   order [ ’ R e l a t i v e f e e s ( as r e l a t i v e ) ’ ] = f e e s [ ’
        percentage_selling_fees_and_spread ’ ]
282                   order [ ’ Total f e e s ( as absolute ) ’ ] = f e e s [ ’
        absolute_fee_sell_order ’ ] + order [ ’ R e l a t i v e f e e s ( as absolute ) ’ ]
283                   o r d e r [ ’ T o t a l f e e s ( a s r e l a t i v e ) ’ ] = round (
284                          o r d e r [ ’ T o t a l f e e s ( a s a b s o l u t e ) ’ ] / ( number_to_be_bought ∗
        p r i c e ) ∗ ( −1) ,
285                          g e n e r a l _ s e t t i n g s [ ’ rounding_decimal_places ’ ]
286                  )
287      else :
288            order [ ’ Absolute f e e s ( as absolute ) ’ ] = 0
289            order [ ’ R e l a t i v e f e e s ( as absolute ) ’ ] = 0
290            order [ ’ R e l a t i v e f e e s ( as r e l a t i v e ) ’ ] = 0
291            order [ ’ Total f e e s ( as absolute ) ’ ] = 0
292            order [ ’ Total f e e s ( as r e l a t i v e ) ’ ] = 0
293
308                                       d f _ t r a d i n g _ j o u r n a l [ ’ P o r t f o l i o v a l u e ’ ] . i l o c [ −1]
309                                       ∗ margin_loan_rate
310                                )
311                         ) ∗∗ (
312                                (
313                                       d a y s _ s i n c e _ l a s t _ o r d e r . t o t a l _ s e c o n d s ( ) / 86400
314                                ) / 365
315                         ),
316                          g e n e r a l _ s e t t i n g s [ ’ rounding_decimal_places ’ ]
317                   ) # " / 8 6 4 0 0 " b e c a u s e one day has 86400 s e c o n d s
318
332      else :
333           # For i n i t i a l row
334            o r d e r [ ’ Number bought ’ ] = number_to_be_bought
335            o r d e r [ ’ Value bought ’ ] = p r i c e ∗ number_to_be_bought
336            order [ ’ I n t e r e s t paid ’ ] = 0.0
337            o r d e r [ ’ Money s p e n t ’ ] = number_to_be_bought ∗ (
338                   price
339            ) + (
340                  + order [ ’ Total f e e s ( as absolute ) ’ ]
341                  + order [ ’ I n t e r e s t paid ’ ]
342            ) # Todo But e v e r y t h i n g t h a t can be bought minus f e e s and o t h e r
        costs
343
         ’ Asset ’ ] ]
355
371       o r d e r [ ’ T o t a l e x p o s u r e ’ ] = round (
372              o r d e r [ ’ P o r t f o l i o v a l u e ’ ] − o r d e r [ ’ Cash ’ ] ,
373              g e n e r a l _ s e t t i n g s [ ’ rounding_decimal_places ’ ]
374       )
375
385       o r d e r [ ’ Gross e x p o s u r e ’ ] = c a l c u l a t e _ r e l a t i v e _ g r o s s _ e x p o s u r e (
386              d f _ p r i c e s=d f _ p r i c e s ,
387              d i c t _ o f _ a s s e t s _ i n _ p o r t f o l i o=o r d e r [ ’ D i c t o f a s s e t s i n p o r t f o l i o ’ ] ,
388              time=o r d e r [ ’ d a t e t i m e ’ ] ,
389              d i s p l a y _ o p t i o n s=d i s p l a y _ o p t i o n s ,
390              c o n s t r a i n t s=c o n s t r a i n t s ,
391              rounding_accuracy=g e n e r a l _ s e t t i n g s [ ’ r o u n d i n g _ d e c i m a l _ p l a c e s ’ ] ,
392              cash_value=o r d e r [ ’ Cash ’ ] ,
393              g e n e r a l _ s e t t i n g s=g e n e r a l _ s e t t i n g s
394       )
395
402      try :
403              try :
404                      d f _ s i g n a l s = read_csv (
405                           file_path_with_signal_data ,
406                           s e p= ’ , ’ ,
407                           p a r s e _ d a t e s=True ,
408                           i n f e r _ d a t e t i m e _ f o r m a t=True ,
409                           i n d e x _ c o l =[ ’ d a t e t i m e ’ , ’ i d ’ ]
410                      )
411              e x c e p t UnicodeDecodeError :
412                      df_signals = read_excel (
413                           file_path_with_signal_data ,
414                           s e p= ’ , ’ ,
415                           p a r s e _ d a t e s=True ,
416                           i n f e r _ d a t e t i m e _ f o r m a t=True ,
417                           i n d e x _ c o l =[ ’ d a t e t i m e ’ , ’ i d ’ ]
418                      )
419      except :
420              try :
421                      d f _ s i g n a l s = read_csv (
422                            ’ b a c k t e s t i n g / ’ + file_path_with_signal_data ,
423                           s e p= ’ , ’ ,
424                           p a r s e _ d a t e s=True ,
425                           i n f e r _ d a t e t i m e _ f o r m a t=True ,
426                           i n d e x _ c o l =[ ’ d a t e t i m e ’ , ’ i d ’ ]
427                      )
428              except :
429                      r a i s e ValueError ( " Please setup a s i g n a l t a b l e . A s i g n a l t a b l e
        n e e d s t h e f o l l o w i n g columns :          ’ d a t e t i m e ’ , ’ i d ’ ( h e x a d e c i m a l ERC20 toke n
        i d e n t i f i e r ) , ’ s i g n a l _ s t r e n g t h ’ ( numeric t h a t i s used t o i n f e r buy o r s e l l
          orders ) " )
430
        ’)
442      comments [ ’ Number o f u niqu e e t h IDs a f t e r ’ ] = l e n ( d f _ s i g n a l s [ ’ i d ’ ] . un i q u e
        () )
443
480      l a s t _ s i g n a l _ d a t e = d f _ s i g n a l s [ ’ d a t e t i m e ’ ] . i l o c [ −1]
481      p r i c e = None
482
483      list_of_assets_dropped_due_to_price_lag_at_last_day = [ ]
484
495             i f p r i c e i s None :
496                     d f _ s i g n a l s = d f _ s i g n a l s [ d f _ s i g n a l s . ID != a s s e t _ i n _ s i g n a l _ t a b l e ]
497                     i f d i s p l a y _ o p t i o n s [ ’ warning_no_price_for_last_day ’ ] :
498                            p r i n t ( f ’ { a s s e t _ i n _ s i g n a l _ t a b l e } was dropped b e c a u s e t h e r e i s
         no p r i c e f o r t h e l a s t day . { p r i c e } ’ )
499                    list_of_assets_dropped_due_to_price_lag_at_last_day . append (
500                            asset_in_signal_table
501                    )
502             e l i f isnan ( price ) :
503                     d f _ s i g n a l s = d f _ s i g n a l s [ d f _ s i g n a l s . ID != a s s e t _ i n _ s i g n a l _ t a b l e ]
504                     i f d i s p l a y _ o p t i o n s [ ’ warning_no_price_for_last_day ’ ] :
505                            p r i n t ( f ’ { a s s e t _ i n _ s i g n a l _ t a b l e } was dropped b e c a u s e t h e r e i s
          o n l y a NaN p r i c e f o r t h e l a s t day . { p r i c e } ’ )
506                    list_of_assets_dropped_due_to_price_lag_at_last_day . append (
507                            asset_in_signal_table
508                    )
509             e l i f p r i c e == 0 :
510                     d f _ s i g n a l s = d f _ s i g n a l s [ d f _ s i g n a l s . ID != a s s e t _ i n _ s i g n a l _ t a b l e ]
511                     i f d i s p l a y _ o p t i o n s [ ’ warning_no_price_for_last_day ’ ] :
      BACKTESTING ALGOR. CRYPTOC. TRADING STRATEGIES                                                                              87
524      p r i n t ( f ’ Number o f s i g n a l s a f t e r d r o p p i n g u n a v a i l a b l e p r i c e s : { l e n (
        df_signals )} ’ )
525      comments [ ’ Number o f s i g n a l s a f t e r d r o p p i n g u n a v a i l a b l e p r i c e s ’ ] = l e n (
        df_signals )
526
         list_of_assets_dropped_due_to_price_lag_at_last_day )
548
549       r e t u r n d f _ s i g n a l s , id_column_name
550
608      df_trading_journal = i n i t i a l i z e _ t r a d i n g _ j o u r n a l ()
609
613             crypto_key = i n d e x [ 1 ]
614
        minimum_cash ’ ] :
630                                         allow_buy_orders = F a l s e
631
632                                 else :
633                                         allow_buy_orders = True
634
635                           else :
636                                 allow_buy_orders = True
637
638                           i f allow_buy_orders :
639                                 try :
640                                         float_budget_in_usd = d f _ t r a d i n g _ j o u r n a l [ ’ Cash ’ ] .
        i l o c [ −1]
641                                 except :
642                                         pass
643
652                           else :
653                                 number_to_be_bought = None
654
662                           i f not (
663                                 (
664                                         boolean_buy == F a l s e
665                                 ) and (
666                                         find_dataframe_value_with_keywords (
667                                                df_trading_journal ,
668                                                search_term_1=crypto_key ,
669                                                search_column_name_1= ’ A s s e t ’
670                                         )
671                                 ) i s None
672                           ):
673                                 number_to_be_bought = ( −1) ∗ (
      BACKTESTING ALGOR. CRYPTOC. TRADING STRATEGIES                                                                           91
674                                      find_dataframe_value_with_keywords (
675                                               df_trading_journal ,
676                                               search_term_1=crypto_key ,
677                                               search_column_name_1= ’ A s s e t ’ ,
678                                               output_column_name= ’ Exposure ( number ) ’ ,
679                                               f i r s t _ l a s t _ o r _ a l l _ e l e m e n t s= ’ L a s t ’
680                                      )
681                              )
682
683                       else :
684                              number_to_be_bought = None
685
686                else :
687                       r a i s e V a l u e E r r o r ( ’ Ambiguous s i g n a l : ’ , row [ ’ s i g n a l _ t y p e ’ ] )
688
689               try :
690                       amount = d f _ t r a d i n g _ j o u r n a l [ ’ Cash ’ ] . i l o c [ −1]
691               except :
692                       amount = float_budget_in_usd
693
718                             c o n s t r a i n t s=c o n s t r a i n t s ,
719                             g e n e r a l _ s e t t i n g s=g e n e r a l _ s e t t i n g s ,
720                             b o o l e a n _ a l l o w _ p a r t i a l l y _ f i l l e d _ o r d e r s=
         strategy_hyperparameters [ ’ boolean_allow_partially_filled_orders ’ ]
721                     )
722
723                     d f _ t r a d i n g _ j o u r n a l = d f _ t r a d i n g _ j o u r n a l . append (
724                             dataseries_trading_journal ,
725                             i g n o r e _ i n d e x=True
726                     )
727
728       comments [ ’ c o n s t r a i n t s ’ ] = c o n s t r a i n t s
729       comments [ ’ g e n e r a l _ s e t t i n g s ’ ] = g e n e r a l _ s e t t i n g s
730
731       dict_return = {
732              ’ df_trading_journal ’ : df_trading_journal ,
733              ’ S t r a t e g y ID ’ :     ’3 ’ ,
734              ’ Strategy label ’ :               ’X ’ ,
735              ’ strategy_hyperparameters ’ : strategy_hyperparameters ,
736              ’ comments ’ : comments
737       }
738
739       save_dataframe_to_csv (
740              df_trading_journal ,
741              ’ trading_journal ’ ,
742              s t r i n g _ d i r e c t o r y=d i s p l a y _ o p t i o n s [ ’ s t r i n g _ r e s u l t s _ d i r e c t o r y ’ ] ,
743       )
744
        gross
766      return of zero .
767      """
768      df_prices = df_prices . l oc [ ( s l i c e ( strategy_hyperparameters [ ’ start_time ’ ] ,
          s t r a t e g y _ h y p e r p a r a m e t e r s [ ’ end_time ’ ] ) , s t r a t e g y _ h y p e r p a r a m e t e r s [ ’ i d ’ ] ) ,
        :]
769
770      df_trading_journal = i n i t i a l i z e _ t r a d i n g _ j o u r n a l ()
771
772      u s d _ s a f e t y _ b u f f e r = 100
773
774     # S i n g l e −a s s e t o n l y
775      crypto_key = s t r a t e g y _ h y p e r p a r a m e t e r s [ ’ i d ’ ]
776
802             else :
803                     # Determine t h e number o f a s s e t s t o be bought o r s o l d
804                     try :
805                             number_to_be_bought = ( −1) ∗ d f _ t r a d i n g _ j o u r n a l [
806                                    ’ Exposure ( number ) ’
807                             ] . i l o c [ −1]
808                     except :
809                             number_to_be_bought = 0
810
      BACKTESTING ALGOR. CRYPTOC. TRADING STRATEGIES                                                                                                94
811                 boolean_buy = F a l s e
812
837          d f _ t r a d i n g _ j o u r n a l = d f _ t r a d i n g _ j o u r n a l . append (
838                 dataseries_trading_journal ,
839                 i g n o r e _ i n d e x=True
840          )
841
842      dict_return = {
843          ’ df_trading_journal ’ : df_trading_journal ,
844          ’ S t r a t e g y ID ’ : i n t _ c h o s e n _ s t r a t e g y ,
845          ’ Strategy label ’ :               ’ White N o i s e ’ ,
846          ’ strategy_hyperparameters ’ : strategy_hyperparameters ,
847          ’ comments ’ :         ’’
848      }
849
850      save_dataframe_to_csv (
851          df_trading_journal ,
852          ’ trading_journal ’ ,
853          s t r i n g _ d i r e c t o r y=d i s p l a y _ o p t i o n s [ ’ s t r i n g _ r e s u l t s _ d i r e c t o r y ’ ] ,
854      )
      BACKTESTING ALGOR. CRYPTOC. TRADING STRATEGIES                                                                                         95
855
888       df_trading_journal = i n i t i a l i z e _ t r a d i n g _ j o u r n a l ()
889
890       p r i c e = None
891
        through s i g n a l s ’ , u n i t =’ s i g n a l ’ ) :
898
960                  e l i f boolean_buy == F a l s e :
961                       i f len ( df_trading_journal ) > 0:
962                               number_to_be_bought = ( −1) ∗ round (
963                                      df_trading_journal . i l o c [ −1][ ’ Dict of a s s e t s in
        p o r t f o l i o ’ ] [ strategy_hyperparameters [ ’ id ’ ] ] ,
964                                      general_settings [ ’
        rounding_decimal_places_for_security_quantities ’ ]
965                               )
966                       else :
967                               number_to_be_bought = 0
968                  else :
969                       number_to_be_bought = 0
970
971                  i f number_to_be_bought != 0 :
972                       dataseries_trading_journal = execute_order (
973                               boolean_buy=boolean_buy ,
974                               i n d e x=current_time ,
975                               d a t e=current_time ,
976                               s t r a t e g y _ i d=i n t _ c h o s e n _ s t r a t e g y ,
977                               crypto_key=s t r a t e g y _ h y p e r p a r a m e t e r s [ ’ i d ’ ] ,
978                               number_to_be_bought=number_to_be_bought ,
979                               d f _ p r i c e s=d f _ p r i c e s ,
980                               d f _ t r a d i n g _ j o u r n a l=d f _ t r a d i n g _ j o u r n a l ,
981                               margin_loan_rate=margin_loan_rate ,
       BACKTESTING ALGOR. CRYPTOC. TRADING STRATEGIES                                                                                     98
 982                                   float_budget_in_usd=float_budget_in_usd ,
 983                                   p r i c e=p r i c e ,
 984                                   f e e s ={
 985                                           ’ absolute_fee_buy_order ’ : d i c t _ c r y p t o _ o p t i o n s [ ’
          g e n e r a l ’ ] [ ’ absolute_fee_buy_order ’ ] ,
 986                                           ’ absolute_fee_sell_order ’ : dict_crypto_options [ ’
          general ’ ] [ ’ absolute_fee_sell_order ’ ] ,
 987                                           ’ percentage_buying_fees_and_spread ’ :
          d i c t _ c r y p t o _ o p t i o n s [ ’ g e n e r a l ’ ] [ ’ percentage_buying_fees_and_spread ’ ] ,
 988                                           ’ percentage_selling_fees_and_spread ’ :
          dict_crypto_options [ ’ general ’ ] [ ’ percentage_selling_fees_and_spread ’ ]
 989                                   },
 990                                   d i s p l a y _ o p t i o n s=d i s p l a y _ o p t i o n s ,
 991                                   c o n s t r a i n t s=c o n s t r a i n t s ,
 992                                   g e n e r a l _ s e t t i n g s=g e n e r a l _ s e t t i n g s ,
 993                                   b o o l e a n _ a l l o w _ p a r t i a l l y _ f i l l e d _ o r d e r s=
          strategy_hyperparameters [ ’ boolean_allow_partially_filled_orders ’ ]
 994                           )
 995
 996                           d f _ t r a d i n g _ j o u r n a l = d f _ t r a d i n g _ j o u r n a l . append (
 997                                   dataseries_trading_journal ,
 998                                   i g n o r e _ i n d e x=True
 999                           )
1000
1001       dict_return = {
1002            ’ df_trading_journal ’ : df_trading_journal ,
1003            ’ S t r a t e g y ID ’ :     ’4 ’ ,
1004            ’ Strategy label ’ :                ’ Moving Average C r o s s o v e r ’ ,
1005            ’ strategy_hyperparameters ’ : strategy_hyperparameters ,
1006            ’ comments ’ : comments
1007       }
1008
1009       save_dataframe_to_csv (
1010            df_trading_journal ,
1011            ’ trading_journal ’ ,
1012            s t r i n g _ d i r e c t o r y=d i s p l a y _ o p t i o n s [ ’ s t r i n g _ r e s u l t s _ d i r e c t o r y ’ ] ,
1013       )
1014
1026             strategy_hyperparameters ,
1027             sell_at_the_end ,
1028             benchmark_data_specifications ,
1029             display_options ,
1030             constraints ,
1031             general_settings ,
1032             start_time ,
1033             f i l e _ p a t h _ w i t h _ s i g n a l _ d a t a=None ,
1034            comments={}
1035      ):
1036      " " " C a l l s u s e r −d e f i n e d s t r a t e g y .
1037
1042      i f i n t _ c h o s e n _ s t r a t e g y == 1 :
1043             dict_execution_results = execute_strategy_white_noise (
1044                    d f _ p r i c e s=d f _ p r i c e s ,
1045                    float_budget_in_usd=float_budget_in_usd ,
1046                    margin_loan_rate=margin_loan_rate ,
1047                    b o o l e a n _ a l l o w _ s h o r t i n g=b o o l e a n _ a l l o w _ s h o r t i n g ,
1048                    l i s t _ t r a d i n g _ e x e c u t i o n _ d e l a y _ a f t e r _ s i g n a l _ i n _ h o u r s=
         list_trading_execution_delay_after_signal_in_hours ,
1049                    i n t _ c h o s e n _ s t r a t e g y=i n t _ c h o s e n _ s t r a t e g y ,
1050                    d i c t _ c r y p t o _ o p t i o n s=d i c t _ c r y p t o _ o p t i o n s ,
1051                    minimum_expected_mispricing_trigger_in_percent=
         minimum_expected_mispricing_trigger_in_percent ,
1052                    s t r a t e g y _ h y p e r p a r a m e t e r s=s t r a t e g y _ h y p e r p a r a m e t e r s ,
1053                    d i s p l a y _ o p t i o n s=d i s p l a y _ o p t i o n s ,
1054                    c o n s t r a i n t s=c o n s t r a i n t s ,
1055                    g e n e r a l _ s e t t i n g s=g e n e r a l _ s e t t i n g s ,
1056                    sell_at_the_end=sell_at_the_end ,
1057                    comments=comments
1058             )
1059      e l i f i n t _ c h o s e n _ s t r a t e g y == 2 :
1060             r a i s e NotImplementedError ( ’ S t r a t e g y 2 i s not implemented . ’ )
1061      e l i f i n t _ c h o s e n _ s t r a t e g y == 3 :
1062             dict_execution_results = execute_strategy_multi_asset (
1063                    d f _ p r i c e s=d f _ p r i c e s ,
1064                    f i l e _ p a t h _ w i t h _ s i g n a l _ d a t a=f i l e _ p a t h _ w i t h _ s i g n a l _ d a t a ,
1065                    float_budget_in_usd=float_budget_in_usd ,
1066                    i n t _ c h o s e n _ s t r a t e g y=i n t _ c h o s e n _ s t r a t e g y ,
1067                    margin_loan_rate=margin_loan_rate ,
1068                    b o o l e a n _ a l l o w _ s h o r t i n g=b o o l e a n _ a l l o w _ s h o r t i n g ,
1069                    l i s t _ t r a d i n g _ e x e c u t i o n _ d e l a y _ a f t e r _ s i g n a l _ i n _ h o u r s=
         list_trading_execution_delay_after_signal_in_hours ,
1070                    d i c t _ c r y p t o _ o p t i o n s=d i c t _ c r y p t o _ o p t i o n s ,
1071                    minimum_expected_mispricing_trigger_in_percent=
       BACKTESTING ALGOR. CRYPTOC. TRADING STRATEGIES                                                                                     100
         minimum_expected_mispricing_trigger_in_percent ,
1072                   s t r a t e g y _ h y p e r p a r a m e t e r s=s t r a t e g y _ h y p e r p a r a m e t e r s ,
1073                   sell_at_the_end=sell_at_the_end ,
1074                   d i s p l a y _ o p t i o n s=d i s p l a y _ o p t i o n s ,
1075                   c o n s t r a i n t s=c o n s t r a i n t s ,
1076                   g e n e r a l _ s e t t i n g s=g e n e r a l _ s e t t i n g s ,
1077                   comments=comments
1078            )
1079      e l i f i n t _ c h o s e n _ s t r a t e g y == 4 :
1080            d i c t _ e x e c u t i o n _ r e s u l t s = execute_strategy_ma_crossover (
1081                   d f _ p r i c e s=d f _ p r i c e s ,
1082                   f i l e _ p a t h _ w i t h _ s i g n a l _ d a t a=f i l e _ p a t h _ w i t h _ s i g n a l _ d a t a ,
1083                   i n t _ c h o s e n _ s t r a t e g y=i n t _ c h o s e n _ s t r a t e g y ,
1084                   float_budget_in_usd=float_budget_in_usd ,
1085                   margin_loan_rate=margin_loan_rate ,
1086                   b o o l e a n _ a l l o w _ s h o r t i n g=b o o l e a n _ a l l o w _ s h o r t i n g ,
1087                   l i s t _ t r a d i n g _ e x e c u t i o n _ d e l a y _ a f t e r _ s i g n a l _ i n _ h o u r s=
         list_trading_execution_delay_after_signal_in_hours ,
1088                   d i c t _ c r y p t o _ o p t i o n s=d i c t _ c r y p t o _ o p t i o n s ,
1089                   minimum_expected_mispricing_trigger_in_percent=
         minimum_expected_mispricing_trigger_in_percent ,
1090                   s t r a t e g y _ h y p e r p a r a m e t e r s=s t r a t e g y _ h y p e r p a r a m e t e r s ,
1091                   sell_at_the_end=sell_at_the_end ,
1092                   d i s p l a y _ o p t i o n s=d i s p l a y _ o p t i o n s ,
1093                   c o n s t r a i n t s=c o n s t r a i n t s ,
1094                   g e n e r a l _ s e t t i n g s=g e n e r a l _ s e t t i n g s ,
1095                   comments=comments
1096            )
1097
1098      save_dataframe_to_csv (
1099            dict_execution_results [ ’ df_trading_journal ’ ] ,
1100            string_name= ’ t r a d i n g _ j o u r n a l ’ ,
1101            s t r i n g _ d i r e c t o r y=d i s p l a y _ o p t i o n s [ ’ s t r i n g _ r e s u l t s _ d i r e c t o r y ’ ] ,
1102      )
1103
1104      df_performance = e v a l u a t e _ p e r f o r m a n c e (
1105            d f _ p r i c e s=d f _ p r i c e s ,
1106            d i c t _ e x e c u t i o n _ r e s u l t s=d i c t _ e x e c u t i o n _ r e s u l t s ,
1107            float_budget_in_usd=float_budget_in_usd ,
1108            b e n c h m a r k _ d a t a _ s p e c i f i c a t i o n s=b e n c h m a r k _ d a t a _ s p e c i f i c a t i o n s ,
1109            s t r a t e g y _ h y p e r p a r a m e t e r s=s t r a t e g y _ h y p e r p a r a m e t e r s ,
1110            d i s p l a y _ o p t i o n s=d i s p l a y _ o p t i o n s ,
1111            c o n s t r a i n t s=c o n s t r a i n t s ,
1112            g e n e r a l _ s e t t i n g s=g e n e r a l _ s e t t i n g s ,
1113            s t a r t _ t i m e=s t a r t _ t i m e ,
1114      )
1115
       BACKTESTING ALGOR. CRYPTOC. TRADING STRATEGIES                                                                                    101
1116 r e t u r n [ df_performance , d i c t _ e x e c u t i o n _ r e s u l t s [ ’ d f _ t r a d i n g _ j o u r n a l ’ ] ]
   5   # For mathematical o p e r a t i o n s
   6   from math import s q r t , exp , l o g
   7
   8   # For t e n s o r c a l c u l a t i o n s
   9   from numpy import cov
  10
  11   # For managing t a b l e s p r o p e r l y
  12   from pandas import to_datetime
  13
17
  18   def calculate_alpha (
  19                 annualized_portfolio_return ,
  20                 risk_free_rate ,
  21                beta_exposure ,
  22                annualized_market_return
  23          ):
  24          " " " C a l c u l a t e s t h e J e n s e n ’ s a l p h a o f a p o r t f o l i o a g a i n s t a benchmark . " " "
  25          market_risk_premium = annualized_market_return − r i s k _ f r e e _ r a t e
  26
  27          alpha = annualized_portfolio_return − (
  28                 r i s k _ f r e e _ r a t e + beta_exposure ∗ market_risk_premium
  29          )
  30
  31          return alpha
  32
  33   d e f c a l c u l a t e _ b e t a ( d f _ d a i l y _ r e t u r n s , df_daily_benchmark_returns ) :
  34          " " " C a l c u l a t e s t h e b e t a o f a p o r t f o l i o a g a i n s t a benchmark . " " "
  35          portfolio_returns = df_daily_returns [ ’ relative_portfolio_return ’ ] .
             to_numpy ( )
  36          benchmark_returns = df_daily_benchmark_returns [ ’
             b e n c h m a r k _ r e l a t i v e _ r e t u r n ’ ] . to_numpy ( )
  37
  38          c o v a r i a n c e _ m a t r i x = cov ( p o r t f o l i o _ r e t u r n s , benchmark_returns )
  39
  40          benchmark_portfolio_covariance = covariance_matrix [ 0 ] [ 1 ]
  41          benchmark_variance = c o v a r i a n c e _ m a t r i x [ 1 ] [ 1 ]
  42
     BACKTESTING ALGOR. CRYPTOC. TRADING STRATEGIES                                                                                   102
43       r e t u r n b e n c h m a r k _ p o r t f o l i o _ c o v a r i a n c e / benchmark_variance
44
45   d e f calculate_maximum_drawdown (
46              df_daily_returns ,
47              column_with_portfolio_values
48       ):
49       " " " C a l c u l a t e s t h e maximum_drawdown o f a p o r t f o l i o u s i n g p o r t f o l i o _ v a l u e
        from d f _ d a i l y _ r e t u r n s .
50
51       Outputs t h e maximum_drawdown r a t i o .
52       """
53       dict_peak_tracking = {
54              ’ first_peak ’ : {
55                      ’ peak ’ : {
56                             column_with_portfolio_values : df_daily_returns [
        column_with_portfolio_values ] . i l o c [ 0 ] ,
57                             ’ datetime ’ : df_daily_returns . index . values [ 0 ]
58                     },
59                      ’ trough ’ : {
60                             column_with_portfolio_values : df_daily_returns [
        column_with_portfolio_values ] . i l o c [ 0 ] ,
61                             ’ datetime ’ : df_daily_returns . index . values [ 0 ]
62                     }
63              },
64              ’ second_peak ’ : {
65                      ’ peak ’ : {
66                             column_with_portfolio_values : df_daily_returns [
        column_with_portfolio_values ] . i l o c [ 0 ] ,
67                             ’ datetime ’ : df_daily_returns . index . values [ 0 ]
68                     },
69                      ’ trough ’ : {
70                             column_with_portfolio_values : df_daily_returns [
        column_with_portfolio_values ] . i l o c [ 0 ] ,
71                             ’ datetime ’ : df_daily_returns . index . values [ 0 ]
72                     }
73              }
74       }
75
76       f o r datetime , row i n d f _ d a i l y _ r e t u r n s . i t e r r o w s ( ) :
77              i f row [ c o l u m n _ w i t h _ p o r t f o l i o _ v a l u e s ] > d i c t _ p e a k _ t r a c k i n g [ ’
        second_peak ’ ] [ ’ peak ’ ] [ c o l u m n _ w i t h _ p o r t f o l i o _ v a l u e s ] :
78                      d i c t _ p e a k _ t r a c k i n g [ ’ second_peak ’ ] [ ’ peak ’ ] [
        c o l u m n _ w i t h _ p o r t f o l i o _ v a l u e s ] = row [ c o l u m n _ w i t h _ p o r t f o l i o _ v a l u e s ]
79                      d i c t _ p e a k _ t r a c k i n g [ ’ second_peak ’ ] [ ’ peak ’ ] [ ’ d a t e t i m e ’ ] = d a t e t i m e
80                      d i c t _ p e a k _ t r a c k i n g [ ’ second_peak ’ ] [ ’ t r o u g h ’ ] [
        c o l u m n _ w i t h _ p o r t f o l i o _ v a l u e s ] = row [ c o l u m n _ w i t h _ p o r t f o l i o _ v a l u e s ]
81                      d i c t _ p e a k _ t r a c k i n g [ ’ second_peak ’ ] [ ’ t r o u g h ’ ] [ ’ d a t e t i m e ’ ] =
        datetime
82              i f row [ c o l u m n _ w i t h _ p o r t f o l i o _ v a l u e s ] < d i c t _ p e a k _ t r a c k i n g [ ’
      BACKTESTING ALGOR. CRYPTOC. TRADING STRATEGIES                                                                                     103
         second_peak ’ ] [ ’ t r o u g h ’ ] [ c o l u m n _ w i t h _ p o r t f o l i o _ v a l u e s ] :
 83                      d i c t _ p e a k _ t r a c k i n g [ ’ second_peak ’ ] [ ’ t r o u g h ’ ] [
         c o l u m n _ w i t h _ p o r t f o l i o _ v a l u e s ] = row [ c o l u m n _ w i t h _ p o r t f o l i o _ v a l u e s ]
 84                      d i c t _ p e a k _ t r a c k i n g [ ’ second_peak ’ ] [ ’ t r o u g h ’ ] [ ’ d a t e t i m e ’ ] =
         datetime
 85              i f row [ c o l u m n _ w i t h _ p o r t f o l i o _ v a l u e s ] < d i c t _ p e a k _ t r a c k i n g [ ’
         f i r s t _ p e a k ’ ] [ ’ trough ’ ] [ column_with_portfolio_values ] :
 86                      dict_peak_tracking [ ’ f i r s t _ p e a k ’ ] [ ’ trough ’ ] [
         c o l u m n _ w i t h _ p o r t f o l i o _ v a l u e s ] = row [ c o l u m n _ w i t h _ p o r t f o l i o _ v a l u e s ]
 87                      dict_peak_tracking [ ’ f i r s t _ p e a k ’ ] [ ’ trough ’ ] [ ’ datetime ’ ] =
         datetime
 88
 89              drawdown_second = (
 90                      d i c t _ p e a k _ t r a c k i n g [ ’ second_peak ’ ] [ ’ peak ’ ] [
         c o l u m n _ w i t h _ p o r t f o l i o _ v a l u e s ] − d i c t _ p e a k _ t r a c k i n g [ ’ second_peak ’ ] [ ’ t r o u g h
          ’ ] [ column_with_portfolio_values ]
 91              ) / d i c t _ p e a k _ t r a c k i n g [ ’ second_peak ’ ] [ ’ peak ’ ] [
         column_with_portfolio_values ]
 92              drawdown_first = (
 93                      d i c t _ p e a k _ t r a c k i n g [ ’ f i r s t _ p e a k ’ ] [ ’ peak ’ ] [
         column_with_portfolio_values ] − dict_peak_tracking [ ’ f i r s t _ p e a k ’ ] [ ’ trough ’
          ] [ column_with_portfolio_values ]
 94              ) / d i c t _ p e a k _ t r a c k i n g [ ’ f i r s t _ p e a k ’ ] [ ’ peak ’ ] [
         column_with_portfolio_values ]
 95              i f drawdown_second > drawdown_first :
 96                      dict_peak_tracking [ ’ first_peak ’ ] = dict_peak_tracking [ ’
         second_peak ’ ]
 97
 98       maximum_drawdown_duration = d i c t _ p e a k _ t r a c k i n g [ ’ f i r s t _ p e a k ’ ] [ ’ t r o u g h ’ ] [ ’
         d a t e t i m e ’ ] − d i c t _ p e a k _ t r a c k i n g [ ’ f i r s t _ p e a k ’ ] [ ’ peak ’ ] [ ’ d a t e t i m e ’ ]
 99       peak_date = d i c t _ p e a k _ t r a c k i n g [ ’ f i r s t _ p e a k ’ ] [ ’ peak ’ ] [ ’ d a t e t i m e ’ ]
100       trough_date = d i c t _ p e a k _ t r a c k i n g [ ’ f i r s t _ p e a k ’ ] [ ’ t r o u g h ’ ] [ ’ d a t e t i m e ’ ]
101
        or
118      r e t u r n s i n r e l a t i o n t o a benchmark . I t can a l s o u s e d i f f e r e n t s t a r t and
        end
119      p o i n t s f o r c a l c u l a t i n g t h e time frame , i . e . , t h e d u r a t i o n between t h e
        first
120      and t h e l a s t t r a d e OR t h e d u r a t i o n between t h e f i r s t and t h e l a s t data
        point
121      o f t h e p r i c e data .
122      """
123      start_time = df_trading_journal [
124            df_trading_journal_time_column_name
125      ]. iloc [0]
126      end_time = d f _ t r a d i n g _ j o u r n a l [
127            df_trading_journal_time_column_name
128      ] . i l o c [ −1]
129
130      p o r t f o l i o _ r o i = calculate_roi_math (
131            end_value=d f _ t r a d i n g _ j o u r n a l [
132                    df_trading_journal_price_column_name
133      ] . i l o c [ −1] ,
134            s t a r t _ v a l u e=float_budget_in_usd ,
135            end_time=end_time ,
136            s t a r t _ t i m e=s t a r t _ t i m e
137      )
138
163              benchmark_roi = c a l c u l a t e _ r o i _ m a t h (
164                      end_value=end_value_benchmark ,
165                      s t a r t _ v a l u e=begin_value_benchmark ,
166                      end_time=end_time ,
167                      s t a r t _ t i m e=s t a r t _ t i m e
168              )
169
172       return {
173              ’ portfolio_roi ’ : portfolio_roi ,
174              ’ benchmark_roi ’ : benchmark_roi ,
175              ’ roi_delta_compared_to_benchmark ’ : roi_delta_compared_to_benchmark
176       }
177
187       r o i = exp (
188              log (
189                      unadjusted_return_factor
190              ) ∗ 365 / ( t i m e _ d u r a t i o n )
191       ) − 1
192
204       i f r i s k _ f r e e _ r a t e i s None :
205              sharpe_ratio = portfolio_roi_usd / v o l a t i l i t y
206       else :
207              sharpe_ratio = ( portfolio_roi_usd − risk_free_rate ) / v o l a t i l i t y
208       return sharpe_ratio
209
210
212          r e t u r n round (
213                sum ( d f _ t r a d i n g _ j o u r n a l [ " T o t a l f e e s ( a s a b s o l u t e ) " ] ) ,
214                 2
215          )
216
217   d e f c a l c u l a t e _ v o l a t i l i t y ( d f _ d a i l y _ r e t u r n s , time_adjustment_in_days=None ) :
218          " " " C a l c u l a t e s the v o l a t i l i t y of a p o r t f o l i o using d a i l y p o r t f o l i o
            returns .
219
230          i f round ( v o l a t i l i t y , 4 ) == 0 :
231                 r a i s e V a l u e E r r o r ( ’ V o l a t i l i t y cannot be z e r o o r c l o s e t o z e r o . P l e a s e
              check i f d a i l y r e t u r n s a r e c o r r e c t l y c a l c u l a t e d and i f any t r a d e s were
            made . \n{ d f _ d a i l y _ r e t u r n s } ’ )
232
233 return v o l a t i l i t y
  5   # For deep−c o p i e d d i c t i o n a r i e s
  6   from copy import deepcopy
  7
  8   # For managing t a b l e s p r o p e r l y
  9   from pandas import DataFrame
 10
 11   # For p l o t t i n g r e t u r n c u r v e
 12   import m a t p l o t l i b . p y p l o t a s p l t
 13   from m a t p l o t l i b . d a t e s import MonthLocator
 14   from m a t p l o t l i b . d a t e s import DateFormatter
 15   import m a t p l o t l i b . t i c k e r a s t i c k e r c a l c u l a t e _ a l p h a
 16
 17   # For managing d a t e s
     BACKTESTING ALGOR. CRYPTOC. TRADING STRATEGIES                                                                                         107
32
33   def calculate_returns_single (
34                  previous_trading_journal_row ,
35                  current_trading_journal_row ,
36                  df_prices ,
37                  strategy_hyperparameters ,
38                  display_options ,
39                  general_settings ,
40                  constraints
41          ):
42          " " " Returns a l i s t o f d i c t s with p o r t f o l i o r e t u r n data f o r a g i v e n
           frequency .
43
44          Dict f i e l d s :      ’ timestamp ’ ( d a t e t i m e . d a t e t i m e ) , ’ p o r t f o l i o _ v a l u e ’ ( f l o a t ) ,
45                                  ’ return ’ ( f l o a t ) , ’ relative_return ’ ( f l o a t ) ,
46                                  ’ d i c t _ o f _ a s s e t s _ i n _ p o r t f o l i o ’ ( d i c t with i t i n s a s k e y s and
47                                  i n t e g e r with t h e number o f p i e c e s h e l d o f t h i s a s s e t a s a s
48                                  values )
49
56         Trades                                               Frequency i n c r e m e n t s
57
58         −No t r a d e −                                      Minute 1
59         Trade 1                                              Minute 2
      BACKTESTING ALGOR. CRYPTOC. TRADING STRATEGIES                                                                                       108
 60     −No t r a d e −                                        Minute 3
 61     −No t r a d e −                                        Minute 4
 62      Trade 2                                               Minute 5
 63
 74      d a t e t i m e . d a t e t i m e o b j e c t s assume ’ t h e r e a r e e x a c t l y 3600∗24 s e c o n d s i n
        every
 75      day ’ . h t t p s : / / d o c s . python . o r g /2/ l i b r a r y / d a t e t i m e . html#datetime −o b j e c t s
 76      """
 77      list_of_dict_returns = [ ]
 78
 82      c a s h = p r e v i o u s _ t r a d i n g _ j o u r n a l _ r o w [ ’ Cash ’ ]
 83
 86      while (
 87                    datetime_counter + strategy_hyperparameters [ ’ frequency ’ ]
 88      ) <= (
 89                    c u r r e n t _ t r a d i n g _ j o u r n a l _ r o w [ ’ d a t e t i m e ’ ] . to_pydatetime ( )
 90      ):
 91             returns = { ’ dict_of_assets_in_portfolio ’ :
        copy_dict_of_assets_in_portfolio }
 92
 93            # The c o u n t e r i s p u r p o s e f u l l y i n c r e m e n t e d a t t h e b e g i n n i n g
 94             d a t e t i m e _ c o u n t e r += s t r a t e g y _ h y p e r p a r a m e t e r s [ ’ f r e q u e n c y ’ ]
 95
 96             r e t u r n s [ ’ datetime ’ ] = datetime_counter
 97
 98             previous_portfolio_value = calculate_portfolio_value (
 99                    d f _ p r i c e s=d f _ p r i c e s ,
100                    d i c t _ o f _ a s s e t s _ i n _ p o r t f o l i o=d i c t _ o f _ a s s e t s _ i n _ p o r t f o l i o ,
      BACKTESTING ALGOR. CRYPTOC. TRADING STRATEGIES                                                                                        109
101                    time=d a t e t i m e _ c o u n t e r − s t r a t e g y _ h y p e r p a r a m e t e r s [ ’ f r e q u e n c y ’ ] ,
102                    cash_value=cash ,
103                    d i s p l a y _ o p t i o n s=d i s p l a y _ o p t i o n s ,
104                    c o n s t r a i n t s=c o n s t r a i n t s ,
105                    rounding_accuracy=g e n e r a l _ s e t t i n g s [ ’ r o u n d i n g _ d e c i m a l _ p l a c e s ’ ]
106            )
107
118            r e t u r n s [ ’ d i c t _ o f _ a s s e t s _ i n _ p o r t f o l i o ’ ] = deepcopy (
        copy_dict_of_assets_in_portfolio )
119
120            try :
121                    r e t u r n s [ ’ p o r t f o l i o _ r e t u r n ’ ] = round (
122                            returns [ ’ portfolio_value ’ ] − previous_portfolio_value ,
123                            g e n e r a l _ s e t t i n g s [ ’ rounding_decimal_places ’ ]
124                    )
125                    try :
126                            r e t u r n s [ ’ r e l a t i v e _ p o r t f o l i o _ r e t u r n ’ ] = round (
127                                   returns [ ’ portfolio_return ’ ] / previous_portfolio_value ,
128                                   g e n e r a l _ s e t t i n g s [ ’ rounding_decimal_places ’ ]
129                            )
130                    except ZeroDivisionError :
131                            raise ZeroDivisionError (
132                                   """
133                                           This i s a r a t h e r u n l i k e l y s c e n a r i o , t h e r e f o r e an
        error
134                                           was r a i s e d . The p r e v i o u s p o r t f o l i o v a l u e i s z e r o .
        This
135                                           cannot be .
136                                   """ +
137                                   ’\ nportfolio_value :                   ’ + str (
138                                           returns [ ’ portfolio_value ’ ]
139                                   ) +
140                                   ’\ nportfolio_return :                    ’ + str (
141                                           returns [ ’ portfolio_return ’ ]
142                                   )
143                            )
144            except IndexError :
145                    r e t u r n s [ ’ p o r t f o l i o _ r e t u r n ’ ] = round (
146                            r e t u r n s [ ’ p o r t f o l i o _ v a l u e ’ ] − d f _ t r a d i n g _ j o u r n a l [ ’ Cash b e f o r e
      BACKTESTING ALGOR. CRYPTOC. TRADING STRATEGIES                                                                                  110
         ’ ]. iloc [0] ,
147                            g e n e r a l _ s e t t i n g s [ ’ rounding_decimal_places ’ ]
148                     )
149                     r e t u r n s [ ’ r e l a t i v e _ p o r t f o l i o _ r e t u r n ’ ] = round (
150                            r e t u r n s [ ’ p o r t f o l i o _ r e t u r n ’ ] / d f _ t r a d i n g _ j o u r n a l [ ’ Cash
         before ’ ] . i l o c [ 0 ] ,
151                            g e n e r a l _ s e t t i n g s [ ’ rounding_decimal_places ’ ]
152                     )
153
154              r e t u r n s = deepcopy ( r e t u r n s )
155
156              l i s t _ o f _ d i c t _ r e t u r n s . append ( r e t u r n s )
157
177       d f _ r e t u r n s = DataFrame (
178             columns =[
179                     ’ datetime ’ ,
180                     ’ portfolio_value ’ ,
181                     ’ portfolio_return ’ ,
182                     ’ relative_portfolio_return ’ ,
183                     ’ dict_of_assets_in_portfolio ’
184              ]
185       )
186
198       current_date = f i r s t _ d a t e
199       f o r index , row i n d f _ t r a d i n g _ j o u r n a l . i t e r r o w s ( ) :
200              i f index > 0 :
201                     previous_trading_journal_row = df_trading_journal . l o c [ index − 1 ]
202                     current_trading_journal_row = df_trading_journal . l o c [ index ]
203
239      d f _ d a i l y _ r e t u r n s = DataFrame (
240            columns =[ ’ benchmark_datetime ’ , ’
        benchmark_dict_of_assets_in_portfolio ’ , ’ benchmark_portfolio_value ’ , ’
        benchmark_return ’ , ’ b e n c h m a r k _ r e l a t i v e _ r e t u r n ’ ]
241      )
242
254      i f i n i t i a l _ b u d g e t i s None :
255            r a i s e V a l u e E r r o r ( f ’ i n i t i a l _ b u d g e t s h o u l d not be None . Benchmark ID :
        { benchmark_id } . No p r i c e found f o r { f i r s t _ d a t e } i n \n{ d f _ p r i c e s } . ’ )
256
257      current_date = f i r s t _ d a t e
258      f o r d a y s _ e l a p s e d i n r a n g e ( 1 , ( ( l a s t _ d a t e − f i r s t _ d a t e ) . days + 1 ) ) :
259            c u r r e n t _ d a t e = f i r s t _ d a t e + t i m e d e l t a ( days=d a y s _ e l a p s e d )
260            # p r e v i o u s _ d a t e = c u r r e n t _ d a t e − t i m e d e l t a ( days =1)
261
262            dict_of_assets_in_portfolio = {
263                    s t r ( benchmark_id ) : 1 . 0
264            }
265
276            try :
277                    p o r t f o l i o _ r e t u r n = round (
278                            portfolio_value − df_daily_returns [ ’
        benchmark_portfolio_value ’ ] . i l o c [ −1] ,
279                            2
280                    )
281                    try :
282                            r e l a t i v e _ p o r t f o l i o _ r e t u r n = round (
283                                   portfolio_return / df_daily_returns [ ’
      BACKTESTING ALGOR. CRYPTOC. TRADING STRATEGIES                                                                          113
         benchmark_portfolio_value ’ ] . i l o c [ −1] ,
284                                  2
285                           )
286                    except ZeroDivisionError :
287                            raise ZeroDivisionError (
288                                   """
289                                         This i s a r a t h e r u n l i k e l y s c e n a r i o , t h e r e f o r e an
         error
290                                         was r a i s e d . The p r e v i o u s p o r t f o l i o v a l u e i s z e r o .
         This
291                                         cannot be .
292                                   """ +
293                                   ’ \ nprevious p o r t f o l i o value :           ’ + str (
294                                         df_daily_returns [ ’ benchmark_portfolio_value ’ ] . i l o c
         [ −1]
295                                  ) +
296                                   ’\ nportfolio_value :              ’ + str (
297                                          portfolio_value
298                                  ) +
299                                   ’\ nportfolio_return :               ’ + str (
300                                          portfolio_return
301                                  )
302                           )
303              except IndexError :
304                     p o r t f o l i o _ r e t u r n = round (
305                            portfolio_value − initial_budget ,
306                           2
307                    )
308                     r e l a t i v e _ p o r t f o l i o _ r e t u r n = round (
309                            portfolio_return / initial_budget ,
310                           2
311                    )
312
313              c o p y _ d i c t _ o f _ a s s e t s _ i n _ p o r t f o l i o = deepcopy (
         dict_of_assets_in_portfolio )
314
329      I n i t i a l i z a t i o n i s i m p o r t a n t f o r d e t e r m i n i n g t h e column o r d e r .
330      """
331      df_performance = DataFrame ( columns =[
332            ’ S t r a t e g y metadata −−−> ’ ,
333            ’ S t r a t e g y ID ’ ,
334            ’ Strategy label ’ ,
335            ’ Trading i n f o −−−> ’ ,
336            ’ Begin time o f t e s t e d i n t e r v a l ’ ,
337            ’ End time o f t e s t e d i n t e r v a l ’ ,
338            ’ Duration o f t h e t e s t e d i n t e r v a l ’ ,
339            ’ Duration o f t h e t e s t e d i n t e r v a l ( i n days ) ’ ,
340            ’ Average c a s h ’ ,
341            ’ Average t i c k e t s i z e ’ ,
342            ’ Number o f t r a d e s ’ ,
343            ’ Number o f un iqu e a s s e t s t r a d e d ’ ,
344            ’ Total t r a n s a c t i o n c o s t ’ ,
345            ’ Return m e t r i c s −−−> ’ ,
346            ’USD a n n u a l i z e d ROI ( from f i r s t t o l a s t t r a d e ) ’ ,
347            ’ C r y p t o c u r r e n c y a n n u a l i z e d ROI d e l t a ( from f i r s t t o l a s t t r a d e ) ’ ,
348            ’ Ending benchmark v a l u e ( f i r s t t o l a s t t r a d e ) ’ ,
349            ’ I n i t i a l budget ’ ,
350            ’ Ending p o r t f o l i o v a l u e ’ ,
351            ’ Risk m e t r i c s −−−> ’ ,
352            ’ H ol d in g p e r i o d v o l a t i l i t y ’ ,
353            ’ Annual v o l a t i l i t y ’ ,
354            ’ Monthly v o l a t i l i t y ’ ,
355            ’ Weekly v o l a t i l i t y ’ ,
356            ’ Beta r e l a t i v e t o benchmark ’ ,
357            ’Maximum drawdown ’ ,
358            ’Maximum drawdown d u r a t i o n ’ ,
359            ’Maximum drawdown peak d a t e ’ ,
360            ’Maximum drawdown t r o u g h d a t e ’ ,
361            ’ Other m e t r i c s −−−> ’ ,
362            ’ Alpha ’ ,
363            ’ Sharpe r a t i o ( h o l d i n g p e r i o d ) ’ ,
364            ’ Sharpe r a t i o ( y e a r l y ) ’ ,
365            ’ B e g i n n i n g benchmark v a l u e ( f i r s t t o l a s t t r a d e ) ’ ,
366            ’ Other i n f o −−−> ’ ,
367            ’ S t a r t time ’ ,
368            ’ End time ’ ,
369            ’ Parameter 1 ’ ,
370            ’ Parameter 2 ’ ,
371            ’ Comments ’ ,
372            ’ Benchmark r e t u r n m e t r i c s −−−> ’ ,
373            ’ Benchmark USD a n n u a l i z e d ROI ( from f i r s t t o l a s t t r a d e ) ’ ,
374            ’ Benchmark c r y p t o c u r r e n c y a n n u a l i z e d ROI d e l t a ( from f i r s t t o l a s t
        trade ) ’ ,
      BACKTESTING ALGOR. CRYPTOC. TRADING STRATEGIES                                                                        115
393       r e t u r n df_performance
394
411       df_performance = i n i t i a l i z e _ p e r f o r m a n c e _ o v e r v i e w ( )
412
415       try :
416               df_daily_returns = calculate_returns_batch (
417                     d f _ t r a d i n g _ j o u r n a l=d f _ t r a d i n g _ j o u r n a l ,
418                     d f _ p r i c e s=d f _ p r i c e s ,
419                     s t r a t e g y _ h y p e r p a r a m e t e r s=s t r a t e g y _ h y p e r p a r a m e t e r s ,
420                     d i s p l a y _ o p t i o n s=d i s p l a y _ o p t i o n s ,
421                     g e n e r a l _ s e t t i n g s=g e n e r a l _ s e t t i n g s ,
422                     c o n s t r a i n t s=c o n s t r a i n t s
423               )
      BACKTESTING ALGOR. CRYPTOC. TRADING STRATEGIES                                                                       116
436      else :
437            begin_time_of_tested_interval = df_trading_journal [
438                    ’ datetime ’
439            ]. iloc [0]
440            end_time_of_tested_interval = dict_execution_results [
441                    ’ df_trading_journal ’
442            ] [ ’ d a t e t i m e ’ ] . i l o c [ −1]
443
450            df_daily_benchmark_returns = c a l c u l a t e _ d a i l y _ r e t u r n s _ f r o m _ b e n c h m a r k (
451                    f i r s t _ d a t e=b e g i n _ t i m e _ o f _ t e s t e d _ i n t e r v a l ,
452                    l a s t _ d a t e=e n d _ t i m e _ o f _ t e s t e d _ i n t e r v a l ,
453                    d f _ p r i c e s=df_benchmark ,
454                    benchmark_id=b e n c h m a r k _ d a t a _ s p e c i f i c a t i o n s [ ’ benchmark_key ’ ] ,
455                    d i s p l a y _ o p t i o n s=d i s p l a y _ o p t i o n s ,
456                    g e n e r a l _ s e t t i n g s=g e n e r a l _ s e t t i n g s ,
457                    s t r a t e g y _ h y p e r p a r a m e t e r s=s t r a t e g y _ h y p e r p a r a m e t e r s ,
458                    c o n s t r a i n t s=c o n s t r a i n t s
459            )
460
470            save_dataframe_to_csv (
471                    df_daily_returns ,
472                    string_name= ’ d f _ d a i l y _ r e t u r n s ’ ,
473                    s t r i n g _ d i r e c t o r y=d i s p l a y _ o p t i o n s [ ’ s t r i n g _ r e s u l t s _ d i r e c t o r y ’ ] ,
474            )
475
502            # Average t i c k e t s i z e
503            d i c t _ e x e c u t i o n _ r e s u l t s [ ’ d f _ t r a d i n g _ j o u r n a l ’ ] [ ’ Value bought a b s o l u t e ’
        ] = d i c t _ e x e c u t i o n _ r e s u l t s [ ’ d f _ t r a d i n g _ j o u r n a l ’ ] [ ’ Value bought ’ ] . abs ( )
504            a v e r a g e _ t i c k e t _ s i z e _ a b s o l u t e _ v a l u e = round (
505                    d i c t _ e x e c u t i o n _ r e s u l t s [ ’ d f _ t r a d i n g _ j o u r n a l ’ ] [ ’ Value bought
        a b s o l u t e ’ ] . mean ( ) ,
506                    2
507            )
      BACKTESTING ALGOR. CRYPTOC. TRADING STRATEGIES                                                                                    118
508
545                                 d e s i r e d _ i n d e x =( e n d _ t i m e _ o f _ t e s t e d _ i n t e r v a l ,
        b e n c h m a r k _ d a t a _ s p e c i f i c a t i o n s [ ’ benchmark_key ’ ] ) ,
546                                 b o o l e a n _ a l l o w _ o l d e r _ p r i c e s=F a l s e ,
547                                 b o o l e a n _ a l l o w _ n e w e r _ p r i c e s=F a l s e ,
548                                 boolean_warnings=True ,
549                                 b o o l e a n _ e r r o r s=d i s p l a y _ o p t i o n s [ ’ errors_on_benchmark_gap ’
        ]
550                          ),
551                          ’ I n i t i a l budget ’ : float_budget_in_usd ,
552                          ’ Ending p o r t f o l i o v a l u e ’ : d f _ t r a d i n g _ j o u r n a l [ ’ P o r t f o l i o
        value ’ ] . i l o c [ −1] ,
553                          ’ H ol d in g p e r i o d v o l a t i l i t y ’ : c a l c u l a t e _ v o l a t i l i t y (
        df_daily_returns , len ( df_daily_returns ) ) ,
554                          ’ Annual v o l a t i l i t y ’ : c a l c u l a t e _ v o l a t i l i t y ( d f _ d a i l y _ r e t u r n s ,
        365) ,
555                          ’ Monthly v o l a t i l i t y ’ : c a l c u l a t e _ v o l a t i l i t y ( d f _ d a i l y _ r e t u r n s ,
            30) ,
556                          ’ Weekly v o l a t i l i t y ’ : c a l c u l a t e _ v o l a t i l i t y ( d f _ d a i l y _ r e t u r n s ,
        7) ,
557                          ’ Alpha ’ : c a l c u l a t e _ a l p h a (
558                                 a n n u a l i z e d _ p o r t f o l i o _ r e t u r n=d i c t _ r o i _ r e s u l t s [ ’
        portfolio_roi ’ ] ,
559                                 r i s k _ f r e e _ r a t e=b e n c h m a r k _ d a t a _ s p e c i f i c a t i o n s [
560                                         ’ risk_free_rate ’
561                                 ],
562                                 beta_exposure=beta ,
563                                 annualized_market_return=d i c t _ r o i _ r e s u l t s [ ’ benchmark_roi
        ’]
564                          ),
565                          ’ Sharpe r a t i o ( h o l d i n g p e r i o d ) ’ : c a l c u l a t e _ s h a r p e _ r a t i o (
566                                 df_daily_returns ,
567                                 p o r t f o l i o _ r o i _ u s d=d i c t _ r o i _ r e s u l t s [ ’ p o r t f o l i o _ r o i ’ ] ,
568                                 r i s k _ f r e e _ r a t e=b e n c h m a r k _ d a t a _ s p e c i f i c a t i o n s [
569                                         ’ risk_free_rate ’
570                                 ]
571                          ),
572                          ’ Sharpe r a t i o ( y e a r l y ) ’ : c a l c u l a t e _ s h a r p e _ r a t i o (
573                                 df_daily_returns ,
574                                 p o r t f o l i o _ r o i _ u s d=d i c t _ r o i _ r e s u l t s [ ’ p o r t f o l i o _ r o i ’ ] ,
575                                 r i s k _ f r e e _ r a t e=b e n c h m a r k _ d a t a _ s p e c i f i c a t i o n s [
576                                         ’ risk_free_rate ’
577                                 ],
578                                 days =365
579                          ),
580                          ’ Beta r e l a t i v e t o benchmark ’ : beta ,
581                          ’Maximum drawdown ’ : maximum_drawdown ,
582                          ’Maximum drawdown d u r a t i o n ’ : maximum_drawdown_duration ,
583                          ’Maximum drawdown peak d a t e ’ : maximum_drawdown_peak_date ,
584                          ’Maximum drawdown t r o u g h d a t e ’ : maximum_drawdown_trough_date
      BACKTESTING ALGOR. CRYPTOC. TRADING STRATEGIES                                                                                          120
        ,
585                           ’ Other m e t r i c s −−−> ’ :            ’ Other m e t r i c s −−−> ’ ,
586                           ’ Total t r a n s a c t i o n c o s t ’ : c a l c u l a t e _ t r a n s a c t i o n _ c o s t (
        df_trading_journal ) ,
587                           ’ Number o f t r a d e s ’ : l e n (
588                                  dict_execution_results [ ’ df_trading_journal ’ ]
589                           ),
590                           ’ Number o f un iqu e a s s e t s t r a d e d ’ : l e n ( d f _ t r a d i n g _ j o u r n a l [ ’
        A s s e t ’ ] . un iqu e ( ) ) ,
591                           ’ Average t i c k e t s i z e ’ : a v e r a g e _ t i c k e t _ s i z e _ a b s o l u t e _ v a l u e ,
592                           ’ Average c a s h ’ : round (
593                                  d i c t _ e x e c u t i o n _ r e s u l t s [ ’ d f _ t r a d i n g _ j o u r n a l ’ ] [ ’ Cash ’ ] .
        mean ( ) ,
594                                  2
595                           ),
596                           ’ Other i n f o −−−> ’ :            ’ Other i n f o −−−> ’ ,
597                           ’ S t a r t time ’ : s t a r t _ t i m e ,
598                           ’ End time ’ : d a t e t i m e . now ( ) ,
599                           ’ Parameter 1 ’ : s t r a t e g y _ h y p e r p a r a m e t e r s [ ’ s e l l _ p a r a m e t e r ’ ] ,
600                           ’ Parameter 2 ’ : s t r a t e g y _ h y p e r p a r a m e t e r s [ ’ buy_parameter ’ ] ,
601                           ’ Comments ’ : d i c t _ e x e c u t i o n _ r e s u l t s [ ’ comments ’ ] ,
602                           ’ Benchmark r e t u r n m e t r i c s −−−> ’ :                 ’ Benchmark r e t u r n m e t r i c s
        −−−> ’ ,
603                           ’ Benchmark USD a n n u a l i z e d ROI ( from f i r s t t o l a s t t r a d e ) ’ :
        d i c t _ r o i _ r e s u l t s [ ’ benchmark_roi ’ ] ,
604                           ’ Benchmark r i s k m e t r i c s −−−> ’ :                 ’ Benchmark r i s k m e t r i c s −−−> ’
        ,
605                           ’ Benchmark h o l d i n g p e r i o d v o l a t i l i t y ’ :              ’NOT IMPLEMENTED ’ ,
606                           ’ Benchmark annual v o l a t i l i t y ’ :                 ’NOT IMPLEMENTED ’ ,
607                           ’ Benchmark monthly v o l a t i l i t y ’ :                  ’NOT IMPLEMENTED ’ ,
608                           ’ Benchmark weekly v o l a t i l i t y ’ :                 ’NOT IMPLEMENTED ’ ,
609                           ’ Benchmark Beta r e l a t i v e t o benchmark ’ :                           ’NOT IMPLEMENTED ’ ,
610                           ’ Benchmark maximum drawdown ’ : benchmark_maximum_drawdown ,
611                           ’ Benchmark maximum drawdown d u r a t i o n ’ :
        benchmark_maximum_drawdown_duration ,
612                           ’ Benchmark maximum drawdown peak d a t e ’ :
        benchmark_maximum_drawdown_peak_date ,
613                           ’ Benchmark maximum drawdown t r o u g h d a t e ’ :
        benchmark_maximum_drawdown_trough_date ,
614                           ’ Benchmark o t h e r m e t r i c s −−−> ’ :                 ’NOT IMPLEMENTED ’ ,
615                           ’ Benchmark Sharpe r a t i o ( h o l d i n g p e r i o d ) ’ :                     ’NOT IMPLEMENTED ’
        ,
616                           ’ Benchmark Sharpe r a t i o ( y e a r l y ) ’ :                   ’NOT IMPLEMENTED ’ ,
617                    },
618                    i g n o r e _ i n d e x=True
619            )
620
621            plot_equity_curve (
622                    df_daily_returns ,
      BACKTESTING ALGOR. CRYPTOC. TRADING STRATEGIES                                                                                         121
623                      df_daily_benchmark_returns ,
624                      d i s p l a y _ o p t i o n s=d i s p l a y _ o p t i o n s ,
625                      b o o l e a n _ p l o t=d i s p l a y _ o p t i o n s [ ’ b o o l e a n _ p l o t _ e q u i t y _ c u r v e ’ ] ,
626                      boolean_save_to_disk=d i s p l a y _ o p t i o n s [ ’
         boolean_save_equity_curve_to_disk ’ ] ,
627              )
628
629       r e t u r n df_performance
630
652       f i g , ax = p l t . s u b p l o t s ( f i g s i z e =(12 , 8 ) )
653       ax . x a x i s . s e t _ m a j o r _ l o c a t o r (
654              MonthLocator (
655                      bymonthday=1,
656                      i n t e r v a l =3,
657                      t z=None
658              )
659       )
660       ax . x a x i s . s e t _ m a j o r _ f o r m a t t e r ( DateFormatter ( "%y−%m−%d " ) )
661
669
670       i f boolean_save_to_disk :
671              p l t . s a v e f i g ( s t r i n g _ d i r e c t o r y + ’ / equity_curve_ ’ + s t r (
         n u m b e r _ o f _ r e s u l t _ f i l e s _ p l u s _ 1 ) + ’ . png ’ )
672
673       i f boolean_plot :
674              p l t . show ( )