Lean  $LEAN_TAG$
OpenInterestFutureUniverseSelectionModel.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;
17 using System.Collections.Generic;
18 using System.Linq;
19 using NodaTime;
20 using Python.Runtime;
21 using QuantConnect.Data;
25 
27 {
28  /// <summary>
29  /// Selects contracts in a futures universe, sorted by open interest. This allows the selection to identifiy current
30  /// active contract.
31  /// </summary>
33  {
34  private readonly int? _chainContractsLookupLimit;
35  private readonly IAlgorithm _algorithm;
36  private readonly int? _resultsLimit;
37  private readonly MarketHoursDatabase _marketHoursDatabase;
38 
39  /// <summary>
40  /// Creates a new instance of <see cref="OpenInterestFutureUniverseSelectionModel" />
41  /// </summary>
42  /// <param name="algorithm">Algorithm</param>
43  /// <param name="futureChainSymbolSelector">Selects symbols from the provided future chain</param>
44  /// <param name="chainContractsLookupLimit">Limit on how many contracts to query for open interest</param>
45  /// <param name="resultsLimit">Limit on how many contracts will be part of the universe</param>
46  public OpenInterestFutureUniverseSelectionModel(IAlgorithm algorithm, Func<DateTime, IEnumerable<Symbol>> futureChainSymbolSelector, int? chainContractsLookupLimit = 6,
47  int? resultsLimit = 1) : base(TimeSpan.FromDays(1), futureChainSymbolSelector)
48  {
49  _marketHoursDatabase = MarketHoursDatabase.FromDataFolder();
50  if (algorithm == null)
51  {
52  throw new ArgumentNullException(nameof(algorithm));
53  }
54 
55  _algorithm = algorithm;
56  _resultsLimit = resultsLimit;
57  _chainContractsLookupLimit = chainContractsLookupLimit;
58  }
59 
60  /// <summary>
61  /// Creates a new instance of <see cref="OpenInterestFutureUniverseSelectionModel" />
62  /// </summary>
63  /// <param name="algorithm">Algorithm</param>
64  /// <param name="futureChainSymbolSelector">Selects symbols from the provided future chain</param>
65  /// <param name="chainContractsLookupLimit">Limit on how many contracts to query for open interest</param>
66  /// <param name="resultsLimit">Limit on how many contracts will be part of the universe</param>
67  public OpenInterestFutureUniverseSelectionModel(IAlgorithm algorithm, PyObject futureChainSymbolSelector, int? chainContractsLookupLimit = 6,
68  int? resultsLimit = 1) : this(algorithm, ConvertFutureChainSymbolSelectorToFunc(futureChainSymbolSelector), chainContractsLookupLimit, resultsLimit)
69  {
70  }
71 
72  /// <summary>
73  /// Defines the future chain universe filter
74  /// </summary>
76  {
77  // Remove duplicated keys
78  return filter.Contracts(FilterByOpenInterest(filter.DistinctBy(x => x).ToDictionary(x => x, x => _marketHoursDatabase.GetEntry(x.ID.Market, x, x.ID.SecurityType))));
79  }
80 
81  /// <summary>
82  /// Filters a set of contracts based on open interest.
83  /// </summary>
84  /// <param name="contracts">Contracts to filter</param>
85  /// <returns>Filtered set</returns>
86  public IEnumerable<Symbol> FilterByOpenInterest(IReadOnlyDictionary<Symbol, MarketHoursDatabase.Entry> contracts)
87  {
88  var symbols = new List<Symbol>(_chainContractsLookupLimit.HasValue ? contracts.Keys.OrderBy(x => x.ID.Date).Take(_chainContractsLookupLimit.Value) : contracts.Keys);
89  var openInterest = symbols.GroupBy(x => contracts[x]).SelectMany(g => GetOpenInterest(g.Key, g.Select(i => i))).ToDictionary(x => x.Key, x => x.Value);
90 
91  if (openInterest.Count == 0)
92  {
93  _algorithm.Error(
94  $"{nameof(OpenInterestFutureUniverseSelectionModel)}.{nameof(FilterByOpenInterest)}: Failed to get historical open interest, no symbol will be selected."
95  );
96  return Enumerable.Empty<Symbol>();
97  }
98 
99  var filtered = openInterest.OrderByDescending(x => x.Value).ThenBy(x => x.Key.ID.Date).Select(x => x.Key);
100  if (_resultsLimit.HasValue)
101  {
102  filtered = filtered.Take(_resultsLimit.Value);
103  }
104 
105  return filtered;
106  }
107 
108  private Dictionary<Symbol, decimal> GetOpenInterest(MarketHoursDatabase.Entry marketHours, IEnumerable<Symbol> symbols)
109  {
110  var current = _algorithm.UtcTime;
111  var exchangeHours = marketHours.ExchangeHours;
112  var endTime = Instant.FromDateTimeUtc(_algorithm.UtcTime).InZone(exchangeHours.TimeZone).ToDateTimeUnspecified();
113  var previousDay = Time.GetStartTimeForTradeBars(exchangeHours, endTime, Time.OneDay, 1, true, marketHours.DataTimeZone);
114  var requests = symbols.Select(
115  symbol => new HistoryRequest(
116  previousDay,
117  current,
118  typeof(Tick),
119  symbol,
120  Resolution.Tick,
121  exchangeHours,
122  exchangeHours.TimeZone,
123  null,
124  true,
125  false,
127  TickType.OpenInterest
128  )
129  )
130  .ToArray();
131  return _algorithm.HistoryProvider.GetHistory(requests, exchangeHours.TimeZone)
132  .Where(s => s.HasData && s.Ticks.Keys.Count > 0)
133  .SelectMany(s => s.Ticks.Select(x => new Tuple<Symbol, Tick>(x.Key, x.Value.LastOrDefault())))
134  .GroupBy(x => x.Item1)
135  .ToDictionary(x => x.Key, x => x.OrderByDescending(i => i.Item2.Time).LastOrDefault().Item2.Value);
136  }
137 
138  /// <summary>
139  /// Converts future chain symbol selector, provided as a Python lambda function, to a managed func
140  /// </summary>
141  /// <param name="futureChainSymbolSelector">Python lambda function that selects symbols from the provided future chain</param>
142  /// <returns>Given Python future chain symbol selector as a func objet</returns>
143  /// <exception cref="ArgumentException"></exception>
144  private static Func<DateTime, IEnumerable<Symbol>> ConvertFutureChainSymbolSelectorToFunc(PyObject futureChainSymbolSelector)
145  {
146  if (futureChainSymbolSelector.TryConvertToDelegate(out Func<DateTime, IEnumerable<Symbol>> futureSelector))
147  {
148  return futureSelector;
149  }
150  else
151  {
152  using (Py.GIL())
153  {
154  throw new ArgumentException($"FutureUniverseSelectionModel.ConvertFutureChainSymbolSelectorToFunc: {futureChainSymbolSelector.Repr()} is not a valid argument.");
155  }
156  }
157  }
158  }
159 }