/*
* 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 QuantConnect.Data;
using QuantConnect.Interfaces;
using QuantConnect.Orders;
namespace QuantConnect.Algorithm.CSharp
{
///
/// Regression algorithm asserting that a short option position is auto exercised even when there is insufficient margin,
/// but triggering a margin call for the underlying stock to cover the assignment.
///
public class InsufficientBuyingPowerForAutomaticExerciseRegressionAlgorithm : QCAlgorithm, IRegressionAlgorithmDefinition
{
private Symbol _stock;
private Symbol _option;
private bool _stockBought;
private bool _optionSold;
private bool _optionAssigned;
private bool _marginCallReceived;
public override void Initialize()
{
SetStartDate(2015, 12, 23);
SetEndDate(2015, 12, 28);
SetCash(100000);
_stock = AddEquity("GOOG").Symbol;
var contracts = OptionChain(_stock).ToList();
_option = contracts
.Where(c => c.ID.OptionRight == OptionRight.Put)
.OrderBy(c => c.ID.Date)
.First(c => c.ID.StrikePrice == 800m);
AddOptionContract(_option);
}
public override void OnData(Slice slice)
{
// We are done with buying
if (_stockBought && _optionSold)
{
return;
}
if (!Portfolio.Invested)
{
// We'll use all our buying power to buy the stock, so when we then open a short put position,
// the margin will not be enough to cover the automatic exercise
SetHoldings(_stock, 1);
}
if (_stockBought && Securities[_option].Price != 0)
{
MarketOrder(_option, -2);
}
}
public override void OnMarginCall(List requests)
{
if (!_optionAssigned)
{
throw new RegressionTestException("Expected option to have been assigned before the margin call " +
"(which should have been triggered by the auto-exercise of the option with inssuficient margin).");
}
if (_marginCallReceived)
{
throw new RegressionTestException("Received multiple margin calls. Expected just one.");
}
var request = requests.Single();
if (request.Symbol != _stock)
{
throw new RegressionTestException("Expected margin call for the stock, but got margin call for: " + request.Symbol);
}
_marginCallReceived = true;
}
public override void OnOrderEvent(OrderEvent orderEvent)
{
var order = Transactions.GetOrderById(orderEvent.OrderId);
Debug($"{Time} :: {order.Id} - {order.Type} - {orderEvent.Symbol}: {orderEvent.Status} - {orderEvent.Quantity} shares at {orderEvent.FillPrice}");
if (orderEvent.Status == OrderStatus.Filled)
{
if (orderEvent.Symbol == _stock)
{
_stockBought = true;
}
else if (orderEvent.Symbol == _option)
{
if (order.Type == OrderType.Market)
{
if (!_stockBought)
{
throw new RegressionTestException("Stock should have been bought first");
}
_optionSold = true;
}
else if (order.Type == OrderType.OptionExercise && orderEvent.IsAssignment)
{
if (!_optionSold)
{
throw new RegressionTestException("Option should have been sold first");
}
_optionAssigned = true;
}
}
else
{
throw new RegressionTestException("Unexpected symbol: " + orderEvent.Symbol);
}
}
}
public override void OnEndOfAlgorithm()
{
if (!_stockBought)
{
throw new RegressionTestException("Stock was not bought");
}
if (!_optionSold)
{
throw new RegressionTestException("Option was not sold");
}
if (!_optionAssigned)
{
throw new RegressionTestException("Option was not assigned");
}
if (!_marginCallReceived)
{
throw new RegressionTestException("Margin call was not received");
}
}
///
/// This is used by the regression test system to indicate if the open source Lean repository has the required data to run this algorithm.
///
public bool CanRunLocally { get; } = true;
///
/// This is used by the regression test system to indicate which languages this algorithm is written in.
///
public List Languages { get; } = new() { Language.CSharp };
///
/// Data Points count of all timeslices of algorithm
///
public long DataPoints => 2821;
///
/// Data Points count of the algorithm history
///
public int AlgorithmHistoryDataPoints => 1;
///
/// Final status of the algorithm
///
public AlgorithmStatus AlgorithmStatus => AlgorithmStatus.Completed;
///
/// This is used by the regression test system to indicate what the expected statistics are from running the algorithm
///
public Dictionary ExpectedStatistics => new Dictionary
{
{"Total Orders", "4"},
{"Average Win", "8.96%"},
{"Average Loss", "-1.95%"},
{"Compounding Annual Return", "-67.963%"},
{"Drawdown", "2.900%"},
{"Expectancy", "-1"},
{"Start Equity", "100000"},
{"End Equity", "98248.35"},
{"Net Profit", "-1.752%"},
{"Sharpe Ratio", "-6.542"},
{"Sortino Ratio", "0"},
{"Probabilistic Sharpe Ratio", "1.125%"},
{"Loss Rate", "100%"},
{"Win Rate", "0%"},
{"Profit-Loss Ratio", "4.60"},
{"Alpha", "-0.007"},
{"Beta", "1.181"},
{"Annual Standard Deviation", "0.036"},
{"Annual Variance", "0.001"},
{"Information Ratio", "-1.422"},
{"Tracking Error", "0.03"},
{"Treynor Ratio", "-0.2"},
{"Total Fees", "$3.30"},
{"Estimated Strategy Capacity", "$2400000.00"},
{"Lowest Capacity Asset", "GOOCV 305RBQ20WHPNQ|GOOCV VP83T1ZUHROL"},
{"Portfolio Turnover", "54.01%"},
{"Drawdown Recovery", "0"},
{"OrderListHash", "97d1c3373a72ec15038949710e7c7a62"}
};
}
}