Lean  $LEAN_TAG$
BacktestingOptionChainProvider.cs
1 /*
2  * QUANTCONNECT.COM - Democratizing Finance, Empowering Individuals.
3  * Lean Algorithmic Trading Engine v2.0. Copyright 2014 QuantConnect Corporation.
4  *
5  * Licensed under the Apache License, Version 2.0 (the "License");
6  * you may not use this file except in compliance with the License.
7  * You may obtain a copy of the License at http://www.apache.org/licenses/LICENSE-2.0
8  *
9  * Unless required by applicable law or agreed to in writing, software
10  * distributed under the License is distributed on an "AS IS" BASIS,
11  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12  * See the License for the specific language governing permissions and
13  * limitations under the License.
14 */
15 
16 using System;
18 using System.Collections.Generic;
20 
22 {
23  /// <summary>
24  /// An implementation of <see cref="IOptionChainProvider"/> that reads the list of contracts from open interest zip data files
25  /// </summary>
27  {
28  /// <summary>
29  /// Gets the list of option contracts for a given underlying symbol
30  /// </summary>
31  /// <param name="symbol">The option or the underlying symbol to get the option chain for.
32  /// Providing the option allows targeting an option ticker different than the default e.g. SPXW</param>
33  /// <param name="date">The date for which to request the option chain (only used in backtesting)</param>
34  /// <returns>The list of option contracts</returns>
35  public virtual IEnumerable<Symbol> GetOptionContractList(Symbol symbol, DateTime date)
36  {
37  Symbol canonicalSymbol;
38  if (!symbol.SecurityType.HasOptions())
39  {
40  // we got an option
41  if (symbol.SecurityType.IsOption() && symbol.Underlying != null)
42  {
43  canonicalSymbol = GetCanonical(symbol, date);
44  }
45  else
46  {
47  throw new NotSupportedException($"BacktestingOptionChainProvider.GetOptionContractList(): " +
48  $"{nameof(SecurityType.Equity)}, {nameof(SecurityType.Future)}, or {nameof(SecurityType.Index)} is expected but was {symbol.SecurityType}");
49  }
50  }
51  else
52  {
53  // we got the underlying
54  var mappedUnderlyingSymbol = MapUnderlyingSymbol(symbol, date);
55  canonicalSymbol = Symbol.CreateCanonicalOption(mappedUnderlyingSymbol);
56  }
57 
58  return GetSymbols(canonicalSymbol, date);
59  }
60 
61  private Symbol GetCanonical(Symbol optionSymbol, DateTime date)
62  {
63  // Resolve any mapping before requesting option contract list for equities
64  // Needs to be done in order for the data file key to be accurate
65  if (optionSymbol.Underlying.RequiresMapping())
66  {
67  var mappedUnderlyingSymbol = MapUnderlyingSymbol(optionSymbol.Underlying, date);
68 
69  return Symbol.CreateCanonicalOption(mappedUnderlyingSymbol);
70  }
71  else
72  {
73  return optionSymbol.Canonical;
74  }
75  }
76 
77  private Symbol MapUnderlyingSymbol(Symbol underlying, DateTime date)
78  {
79  if (underlying.RequiresMapping())
80  {
81  var mapFileResolver = MapFileProvider.Get(AuxiliaryDataKey.Create(underlying));
82  var mapFile = mapFileResolver.ResolveMapFile(underlying);
83  var ticker = mapFile.GetMappedSymbol(date, underlying.Value);
84  return underlying.UpdateMappedSymbol(ticker);
85  }
86  else
87  {
88  return underlying;
89  }
90  }
91  }
92 }