/*
* 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.Threading;
using QuantConnect.Util;
namespace QuantConnect.Scheduling
{
///
/// Helper class that will monitor timer consumers and request more time if required.
/// Used by
///
public class TimeMonitor : IDisposable
{
private readonly Timer _timer;
///
/// List to store the coming TimeConsumer objects
///
/// This field is protected because it's used in a test class
/// in `IsolatorLimitResultProviderTests.cs
protected List TimeConsumers { get; init; }
///
/// Returns the number of time consumers currently being monitored
///
public int Count
{
get
{
lock (TimeConsumers)
{
return TimeConsumers.Count;
}
}
}
///
/// Creates a new instance
///
public TimeMonitor(int monitorIntervalMs = 100)
{
TimeConsumers = new List();
_timer = new Timer(state =>
{
try
{
lock (TimeConsumers)
{
RemoveAll();
foreach (var consumer in TimeConsumers)
{
ProcessConsumer(consumer);
}
}
}
finally
{
try
{
_timer.Change(Time.GetSecondUnevenWait(monitorIntervalMs), Timeout.Infinite);
}
catch (ObjectDisposedException)
{
// ignored disposed
}
}
}, null, monitorIntervalMs, Timeout.Infinite);
}
///
/// Process the TimeConsumer object in TimeConsumers list
///
/// The TimeConsumer object to be processed
/// This method is protected because it's overrode by a test class
/// in `IsolatorLimitResultProviderTests.cs`
protected virtual void ProcessConsumer(TimeConsumer consumer)
{
if (consumer.NextTimeRequest == null)
{
// first time, for performance we register this here and not the time consumer
consumer.NextTimeRequest = consumer.TimeProvider.GetUtcNow().AddMinutes(1);
}
else if (consumer.TimeProvider.GetUtcNow() >= consumer.NextTimeRequest)
{
// each minute request additional time from the isolator
consumer.NextTimeRequest = consumer.NextTimeRequest.Value.AddMinutes(1);
try
{
// this will notify the isolator that we've exceed the limits
consumer.IsolatorLimitProvider.RequestAdditionalTime(minutes: 1);
}
catch
{
// pass
}
}
}
///
/// Remove all TimeConsumer objects where the `Finished` field is marked as true
///
/// This method is protected because it's overrode by a test class in
/// `IsolatorLimitResultProviderTests.cs`
protected virtual void RemoveAll()
{
TimeConsumers.RemoveAll(time => time.Finished);
}
///
/// Adds a new time consumer element to be monitored
///
/// Time consumer instance
public void Add(TimeConsumer consumer)
{
lock (TimeConsumers)
{
TimeConsumers.Add(consumer);
}
}
///
/// Disposes of the inner timer
///
public void Dispose()
{
_timer.DisposeSafely();
}
}
}