From a99a51c93d565410d9f52196654b66feab87a5a0 Mon Sep 17 00:00:00 2001 From: Nikolay Baraboshkin Date: Fri, 6 Dec 2024 23:11:24 +0400 Subject: [PATCH 1/2] Improve the performance of ConsolidatorScanPriority comparison in SubscriptionManager --- Common/Data/ConsolidatorWrapper.cs | 33 ++++++++++--------- Common/Data/SubscriptionManager.cs | 2 +- Tests/Common/Data/ConsolidatorWrapperTests.cs | 20 +++++++++++ 3 files changed, 39 insertions(+), 16 deletions(-) diff --git a/Common/Data/ConsolidatorWrapper.cs b/Common/Data/ConsolidatorWrapper.cs index f383e1bd91a5..7c06aa3b1d45 100644 --- a/Common/Data/ConsolidatorWrapper.cs +++ b/Common/Data/ConsolidatorWrapper.cs @@ -14,6 +14,7 @@ */ using System; +using System.Collections.Generic; using System.Threading; using QuantConnect.Interfaces; using QuantConnect.Data.Consolidators; @@ -138,8 +139,24 @@ private void AdvanceScanTime(object _ = null, IBaseData consolidated = null) } } - internal class ConsolidatorScanPriority : IComparable + internal class ConsolidatorScanPriority { + private sealed class UtcScanTimeIdRelationalComparer : IComparer + { + public int Compare(ConsolidatorScanPriority? x, ConsolidatorScanPriority? y) + { + if (ReferenceEquals(x, y)) return 0; + if (y is null) return 1; + if (x is null) return -1; + var utcScanTimeComparison = x.UtcScanTime.CompareTo(y.UtcScanTime); + if (utcScanTimeComparison != 0) return utcScanTimeComparison; + return x.Id.CompareTo(y.Id); + } + } + + public static IComparer Comparer { get; } = + new UtcScanTimeIdRelationalComparer(); + /// /// The next utc scan time /// @@ -155,19 +172,5 @@ public ConsolidatorScanPriority(DateTime utcScanTime, long id) Id = id; UtcScanTime = utcScanTime; } - - public int CompareTo(object obj) - { - if (obj == null) return 1; - - var other = (ConsolidatorScanPriority)obj; - var result = UtcScanTime.CompareTo(other.UtcScanTime); - if (result == 0) - { - // if they are the same let's compare Ids too - return Id.CompareTo(other.Id); - } - return result; - } } } diff --git a/Common/Data/SubscriptionManager.cs b/Common/Data/SubscriptionManager.cs index f068c68912c6..45b06159b8ed 100644 --- a/Common/Data/SubscriptionManager.cs +++ b/Common/Data/SubscriptionManager.cs @@ -66,7 +66,7 @@ public SubscriptionManager(ITimeKeeper timeKeeper) { _consolidators = new(); _timeKeeper = timeKeeper; - _consolidatorsSortedByScanTime = new(1000); + _consolidatorsSortedByScanTime = new(1000, ConsolidatorScanPriority.Comparer); _threadSafeCollectionLock = new object(); } diff --git a/Tests/Common/Data/ConsolidatorWrapperTests.cs b/Tests/Common/Data/ConsolidatorWrapperTests.cs index b130fe040b67..8763972488a2 100644 --- a/Tests/Common/Data/ConsolidatorWrapperTests.cs +++ b/Tests/Common/Data/ConsolidatorWrapperTests.cs @@ -163,6 +163,26 @@ public void ScanTimeOnWorkingBarDayLightSavings(int hoursShift, bool savingsStar Assert.AreEqual(consolidator.WorkingData.EndTime.ConvertToUtc(tz), wrapper.UtcScanTime); } + [Test] + public void ConsolidatorScanPriorityComparerComparesByUtcScanDateThenById() + { + var id1 = Random.Shared.NextInt64(); + var utcScanTime = DateTime.UtcNow; + var priority1 = new ConsolidatorScanPriority(utcScanTime, id1); + var priority2 = new ConsolidatorScanPriority(utcScanTime.AddSeconds(1), id1 + 1); + var priority3 = new ConsolidatorScanPriority(utcScanTime, id1 + 1); + var priority4 = new ConsolidatorScanPriority(utcScanTime, id1); + + Assert.AreEqual(-1, ConsolidatorScanPriority.Comparer.Compare(priority1, priority2)); + Assert.AreEqual(1, ConsolidatorScanPriority.Comparer.Compare(priority2, priority1)); + Assert.AreEqual(1, ConsolidatorScanPriority.Comparer.Compare(priority3, priority1)); + Assert.AreEqual(-1, ConsolidatorScanPriority.Comparer.Compare(priority3, priority2)); + Assert.AreEqual(0, ConsolidatorScanPriority.Comparer.Compare(priority1, priority4)); + Assert.AreEqual(0, ConsolidatorScanPriority.Comparer.Compare(priority1, priority1)); + Assert.AreEqual(1, ConsolidatorScanPriority.Comparer.Compare(priority1, null)); + Assert.AreEqual(-1, ConsolidatorScanPriority.Comparer.Compare(null, priority1)); + Assert.AreEqual(0, ConsolidatorScanPriority.Comparer.Compare(null, null)); + } private class TestConsolidator : IDataConsolidator { public IBaseData Consolidated { get; set; } From ee66d96ce2fe5ce0da6859699a4af74942eb2177 Mon Sep 17 00:00:00 2001 From: Nikolay Baraboshkin Date: Wed, 11 Dec 2024 21:44:25 +0400 Subject: [PATCH 2/2] address review comments --- Tests/Common/Data/ConsolidatorWrapperTests.cs | 37 +++++++++++++++---- 1 file changed, 29 insertions(+), 8 deletions(-) diff --git a/Tests/Common/Data/ConsolidatorWrapperTests.cs b/Tests/Common/Data/ConsolidatorWrapperTests.cs index 8763972488a2..1bae9de42a57 100644 --- a/Tests/Common/Data/ConsolidatorWrapperTests.cs +++ b/Tests/Common/Data/ConsolidatorWrapperTests.cs @@ -164,25 +164,46 @@ public void ScanTimeOnWorkingBarDayLightSavings(int hoursShift, bool savingsStar } [Test] - public void ConsolidatorScanPriorityComparerComparesByUtcScanDateThenById() + public void ConsolidatorScanPriorityComparerComparesByUtcScanDate() { - var id1 = Random.Shared.NextInt64(); - var utcScanTime = DateTime.UtcNow; - var priority1 = new ConsolidatorScanPriority(utcScanTime, id1); - var priority2 = new ConsolidatorScanPriority(utcScanTime.AddSeconds(1), id1 + 1); - var priority3 = new ConsolidatorScanPriority(utcScanTime, id1 + 1); - var priority4 = new ConsolidatorScanPriority(utcScanTime, id1); + const int id = 1; + var utcScanTime = new DateTime(2024, 12, 10, 0, 0, 0, DateTimeKind.Utc); + + var priority1 = new ConsolidatorScanPriority(utcScanTime, id); + var priority2 = new ConsolidatorScanPriority(utcScanTime.AddSeconds(1), id + 1); + var priority3 = new ConsolidatorScanPriority(utcScanTime, id + 1); Assert.AreEqual(-1, ConsolidatorScanPriority.Comparer.Compare(priority1, priority2)); Assert.AreEqual(1, ConsolidatorScanPriority.Comparer.Compare(priority2, priority1)); Assert.AreEqual(1, ConsolidatorScanPriority.Comparer.Compare(priority3, priority1)); Assert.AreEqual(-1, ConsolidatorScanPriority.Comparer.Compare(priority3, priority2)); - Assert.AreEqual(0, ConsolidatorScanPriority.Comparer.Compare(priority1, priority4)); Assert.AreEqual(0, ConsolidatorScanPriority.Comparer.Compare(priority1, priority1)); + } + + [Test] + public void ConsolidatorScanPriorityComparerComparesByIdIfUtcScanTimesAreEqual() + { + const int id = 1; + var utcScanTime = new DateTime(2024, 12, 10, 0, 0, 0, DateTimeKind.Utc); + + var priority1 = new ConsolidatorScanPriority(utcScanTime, id); + var priority2 = new ConsolidatorScanPriority(utcScanTime, id + 1); + + Assert.AreEqual(1, ConsolidatorScanPriority.Comparer.Compare(priority2, priority1)); + } + + [Test] + public void ConsolidatorScanPriorityComparerTreatsNullsRight() + { + const int id = 1; + var utcScanTime = new DateTime(2024, 12, 10, 0, 0, 0, DateTimeKind.Utc); + var priority1 = new ConsolidatorScanPriority(utcScanTime, id); + Assert.AreEqual(1, ConsolidatorScanPriority.Comparer.Compare(priority1, null)); Assert.AreEqual(-1, ConsolidatorScanPriority.Comparer.Compare(null, priority1)); Assert.AreEqual(0, ConsolidatorScanPriority.Comparer.Compare(null, null)); } + private class TestConsolidator : IDataConsolidator { public IBaseData Consolidated { get; set; }