/*
* QUANTCONNECT.COM - Democratizing Finance, Empowering Individuals.
* Lean Algorithmic Trading Engine v2.0. Copyright 2014 QuantConnect Corporation.
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
using System;
using System.Collections.Generic;
using System.Linq;
using Deedle;
using Python.Runtime;
using QuantConnect.Packets;
namespace QuantConnect.Report.ReportElements
{
internal sealed class CrisisReportElement : ChartReportElement
{
private const string _emptyChart = "";
private LiveResult _live;
private BacktestResult _backtest;
private string _template;
///
/// Create a new array of crisis event plots
///
/// Name of the widget
/// Location of injection
/// Backtest result object
/// Live result object
/// HTML template to use
public CrisisReportElement(string name, string key, BacktestResult backtest, LiveResult live, string template)
{
_live = live;
_backtest = backtest;
_template = template;
Name = name;
Key = key;
}
///
/// The generated output string to be injected
///
public override string Render()
{
var backtestPoints = ResultsUtil.EquityPoints(_backtest);
var backtestBenchmarkPoints = ResultsUtil.BenchmarkPoints(_backtest);
var backtestSeries = new Series(backtestPoints.Keys, backtestPoints.Values);
var backtestBenchmarkSeries = new Series(backtestBenchmarkPoints.Keys, backtestBenchmarkPoints.Values);
var html = new List();
foreach (var crisisEvent in Crisis.Events)
{
using (Py.GIL())
{
var crisis = crisisEvent.Value;
var data = new PyList();
var frame = Frame.CreateEmpty();
// The two following operations are equivalent to Pandas' `df.resample("D").sum()`
frame["Backtest"] = backtestSeries.ResampleEquivalence(date => date.Date, s => s.LastValue());
frame["Benchmark"] = backtestBenchmarkSeries.ResampleEquivalence(date => date.Date, s => s.LastValue());
var crisisFrame = frame.Where(kvp => kvp.Key >= crisis.Start && kvp.Key <= crisis.End);
crisisFrame = crisisFrame.Join("BacktestPercent", crisisFrame["Backtest"].CumulativeReturns());
crisisFrame = crisisFrame.Join("BenchmarkPercent", crisisFrame["Benchmark"].CumulativeReturns());
// Pad out all missing values to start from 0 for nice plots
crisisFrame = crisisFrame.FillMissing(Direction.Forward).FillMissing(0.0);
data.Append(crisisFrame.RowKeys.ToList().ToPython());
data.Append(crisisFrame["BacktestPercent"].Values.ToList().ToPython());
data.Append(crisisFrame["BenchmarkPercent"].Values.ToList().ToPython());
var base64 = (string)Charting.GetCrisisEventsPlots(data, crisis.Name.Replace("/", "").Replace(".", "").Replace(" ", ""));
if (base64 == _emptyChart)
{
continue;
}
if (!crisisFrame.IsEmpty)
{
var contents = _template.Replace(ReportKey.CrisisTitle, crisis.ToString(crisisFrame.GetRowKeyAt(0), crisisFrame.GetRowKeyAt(crisisFrame.RowCount - 1)));
contents = contents.Replace(ReportKey.CrisisContents, base64);
html.Add(contents);
}
}
}
if (Key == ReportKey.CrisisPageStyle)
{
if (html.Count == 0)
{
return "display: none;";
}
return string.Empty;
}
return string.Join("\n", html);
}
}
}