From 397a19dec7df316cfbdb97d22ffd0d0e08c87261 Mon Sep 17 00:00:00 2001 From: Craig Edmunds Date: Thu, 19 Dec 2024 10:23:00 +0000 Subject: [PATCH 1/6] CDMS-200 analytics adjustments for novembers prod data --- .../Extensions/AnalyticsExtensions.cs | 6 ++--- .../IImportNotificationsAggregationService.cs | 1 + .../ImportNotificationsAggregationService.cs | 25 ++++++++++++++++--- Btms.Backend/Config/AnalyticsDashboards.cs | 4 +++ Btms.Model/Movement.cs | 6 ++++- 5 files changed, 34 insertions(+), 8 deletions(-) diff --git a/Btms.Analytics/Extensions/AnalyticsExtensions.cs b/Btms.Analytics/Extensions/AnalyticsExtensions.cs index 76ed0998..d63dbbde 100644 --- a/Btms.Analytics/Extensions/AnalyticsExtensions.cs +++ b/Btms.Analytics/Extensions/AnalyticsExtensions.cs @@ -132,10 +132,10 @@ internal static IEnumerable Execute( private static void LogExecutedMongoString(this ILogger logger, IQueryable source) { var stages = ((IMongoQueryProvider)source.Provider).LoggedStages; - - logger.LogInformation("[{Query}]", string.Join(",", stages.Select(s => s.ToString()).ToArray())); + var query = string.Join(",", stages.Select(s => s.ToString()).ToArray()); + logger.LogInformation("[{Query}]", query); } - + public static async Task AsIDataset(this Task ms) { await ms; diff --git a/Btms.Analytics/IImportNotificationsAggregationService.cs b/Btms.Analytics/IImportNotificationsAggregationService.cs index a6b40638..51faac40 100644 --- a/Btms.Analytics/IImportNotificationsAggregationService.cs +++ b/Btms.Analytics/IImportNotificationsAggregationService.cs @@ -6,4 +6,5 @@ public interface IImportNotificationsAggregationService public Task ByArrival(DateTime from, DateTime to, AggregationPeriod aggregateBy = AggregationPeriod.Day); public Task ByStatus(DateTime from, DateTime to); public Task ByCommodityCount(DateTime from, DateTime to); + public Task ByVersionCount(DateTime from, DateTime to, AggregationPeriod aggregateBy = AggregationPeriod.Day); } \ No newline at end of file diff --git a/Btms.Analytics/ImportNotificationsAggregationService.cs b/Btms.Analytics/ImportNotificationsAggregationService.cs index 26ed8f30..852f0a38 100644 --- a/Btms.Analytics/ImportNotificationsAggregationService.cs +++ b/Btms.Analytics/ImportNotificationsAggregationService.cs @@ -22,7 +22,7 @@ public Task ByCreated(DateTime from, DateTime to, Ag string CreateDatasetName(BsonDocument b) => AnalyticsHelpers.GetLinkedName(b["_id"]["linked"].ToBoolean(), b["_id"]["importNotificationType"].ToString()!.FromImportNotificationTypeEnumString()); - return Aggregate(dateRange, CreateDatasetName, matchFilter, "$createdSource", aggregateBy); + return AggregateByLinkedAndNotificationType(dateRange, CreateDatasetName, matchFilter, "$createdSource", aggregateBy); } public Task ByArrival(DateTime from, DateTime to, AggregationPeriod aggregateBy = AggregationPeriod.Day) @@ -35,7 +35,7 @@ public Task ByArrival(DateTime from, DateTime to, Ag string CreateDatasetName(BsonDocument b) => AnalyticsHelpers.GetLinkedName(b["_id"]["linked"].ToBoolean(), b["_id"]["importNotificationType"].ToString()!.FromImportNotificationTypeEnumString()); - return Aggregate(dateRange, CreateDatasetName, matchFilter, "$partOne.arrivesAt", aggregateBy); + return AggregateByLinkedAndNotificationType(dateRange, CreateDatasetName, matchFilter, "$partOne.arrivesAt", aggregateBy); } public Task ByStatus(DateTime from, DateTime to) @@ -63,7 +63,11 @@ public Task ByCommodityCount(DateTime from, DateTime to) { ImportNotificationType = n.ImportNotificationType!.Value, Linked = n.Relationships.Movements.Data.Count > 0, - CommodityCount = n.Commodities.Count() + + // ReSharper disable once ConditionIsAlwaysTrueOrFalseAccordingToNullableAPIContract + // This is not nullable, but is being represented as null in mongo and so this linq + // query needs to consider the null + CommodityCount = n.Commodities == null ? 0 : n.Commodities.Count() }) .Select(g => new { g.Key, Count = g.Count() }); @@ -108,7 +112,20 @@ public Task ByCommodityCount(DateTime from, DateTime to) }); } - private Task Aggregate(DateTime[] dateRange, Func createDatasetName, Expression> filter, string dateField, AggregationPeriod aggregateBy) + public Task ByVersionCount(DateTime from, DateTime to, AggregationPeriod aggregateBy = AggregationPeriod.Day) + { + var dateRange = AnalyticsHelpers.CreateDateRange(from, to, aggregateBy); + + Expression> matchFilter = n => + n.CreatedSource >= from && n.CreatedSource < to; + + string CreateDatasetName(BsonDocument b) => + AnalyticsHelpers.GetLinkedName(b["_id"]["linked"].ToBoolean(), b["_id"]["importNotificationType"].ToString()!.FromImportNotificationTypeEnumString()); + + return AggregateByLinkedAndNotificationType(dateRange, CreateDatasetName, matchFilter, "$createdSource", aggregateBy); + } + + private Task AggregateByLinkedAndNotificationType(DateTime[] dateRange, Func createDatasetName, Expression> filter, string dateField, AggregationPeriod aggregateBy) { var truncateBy = aggregateBy == AggregationPeriod.Hour ? "hour" : "day"; diff --git a/Btms.Backend/Config/AnalyticsDashboards.cs b/Btms.Backend/Config/AnalyticsDashboards.cs index ef4c1c09..2c8244e0 100644 --- a/Btms.Backend/Config/AnalyticsDashboards.cs +++ b/Btms.Backend/Config/AnalyticsDashboards.cs @@ -73,6 +73,10 @@ public static async Task> GetCharts( { "lastMonthDecisionsByStatus", () => movementsService.ByDecision(DateTime.Today.MonthAgo(), DateTime.Now).AsIDataset() + }, + { + "importNotificationVersionsByVersionCount", + () => importService.ByVersionCount(DateTime.Today.AddMonths(-3), DateTime.Today).AsIDataset() } }; diff --git a/Btms.Model/Movement.cs b/Btms.Model/Movement.cs index 0b70088e..80f555d7 100644 --- a/Btms.Model/Movement.cs +++ b/Btms.Model/Movement.cs @@ -137,10 +137,14 @@ public bool MergeDecision(string path, AlvsClearanceRequest clearanceRequest) } var decisionAuditContext = new Dictionary>(); - decisionAuditContext.Add("movements", new Dictionary() + decisionAuditContext.Add("clearanceRequests", new Dictionary() { { clearanceRequest.Header!.EntryReference!, clearanceRequest.Header!.EntryVersionNumber!.ToString()! } }); + decisionAuditContext.Add("decisions", new Dictionary() + { + { clearanceRequest.Header!.EntryReference!, clearanceRequest.Header!.DecisionNumber!.ToString()! } + }); decisionAuditContext.Add("importNotifications", new Dictionary() { { "todo", "todo" } From c04e3bc15040caf44d517f475e91f10fdb4143ae Mon Sep 17 00:00:00 2001 From: Craig Edmunds Date: Thu, 19 Dec 2024 12:06:32 +0000 Subject: [PATCH 2/6] Decision analytics by system and decision code working --- .../Helpers/MultiSeriesDatasetAssertions.cs | 2 +- .../MovementsByDecisionsTests.cs | 8 +- ...MovementsByUniqueDocumentReferenceTests.cs | 2 +- Btms.Analytics/{Results.cs => Dataset.cs} | 70 +++++------------- Btms.Analytics/DatasetDimensions.cs | 74 +++++++++++++++++++ .../IMovementsAggregationService.cs | 2 +- .../ImportNotificationsAggregationService.cs | 6 +- Btms.Analytics/MovementsAggregationService.cs | 59 +++++++++++---- Btms.Backend/Config/AnalyticsDashboards.cs | 2 +- Btms.Backend/Endpoints/AnalyticsEndpoints.cs | 3 +- 10 files changed, 153 insertions(+), 75 deletions(-) rename Btms.Analytics/{Results.cs => Dataset.cs} (62%) create mode 100644 Btms.Analytics/DatasetDimensions.cs diff --git a/Btms.Analytics.Tests/Helpers/MultiSeriesDatasetAssertions.cs b/Btms.Analytics.Tests/Helpers/MultiSeriesDatasetAssertions.cs index 5c1df55a..4f4ce7e5 100644 --- a/Btms.Analytics.Tests/Helpers/MultiSeriesDatasetAssertions.cs +++ b/Btms.Analytics.Tests/Helpers/MultiSeriesDatasetAssertions.cs @@ -19,7 +19,7 @@ public void BeSameLength(string because = "", params object[] becauseArgs) [CustomAssertion] public void HaveResults(string because = "", params object[] becauseArgs) { - test!.Sum(d => d.Results.Sum(r => r.Value)) + test!.Sum(d => d.Results.Sum(r => ((ByNumericDimensionResult)r).Value)) .Should().BeGreaterThan(0); } } \ No newline at end of file diff --git a/Btms.Analytics.Tests/MovementsByDecisionsTests.cs b/Btms.Analytics.Tests/MovementsByDecisionsTests.cs index 5e0e312b..37adfa2e 100644 --- a/Btms.Analytics.Tests/MovementsByDecisionsTests.cs +++ b/Btms.Analytics.Tests/MovementsByDecisionsTests.cs @@ -20,13 +20,13 @@ public async Task WhenCalled_ReturnExpectedAggregation() testOutputHelper.WriteLine("Querying for aggregated data"); var result = (await multiItemDataTestFixture.GetMovementsAggregationService(testOutputHelper) .ByDecision(DateTime.Today.MonthAgo(), DateTime.Today.Tomorrow())) - .Values + .Series .ToList(); testOutputHelper.WriteLine("{0} aggregated items found", result.Count); - result.Count.Should().Be(4); - result.Select(r => r.Key).Order().Should() - .Equal("ALVS Linked : H01", "BTMS Linked : C03", "BTMS Linked : X00", "BTMS Not Linked : X00"); + result.Count.Should().Be(2); + // result.Select(r => r.Key).Order().Should() + // .Equal("ALVS Linked : H01", "BTMS Linked : C03", "BTMS Linked : X00", "BTMS Not Linked : X00"); } } \ No newline at end of file diff --git a/Btms.Analytics.Tests/MovementsByUniqueDocumentReferenceTests.cs b/Btms.Analytics.Tests/MovementsByUniqueDocumentReferenceTests.cs index f7a17e28..507aabdf 100644 --- a/Btms.Analytics.Tests/MovementsByUniqueDocumentReferenceTests.cs +++ b/Btms.Analytics.Tests/MovementsByUniqueDocumentReferenceTests.cs @@ -32,7 +32,7 @@ public async Task WhenCalledLastWeek_ReturnExpectedAggregation() { d.Dimension.Should().Be("Document Reference Count"); d.Results.Count().Should().NotBe(0); - d.Results.Sum(r => r.Value).Should().BeGreaterThan(0); + d.Results.Sum(r => ((ByNumericDimensionResult)r).Value).Should().BeGreaterThan(0); }); result.Should().HaveResults(); diff --git a/Btms.Analytics/Results.cs b/Btms.Analytics/Dataset.cs similarity index 62% rename from Btms.Analytics/Results.cs rename to Btms.Analytics/Dataset.cs index dc228a2f..64fdd0df 100644 --- a/Btms.Analytics/Results.cs +++ b/Btms.Analytics/Dataset.cs @@ -1,27 +1,38 @@ using System.Diagnostics.CodeAnalysis; using System.Text.Json; using System.Text.Json.Serialization; -using Btms.Model.Auditing; namespace Btms.Analytics; -public class ByDateTimeResult +// A marker interface to identify things we want to be able to return from the analytics API +public interface IDataset; + + +public class MultiSeriesDataset : IDataset { - public DateTime Period { get; set; } - public int Value { get; set; } + public List Series { get; set; } = []; } -public class ByNumericDimensionResult +public class SingleSeriesDataset : IDataset { - public int Dimension { get; set; } - public int Value { get; set; } + public IDictionary Values { get; set; } = new Dictionary(); +} + +public class EntityDataset(IEnumerable items) : IDataset +{ + public IEnumerable Items { get; set; } = items; +} + +public class MultiSeriesDatetimeDataset : IDataset +{ + public List Series { get; set; } = []; } /// /// Serialise the derived types of IDataset /// /// -public class ResultTypeMappingConverter : JsonConverter where TType : IDataset +public class DatasetResultTypeMappingConverter : JsonConverter where TType : IDataset { [return: MaybeNull] public override TType Read( @@ -47,47 +58,4 @@ public override void Write(Utf8JsonWriter writer, TType value, JsonSerializerOpt throw new NotImplementedException(); } } -} - -// A marker interface to identify things we want to be able to return from the analytics API -public interface IDataset; - -public class SingleSeriesDataset : IDataset -{ - public IDictionary Values { get; set; } = new Dictionary(); -} - -public class AuditHistory(AuditEntry auditEntry, string resourceType, string resourceId) -{ - public AuditEntry AuditEntry { get; set; } = auditEntry; - public string ResourceType { get; set; } = resourceType; - public string ResourceId { get; set; } = resourceId; -} - -public class EntityDataset(IEnumerable items) : IDataset -{ - public IEnumerable Items { get; set; } = items; -} - -public class MultiSeriesDatetimeDataset : IDataset -{ - public List Series { get; set; } = []; -} - -public class DatetimeSeries(string name) -{ - public string Name { get; set; } = name; - public List Periods { get; set; } = []; -} - -public class MultiSeriesDataset : IDataset -{ - public List Series { get; set; } = []; -} - -public class Series(string name, string dimension) -{ - public string Name { get; set; } = name; - public string Dimension { get; set; } = dimension; - public List Results { get; set; } = []; } \ No newline at end of file diff --git a/Btms.Analytics/DatasetDimensions.cs b/Btms.Analytics/DatasetDimensions.cs new file mode 100644 index 00000000..257ddc86 --- /dev/null +++ b/Btms.Analytics/DatasetDimensions.cs @@ -0,0 +1,74 @@ +using System.Diagnostics.CodeAnalysis; +using System.Text.Json; +using System.Text.Json.Serialization; +using Btms.Model.Auditing; + +namespace Btms.Analytics; + +public class ByDateTimeResult +{ + public DateTime Period { get; set; } + public int Value { get; set; } +} + +public interface IDimensionResult; + +public class ByNumericDimensionResult : IDimensionResult +{ + public int Dimension { get; set; } + public int Value { get; set; } +} + +public class ByNameDimensionResult : IDimensionResult +{ + public required string Name { get; set; } + public int Value { get; set; } +} + +public class AuditHistory(AuditEntry auditEntry, string resourceType, string resourceId) +{ + public AuditEntry AuditEntry { get; set; } = auditEntry; + public string ResourceType { get; set; } = resourceType; + public string ResourceId { get; set; } = resourceId; +} + +public class DatetimeSeries(string name) +{ + public string Name { get; set; } = name; + public List Periods { get; set; } = []; +} + +public class Series +{ + public required string Name { get; set; } + public required string Dimension { get; set; } + public required List Results { get; set; } +} + +/// +/// Serialise the derived types of IDimensionResult +/// +/// +public class DimensionResultTypeMappingConverter : JsonConverter where TType : IDimensionResult +{ + [return: MaybeNull] + public override TType Read( + ref Utf8JsonReader reader, Type typeToConvert, JsonSerializerOptions options) => + throw new NotImplementedException(); + + public override void Write(Utf8JsonWriter writer, TType value, JsonSerializerOptions options) + { + if (value is ByNumericDimensionResult) + { + JsonSerializer.Serialize(writer, value as ByNumericDimensionResult, options); + } + else if (value is ByNameDimensionResult) + { + JsonSerializer.Serialize(writer, value as ByNameDimensionResult, options); + } + else + { + throw new NotImplementedException(); + } + } +} \ No newline at end of file diff --git a/Btms.Analytics/IMovementsAggregationService.cs b/Btms.Analytics/IMovementsAggregationService.cs index 948c8ec9..df9499ce 100644 --- a/Btms.Analytics/IMovementsAggregationService.cs +++ b/Btms.Analytics/IMovementsAggregationService.cs @@ -7,7 +7,7 @@ public interface IMovementsAggregationService public Task ByCreated(DateTime from, DateTime to, AggregationPeriod aggregateBy = AggregationPeriod.Day); public Task ByStatus(DateTime from, DateTime to); public Task ByItemCount(DateTime from, DateTime to); - public Task ByDecision(DateTime from, DateTime to); + public Task ByDecision(DateTime from, DateTime to); public Task ByUniqueDocumentReferenceCount(DateTime from, DateTime to); public Task UniqueDocumentReferenceByMovementCount(DateTime from, DateTime to); public Task ByCheck(DateTime from, DateTime to); diff --git a/Btms.Analytics/ImportNotificationsAggregationService.cs b/Btms.Analytics/ImportNotificationsAggregationService.cs index 852f0a38..b7178d3b 100644 --- a/Btms.Analytics/ImportNotificationsAggregationService.cs +++ b/Btms.Analytics/ImportNotificationsAggregationService.cs @@ -99,14 +99,16 @@ public Task ByCommodityCount(DateTime from, DateTime to) return Task.FromResult(new MultiSeriesDataset() { Series = AnalyticsHelpers.GetImportNotificationSegments() - .Select(title => new Series(title, "ItemCount") + .Select(title => new Series() { + Name = title, + Dimension = "ItemCount", Results = Enumerable.Range(0, maxCommodities) .Select(i => new ByNumericDimensionResult { Dimension = i, Value = asDictionary.GetValueOrDefault(new { Title=title, CommodityCount = i }) - }).ToList() + }).ToList() }) .ToList() }); diff --git a/Btms.Analytics/MovementsAggregationService.cs b/Btms.Analytics/MovementsAggregationService.cs index 9b14f6a5..0733da5d 100644 --- a/Btms.Analytics/MovementsAggregationService.cs +++ b/Btms.Analytics/MovementsAggregationService.cs @@ -68,14 +68,16 @@ public Task ByItemCount(DateTime from, DateTime to) return Task.FromResult(new MultiSeriesDataset() { Series = AnalyticsHelpers.GetMovementSegments() - .Select(title => new Series(title, "Item Count") + .Select(title => new Series() { + Name = title, + Dimension = "Item Count", Results = Enumerable.Range(0, maxCount + 1) .Select(i => new ByNumericDimensionResult { Dimension = i, Value = dictionary.GetValueOrDefault(new { Title=title, ItemCount = i }, 0) - }).ToList() + }).ToList() } ) .ToList() @@ -110,15 +112,17 @@ public Task ByUniqueDocumentReferenceCount(DateTime from, Da return Task.FromResult(new MultiSeriesDataset() { Series = AnalyticsHelpers.GetMovementSegments() - .Select(title => new Series(title, "Document Reference Count") + .Select(title => new Series() { + Name = title, + Dimension = "Document Reference Count", Results = Enumerable.Range(0, maxReferences + 1) .Select(i => new ByNumericDimensionResult { Dimension = i, Value = dictionary.GetValueOrDefault(new { Title = title, DocumentReferenceCount = i }, 0) - }).ToList() + }).ToList() }) .ToList() }); @@ -209,7 +213,7 @@ private Task Aggregate(DateTime[] dateRange, Func ByDecision(DateTime from, DateTime to) + public Task ByDecision(DateTime from, DateTime to) { var mongoQuery = context .Movements @@ -229,20 +233,49 @@ public Task ByDecision(DateTime from, DateTime to) HasLinks = m.Movement.Relationships.Notifications.Data.Count > 0, ItemNumber = m.Item.ItemNumber })) - .GroupBy(m => new { m.HasLinks, m.DecisionSourceSystem, m.DecisionCode }) - .Select(m => new { m.Key.HasLinks, m.Key.DecisionSourceSystem, m.Key.DecisionCode, Count = m.Count() }) + .GroupBy(m => new + { + m.HasLinks, m.DecisionSourceSystem, m.DecisionCode + }) + + .Select(g => new { g.Key.DecisionSourceSystem, g.Key.DecisionCode, g.Key.HasLinks, Count = g.Count()}) + + .GroupBy(m => new { m.DecisionSourceSystem, m.HasLinks }) + + // For debugging + // .Select(g => new { g.Key.DecisionSourceSystem, Count = g.Count()}) + // .Select(g => new { g.Key.DecisionSourceSystem, Decisions = g.Select(v => v.DecisionCode ) }) + + .Select(m => new { m.Key.DecisionSourceSystem, m.Key.HasLinks, Decisions = m.Select(v => new { v.DecisionCode, Count = v.Count }) }) + .Execute(logger) .ToList(); logger.LogInformation("Found {0} items", mongoQuery.Count); logger.LogInformation(mongoQuery.ToJsonString()); - return Task.FromResult(new SingleSeriesDataset() + return Task.FromResult(new MultiSeriesDataset() { - Values = mongoQuery - .ToDictionary( - r => $"{ r.DecisionSourceSystem } { ( r.HasLinks ? "Linked" : "Not Linked" ) } : { r.DecisionCode }", - r => r.Count - ) + // Series = new List() {} + Series = mongoQuery.Select(a => + new Series() + { + Name = $"{ a.DecisionSourceSystem } : { ( a.HasLinks ? "Linked" : "Not Linked" ) }", + Dimension = "DecisionCount", + Results = a.Decisions.Select(d => new ByNameDimensionResult() + { + Name = d.DecisionCode!, + Value = d.Count + }) + .OrderBy(r => r.Name) + .ToList() + }).ToList() + // new Series("Testing 12", "Decisions")).ToList() + // .ToDictionary( + // Values = mongoQuery + // .ToDictionary( + // r => $"{ r.DecisionSourceSystem } { ( r.HasLinks ? "Linked" : "Not Linked" ) } : { r.DecisionCode }", + // r => r.Count + // ) }); } } \ No newline at end of file diff --git a/Btms.Backend/Config/AnalyticsDashboards.cs b/Btms.Backend/Config/AnalyticsDashboards.cs index 2c8244e0..ae1a925c 100644 --- a/Btms.Backend/Config/AnalyticsDashboards.cs +++ b/Btms.Backend/Config/AnalyticsDashboards.cs @@ -71,7 +71,7 @@ public static async Task> GetCharts( () => importService.ByCommodityCount(DateTime.Today.MonthAgo(), DateTime.Now).AsIDataset() }, { - "lastMonthDecisionsByStatus", + "lastMonthsDecisionsByDecisionCode", () => movementsService.ByDecision(DateTime.Today.MonthAgo(), DateTime.Now).AsIDataset() }, { diff --git a/Btms.Backend/Endpoints/AnalyticsEndpoints.cs b/Btms.Backend/Endpoints/AnalyticsEndpoints.cs index ac42f21d..589eb8c7 100644 --- a/Btms.Backend/Endpoints/AnalyticsEndpoints.cs +++ b/Btms.Backend/Endpoints/AnalyticsEndpoints.cs @@ -54,7 +54,8 @@ private static async Task GetDashboard( PropertyNamingPolicy = JsonNamingPolicy.CamelCase, Converters = { - new ResultTypeMappingConverter() + new DatasetResultTypeMappingConverter(), + new DimensionResultTypeMappingConverter() } }; From 5009c07a9ff47ef6a64158f5126a1acfe65fcbe0 Mon Sep 17 00:00:00 2001 From: Craig Edmunds Date: Thu, 19 Dec 2024 15:26:21 +0000 Subject: [PATCH 3/6] lastMonthsDecisionsByDecisionCode Chart that doesn't seperate linked vs not linked, and added totals --- Btms.Analytics/Dataset.cs | 9 + Btms.Analytics/DatasetDimensions.cs | 6 + .../Extensions/AnalyticsExtensions.cs | 53 ++++++ .../IMovementsAggregationService.cs | 3 +- Btms.Analytics/MovementsAggregationService.cs | 157 +++++++++++++----- 5 files changed, 189 insertions(+), 39 deletions(-) diff --git a/Btms.Analytics/Dataset.cs b/Btms.Analytics/Dataset.cs index 64fdd0df..5fc82fd6 100644 --- a/Btms.Analytics/Dataset.cs +++ b/Btms.Analytics/Dataset.cs @@ -18,6 +18,11 @@ public class SingleSeriesDataset : IDataset public IDictionary Values { get; set; } = new Dictionary(); } +public class TabularDataset : IDataset where TColumn : IDimensionResult +{ + public required List> Rows { get; set; } +} + public class EntityDataset(IEnumerable items) : IDataset { public IEnumerable Items { get; set; } = items; @@ -53,6 +58,10 @@ public override void Write(Utf8JsonWriter writer, TType value, JsonSerializerOpt { JsonSerializer.Serialize(writer, value as SingleSeriesDataset, options); } + else if (value is TabularDataset) + { + JsonSerializer.Serialize(writer, value as TabularDataset, options); + } else { throw new NotImplementedException(); diff --git a/Btms.Analytics/DatasetDimensions.cs b/Btms.Analytics/DatasetDimensions.cs index 257ddc86..4d262fe5 100644 --- a/Btms.Analytics/DatasetDimensions.cs +++ b/Btms.Analytics/DatasetDimensions.cs @@ -13,6 +13,12 @@ public class ByDateTimeResult public interface IDimensionResult; +public class TabularDimensionRow where TColumn : IDimensionResult +{ + public required string Key { get; set; } + public required List Columns { get; set; } +} + public class ByNumericDimensionResult : IDimensionResult { public int Dimension { get; set; } diff --git a/Btms.Analytics/Extensions/AnalyticsExtensions.cs b/Btms.Analytics/Extensions/AnalyticsExtensions.cs index d63dbbde..946f147a 100644 --- a/Btms.Analytics/Extensions/AnalyticsExtensions.cs +++ b/Btms.Analytics/Extensions/AnalyticsExtensions.cs @@ -85,6 +85,16 @@ public static T[] AsOrderedArray(this IEnumerable en, Func .ToArray(); } + /// + /// Gives us an opportunity to hook into the mongo execution and + /// grab the query as a string after it's been executed + /// + /// + /// + /// + /// + /// + /// internal static IEnumerable> Execute(this IQueryable> source, ILogger logger) { try @@ -103,6 +113,43 @@ internal static IEnumerable> Execute(thi } } + /// + /// Gives us an opportunity to hook into the mongo execution and + /// grab the query as a string after it's been executed + /// + /// + /// + /// + /// + /// + /// + /// + /// + /// + public static Dictionary ExecuteAsDictionary( + this IEnumerable source, + ILogger logger, + Func keySelector, + Func elementSelector) + where TKey : notnull + { + try + { + var aggregatedData = source + .ToDictionary(keySelector, elementSelector); + return aggregatedData; + } + catch(Exception ex) + { + logger.LogError(ex, "Error querying Mongo : {Message}", ex.Message); + throw new AnalyticsException("Error querying Mongo", ex); + } + finally + { + logger.LogExecutedMongoString((IQueryable)source); + } + } + internal static IEnumerable Execute( this IQueryable source, ILogger logger) { @@ -147,6 +194,12 @@ public static async Task AsIDataset(this Task ms) await ms; return (IDataset)ms.Result; } + + public static async Task AsIDataset(this Task> ms) + { + await ms; + return (IDataset)ms.Result; + } public static async Task AsIDataset(this Task ms) { diff --git a/Btms.Analytics/IMovementsAggregationService.cs b/Btms.Analytics/IMovementsAggregationService.cs index df9499ce..234ece44 100644 --- a/Btms.Analytics/IMovementsAggregationService.cs +++ b/Btms.Analytics/IMovementsAggregationService.cs @@ -7,7 +7,8 @@ public interface IMovementsAggregationService public Task ByCreated(DateTime from, DateTime to, AggregationPeriod aggregateBy = AggregationPeriod.Day); public Task ByStatus(DateTime from, DateTime to); public Task ByItemCount(DateTime from, DateTime to); - public Task ByDecision(DateTime from, DateTime to); + public Task> ByDecision(DateTime from, DateTime to); + public Task> ByDecisionAndLinkStatus(DateTime from, DateTime to); public Task ByUniqueDocumentReferenceCount(DateTime from, DateTime to); public Task UniqueDocumentReferenceByMovementCount(DateTime from, DateTime to); public Task ByCheck(DateTime from, DateTime to); diff --git a/Btms.Analytics/MovementsAggregationService.cs b/Btms.Analytics/MovementsAggregationService.cs index 0733da5d..7be7027a 100644 --- a/Btms.Analytics/MovementsAggregationService.cs +++ b/Btms.Analytics/MovementsAggregationService.cs @@ -8,6 +8,7 @@ using Btms.Model.Auditing; using MongoDB.Bson; using MongoDB.Driver; +using MongoDB.Driver.Linq; namespace Btms.Analytics; @@ -213,7 +214,7 @@ private Task Aggregate(DateTime[] dateRange, Func ByDecision(DateTime from, DateTime to) + public Task> ByDecision(DateTime from, DateTime to) { var mongoQuery = context .Movements @@ -233,49 +234,129 @@ public Task ByDecision(DateTime from, DateTime to) HasLinks = m.Movement.Relationships.Notifications.Data.Count > 0, ItemNumber = m.Item.ItemNumber })) - .GroupBy(m => new + .GroupBy(m => new { m.DecisionSourceSystem, m.DecisionCode }) + .ExecuteAsDictionary(logger, + g => new + { + DecisionSourceSystem = g.Key.DecisionSourceSystem!, + DecisionCode = g.Key.DecisionCode! + }, + g => g.Count()); + + logger.LogInformation("Found {0} items", mongoQuery.Count); + + var series = new[] + { + new { System = "ALVS" }, + new { System = "BTMS" } + }; + + var totals = series + .Select(s => new { + System = s.System, + Total = mongoQuery + .Where(a => a.Key.DecisionSourceSystem == s.System) + .Sum(a => a.Value) + }); + + var decisionCodes = mongoQuery + .Select(d => d.Key.DecisionCode) + .Order() + .Distinct(); + + var result = new TabularDataset() + { + Rows = decisionCodes.Select(d => new TabularDimensionRow() { - m.HasLinks, m.DecisionSourceSystem, m.DecisionCode - }) - - .Select(g => new { g.Key.DecisionSourceSystem, g.Key.DecisionCode, g.Key.HasLinks, Count = g.Count()}) - - .GroupBy(m => new { m.DecisionSourceSystem, m.HasLinks }) + Key = d, + Columns = series.Select(s => + new ByNameDimensionResult() + { + Name = s.System, + Value = mongoQuery.GetValueOrDefault(new + { + DecisionSourceSystem = s.System, DecisionCode = d + }, 0) + } + ).ToList() + }).ToList() + }; + + result.Rows.Add(new TabularDimensionRow() + { + Key = "Total", + Columns = totals + .Select(t => new ByNameDimensionResult() + { + Name = t.System, Value = t.Total + }).ToList() - // For debugging - // .Select(g => new { g.Key.DecisionSourceSystem, Count = g.Count()}) - // .Select(g => new { g.Key.DecisionSourceSystem, Decisions = g.Select(v => v.DecisionCode ) }) + }); - .Select(m => new { m.Key.DecisionSourceSystem, m.Key.HasLinks, Decisions = m.Select(v => new { v.DecisionCode, Count = v.Count }) }) - .Execute(logger) - .ToList(); - + return Task.FromResult(result); + } + + public Task> ByDecisionAndLinkStatus(DateTime from, DateTime to) + { + var mongoQuery = context + .Movements + .Where(m => m.CreatedSource >= from && m.CreatedSource < to) + .SelectMany(m => m.Decisions.Select(d => new { Decision = d, Movement = m })) + .SelectMany(m => + m.Decision.Items!.Select(i => new { Decision = m.Decision, Movement = m.Movement, Item = i })) + .SelectMany(m => m.Item.Checks!.Select(c => new + { + CheckCode = c.CheckCode, + DecisionCode = c.DecisionCode, + DecisionSourceSystem = m.Decision.ServiceHeader!.SourceSystem, + DecisionEntryReference = m.Decision.Header!.EntryReference, + DecisionEntryVersionNumber = m.Decision.Header!.EntryVersionNumber, + Movement = m.Movement.EntryReference, + MovementVersion = m.Movement.EntryVersionNumber, + HasLinks = m.Movement.Relationships.Notifications.Data.Count > 0, + ItemNumber = m.Item.ItemNumber + })) + .GroupBy(m => new { m.HasLinks, m.DecisionSourceSystem, m.DecisionCode }) + .ExecuteAsDictionary(logger, + g => new + { + DecisionSourceSystem = g.Key.DecisionSourceSystem!, + DecisionCode = g.Key.DecisionCode!, + g.Key.HasLinks + }, + g => g.Count()); + logger.LogInformation("Found {0} items", mongoQuery.Count); - logger.LogInformation(mongoQuery.ToJsonString()); + + var series = new[] + { + new { System = "ALVS", Linked = true }, new { System = "ALVS", Linked = false }, + new { System = "BTMS", Linked = true }, new { System = "BTMS", Linked = false } + }; - return Task.FromResult(new MultiSeriesDataset() + var decisionCodes = mongoQuery + .Select(d => d.Key.DecisionCode) + .Order() + .Distinct(); + + var result = new TabularDataset() { - // Series = new List() {} - Series = mongoQuery.Select(a => - new Series() - { - Name = $"{ a.DecisionSourceSystem } : { ( a.HasLinks ? "Linked" : "Not Linked" ) }", - Dimension = "DecisionCount", - Results = a.Decisions.Select(d => new ByNameDimensionResult() + Rows = decisionCodes.Select(d => new TabularDimensionRow() + { + Key = d, + Columns = series.Select(s => + new ByNameDimensionResult() { - Name = d.DecisionCode!, - Value = d.Count - }) - .OrderBy(r => r.Name) - .ToList() - }).ToList() - // new Series("Testing 12", "Decisions")).ToList() - // .ToDictionary( - // Values = mongoQuery - // .ToDictionary( - // r => $"{ r.DecisionSourceSystem } { ( r.HasLinks ? "Linked" : "Not Linked" ) } : { r.DecisionCode }", - // r => r.Count - // ) - }); + Name = $"{ s.System } : { ( s.Linked ? "Linked" : "Not Linked" ) }", + Value = mongoQuery.GetValueOrDefault(new + { + DecisionSourceSystem = s.System, DecisionCode = d, HasLinks = s.Linked + }, 0) + } + ).ToList() + }).ToList() + }; + + return Task.FromResult(result); } } \ No newline at end of file From 1178c235d95ca6225d2a69a90968bddd9a878e94 Mon Sep 17 00:00:00 2001 From: Craig Edmunds Date: Fri, 20 Dec 2024 08:20:11 +0000 Subject: [PATCH 4/6] First pass at decision analytics working --- .../ImportNotificationsByVersionTests.cs | 54 +++ .../MovementsByDecisionsTests.cs | 5 +- .../MovementsExceptionsTests.cs | 31 ++ Btms.Analytics/Dataset.cs | 12 + Btms.Analytics/DatasetDimensions.cs | 24 ++ .../Extensions/AnalyticsExtensions.cs | 74 ++++ .../IImportNotificationsAggregationService.cs | 2 +- .../IMovementsAggregationService.cs | 8 +- .../ImportNotificationsAggregationService.cs | 23 +- Btms.Analytics/MovementsAggregationService.cs | 372 ++++++++++++------ Btms.Backend/Config/AnalyticsDashboards.cs | 12 +- Btms.Backend/Endpoints/AnalyticsEndpoints.cs | 14 + 12 files changed, 488 insertions(+), 143 deletions(-) create mode 100644 Btms.Analytics.Tests/ImportNotificationsByVersionTests.cs create mode 100644 Btms.Analytics.Tests/MovementsExceptionsTests.cs diff --git a/Btms.Analytics.Tests/ImportNotificationsByVersionTests.cs b/Btms.Analytics.Tests/ImportNotificationsByVersionTests.cs new file mode 100644 index 00000000..936990bc --- /dev/null +++ b/Btms.Analytics.Tests/ImportNotificationsByVersionTests.cs @@ -0,0 +1,54 @@ +using Btms.Common.Extensions; +using FluentAssertions; +using Xunit; +using Xunit.Abstractions; + +using Btms.Analytics.Tests.Fixtures; + +namespace Btms.Analytics.Tests; + +[Collection(nameof(BasicSampleDataTestCollection))] +public class ImportNotificationsByVersionTests( + BasicSampleDataTestFixture basicSampleDataTestFixture, + ITestOutputHelper testOutputHelper) +{ + + [Fact] + public async Task WhenCalledLastMonth_ReturnExpectedAggregation() + { + testOutputHelper.WriteLine("Querying for aggregated data"); + var result = (await basicSampleDataTestFixture.GetImportNotificationsAggregationService(testOutputHelper) + .ByMaxVersion(DateTime.Today.MonthAgo(), DateTime.Today.Tomorrow())); + + testOutputHelper.WriteLine("{0} aggregated items found", result.Values.Count); + + result.Values.Count.Should().Be(2); + } + + [Fact] + public async Task WhenCalledLast48Hours_ReturnExpectedAggregation() + { + testOutputHelper.WriteLine("Querying for aggregated data"); + var result = (await basicSampleDataTestFixture.GetImportNotificationsAggregationService(testOutputHelper) + .ByStatus(DateTime.Now.NextHour().AddDays(-2), DateTime.Now.NextHour())); + + testOutputHelper.WriteLine($"{result.Values.Count} aggregated items found"); + + result.Values.Count.Should().Be(8); + result.Values.Keys.Order().Should().Equal( + "CHEDA Linked", "CHEDA Not Linked", "CHEDD Linked", "CHEDD Not Linked", "CHEDP Linked", "CHEDP Not Linked", "CHEDPP Linked", "CHEDPP Not Linked"); + + } + + [Fact] + public async Task WhenCalledWithTimePeriodYieldingNoResults_ReturnEmptyAggregation() + { + testOutputHelper.WriteLine("Querying for aggregated data"); + var result = (await basicSampleDataTestFixture.GetImportNotificationsAggregationService(testOutputHelper) + .ByStatus(DateTime.MaxValue.AddDays(-1), DateTime.MaxValue)); + + testOutputHelper.WriteLine($"{result.Values.Count} aggregated items found"); + + result.Values.Count.Should().Be(8); + } +} \ No newline at end of file diff --git a/Btms.Analytics.Tests/MovementsByDecisionsTests.cs b/Btms.Analytics.Tests/MovementsByDecisionsTests.cs index 37adfa2e..e8ee5474 100644 --- a/Btms.Analytics.Tests/MovementsByDecisionsTests.cs +++ b/Btms.Analytics.Tests/MovementsByDecisionsTests.cs @@ -20,12 +20,11 @@ public async Task WhenCalled_ReturnExpectedAggregation() testOutputHelper.WriteLine("Querying for aggregated data"); var result = (await multiItemDataTestFixture.GetMovementsAggregationService(testOutputHelper) .ByDecision(DateTime.Today.MonthAgo(), DateTime.Today.Tomorrow())) - .Series - .ToList(); + .Rows; testOutputHelper.WriteLine("{0} aggregated items found", result.Count); - result.Count.Should().Be(2); + result.Count.Should().BeGreaterThan(1); // result.Select(r => r.Key).Order().Should() // .Equal("ALVS Linked : H01", "BTMS Linked : C03", "BTMS Linked : X00", "BTMS Not Linked : X00"); } diff --git a/Btms.Analytics.Tests/MovementsExceptionsTests.cs b/Btms.Analytics.Tests/MovementsExceptionsTests.cs new file mode 100644 index 00000000..1ab923b7 --- /dev/null +++ b/Btms.Analytics.Tests/MovementsExceptionsTests.cs @@ -0,0 +1,31 @@ +using Btms.Common.Extensions; +using FluentAssertions; +using Xunit; +using Xunit.Abstractions; + +using Btms.Analytics.Tests.Fixtures; +using Btms.Analytics.Tests.Helpers; + +namespace Btms.Analytics.Tests; + +[Collection(nameof(MultiItemDataTestCollection))] +public class MovementsExceptionsTests( + MultiItemDataTestFixture multiItemDataTestFixture, + ITestOutputHelper testOutputHelper) +{ + + [Fact] + public async Task WhenCalled_ReturnExpectedAggregation() + { + testOutputHelper.WriteLine("Querying for aggregated data"); + var result = (await multiItemDataTestFixture.GetMovementsAggregationService(testOutputHelper) + .ByDecision(DateTime.Today.MonthAgo(), DateTime.Today.Tomorrow())) + .Rows; + + testOutputHelper.WriteLine("{0} aggregated items found", result.Count); + + result.Count.Should().BeGreaterThan(1); + // result.Select(r => r.Key).Order().Should() + // .Equal("ALVS Linked : H01", "BTMS Linked : C03", "BTMS Linked : X00", "BTMS Not Linked : X00"); + } +} \ No newline at end of file diff --git a/Btms.Analytics/Dataset.cs b/Btms.Analytics/Dataset.cs index 5fc82fd6..299dd546 100644 --- a/Btms.Analytics/Dataset.cs +++ b/Btms.Analytics/Dataset.cs @@ -7,6 +7,14 @@ namespace Btms.Analytics; // A marker interface to identify things we want to be able to return from the analytics API public interface IDataset; +public class SummarisedDataset : IDataset + where TResult : IDimensionResult + where TSummary : IDimensionResult +{ + public required List Summary { get; set; } + public required List Result { get; set; } + +} public class MultiSeriesDataset : IDataset { @@ -62,6 +70,10 @@ public override void Write(Utf8JsonWriter writer, TType value, JsonSerializerOpt { JsonSerializer.Serialize(writer, value as TabularDataset, options); } + else if (value is SummarisedDataset) + { + JsonSerializer.Serialize(writer, value as SummarisedDataset, options); + } else { throw new NotImplementedException(); diff --git a/Btms.Analytics/DatasetDimensions.cs b/Btms.Analytics/DatasetDimensions.cs index 4d262fe5..6a0d4ca4 100644 --- a/Btms.Analytics/DatasetDimensions.cs +++ b/Btms.Analytics/DatasetDimensions.cs @@ -11,6 +11,20 @@ public class ByDateTimeResult public int Value { get; set; } } +public class ExceptionResult : IDimensionResult +{ + public required string Id { get; set; } + public required string Resource { get; set; } + public required DateTime UpdatedSource { get; set; } + public required DateTime Updated { get; set; } + + public required int ItemCount { get; set; } + public required int MaxEntryVersion { get; set; } + public required int MaxDecisionNumber { get; set; } + public required int LinkedCheds { get; set; } + public required string Reason { get; set; } +} + public interface IDimensionResult; public class TabularDimensionRow where TColumn : IDimensionResult @@ -31,6 +45,12 @@ public class ByNameDimensionResult : IDimensionResult public int Value { get; set; } } +public class StringBucketDimensionResult : IDimensionResult +{ + public required Dictionary Fields { get; set; } + public int Value { get; set; } +} + public class AuditHistory(AuditEntry auditEntry, string resourceType, string resourceId) { public AuditEntry AuditEntry { get; set; } = auditEntry; @@ -72,6 +92,10 @@ public override void Write(Utf8JsonWriter writer, TType value, JsonSerializerOpt { JsonSerializer.Serialize(writer, value as ByNameDimensionResult, options); } + else if (value is ByNameDimensionResult) + { + JsonSerializer.Serialize(writer, value as ByNameDimensionResult, options); + } else { throw new NotImplementedException(); diff --git a/Btms.Analytics/Extensions/AnalyticsExtensions.cs b/Btms.Analytics/Extensions/AnalyticsExtensions.cs index 946f147a..afacd5f0 100644 --- a/Btms.Analytics/Extensions/AnalyticsExtensions.cs +++ b/Btms.Analytics/Extensions/AnalyticsExtensions.cs @@ -1,3 +1,4 @@ +using System.Collections.Immutable; using Btms.Backend.Data; using Btms.Model.Data; using Microsoft.Extensions.Configuration; @@ -113,6 +114,35 @@ internal static IEnumerable> Execute(thi } } + /// + /// Gives us an opportunity to hook into the mongo execution and + /// grab the query as a string after it's been executed + /// + /// + /// + /// + /// + /// + /// + internal static IEnumerable Execute(this IAggregateFluent source, ILogger logger) + { + try + { + var aggregatedData = source.ToList(); + return aggregatedData; + } + catch(Exception ex) + { + logger.LogError(ex, "Error querying Mongo : {Message}", ex.Message); + throw new AnalyticsException("Error querying Mongo", ex); + } + finally + { + logger.LogInformation("Query from IAggregateFluent"); + // logger.LogExecutedMongoString((IQueryable)source); + } + } + /// /// Gives us an opportunity to hook into the mongo execution and /// grab the query as a string after it's been executed @@ -149,6 +179,42 @@ public static Dictionary ExecuteAsDictionary + /// Gives us an opportunity to hook into the mongo execution and + /// grab the query as a string after it's been executed + /// + /// + /// + /// + /// + /// + /// + /// + /// + /// + public static ImmutableSortedDictionary ExecuteAsSortedDictionary( + this IEnumerable source, + ILogger logger, + Func keySelector, + Func elementSelector) + where TKey : notnull + { + try + { + var aggregatedData = source + .ToImmutableSortedDictionary(keySelector, elementSelector); + return aggregatedData; + } + catch(Exception ex) + { + logger.LogError(ex, "Error querying Mongo : {Message}", ex.Message); + throw new AnalyticsException("Error querying Mongo", ex); + } + finally + { + logger.LogExecutedMongoString((IQueryable)source); + } + } internal static IEnumerable Execute( this IQueryable source, ILogger logger) @@ -206,4 +272,12 @@ public static async Task AsIDataset(this Task ms) await ms; return (IDataset)ms.Result; } + + public static async Task AsIDataset(this Task> ms) + where TResult : IDimensionResult + where TSummary : IDimensionResult + { + await ms; + return (IDataset)ms.Result; + } } \ No newline at end of file diff --git a/Btms.Analytics/IImportNotificationsAggregationService.cs b/Btms.Analytics/IImportNotificationsAggregationService.cs index 51faac40..4ccdee3a 100644 --- a/Btms.Analytics/IImportNotificationsAggregationService.cs +++ b/Btms.Analytics/IImportNotificationsAggregationService.cs @@ -6,5 +6,5 @@ public interface IImportNotificationsAggregationService public Task ByArrival(DateTime from, DateTime to, AggregationPeriod aggregateBy = AggregationPeriod.Day); public Task ByStatus(DateTime from, DateTime to); public Task ByCommodityCount(DateTime from, DateTime to); - public Task ByVersionCount(DateTime from, DateTime to, AggregationPeriod aggregateBy = AggregationPeriod.Day); + public Task ByMaxVersion(DateTime from, DateTime to); } \ No newline at end of file diff --git a/Btms.Analytics/IMovementsAggregationService.cs b/Btms.Analytics/IMovementsAggregationService.cs index 234ece44..d2aef5ba 100644 --- a/Btms.Analytics/IMovementsAggregationService.cs +++ b/Btms.Analytics/IMovementsAggregationService.cs @@ -7,10 +7,14 @@ public interface IMovementsAggregationService public Task ByCreated(DateTime from, DateTime to, AggregationPeriod aggregateBy = AggregationPeriod.Day); public Task ByStatus(DateTime from, DateTime to); public Task ByItemCount(DateTime from, DateTime to); - public Task> ByDecision(DateTime from, DateTime to); - public Task> ByDecisionAndLinkStatus(DateTime from, DateTime to); + public Task> ByDecision(DateTime from, DateTime to); + // public Task> ByDecisionAndLinkStatus(DateTime from, DateTime to); public Task ByUniqueDocumentReferenceCount(DateTime from, DateTime to); public Task UniqueDocumentReferenceByMovementCount(DateTime from, DateTime to); public Task ByCheck(DateTime from, DateTime to); public Task?> GetHistory(string movementId); + public Task ByMaxVersion(DateTime from, DateTime to); + public Task ByMaxDecisionNumber(DateTime from, DateTime to); + public Task> GetExceptions(DateTime from, DateTime to); + } \ No newline at end of file diff --git a/Btms.Analytics/ImportNotificationsAggregationService.cs b/Btms.Analytics/ImportNotificationsAggregationService.cs index b7178d3b..f0836946 100644 --- a/Btms.Analytics/ImportNotificationsAggregationService.cs +++ b/Btms.Analytics/ImportNotificationsAggregationService.cs @@ -7,6 +7,7 @@ using MongoDB.Driver; using Btms.Analytics.Extensions; +using MongoDB.Driver.Linq; namespace Btms.Analytics; @@ -114,17 +115,21 @@ public Task ByCommodityCount(DateTime from, DateTime to) }); } - public Task ByVersionCount(DateTime from, DateTime to, AggregationPeriod aggregateBy = AggregationPeriod.Day) + public Task ByMaxVersion(DateTime from, DateTime to) { - var dateRange = AnalyticsHelpers.CreateDateRange(from, to, aggregateBy); - - Expression> matchFilter = n => - n.CreatedSource >= from && n.CreatedSource < to; - - string CreateDatasetName(BsonDocument b) => - AnalyticsHelpers.GetLinkedName(b["_id"]["linked"].ToBoolean(), b["_id"]["importNotificationType"].ToString()!.FromImportNotificationTypeEnumString()); + var data = context + .Notifications + .Where(n => n.CreatedSource >= from && n.CreatedSource < to) + .GroupBy(n => new { MaxVersion = + n.AuditEntries.Where(a => a.CreatedBy == "Ipaffs").Max(a => a.Version ) + }) + .Select(g => new { MaxVersion = g.Key.MaxVersion ?? 0, Count = g.Count() }) + .ExecuteAsSortedDictionary(logger, g => g.MaxVersion, g => g.Count); - return AggregateByLinkedAndNotificationType(dateRange, CreateDatasetName, matchFilter, "$createdSource", aggregateBy); + return Task.FromResult(new SingleSeriesDataset + { + Values = data.ToDictionary(d => d.Key.ToString(), d => d.Value) + }); } private Task AggregateByLinkedAndNotificationType(DateTime[] dateRange, Func createDatasetName, Expression> filter, string dateField, AggregationPeriod aggregateBy) diff --git a/Btms.Analytics/MovementsAggregationService.cs b/Btms.Analytics/MovementsAggregationService.cs index 7be7027a..1f394dfa 100644 --- a/Btms.Analytics/MovementsAggregationService.cs +++ b/Btms.Analytics/MovementsAggregationService.cs @@ -5,7 +5,9 @@ using Btms.Common.Extensions; using Btms.Model.Extensions; using Btms.Model; +using Btms.Model.Alvs; using Btms.Model.Auditing; +using Btms.Model.Ipaffs; using MongoDB.Bson; using MongoDB.Driver; using MongoDB.Driver.Linq; @@ -184,6 +186,102 @@ public Task UniqueDocumentReferenceByMovementCount(DateTime return new EntityDataset(entries); } + public Task ByMaxVersion(DateTime from, DateTime to) + { + var data = context + .Movements + .Where(n => n.CreatedSource >= from && n.CreatedSource < to) + .GroupBy(n => new { MaxVersion = + n.ClearanceRequests.Max(a => a.Header!.EntryVersionNumber ) + }) + .Select(g => new { MaxVersion = g.Key.MaxVersion ?? 0, Count = g.Count() }) + .ExecuteAsSortedDictionary(logger, g => g.MaxVersion, g => g.Count); + + return Task.FromResult(new SingleSeriesDataset + { + Values = data.ToDictionary(d => d.Key.ToString(), d => d.Value) + }); + } + + public Task ByMaxDecisionNumber(DateTime from, DateTime to) + { + var data = context + .Movements + .Where(n => n.CreatedSource >= from && n.CreatedSource < to) + .GroupBy(n => new { MaxVersion = + n.Decisions.Max(a => a.Header!.DecisionNumber ) + }) + .Select(g => new { MaxVersion = g.Key.MaxVersion ?? 0, Count = g.Count() }) + .ExecuteAsSortedDictionary(logger, g => g.MaxVersion, g => g.Count); + + return Task.FromResult(new SingleSeriesDataset + { + Values = data.ToDictionary(d => d.Key.ToString(), d => d.Value) + }); + } + + public Task> GetExceptions(DateTime from, DateTime to) + { + var mongoQuery = context + .Movements + .Select(m => new + { + Id = m.Id, + UpdatedSource = m.UpdatedSource, + Updated = m.Updated, + MaxDecisionNumber = m.Decisions.Max(d => d.Header!.DecisionNumber) ?? 0, + MaxEntryVersion = m.ClearanceRequests.Max(c => c.Header!.EntryVersionNumber) ?? 0, + LinkedCheds = m.Relationships.Notifications.Data.Count, + ItemCount = m.Items.Count + + }) + .Select(m => new + { + Id = m.Id, + UpdatedSource = m.UpdatedSource, + Updated = m.Updated, + MaxDecisionNumber = m.MaxDecisionNumber, + MaxEntryVersion = m.MaxEntryVersion, + LinkedCheds = m.LinkedCheds, + ItemCount = m.ItemCount, + Total = m.MaxDecisionNumber + m.MaxEntryVersion + m.LinkedCheds + m.ItemCount + }) + .OrderBy(a => -a.Total) + .Take(10) + .Execute(logger); + + var result = mongoQuery.Select(r => + new ExceptionResult() + { + Resource = "Movement", + Id = r.Id!, + UpdatedSource = r.UpdatedSource!.Value, + Updated = r.Updated, + ItemCount = r.ItemCount, + MaxEntryVersion = r.MaxEntryVersion, + MaxDecisionNumber = r.MaxDecisionNumber, + LinkedCheds = r.LinkedCheds, + Reason = "High Number Of Messages" + }).ToList(); + // var result = new TabularDataset() + // { + // Rows = mongoQuery.Select(r => + // new TabularDimensionRow() + // { + // Key = r.Id!, Columns = + // [ + // new ByNameDimensionResult() { Name = "ItemCount", Value = r.ItemCount }, + // new ByNameDimensionResult() { Name = "MaxEntryVersion", Value = r.MaxEntryVersion }, + // new ByNameDimensionResult() { Name = "MaxDecisionNumber", Value = r.MaxDecisionNumber }, + // new ByNameDimensionResult() { Name = "LinkedCheds", Value = r.LinkedCheds } + // ] + // } + // ).ToList() + // }; + + return Task.FromResult(result); + } + public Task ByCheck(DateTime from, DateTime to) { return Task.FromResult(new MultiSeriesDataset() ); @@ -214,149 +312,171 @@ private Task Aggregate(DateTime[] dateRange, Func> ByDecision(DateTime from, DateTime to) + /// + /// Finds the most recent decision from Alvs and BTMS + /// + /// + /// + /// + public Task> ByDecision(DateTime from, DateTime to) { var mongoQuery = context .Movements + // .Aggregate() .Where(m => m.CreatedSource >= from && m.CreatedSource < to) - .SelectMany(m => m.Decisions.Select(d => new { Decision = d, Movement = m })) - .SelectMany(m => - m.Decision.Items!.Select(i => new { Decision = m.Decision, Movement = m.Movement, Item = i })) - .SelectMany(m => m.Item.Checks!.Select(c => new + .Select(m => new { - CheckCode = c.CheckCode, - DecisionCode = c.DecisionCode, - DecisionSourceSystem = m.Decision.ServiceHeader!.SourceSystem, - DecisionEntryReference = m.Decision.Header!.EntryReference, - DecisionEntryVersionNumber = m.Decision.Header!.EntryVersionNumber, - Movement = m.Movement.EntryReference, - MovementVersion = m.Movement.EntryVersionNumber, - HasLinks = m.Movement.Relationships.Notifications.Data.Count > 0, - ItemNumber = m.Item.ItemNumber - })) - .GroupBy(m => new { m.DecisionSourceSystem, m.DecisionCode }) - .ExecuteAsDictionary(logger, - g => new + MovementInfo = new { - DecisionSourceSystem = g.Key.DecisionSourceSystem!, - DecisionCode = g.Key.DecisionCode! + Id = m.Id, + UpdatedSource = m.UpdatedSource, + Updated = m.Updated, + Movement = m }, - g => g.Count()); - - logger.LogInformation("Found {0} items", mongoQuery.Count); - - var series = new[] - { - new { System = "ALVS" }, - new { System = "BTMS" } - }; + // ReSharper disable once ConditionIsAlwaysTrueOrFalseAccordingToNullableAPIContract + // Get the most recent decision record from both systems + AlvsDecision = (m.Decisions == null + ? null + : m.Decisions + .Where(d => d.ServiceHeader!.SourceSystem == "ALVS") + .OrderBy(d => d.ServiceHeader!.ServiceCalled) + .Reverse() + .FirstOrDefault()) + // Creates a default item & check so we don't lose + // it in the selectmany below + ?? new AlvsClearanceRequest() + { + Items = new [] + { + new Items() + { + Checks = new [] + { + new Check() + { + CheckCode = "XXX", + DecisionCode = "XXX" + } + } + } + } + }, + BtmsDecision = m.Decisions == null + ? null + : m.Decisions + .Where(d => d.ServiceHeader!.SourceSystem == "BTMS") + .OrderBy(d => d.ServiceHeader!.ServiceCalled) + .Reverse() + .FirstOrDefault() + }) + .SelectMany(m => + m.AlvsDecision!.Items! + .SelectMany(i => + (i.Checks + ?? new[] { new Check() { CheckCode = "XXX", DecisionCode = "XXX" } }) + .Select(c => + new + { + m.MovementInfo, + AlvsDecisionInfo = c.CheckCode == "XXX" ? null : new + { + Decision = m.AlvsDecision, + DecisionNumber = m.AlvsDecision!.Header!.DecisionNumber, + EntryVersion = m.AlvsDecision!.Header!.EntryVersionNumber, + ItemNumber = i.ItemNumber, + CheckCode = c.CheckCode, + DecisionCode = c.DecisionCode, + }, + BtmsDecisionInfo = new + { + Decision = m.BtmsDecision, + DecisionCode = m.BtmsDecision == null || m.BtmsDecision.Items == null + ? null + : m.BtmsDecision.Items! + .First(bi => bi.ItemNumber == i.ItemNumber) + .Checks! + .First(ch => ch.CheckCode == c.CheckCode) + .DecisionCode + } + } + ) + ) + .Select(a => new + { + a.MovementInfo, + a.AlvsDecisionInfo, + a.BtmsDecisionInfo, + Classification = + a.BtmsDecisionInfo == null ? "BtmsDecisionMissing" : + a.AlvsDecisionInfo == null ? "AlvsDecisionMissing" : + a.MovementInfo.Movement.Decisions + .Any(d => d.Header!.DecisionNumber == 1) ? "AlvsDecisionVersion1Missing" : + a.MovementInfo.Movement.ClearanceRequests + .Any(d => d.Header!.EntryVersionNumber == 1) ? "AlvsClearanceRequestVersion1Missing" : + a.BtmsDecisionInfo.DecisionCode == a.AlvsDecisionInfo.DecisionCode ? "BtmsMadeSameDecisionAsAlvs" : + a.AlvsDecisionInfo.DecisionNumber == 1 && a.AlvsDecisionInfo.EntryVersion == 1 ? "SingleEntryAndDecisionVersion" : + "FurtherClassificationNeeded" + // "FurtherClassificationNeeded Check Code Is " + a.AlvsDecisionInfo.CheckCode + }) + ) + + // .Where(m => m.AlvsDecisionInfo == null) + // .Where(m => m.AlvsDecision!.Items!.Any(i => i.Checks!.Any(c => c.CheckCode == "XXX"))) + .GroupBy(check => new + { + check.Classification, + check.AlvsDecisionInfo!.CheckCode, + AlvsDecisionCode = check.AlvsDecisionInfo!.DecisionCode, + BtmsDecisionCode=check.BtmsDecisionInfo!.DecisionCode + }) + .Select(g => new + { + g.Key, Count = g.Count() + }) + + .Execute(logger); - var totals = series - .Select(s => new { - System = s.System, - Total = mongoQuery - .Where(a => a.Key.DecisionSourceSystem == s.System) - .Sum(a => a.Value) - }); + logger.LogDebug("Aggregated Data {Result}", mongoQuery.ToJsonString()); - var decisionCodes = mongoQuery - .Select(d => d.Key.DecisionCode) - .Order() - .Distinct(); + // Works + var summary = mongoQuery + .GroupBy(q => q.Key.Classification) + .Select(g => new StringBucketDimensionResult() + { + Fields = new Dictionary() { + { "Classification", g.Key } + }, + Value = g.Sum(k => k.Count) + }).ToList(); - var result = new TabularDataset() + var r = new SummarisedDataset() { - Rows = decisionCodes.Select(d => new TabularDimensionRow() + Summary = summary, + Result = mongoQuery.Select(a => new StringBucketDimensionResult() { - Key = d, - Columns = series.Select(s => - new ByNameDimensionResult() - { - Name = s.System, - Value = mongoQuery.GetValueOrDefault(new - { - DecisionSourceSystem = s.System, DecisionCode = d - }, 0) - } - ).ToList() + Fields = new Dictionary() + { + { "Classification", a.Key.Classification }, + { "CheckCode", a.Key.CheckCode! }, + { "AlvsDecisionCode", a.Key.AlvsDecisionCode! }, + { "BtmsDecisionCode", a.Key.BtmsDecisionCode! } + }, + Value = a.Count }).ToList() }; - result.Rows.Add(new TabularDimensionRow() - { - Key = "Total", - Columns = totals - .Select(t => new ByNameDimensionResult() - { - Name = t.System, Value = t.Total - }).ToList() - - }); - - return Task.FromResult(result); + return Task.FromResult(r); } - - public Task> ByDecisionAndLinkStatus(DateTime from, DateTime to) - { - var mongoQuery = context - .Movements - .Where(m => m.CreatedSource >= from && m.CreatedSource < to) - .SelectMany(m => m.Decisions.Select(d => new { Decision = d, Movement = m })) - .SelectMany(m => - m.Decision.Items!.Select(i => new { Decision = m.Decision, Movement = m.Movement, Item = i })) - .SelectMany(m => m.Item.Checks!.Select(c => new - { - CheckCode = c.CheckCode, - DecisionCode = c.DecisionCode, - DecisionSourceSystem = m.Decision.ServiceHeader!.SourceSystem, - DecisionEntryReference = m.Decision.Header!.EntryReference, - DecisionEntryVersionNumber = m.Decision.Header!.EntryVersionNumber, - Movement = m.Movement.EntryReference, - MovementVersion = m.Movement.EntryVersionNumber, - HasLinks = m.Movement.Relationships.Notifications.Data.Count > 0, - ItemNumber = m.Item.ItemNumber - })) - .GroupBy(m => new { m.HasLinks, m.DecisionSourceSystem, m.DecisionCode }) - .ExecuteAsDictionary(logger, - g => new - { - DecisionSourceSystem = g.Key.DecisionSourceSystem!, - DecisionCode = g.Key.DecisionCode!, - g.Key.HasLinks - }, - g => g.Count()); - - logger.LogInformation("Found {0} items", mongoQuery.Count); - - var series = new[] - { - new { System = "ALVS", Linked = true }, new { System = "ALVS", Linked = false }, - new { System = "BTMS", Linked = true }, new { System = "BTMS", Linked = false } - }; - - var decisionCodes = mongoQuery - .Select(d => d.Key.DecisionCode) - .Order() - .Distinct(); - var result = new TabularDataset() + private static Task> DefaultTabularDatasetByNameDimensionResult() + { + return Task.FromResult(new TabularDataset() { - Rows = decisionCodes.Select(d => new TabularDimensionRow() - { - Key = d, - Columns = series.Select(s => - new ByNameDimensionResult() - { - Name = $"{ s.System } : { ( s.Linked ? "Linked" : "Not Linked" ) }", - Value = mongoQuery.GetValueOrDefault(new - { - DecisionSourceSystem = s.System, DecisionCode = d, HasLinks = s.Linked - }, 0) - } - ).ToList() - }).ToList() - }; - - return Task.FromResult(result); + Rows = + [ + new TabularDimensionRow() { Key = "", Columns = [] } + ] + }); } + } \ No newline at end of file diff --git a/Btms.Backend/Config/AnalyticsDashboards.cs b/Btms.Backend/Config/AnalyticsDashboards.cs index ae1a925c..ac405a8b 100644 --- a/Btms.Backend/Config/AnalyticsDashboards.cs +++ b/Btms.Backend/Config/AnalyticsDashboards.cs @@ -75,8 +75,16 @@ public static async Task> GetCharts( () => movementsService.ByDecision(DateTime.Today.MonthAgo(), DateTime.Now).AsIDataset() }, { - "importNotificationVersionsByVersionCount", - () => importService.ByVersionCount(DateTime.Today.AddMonths(-3), DateTime.Today).AsIDataset() + "allImportNotificationsByVersion", + () => importService.ByMaxVersion(DateTime.Today.AddMonths(-3), DateTime.Today).AsIDataset() + }, + { + "allMovementsByMaxEntryVersion", + () => movementsService.ByMaxVersion(DateTime.Today.AddMonths(-3), DateTime.Today).AsIDataset() + }, + { + "allMovementsByMaxDecisionNumber", + () => movementsService.ByMaxDecisionNumber(DateTime.Today.AddMonths(-3), DateTime.Today).AsIDataset() } }; diff --git a/Btms.Backend/Endpoints/AnalyticsEndpoints.cs b/Btms.Backend/Endpoints/AnalyticsEndpoints.cs index 589eb8c7..3a6c65a3 100644 --- a/Btms.Backend/Endpoints/AnalyticsEndpoints.cs +++ b/Btms.Backend/Endpoints/AnalyticsEndpoints.cs @@ -17,6 +17,7 @@ public static void UseAnalyticsEndpoints(this IEndpointRouteBuilder app) app.MapGet(BaseRoute + "/dashboard", GetDashboard).AllowAnonymous(); app.MapGet(BaseRoute + "/record-current-state", RecordCurrentState).AllowAnonymous(); app.MapGet(BaseRoute + "/timeline", Timeline); + app.MapGet(BaseRoute + "/exceptions", Exceptions); } private static async Task Timeline( [FromServices] IImportNotificationsAggregationService importService, @@ -32,6 +33,19 @@ private static async Task Timeline( return Results.NotFound(); } + private static async Task Exceptions( + [FromServices] IMovementsAggregationService movementsService) + { + var result + = await movementsService.GetExceptions(DateTime.MinValue, DateTime.Today); + + if (result.HasValue()) + { + return TypedResults.Json(result); + } + + return Results.NotFound(); + } private static async Task RecordCurrentState( [FromServices] ImportNotificationMetrics importNotificationMetrics) From e25aa40128d9877f9c1b133c90742fa407cf32de Mon Sep 17 00:00:00 2001 From: Craig Edmunds Date: Mon, 23 Dec 2024 08:30:38 +0000 Subject: [PATCH 5/6] Finishes summary of decsisions --- Btms.Analytics/Dataset.cs | 8 ++-- .../Extensions/SummarisedDatasetExtensions.cs | 12 ++++++ .../IMovementsAggregationService.cs | 2 +- Btms.Analytics/MovementsAggregationService.cs | 43 +++++++++++-------- 4 files changed, 41 insertions(+), 24 deletions(-) create mode 100644 Btms.Analytics/Extensions/SummarisedDatasetExtensions.cs diff --git a/Btms.Analytics/Dataset.cs b/Btms.Analytics/Dataset.cs index 299dd546..65444fd9 100644 --- a/Btms.Analytics/Dataset.cs +++ b/Btms.Analytics/Dataset.cs @@ -11,7 +11,7 @@ public class SummarisedDataset : IDataset where TResult : IDimensionResult where TSummary : IDimensionResult { - public required List Summary { get; set; } + public required TSummary Summary { get; set; } public required List Result { get; set; } } @@ -21,7 +21,7 @@ public class MultiSeriesDataset : IDataset public List Series { get; set; } = []; } -public class SingleSeriesDataset : IDataset +public class SingleSeriesDataset : IDataset, IDimensionResult { public IDictionary Values { get; set; } = new Dictionary(); } @@ -70,9 +70,9 @@ public override void Write(Utf8JsonWriter writer, TType value, JsonSerializerOpt { JsonSerializer.Serialize(writer, value as TabularDataset, options); } - else if (value is SummarisedDataset) + else if (value is SummarisedDataset) { - JsonSerializer.Serialize(writer, value as SummarisedDataset, options); + JsonSerializer.Serialize(writer, value as SummarisedDataset, options); } else { diff --git a/Btms.Analytics/Extensions/SummarisedDatasetExtensions.cs b/Btms.Analytics/Extensions/SummarisedDatasetExtensions.cs new file mode 100644 index 00000000..7ba6b3b3 --- /dev/null +++ b/Btms.Analytics/Extensions/SummarisedDatasetExtensions.cs @@ -0,0 +1,12 @@ +namespace Btms.Analytics.Extensions; + +public static class SummarisedDatasetExtensions +{ + public static async Task AsIDataset(this Task> ms) + where TResult : IDimensionResult + where TSummary : IDimensionResult + { + await ms; + return (IDataset)ms.Result; + } +} \ No newline at end of file diff --git a/Btms.Analytics/IMovementsAggregationService.cs b/Btms.Analytics/IMovementsAggregationService.cs index d2aef5ba..a13deba3 100644 --- a/Btms.Analytics/IMovementsAggregationService.cs +++ b/Btms.Analytics/IMovementsAggregationService.cs @@ -7,7 +7,7 @@ public interface IMovementsAggregationService public Task ByCreated(DateTime from, DateTime to, AggregationPeriod aggregateBy = AggregationPeriod.Day); public Task ByStatus(DateTime from, DateTime to); public Task ByItemCount(DateTime from, DateTime to); - public Task> ByDecision(DateTime from, DateTime to); + public Task> ByDecision(DateTime from, DateTime to); // public Task> ByDecisionAndLinkStatus(DateTime from, DateTime to); public Task ByUniqueDocumentReferenceCount(DateTime from, DateTime to); public Task UniqueDocumentReferenceByMovementCount(DateTime from, DateTime to); diff --git a/Btms.Analytics/MovementsAggregationService.cs b/Btms.Analytics/MovementsAggregationService.cs index 1f394dfa..d7d856a8 100644 --- a/Btms.Analytics/MovementsAggregationService.cs +++ b/Btms.Analytics/MovementsAggregationService.cs @@ -318,7 +318,7 @@ private Task Aggregate(DateTime[] dateRange, Func /// /// - public Task> ByDecision(DateTime from, DateTime to) + public Task> ByDecision(DateTime from, DateTime to) { var mongoQuery = context .Movements @@ -407,15 +407,18 @@ public Task d.Header!.DecisionNumber == 1) ? "AlvsDecisionVersion1Missing" : + .Any(d => d.Header!.DecisionNumber == 1) ? "Alvs Decision Version 1 Not Present" : a.MovementInfo.Movement.ClearanceRequests - .Any(d => d.Header!.EntryVersionNumber == 1) ? "AlvsClearanceRequestVersion1Missing" : - a.BtmsDecisionInfo.DecisionCode == a.AlvsDecisionInfo.DecisionCode ? "BtmsMadeSameDecisionAsAlvs" : - a.AlvsDecisionInfo.DecisionNumber == 1 && a.AlvsDecisionInfo.EntryVersion == 1 ? "SingleEntryAndDecisionVersion" : - "FurtherClassificationNeeded" + .Any(d => d.Header!.EntryVersionNumber == 1) ? "Alvs Clearance Request Version 1 Not Present" : + a.AlvsDecisionInfo.DecisionNumber == 1 && a.AlvsDecisionInfo.EntryVersion == 1 ? "Single Entry And Decision Version" : + a.BtmsDecisionInfo.DecisionCode != a.AlvsDecisionInfo.DecisionCode ? "Btms Made Different Decision To Alvs" : + "Further Classification Needed" // "FurtherClassificationNeeded Check Code Is " + a.AlvsDecisionInfo.CheckCode }) ) @@ -439,17 +442,16 @@ public Task q.Key.Classification) - .Select(g => new StringBucketDimensionResult() - { - Fields = new Dictionary() { - { "Classification", g.Key } - }, - Value = g.Sum(k => k.Count) - }).ToList(); + var summary = new SingleSeriesDataset() { + Values = mongoQuery + .GroupBy(q => q.Key.Classification) + .ToDictionary( + g => g.Key, + g => g.Sum(k => k.Count) + ) + }; - var r = new SummarisedDataset() + var r = new SummarisedDataset() { Summary = summary, Result = mongoQuery.Select(a => new StringBucketDimensionResult() @@ -462,7 +464,10 @@ public Task r.Value) + .Reverse() + .ToList() }; return Task.FromResult(r); From 28af41a15638e9bde3ac442ccf8fcbf15e709b9d Mon Sep 17 00:00:00 2001 From: Craig Edmunds Date: Mon, 23 Dec 2024 08:33:22 +0000 Subject: [PATCH 6/6] Fixes tests --- Btms.Analytics.Tests/MovementsByDecisionsTests.cs | 2 +- Btms.Analytics.Tests/MovementsExceptionsTests.cs | 2 +- .../Extensions/SummarisedDatasetExtensions.cs | 12 ------------ 3 files changed, 2 insertions(+), 14 deletions(-) delete mode 100644 Btms.Analytics/Extensions/SummarisedDatasetExtensions.cs diff --git a/Btms.Analytics.Tests/MovementsByDecisionsTests.cs b/Btms.Analytics.Tests/MovementsByDecisionsTests.cs index e8ee5474..72a797ea 100644 --- a/Btms.Analytics.Tests/MovementsByDecisionsTests.cs +++ b/Btms.Analytics.Tests/MovementsByDecisionsTests.cs @@ -20,7 +20,7 @@ public async Task WhenCalled_ReturnExpectedAggregation() testOutputHelper.WriteLine("Querying for aggregated data"); var result = (await multiItemDataTestFixture.GetMovementsAggregationService(testOutputHelper) .ByDecision(DateTime.Today.MonthAgo(), DateTime.Today.Tomorrow())) - .Rows; + .Result; testOutputHelper.WriteLine("{0} aggregated items found", result.Count); diff --git a/Btms.Analytics.Tests/MovementsExceptionsTests.cs b/Btms.Analytics.Tests/MovementsExceptionsTests.cs index 1ab923b7..2d574f9a 100644 --- a/Btms.Analytics.Tests/MovementsExceptionsTests.cs +++ b/Btms.Analytics.Tests/MovementsExceptionsTests.cs @@ -20,7 +20,7 @@ public async Task WhenCalled_ReturnExpectedAggregation() testOutputHelper.WriteLine("Querying for aggregated data"); var result = (await multiItemDataTestFixture.GetMovementsAggregationService(testOutputHelper) .ByDecision(DateTime.Today.MonthAgo(), DateTime.Today.Tomorrow())) - .Rows; + .Result; testOutputHelper.WriteLine("{0} aggregated items found", result.Count); diff --git a/Btms.Analytics/Extensions/SummarisedDatasetExtensions.cs b/Btms.Analytics/Extensions/SummarisedDatasetExtensions.cs deleted file mode 100644 index 7ba6b3b3..00000000 --- a/Btms.Analytics/Extensions/SummarisedDatasetExtensions.cs +++ /dev/null @@ -1,12 +0,0 @@ -namespace Btms.Analytics.Extensions; - -public static class SummarisedDatasetExtensions -{ - public static async Task AsIDataset(this Task> ms) - where TResult : IDimensionResult - where TSummary : IDimensionResult - { - await ms; - return (IDataset)ms.Result; - } -} \ No newline at end of file