From ef2d5799fc62144ae7c4adf482a5d616cd9f39be Mon Sep 17 00:00:00 2001 From: Maksym Kucherov Date: Wed, 29 Nov 2023 12:16:56 +0200 Subject: [PATCH 01/13] Convert constructor-initialized field into lazy `HeadersAffectingJavaScript` property. --- .../Services/ClientsidePropertyService.cs | 65 ++++++++++++++++--- 1 file changed, 55 insertions(+), 10 deletions(-) diff --git a/FiftyOne.Pipeline.Web.Shared/Services/ClientsidePropertyService.cs b/FiftyOne.Pipeline.Web.Shared/Services/ClientsidePropertyService.cs index 2e21407e..13fee760 100644 --- a/FiftyOne.Pipeline.Web.Shared/Services/ClientsidePropertyService.cs +++ b/FiftyOne.Pipeline.Web.Shared/Services/ClientsidePropertyService.cs @@ -48,15 +48,43 @@ public class ClientsidePropertyService : IClientsidePropertyService /// /// Pipeline /// - private IPipeline _pipeline; + private readonly IPipeline _pipeline; - private ILogger _logger; + private readonly ILogger _logger; /// /// A list of all the HTTP headers that are requested evidence /// for elements that populate JavaScript properties /// - private StringValues _headersAffectingJavaScript; + private StringValues? _headersAffectingJavaScript = null; + private readonly object _headersAffectingJavaScriptLock = new object(); + + private StringValues HeadersAffectingJavaScript + { + get + { + if (_headersAffectingJavaScript.HasValue) + { + return _headersAffectingJavaScript.Value; + } + + lock (_headersAffectingJavaScriptLock) + { + if (_headersAffectingJavaScript.HasValue) + { + return _headersAffectingJavaScript.Value; + } + + CollectHeadersAffectingJavaScript(out StringValues newHeaders, out bool gotExceptions); + + if (!gotExceptions) + { + _headersAffectingJavaScript = newHeaders; + } + return newHeaders; + } + } + } private enum ContentType { @@ -68,7 +96,7 @@ private enum ContentType /// The cache control values that will be set for the JavaScript and /// JSON. /// - private StringValues _cacheControl = new StringValues( + private readonly StringValues _cacheControl = new StringValues( new string[] { "private", "max-age=1800", @@ -90,11 +118,27 @@ public ClientsidePropertyService( _pipeline = pipeline; _logger = logger; + } - var headersAffectingJavaScript = new List(); + private void CollectHeadersAffectingJavaScript(out StringValues headers, out bool gotExceptions) + { + var headersAffectingJavaScript = new List(); + gotExceptions = false; + + foreach (var flowElement in _pipeline.FlowElements) + { + IEvidenceKeyFilter filter; + try + { + filter = flowElement.EvidenceKeyFilter; + } + catch (Exception ex) + { + _logger.LogError(ex, $"Failed to get {nameof(flowElement.EvidenceKeyFilter)} from {{flowElementType}}", flowElement.GetType().Name); + gotExceptions = true; + continue; + } - foreach (var filter in _pipeline.FlowElements.Select(e => e.EvidenceKeyFilter)) - { // If the filter is a white list or derived type then // get all HTTP header evidence keys from white list // and add them to the headers that could affect the @@ -120,7 +164,7 @@ public ClientsidePropertyService( .Distinct(StringComparer.OrdinalIgnoreCase)); } } - _headersAffectingJavaScript = new StringValues(headersAffectingJavaScript.ToArray()); + headers = new StringValues(headersAffectingJavaScript.ToArray()); } /// @@ -240,9 +284,10 @@ private void SetHeaders(IContextAdapter context, string hash, int contentLength, context.Response.SetHeader("Content-Length", contentLength.ToString(CultureInfo.InvariantCulture)); context.Response.SetHeader("Cache-Control", _cacheControl); - if (string.IsNullOrEmpty(_headersAffectingJavaScript.ToString()) == false) + var headersAffectingJavaScript = HeadersAffectingJavaScript; + if (string.IsNullOrEmpty(headersAffectingJavaScript.ToString()) == false) { - context.Response.SetHeader("Vary", _headersAffectingJavaScript); + context.Response.SetHeader("Vary", headersAffectingJavaScript); } context.Response.SetHeader("ETag", new StringValues( new string[] { From 8386448d0ba52c97685bbca9063411a64300e886 Mon Sep 17 00:00:00 2001 From: Maksym Kucherov Date: Wed, 29 Nov 2023 13:11:34 +0200 Subject: [PATCH 02/13] Add `org-name` to workflow files. --- .github/workflows/monthly-copyright-update.yml | 1 + .github/workflows/nightly-documentation-update.yml | 1 + .github/workflows/nightly-package-update.yml | 1 + .github/workflows/nightly-prs-to-main.yml | 1 + .github/workflows/nightly-publish-main.yml | 1 + .github/workflows/nightly-submodule-update.yml | 1 + 6 files changed, 6 insertions(+) diff --git a/.github/workflows/monthly-copyright-update.yml b/.github/workflows/monthly-copyright-update.yml index a45e0138..152012af 100644 --- a/.github/workflows/monthly-copyright-update.yml +++ b/.github/workflows/monthly-copyright-update.yml @@ -12,5 +12,6 @@ jobs: uses: 51Degrees/common-ci/.github/workflows/monthly-copyright-update.yml@main with: repo-name: ${{ github.event.repository.name }} + org-name: ${{ github.event.repository.owner.login }} secrets: token: ${{ secrets.ACCESS_TOKEN }} diff --git a/.github/workflows/nightly-documentation-update.yml b/.github/workflows/nightly-documentation-update.yml index 775060f0..bb65c351 100644 --- a/.github/workflows/nightly-documentation-update.yml +++ b/.github/workflows/nightly-documentation-update.yml @@ -13,5 +13,6 @@ jobs: uses: 51Degrees/common-ci/.github/workflows/nightly-documentation-update.yml@main with: repo-name: ${{ github.event.repository.name }} + org-name: ${{ github.event.repository.owner.login }} secrets: token: ${{ secrets.ACCESS_TOKEN }} diff --git a/.github/workflows/nightly-package-update.yml b/.github/workflows/nightly-package-update.yml index 440ba369..d1642ad1 100644 --- a/.github/workflows/nightly-package-update.yml +++ b/.github/workflows/nightly-package-update.yml @@ -11,5 +11,6 @@ jobs: uses: 51Degrees/common-ci/.github/workflows/nightly-package-update.yml@main with: repo-name: ${{ github.event.repository.name }} + org-name: ${{ github.event.repository.owner.login }} secrets: token: ${{ secrets.ACCESS_TOKEN }} diff --git a/.github/workflows/nightly-prs-to-main.yml b/.github/workflows/nightly-prs-to-main.yml index 2ae4edd4..e02ac99f 100644 --- a/.github/workflows/nightly-prs-to-main.yml +++ b/.github/workflows/nightly-prs-to-main.yml @@ -12,5 +12,6 @@ jobs: uses: 51Degrees/common-ci/.github/workflows/nightly-prs-to-main.yml@main with: repo-name: ${{ github.event.repository.name }} + org-name: ${{ github.event.repository.owner.login }} secrets: token: ${{ secrets.ACCESS_TOKEN }} diff --git a/.github/workflows/nightly-publish-main.yml b/.github/workflows/nightly-publish-main.yml index 4f62866b..08ad172d 100644 --- a/.github/workflows/nightly-publish-main.yml +++ b/.github/workflows/nightly-publish-main.yml @@ -12,6 +12,7 @@ jobs: uses: 51Degrees/common-ci/.github/workflows/nightly-publish-main.yml@main with: repo-name: ${{ github.event.repository.name }} + org-name: ${{ github.event.repository.owner.login }} build-platform: windows-latest secrets: token: ${{ secrets.ACCESS_TOKEN }} diff --git a/.github/workflows/nightly-submodule-update.yml b/.github/workflows/nightly-submodule-update.yml index b2ec1118..f3c4bfb5 100644 --- a/.github/workflows/nightly-submodule-update.yml +++ b/.github/workflows/nightly-submodule-update.yml @@ -12,5 +12,6 @@ jobs: uses: 51Degrees/common-ci/.github/workflows/nightly-submodule-update.yml@main with: repo-name: ${{ github.event.repository.name }} + org-name: ${{ github.event.repository.owner.login }} secrets: token: ${{ secrets.ACCESS_TOKEN }} From b42325045db3a31c34328c49d9b7329b43d3ed9d Mon Sep 17 00:00:00 2001 From: Maksym Kucherov Date: Wed, 29 Nov 2023 14:02:17 +0200 Subject: [PATCH 03/13] Wrap `_pipeline.EvidenceKeyFilter` call in try-catch. --- .../Services/ClientsidePropertyService.cs | 33 ++++++++++++++----- 1 file changed, 25 insertions(+), 8 deletions(-) diff --git a/FiftyOne.Pipeline.Web.Shared/Services/ClientsidePropertyService.cs b/FiftyOne.Pipeline.Web.Shared/Services/ClientsidePropertyService.cs index 13fee760..22bfa3cb 100644 --- a/FiftyOne.Pipeline.Web.Shared/Services/ClientsidePropertyService.cs +++ b/FiftyOne.Pipeline.Web.Shared/Services/ClientsidePropertyService.cs @@ -211,10 +211,24 @@ private void ServeContent(IContextAdapter context, IFlowData flowData, ContentTy context.Response.Clear(); context.Response.ClearHeaders(); - // Get the hash code. - var hash = flowData.GenerateKey(_pipeline.EvidenceKeyFilter).GetHashCode(); + IEvidenceKeyFilter pipelineEvidenceKeyFilter = null; + try + { + pipelineEvidenceKeyFilter = _pipeline.EvidenceKeyFilter; + } + catch (PipelineException ex) + { + _logger?.LogError(ex, $"Failed to get {nameof(_pipeline.EvidenceKeyFilter)} from {nameof(_pipeline)}"); + } + + // Get the hash code. + int? hash = null; + if (pipelineEvidenceKeyFilter != null) { + hash = flowData.GenerateKey(pipelineEvidenceKeyFilter).GetHashCode(); + } - if (int.TryParse(context.Request.GetHeaderValue("If-None-Match"), + if (hash.HasValue && + int.TryParse(context.Request.GetHeaderValue("If-None-Match"), out int previousHash) && hash == previousHash) { @@ -259,7 +273,7 @@ private void ServeContent(IContextAdapter context, IFlowData flowData, ContentTy context.Response.StatusCode = 200; SetHeaders(context, - hash.ToString(CultureInfo.InvariantCulture), + hash.HasValue ? hash.Value.ToString(CultureInfo.InvariantCulture) : null, length, contentType == ContentType.JavaScript ? "x-javascript" : "json"); @@ -289,10 +303,13 @@ private void SetHeaders(IContextAdapter context, string hash, int contentLength, { context.Response.SetHeader("Vary", headersAffectingJavaScript); } - context.Response.SetHeader("ETag", new StringValues( - new string[] { - hash, - })); + if (!string.IsNullOrEmpty(hash)) + { + context.Response.SetHeader("ETag", new StringValues( + new string[] { + hash, + })); + } var origin = GetAllowOrigin(context.Request); if (origin != null) { From 823a2c2bc45655708d68e12c189c4bca17fb47ac Mon Sep 17 00:00:00 2001 From: Maksym Kucherov Date: Wed, 29 Nov 2023 14:20:31 +0200 Subject: [PATCH 04/13] Mock `pipeline.EvidenceKeyFilter` in tests to return non-null. --- .../ClientsidePropertyServiceTests.cs | 2 ++ 1 file changed, 2 insertions(+) diff --git a/Web Integration/Tests/FiftyOne.Pipeline.Web.Shared.Tests/ClientsidePropertyServiceTests.cs b/Web Integration/Tests/FiftyOne.Pipeline.Web.Shared.Tests/ClientsidePropertyServiceTests.cs index 05e8bba4..f15d1118 100644 --- a/Web Integration/Tests/FiftyOne.Pipeline.Web.Shared.Tests/ClientsidePropertyServiceTests.cs +++ b/Web Integration/Tests/FiftyOne.Pipeline.Web.Shared.Tests/ClientsidePropertyServiceTests.cs @@ -437,6 +437,8 @@ private void Configure(List flowElements = null) } _pipeline.Setup(p => p.FlowElements) .Returns(flowElements); + _pipeline.Setup(p => p.EvidenceKeyFilter) + .Returns(new EvidenceKeyFilterWhitelist(new List())); // Configure the key for this flow data to contain a fake value // that we can use to test the cached response handling. _defaultDataKey = new DataKeyBuilder().Add(1, "test", "value").Build(); From 9e03d019dd66eb0f4b479b33c1c476330bf052da Mon Sep 17 00:00:00 2001 From: Maksym Kucherov Date: Wed, 29 Nov 2023 15:00:09 +0200 Subject: [PATCH 05/13] Wrap `flowData.GetFromElement` calls in try-catch. --- .../Services/ClientsidePropertyService.cs | 22 +++++++++++++++++-- 1 file changed, 20 insertions(+), 2 deletions(-) diff --git a/FiftyOne.Pipeline.Web.Shared/Services/ClientsidePropertyService.cs b/FiftyOne.Pipeline.Web.Shared/Services/ClientsidePropertyService.cs index 22bfa3cb..bbb14fb0 100644 --- a/FiftyOne.Pipeline.Web.Shared/Services/ClientsidePropertyService.cs +++ b/FiftyOne.Pipeline.Web.Shared/Services/ClientsidePropertyService.cs @@ -35,6 +35,8 @@ using System.Text; using FiftyOne.Pipeline.Web.Shared.Adapters; using Microsoft.Extensions.Logging; +using FiftyOne.Pipeline.JavaScriptBuilder.Data; +using FiftyOne.Pipeline.JsonBuilder.Data; namespace FiftyOne.Pipeline.Web.Shared.Services { @@ -248,7 +250,15 @@ private void ServeContent(IContextAdapter context, IFlowData flowData, ContentTy throw new PipelineConfigurationException( Messages.ExceptionNoJavaScriptBuilder); } - var jsData = flowData.GetFromElement(jsElement); + IJavaScriptBuilderElementData jsData = null; + try + { + jsData = flowData.GetFromElement(jsElement); + } + catch (PipelineException ex) + { + _logger?.LogError(ex, "Failed to get data from {flowElementType}", jsElement.GetType().Name); + } content = jsData?.JavaScript; break; case ContentType.Json: @@ -258,7 +268,15 @@ private void ServeContent(IContextAdapter context, IFlowData flowData, ContentTy throw new PipelineConfigurationException( Messages.ExceptionNoJsonBuilder); } - var jsonData = flowData.GetFromElement(jsonElement); + IJsonBuilderElementData jsonData = null; + try + { + jsonData = flowData.GetFromElement(jsonElement); + } + catch (PipelineException ex) + { + _logger?.LogError(ex, "Failed to get data from {flowElementType}", jsonElement.GetType().Name); + } content = jsonData?.Json; break; default: From 9373b54c82a9b152e04dde9b0de4019a53de1bac Mon Sep 17 00:00:00 2001 From: Maksym Kucherov Date: Wed, 29 Nov 2023 16:50:10 +0200 Subject: [PATCH 06/13] Introduce `JavaScriptBuilderElement.GetFallbackResponse`. --- .../FlowElement/JavaScriptBuilderElement.cs | 58 +++++++++++++++---- .../Services/ClientsidePropertyService.cs | 3 +- 2 files changed, 48 insertions(+), 13 deletions(-) diff --git a/FiftyOne.Pipeline.Elements/FiftyOne.Pipeline.JavaScriptBuilderElement/FlowElement/JavaScriptBuilderElement.cs b/FiftyOne.Pipeline.Elements/FiftyOne.Pipeline.JavaScriptBuilderElement/FlowElement/JavaScriptBuilderElement.cs index d129f8b9..1358ccb6 100644 --- a/FiftyOne.Pipeline.Elements/FiftyOne.Pipeline.JavaScriptBuilderElement/FlowElement/JavaScriptBuilderElement.cs +++ b/FiftyOne.Pipeline.Elements/FiftyOne.Pipeline.JavaScriptBuilderElement/FlowElement/JavaScriptBuilderElement.cs @@ -245,12 +245,27 @@ protected override void ManagedResourcesCleanup() /// protected override void ProcessInternal(IFlowData data) { - if (data == null) throw new ArgumentNullException(nameof(data)); - SetUp(data); + SetUp(data, GetOrAddToData(data)); + } + + + /// + /// Default process method. + /// + /// + /// + /// Thrown if the supplied flow data is null. + /// + public JavaScriptBuilderElementData GetFallbackResponse(IFlowData data) + { + JavaScriptBuilderElementData result = (JavaScriptBuilderElementData)CreateElementData(data.Pipeline); + SetUp(data, () => result); + return result; } - private void SetUp(IFlowData data) + private void SetUp(IFlowData data, Func targetElementDataProvider) { + if (data == null) throw new ArgumentNullException(nameof(data)); var host = Host; var protocol = Protocol; bool supportsPromises = false; @@ -380,7 +395,7 @@ private void SetUp(IFlowData data) } // With the gathered resources, build a new JavaScriptResource. - BuildJavaScript(data, jsonObject, sessionId, sequence, supportsPromises, supportsFetch, url, paramsObject); + BuildJavaScript(data, targetElementDataProvider, jsonObject, sessionId, sequence, supportsPromises, supportsFetch, url, paramsObject); } @@ -474,6 +489,12 @@ protected virtual string GetSessionId(IFlowData data) /// /// The instance to populate with the /// resulting + /// and additional evidence source + /// + /// + /// The method the will inject the resulting + /// + /// into the response (even if differs from `data` above) /// /// /// The JSON data object to include in the JavaScript. @@ -503,6 +524,7 @@ protected virtual string GetSessionId(IFlowData data) /// protected void BuildJavaScript( IFlowData data, + Func targetElementDataProvider, string jsonObject, string sessionId, int sequence, @@ -511,7 +533,15 @@ protected void BuildJavaScript( string url, string parameters) { - BuildJavaScript(data, jsonObject, sessionId, sequence, supportsPromises, supportsFetch, new Uri(url), parameters); + BuildJavaScript(data, targetElementDataProvider, jsonObject, sessionId, sequence, supportsPromises, supportsFetch, new Uri(url), parameters); + } + + private Func GetOrAddToData(IFlowData data) + { + return () => (JavaScriptBuilderElementData) + data.GetOrAdd( + ElementDataKeyTyped, + CreateElementData); } /// @@ -521,6 +551,12 @@ protected void BuildJavaScript( /// /// The instance to populate with the /// resulting + /// and additional evidence source + /// + /// + /// The method the will inject the resulting + /// + /// into the response (even if differs from `data` above) /// /// /// The JSON data object to include in the JavaScript. @@ -549,7 +585,8 @@ protected void BuildJavaScript( /// Thrown if the supplied flow data is null. /// protected void BuildJavaScript( - IFlowData data, + IFlowData data, + Func targetElementDataProvider, string jsonObject, string sessionId, int sequence, @@ -558,12 +595,9 @@ protected void BuildJavaScript( Uri url, string parameters) { - if (data == null) throw new ArgumentNullException(nameof(data)); - - JavaScriptBuilderElementData elementData = (JavaScriptBuilderElementData) - data.GetOrAdd( - ElementDataKeyTyped, - CreateElementData); + if (data == null) throw new ArgumentNullException(nameof(data)); + + JavaScriptBuilderElementData elementData = targetElementDataProvider(); string objectName = ObjName; // Try and get the requested object name from evidence. diff --git a/FiftyOne.Pipeline.Web.Shared/Services/ClientsidePropertyService.cs b/FiftyOne.Pipeline.Web.Shared/Services/ClientsidePropertyService.cs index bbb14fb0..461684f4 100644 --- a/FiftyOne.Pipeline.Web.Shared/Services/ClientsidePropertyService.cs +++ b/FiftyOne.Pipeline.Web.Shared/Services/ClientsidePropertyService.cs @@ -250,7 +250,7 @@ private void ServeContent(IContextAdapter context, IFlowData flowData, ContentTy throw new PipelineConfigurationException( Messages.ExceptionNoJavaScriptBuilder); } - IJavaScriptBuilderElementData jsData = null; + IJavaScriptBuilderElementData jsData; try { jsData = flowData.GetFromElement(jsElement); @@ -258,6 +258,7 @@ private void ServeContent(IContextAdapter context, IFlowData flowData, ContentTy catch (PipelineException ex) { _logger?.LogError(ex, "Failed to get data from {flowElementType}", jsElement.GetType().Name); + jsData = jsElement.GetFallbackResponse(flowData); } content = jsData?.JavaScript; break; From 612f30b68d366808ee7e868c83c3251c032d8e38 Mon Sep 17 00:00:00 2001 From: Maksym Kucherov Date: Wed, 29 Nov 2023 17:36:24 +0200 Subject: [PATCH 07/13] Add a test of `GetFallbackResponse` method. --- .../FlowElement/JavaScriptBuilderElement.cs | 41 ++++++++++++------- .../JavaScriptBuilderElementTests.cs | 28 +++++++++++++ 2 files changed, 55 insertions(+), 14 deletions(-) diff --git a/FiftyOne.Pipeline.Elements/FiftyOne.Pipeline.JavaScriptBuilderElement/FlowElement/JavaScriptBuilderElement.cs b/FiftyOne.Pipeline.Elements/FiftyOne.Pipeline.JavaScriptBuilderElement/FlowElement/JavaScriptBuilderElement.cs index 1358ccb6..8b6c0f14 100644 --- a/FiftyOne.Pipeline.Elements/FiftyOne.Pipeline.JavaScriptBuilderElement/FlowElement/JavaScriptBuilderElement.cs +++ b/FiftyOne.Pipeline.Elements/FiftyOne.Pipeline.JavaScriptBuilderElement/FlowElement/JavaScriptBuilderElement.cs @@ -245,7 +245,7 @@ protected override void ManagedResourcesCleanup() /// protected override void ProcessInternal(IFlowData data) { - SetUp(data, GetOrAddToData(data)); + SetUp(data, GetJSONFromData(data), GetOrAddToData(data)); } @@ -253,17 +253,38 @@ protected override void ProcessInternal(IFlowData data) /// Default process method. /// /// + /// /// /// Thrown if the supplied flow data is null. /// - public JavaScriptBuilderElementData GetFallbackResponse(IFlowData data) + public JavaScriptBuilderElementData GetFallbackResponse(IFlowData data, IJsonBuilderElementData jsonData) { + if (jsonData == null) + { + throw new ArgumentNullException(nameof(jsonData)); + } JavaScriptBuilderElementData result = (JavaScriptBuilderElementData)CreateElementData(data.Pipeline); - SetUp(data, () => result); + SetUp(data, () => jsonData, () => result); return result; } - private void SetUp(IFlowData data, Func targetElementDataProvider) + private static Func GetJSONFromData(IFlowData data) => () => + { + try + { + return data.Get(); + } + catch (KeyNotFoundException ex) + { + throw new PipelineConfigurationException( + Messages.ExceptionJsonBuilderNotRun, ex); + } + }; + + private void SetUp( + IFlowData data, + Func jsonDataProvider, + Func targetElementDataProvider) { if (data == null) throw new ArgumentNullException(nameof(data)); var host = Host; @@ -342,16 +363,8 @@ private void SetUp(IFlowData data, Func targetElem } // Get the JSON include to embed into the JavaScript include. - string jsonObject = string.Empty; - try - { - jsonObject = data.Get().Json; - } - catch (KeyNotFoundException ex) - { - throw new PipelineConfigurationException( - Messages.ExceptionJsonBuilderNotRun, ex); - } + string jsonObject = jsonDataProvider().Json; + var parameters = GetParameters(data); var paramsObject = JsonConvert.SerializeObject(parameters); diff --git a/FiftyOne.Pipeline.Elements/FiftyOne.Pipeline.JavaScriptBuilderElementTests/JavaScriptBuilderElementTests.cs b/FiftyOne.Pipeline.Elements/FiftyOne.Pipeline.JavaScriptBuilderElementTests/JavaScriptBuilderElementTests.cs index 44d78872..66ede0b3 100644 --- a/FiftyOne.Pipeline.Elements/FiftyOne.Pipeline.JavaScriptBuilderElementTests/JavaScriptBuilderElementTests.cs +++ b/FiftyOne.Pipeline.Elements/FiftyOne.Pipeline.JavaScriptBuilderElementTests/JavaScriptBuilderElementTests.cs @@ -196,6 +196,34 @@ public void JavaScriptBuilder_VerifySession() $"JavaScript does not contain expected sequence '1'."); } + /// + /// Check that the callback URL is generated correctly. + /// + [TestMethod] + public void JavaScriptBuilder_VerifyFallbackResponse() + { + _javaScriptBuilderElement = + new JavaScriptBuilderElementBuilder(_loggerFactory) + .SetEndpoint("/json") + .Build(); + + var flowData = new Mock(); + + flowData.Setup(d => d.Get()) + .Throws(); + var evidence = new Evidence(_loggerFactory.CreateLogger()); + flowData.Setup(d => d.GetEvidence()) + .Returns(evidence); + + var jsonData = new Mock(); + jsonData.Setup(j => j.Json) + .Returns("{}"); + + IJavaScriptBuilderElementData result = _javaScriptBuilderElement.GetFallbackResponse(flowData.Object, jsonData.Object); + + Assert.IsFalse(string.IsNullOrWhiteSpace(result.JavaScript)); + } + /// /// Check that the callback URL is generated correctly. /// From 855883cb96d219084aa9b17a55c168a4fc6cc412 Mon Sep 17 00:00:00 2001 From: Maksym Kucherov Date: Wed, 29 Nov 2023 18:08:36 +0200 Subject: [PATCH 08/13] Introduce `JsonBuilderElement.GetFallbackResponse`. --- .../FlowElement/JsonBuilderElement.cs | 42 +++++++++++++++++-- .../JsonBuilderElementTests.cs | 19 ++++++++- 2 files changed, 56 insertions(+), 5 deletions(-) diff --git a/FiftyOne.Pipeline.Elements/FiftyOne.Pipeline.JsonBuilderElement/FlowElement/JsonBuilderElement.cs b/FiftyOne.Pipeline.Elements/FiftyOne.Pipeline.JsonBuilderElement/FlowElement/JsonBuilderElement.cs index fc34e3e0..96a66142 100644 --- a/FiftyOne.Pipeline.Elements/FiftyOne.Pipeline.JsonBuilderElement/FlowElement/JsonBuilderElement.cs +++ b/FiftyOne.Pipeline.Elements/FiftyOne.Pipeline.JsonBuilderElement/FlowElement/JsonBuilderElement.cs @@ -215,7 +215,27 @@ protected override void ManagedResourcesCleanup() /// /// Thrown if the supplied flow data is null. /// - protected override void ProcessInternal(IFlowData data) + protected override void ProcessInternal(IFlowData data) => + BuildAndInjectJSON( + data, + data.GetOrAdd( + ElementDataKeyTyped, + CreateElementData)); + + /// + /// Transform the data in the flow data instance into a + /// JSON object. + /// + /// + /// The + /// + /// + /// The + /// + /// + /// Thrown if the supplied flow data is null. + /// + private void BuildAndInjectJSON(IFlowData data, IJsonBuilderElementData elementData) { if (data == null) throw new ArgumentNullException(nameof(data)); @@ -225,13 +245,27 @@ protected override void ProcessInternal(IFlowData data) config = _pipelineConfigs.GetOrAdd(data.Pipeline, config); } - var elementData = data.GetOrAdd( - ElementDataKeyTyped, - CreateElementData); var jsonString = BuildJson(data, config); elementData.Json = jsonString; } + /// + /// Transform the data in the flow data instance into a + /// JSON object. + /// + /// + /// The + /// + /// + /// Thrown if the supplied flow data is null. + /// + public IJsonBuilderElementData GetFallbackResponse(IFlowData data) + { + var elementData = CreateElementData(data.Pipeline); + BuildAndInjectJSON(data, elementData); + return elementData; + } + /// /// Create and populate a JSON string from the specified data. /// diff --git a/FiftyOne.Pipeline.Elements/FiftyOne.Pipeline.JsonBuilderElementTests/JsonBuilderElementTests.cs b/FiftyOne.Pipeline.Elements/FiftyOne.Pipeline.JsonBuilderElementTests/JsonBuilderElementTests.cs index 8ba47e3d..58de5c52 100644 --- a/FiftyOne.Pipeline.Elements/FiftyOne.Pipeline.JsonBuilderElementTests/JsonBuilderElementTests.cs +++ b/FiftyOne.Pipeline.Elements/FiftyOne.Pipeline.JsonBuilderElementTests/JsonBuilderElementTests.cs @@ -98,6 +98,17 @@ public void JsonBuilder_ValidJson() Assert.IsTrue(IsExpectedJson(json)); } + /// + /// Check that the JSON produced by the JsonBuilder is valid. + /// + [TestMethod] + public void JsonBuilder_NonEmptyFallback() + { + var json = TestIteration(1, null, null, (e, fd) => ((JsonBuilderElement)e).GetFallbackResponse(fd).Json); + + Assert.IsFalse(string.IsNullOrWhiteSpace(json)); + } + /// /// Check that the JSON element removes JavaScript properties from the /// response after max number of iterations has been reached. @@ -731,7 +742,8 @@ public class JsonBuilder private string TestIteration(int iteration, Dictionary data = null, - Mock flowData = null) + Mock flowData = null, + Func processFunc = null) { if(data == null) { @@ -765,6 +777,11 @@ private string TestIteration(int iteration, }); flowData.Setup(d => d.Pipeline).Returns(_pipeline.Object); + if (processFunc != null) + { + return processFunc(_jsonBuilderElement, flowData.Object); + } + _jsonBuilderElement.Process(flowData.Object); var json = result["json"].ToString(); From e681799cabcddc692ff64dfe480bc8a7875da3c8 Mon Sep 17 00:00:00 2001 From: Maksym Kucherov Date: Wed, 29 Nov 2023 18:09:06 +0200 Subject: [PATCH 09/13] Use `GetJsonData` in `GetJsData` --- .../Services/ClientsidePropertyService.cs | 76 +++++++++++-------- 1 file changed, 43 insertions(+), 33 deletions(-) diff --git a/FiftyOne.Pipeline.Web.Shared/Services/ClientsidePropertyService.cs b/FiftyOne.Pipeline.Web.Shared/Services/ClientsidePropertyService.cs index 461684f4..9e0c8530 100644 --- a/FiftyOne.Pipeline.Web.Shared/Services/ClientsidePropertyService.cs +++ b/FiftyOne.Pipeline.Web.Shared/Services/ClientsidePropertyService.cs @@ -241,44 +241,54 @@ private void ServeContent(IContextAdapter context, IFlowData flowData, ContentTy { // Otherwise, return the requested content to the client. string content = null; + Func GetJsonData = () => + { + var jsonElement = flowData.Pipeline.GetElement(); + if (jsonElement == null) + { + throw new PipelineConfigurationException( + Messages.ExceptionNoJsonBuilder); + } + IJsonBuilderElementData jsonData = null; + try + { + jsonData = flowData.GetFromElement(jsonElement); + } + catch (PipelineException ex) + { + _logger?.LogError(ex, "Failed to get data from {flowElementType}", jsonElement.GetType().Name); + } + return jsonData; + }; + + Func GetJsData = () => + { + var jsElement = flowData.Pipeline.GetElement(); + if (jsElement == null) + { + throw new PipelineConfigurationException( + Messages.ExceptionNoJavaScriptBuilder); + } + IJavaScriptBuilderElementData jsData; + try + { + jsData = flowData.GetFromElement(jsElement); + } + catch (PipelineException ex) + { + _logger?.LogError(ex, "Failed to get data from {flowElementType}", jsElement.GetType().Name); + jsData = jsElement.GetFallbackResponse(flowData, GetJsonData()); + } + return jsData; + }; + switch (contentType) { case ContentType.JavaScript: - var jsElement = flowData.Pipeline.GetElement(); - if (jsElement == null) - { - throw new PipelineConfigurationException( - Messages.ExceptionNoJavaScriptBuilder); - } - IJavaScriptBuilderElementData jsData; - try - { - jsData = flowData.GetFromElement(jsElement); - } - catch (PipelineException ex) - { - _logger?.LogError(ex, "Failed to get data from {flowElementType}", jsElement.GetType().Name); - jsData = jsElement.GetFallbackResponse(flowData); - } - content = jsData?.JavaScript; + content = GetJsData()?.JavaScript; break; case ContentType.Json: - var jsonElement = flowData.Pipeline.GetElement(); - if (jsonElement == null) - { - throw new PipelineConfigurationException( - Messages.ExceptionNoJsonBuilder); - } - IJsonBuilderElementData jsonData = null; - try - { - jsonData = flowData.GetFromElement(jsonElement); - } - catch (PipelineException ex) - { - _logger?.LogError(ex, "Failed to get data from {flowElementType}", jsonElement.GetType().Name); - } - content = jsonData?.Json; + content = GetJsonData()?.Json; break; default: break; From 9590695dfc325e559f50fb5b8718ddcf62a63d9e Mon Sep 17 00:00:00 2001 From: Maksym Kucherov Date: Thu, 30 Nov 2023 11:15:15 +0200 Subject: [PATCH 10/13] Add `jsonElement.GetFallbackResponse` call. --- .../Services/ClientsidePropertyService.cs | 1 + 1 file changed, 1 insertion(+) diff --git a/FiftyOne.Pipeline.Web.Shared/Services/ClientsidePropertyService.cs b/FiftyOne.Pipeline.Web.Shared/Services/ClientsidePropertyService.cs index 9e0c8530..1f4e33c9 100644 --- a/FiftyOne.Pipeline.Web.Shared/Services/ClientsidePropertyService.cs +++ b/FiftyOne.Pipeline.Web.Shared/Services/ClientsidePropertyService.cs @@ -257,6 +257,7 @@ private void ServeContent(IContextAdapter context, IFlowData flowData, ContentTy catch (PipelineException ex) { _logger?.LogError(ex, "Failed to get data from {flowElementType}", jsonElement.GetType().Name); + jsonData = jsonElement.GetFallbackResponse(flowData); } return jsonData; }; From b3ae11208ce04b4236d30d3f69e83c0264ca951a Mon Sep 17 00:00:00 2001 From: Maksym Kucherov Date: Thu, 30 Nov 2023 12:02:32 +0200 Subject: [PATCH 11/13] Suppress throwing due to no sequence number when evaluating JSON for fallback response. --- .../FlowElement/JsonBuilderElement.cs | 31 ++++++++++++++----- 1 file changed, 24 insertions(+), 7 deletions(-) diff --git a/FiftyOne.Pipeline.Elements/FiftyOne.Pipeline.JsonBuilderElement/FlowElement/JsonBuilderElement.cs b/FiftyOne.Pipeline.Elements/FiftyOne.Pipeline.JsonBuilderElement/FlowElement/JsonBuilderElement.cs index 96a66142..a74f0ec5 100644 --- a/FiftyOne.Pipeline.Elements/FiftyOne.Pipeline.JsonBuilderElement/FlowElement/JsonBuilderElement.cs +++ b/FiftyOne.Pipeline.Elements/FiftyOne.Pipeline.JsonBuilderElement/FlowElement/JsonBuilderElement.cs @@ -220,7 +220,8 @@ protected override void ProcessInternal(IFlowData data) => data, data.GetOrAdd( ElementDataKeyTyped, - CreateElementData)); + CreateElementData), + true); /// /// Transform the data in the flow data instance into a @@ -232,10 +233,13 @@ protected override void ProcessInternal(IFlowData data) => /// /// The /// + /// + /// Whether to throw if sequence number was not found + /// /// /// Thrown if the supplied flow data is null. /// - private void BuildAndInjectJSON(IFlowData data, IJsonBuilderElementData elementData) + private void BuildAndInjectJSON(IFlowData data, IJsonBuilderElementData elementData, bool requireSequenceNumber) { if (data == null) throw new ArgumentNullException(nameof(data)); @@ -245,7 +249,7 @@ private void BuildAndInjectJSON(IFlowData data, IJsonBuilderElementData elementD config = _pipelineConfigs.GetOrAdd(data.Pipeline, config); } - var jsonString = BuildJson(data, config); + var jsonString = BuildJson(data, config, requireSequenceNumber); elementData.Json = jsonString; } @@ -262,7 +266,7 @@ private void BuildAndInjectJSON(IFlowData data, IJsonBuilderElementData elementD public IJsonBuilderElementData GetFallbackResponse(IFlowData data) { var elementData = CreateElementData(data.Pipeline); - BuildAndInjectJSON(data, elementData); + BuildAndInjectJSON(data, elementData, false); return elementData; } @@ -271,12 +275,25 @@ public IJsonBuilderElementData GetFallbackResponse(IFlowData data) /// /// /// The configuration to use + /// The configuration to use /// /// A string containing the data in JSON format. /// - protected virtual string BuildJson(IFlowData data, PipelineConfig config) + protected virtual string BuildJson(IFlowData data, PipelineConfig config, bool requireSequenceNumber) { - int sequenceNumber = GetSequenceNumber(data); + int? sequenceNumber = null; + try + { + sequenceNumber = GetSequenceNumber(data); + } + catch (PipelineException e) + { + if (requireSequenceNumber) + { + throw; + } + Logger.LogError(e, $"Failed to get {nameof(sequenceNumber)}."); + } // Get property values from all the elements and add the ones that // are accessible to a dictionary. @@ -284,7 +301,7 @@ protected virtual string BuildJson(IFlowData data, PipelineConfig config) // Only populate the JavaScript properties if the sequence // has not reached max iterations. - if (sequenceNumber < Constants.MAX_JAVASCRIPT_ITERATIONS) + if (!sequenceNumber.HasValue || sequenceNumber.Value < Constants.MAX_JAVASCRIPT_ITERATIONS) { AddJavaScriptProperties(data, allProperties); } From f770bab0bdb831665dfd062e7571d23e2c3ce6f4 Mon Sep 17 00:00:00 2001 From: Maksym Kucherov Date: Thu, 30 Nov 2023 12:36:41 +0200 Subject: [PATCH 12/13] Wrap `data.GetAs` calls in try-catch. --- .../FlowElement/JavaScriptBuilderElement.cs | 37 ++++++++++++++++--- 1 file changed, 32 insertions(+), 5 deletions(-) diff --git a/FiftyOne.Pipeline.Elements/FiftyOne.Pipeline.JavaScriptBuilderElement/FlowElement/JavaScriptBuilderElement.cs b/FiftyOne.Pipeline.Elements/FiftyOne.Pipeline.JavaScriptBuilderElement/FlowElement/JavaScriptBuilderElement.cs index 8b6c0f14..356ceb7f 100644 --- a/FiftyOne.Pipeline.Elements/FiftyOne.Pipeline.JavaScriptBuilderElement/FlowElement/JavaScriptBuilderElement.cs +++ b/FiftyOne.Pipeline.Elements/FiftyOne.Pipeline.JavaScriptBuilderElement/FlowElement/JavaScriptBuilderElement.cs @@ -245,7 +245,7 @@ protected override void ManagedResourcesCleanup() /// protected override void ProcessInternal(IFlowData data) { - SetUp(data, GetJSONFromData(data), GetOrAddToData(data)); + SetUp(data, GetJSONFromData(data), GetOrAddToData(data), true); } @@ -264,7 +264,7 @@ public JavaScriptBuilderElementData GetFallbackResponse(IFlowData data, IJsonBui throw new ArgumentNullException(nameof(jsonData)); } JavaScriptBuilderElementData result = (JavaScriptBuilderElementData)CreateElementData(data.Pipeline); - SetUp(data, () => jsonData, () => result); + SetUp(data, () => jsonData, () => result, false); return result; } @@ -284,7 +284,8 @@ private static Func GetJSONFromData(IFlowData data) => private void SetUp( IFlowData data, Func jsonDataProvider, - Func targetElementDataProvider) + Func targetElementDataProvider, + bool throwOnGetAsFailure) { if (data == null) throw new ArgumentNullException(nameof(data)); var host = Host; @@ -312,6 +313,8 @@ private void SetUp( protocol = Constants.FALLBACK_PROTOCOL; } + const string errorFormat_GetAsFailed = "Failed to get property {propertyName}"; + // If device detection is in the Pipeline then we can check // if the client's browser supports promises. // This can be used to customize the JavaScript response. @@ -328,7 +331,19 @@ private void SetUp( try { - var promise = data.GetAs>("Promise"); + IAspectPropertyValue promise = null; + try + { + promise = data.GetAs>("Promise"); + } + catch (Exception ex) + { + if (throwOnGetAsFailure) + { + throw; + } + Logger.LogError(ex, errorFormat_GetAsFailed, "Promise"); + } supportsPromises = promise != null && promise.HasValue && promise.Value == "Full"; } catch (PropertyMissingException) { promisesNotAvailable(); } @@ -353,7 +368,19 @@ private void SetUp( try { - var fetch = data.GetAs>("Fetch"); + IAspectPropertyValue fetch = null; + try + { + fetch = data.GetAs>("Fetch"); + } + catch (Exception ex) + { + if (throwOnGetAsFailure) + { + throw; + } + Logger.LogError(ex, errorFormat_GetAsFailed, "Fetch"); + } supportsFetch = fetch != null && fetch.HasValue && fetch.Value; } catch (PropertyMissingException) { fetchNotAvailable(); } From fb21348f4020b51c71c409772bae37e8e16ac176 Mon Sep 17 00:00:00 2001 From: Maksym Kucherov Date: Fri, 1 Dec 2023 12:02:08 +0200 Subject: [PATCH 13/13] Set "Cache-Control" header to "no-cache" if `GetFallbackResponse` method was used. --- .../Services/ClientsidePropertyService.cs | 13 ++++++++++--- 1 file changed, 10 insertions(+), 3 deletions(-) diff --git a/FiftyOne.Pipeline.Web.Shared/Services/ClientsidePropertyService.cs b/FiftyOne.Pipeline.Web.Shared/Services/ClientsidePropertyService.cs index 1f4e33c9..12911648 100644 --- a/FiftyOne.Pipeline.Web.Shared/Services/ClientsidePropertyService.cs +++ b/FiftyOne.Pipeline.Web.Shared/Services/ClientsidePropertyService.cs @@ -213,6 +213,8 @@ private void ServeContent(IContextAdapter context, IFlowData flowData, ContentTy context.Response.Clear(); context.Response.ClearHeaders(); + bool hadFailures = false; + IEvidenceKeyFilter pipelineEvidenceKeyFilter = null; try { @@ -221,6 +223,7 @@ private void ServeContent(IContextAdapter context, IFlowData flowData, ContentTy catch (PipelineException ex) { _logger?.LogError(ex, $"Failed to get {nameof(_pipeline.EvidenceKeyFilter)} from {nameof(_pipeline)}"); + hadFailures = true; } // Get the hash code. @@ -258,6 +261,7 @@ private void ServeContent(IContextAdapter context, IFlowData flowData, ContentTy { _logger?.LogError(ex, "Failed to get data from {flowElementType}", jsonElement.GetType().Name); jsonData = jsonElement.GetFallbackResponse(flowData); + hadFailures = true; } return jsonData; }; @@ -279,6 +283,7 @@ private void ServeContent(IContextAdapter context, IFlowData flowData, ContentTy { _logger?.LogError(ex, "Failed to get data from {flowElementType}", jsElement.GetType().Name); jsData = jsElement.GetFallbackResponse(flowData, GetJsonData()); + hadFailures = true; } return jsData; }; @@ -305,7 +310,8 @@ private void ServeContent(IContextAdapter context, IFlowData flowData, ContentTy SetHeaders(context, hash.HasValue ? hash.Value.ToString(CultureInfo.InvariantCulture) : null, length, - contentType == ContentType.JavaScript ? "x-javascript" : "json"); + contentType == ContentType.JavaScript ? "x-javascript" : "json", + !hadFailures); context.Response.Write(content); } @@ -319,7 +325,8 @@ private void ServeContent(IContextAdapter context, IFlowData flowData, ContentTy /// /// /// - private void SetHeaders(IContextAdapter context, string hash, int contentLength, string contentType) + /// + private void SetHeaders(IContextAdapter context, string hash, int contentLength, string contentType, bool shouldCache) { try { @@ -327,7 +334,7 @@ private void SetHeaders(IContextAdapter context, string hash, int contentLength, $"application/{contentType}"); context.Response.SetHeader("Content-Length", contentLength.ToString(CultureInfo.InvariantCulture)); - context.Response.SetHeader("Cache-Control", _cacheControl); + context.Response.SetHeader("Cache-Control", shouldCache ? _cacheControl.ToString() : "no-cache"); var headersAffectingJavaScript = HeadersAffectingJavaScript; if (string.IsNullOrEmpty(headersAffectingJavaScript.ToString()) == false) {