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  private IMapFileProvider _mapFileProvider;
29 
30  /// <summary>
31  /// Creates a new instance
32  /// </summary>
33  /// <param name="dataCacheProvider">The data cache provider instance to use</param>
34  /// <param name="mapFileProvider">The map file provider instance to use</param>
35  public BacktestingOptionChainProvider(IDataCacheProvider dataCacheProvider, IMapFileProvider mapFileProvider)
36  : base(dataCacheProvider)
37  {
38  _mapFileProvider = mapFileProvider;
39  }
40 
41  /// <summary>
42  /// Gets the list of option contracts for a given underlying symbol
43  /// </summary>
44  /// <param name="symbol">The option or the underlying symbol to get the option chain for.
45  /// Providing the option allows targetting an option ticker different than the default e.g. SPXW</param>
46  /// <param name="date">The date for which to request the option chain (only used in backtesting)</param>
47  /// <returns>The list of option contracts</returns>
48  public virtual IEnumerable<Symbol> GetOptionContractList(Symbol symbol, DateTime date)
49  {
50  Symbol canonicalSymbol;
51  if (!symbol.SecurityType.HasOptions())
52  {
53  // we got an option
54  if (symbol.SecurityType.IsOption() && symbol.Underlying != null)
55  {
56  canonicalSymbol = GetCanonical(symbol, date);
57  }
58  else
59  {
60  throw new NotSupportedException($"BacktestingOptionChainProvider.GetOptionContractList(): " +
61  $"{nameof(SecurityType.Equity)}, {nameof(SecurityType.Future)}, or {nameof(SecurityType.Index)} is expected but was {symbol.SecurityType}");
62  }
63  }
64  else
65  {
66  // we got the underlying
67  var mappedUnderlyingSymbol = MapUnderlyingSymbol(symbol, date);
68  canonicalSymbol = Symbol.CreateCanonicalOption(mappedUnderlyingSymbol);
69  }
70 
71  return GetSymbols(canonicalSymbol, date);
72  }
73 
74  private Symbol GetCanonical(Symbol optionSymbol, DateTime date)
75  {
76  // Resolve any mapping before requesting option contract list for equities
77  // Needs to be done in order for the data file key to be accurate
78  if (optionSymbol.Underlying.RequiresMapping())
79  {
80  var mappedUnderlyingSymbol = MapUnderlyingSymbol(optionSymbol.Underlying, date);
81 
82  return Symbol.CreateCanonicalOption(mappedUnderlyingSymbol);
83  }
84  else
85  {
86  return optionSymbol.Canonical;
87  }
88  }
89 
90  private Symbol MapUnderlyingSymbol(Symbol underlying, DateTime date)
91  {
92  if (underlying.RequiresMapping())
93  {
94  var mapFileResolver = _mapFileProvider.Get(AuxiliaryDataKey.Create(underlying));
95  var mapFile = mapFileResolver.ResolveMapFile(underlying);
96  var ticker = mapFile.GetMappedSymbol(date, underlying.Value);
97  return underlying.UpdateMappedSymbol(ticker);
98  }
99  else
100  {
101  return underlying;
102  }
103  }
104  }
105 }