From 5b9904d4f1b3ec222fb4bf7d22e630bb122fe9cb Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Tom=C3=A1=C5=A1=20Herceg?= Date: Wed, 6 Nov 2024 09:12:11 +0100 Subject: [PATCH 01/14] Upgraded actions/upload-artifact from deprecated version --- .github/test-report/action.yml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/.github/test-report/action.yml b/.github/test-report/action.yml index d4b2243338..32c421934d 100644 --- a/.github/test-report/action.yml +++ b/.github/test-report/action.yml @@ -21,7 +21,7 @@ inputs: runs: using: composite steps: - - uses: actions/upload-artifact@v3 + - uses: actions/upload-artifact@v4 with: name: ${{ inputs.report-name }} path: ${{ inputs.trx-path }} From 3724a7d77f071342397f7c3c660c91b90ccb8833 Mon Sep 17 00:00:00 2001 From: "dependabot[bot]" <49699333+dependabot[bot]@users.noreply.github.com> Date: Tue, 19 Nov 2024 02:22:35 +0000 Subject: [PATCH 02/14] Bump cross-spawn from 7.0.3 to 7.0.6 in /src/Framework/Framework Bumps [cross-spawn](https://github.com/moxystudio/node-cross-spawn) from 7.0.3 to 7.0.6. - [Changelog](https://github.com/moxystudio/node-cross-spawn/blob/master/CHANGELOG.md) - [Commits](https://github.com/moxystudio/node-cross-spawn/compare/v7.0.3...v7.0.6) --- updated-dependencies: - dependency-name: cross-spawn dependency-type: indirect ... Signed-off-by: dependabot[bot] --- src/Framework/Framework/yarn.lock | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/src/Framework/Framework/yarn.lock b/src/Framework/Framework/yarn.lock index b1031bed0b..e12edd24fa 100644 --- a/src/Framework/Framework/yarn.lock +++ b/src/Framework/Framework/yarn.lock @@ -1972,13 +1972,13 @@ __metadata: linkType: hard "cross-spawn@npm:^7.0.3": - version: 7.0.3 - resolution: "cross-spawn@npm:7.0.3" + version: 7.0.6 + resolution: "cross-spawn@npm:7.0.6" dependencies: path-key: ^3.1.0 shebang-command: ^2.0.0 which: ^2.0.1 - checksum: 671cc7c7288c3a8406f3c69a3ae2fc85555c04169e9d611def9a675635472614f1c0ed0ef80955d5b6d4e724f6ced67f0ad1bb006c2ea643488fcfef994d7f52 + checksum: 8d306efacaf6f3f60e0224c287664093fa9185680b2d195852ba9a863f85d02dcc737094c6e512175f8ee0161f9b87c73c6826034c2422e39de7d6569cf4503b languageName: node linkType: hard From e67a0699918332fa3ee5b06d7bd12253e3ab27a6 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Standa=20Luke=C5=A1?= Date: Sat, 23 Nov 2024 20:10:03 +0100 Subject: [PATCH 03/14] Fix excessive allocations in EmbeddedResourceFileLoader Previously we'd load the entire resource on every invocation of GetControlBuilder, which might be a number of times per request in case of markup controls. --- .../Hosting/EmbeddedMarkupFileLoader.cs | 19 +-- src/Framework/Framework/Hosting/MarkupFile.cs | 8 ++ src/Framework/Testing/DotvvmTestHelper.cs | 12 ++ src/Tests/Runtime/MarkupLoaderTests.cs | 112 ++++++++++++++++++ 4 files changed, 144 insertions(+), 7 deletions(-) create mode 100644 src/Tests/Runtime/MarkupLoaderTests.cs diff --git a/src/Framework/Framework/Hosting/EmbeddedMarkupFileLoader.cs b/src/Framework/Framework/Hosting/EmbeddedMarkupFileLoader.cs index c424c37c7a..05e8e29dcf 100644 --- a/src/Framework/Framework/Hosting/EmbeddedMarkupFileLoader.cs +++ b/src/Framework/Framework/Hosting/EmbeddedMarkupFileLoader.cs @@ -1,12 +1,15 @@ using System; +using System.Buffers; using System.Collections.Generic; using System.IO; using System.Reflection; using System.Text; using DotVVM.Framework.Configuration; +using DotVVM.Framework.Utils; namespace DotVVM.Framework.Hosting { + /// Read markup files from embedded resources, if the virtual path has the following format: embedded://{AssemblyName}/{ResourceName} public class EmbeddedMarkupFileLoader : IMarkupFileLoader { /// @@ -23,7 +26,7 @@ public class EmbeddedMarkupFileLoader : IMarkupFileLoader if (resourceName.IndexOf('/') == -1 || resourceName.IndexOf('/') == 0) { - throw new ArgumentException("Wrong string format", "virtualPath"); + throw new ArgumentException("Wrong embedded file format. Use `embedded://{AssemblyName}/{ResourceName}`", "virtualPath"); } string assemblyName = resourceName.Substring(0, resourceName.IndexOf('/')); @@ -37,20 +40,22 @@ public class EmbeddedMarkupFileLoader : IMarkupFileLoader //no such assembly found catch (FileLoadException) { - throw new ArgumentException("No such assembly was found", "virtualPath"); + throw new ArgumentException($"Assembly '{assemblyName}' was not found", "virtualPath"); } //no such resource found resourceName = resourceName.Replace('/', '.'); if (assembly.GetManifestResourceInfo(resourceName) == null) { - throw new ArgumentException("No such resource was found", "virtualPath"); + throw new ArgumentException($"Resource '{resourceName}' was not found in assembly '{assembly.FullName}'", "virtualPath"); } - //load the file - using (Stream stream = assembly.GetManifestResourceStream(resourceName)!) - using (StreamReader sr = new StreamReader(stream)) - return new MarkupFile(virtualPath, virtualPath, sr.ReadToEnd()); + return new MarkupFile(virtualPath, virtualPath, () => { + //load the file + using (Stream stream = assembly.GetManifestResourceStream(resourceName)!) + using (var reader = new StreamReader(stream)) + return reader.ReadToEnd(); + }); } /// diff --git a/src/Framework/Framework/Hosting/MarkupFile.cs b/src/Framework/Framework/Hosting/MarkupFile.cs index 23d9870730..e7b190a52f 100644 --- a/src/Framework/Framework/Hosting/MarkupFile.cs +++ b/src/Framework/Framework/Hosting/MarkupFile.cs @@ -58,6 +58,14 @@ public MarkupFile(string fileName, string fullPath) }; } + public MarkupFile(string fileName, string fullPath, Func readContent, DateTime lastWriteDateTimeUtc = default) + { + FileName = fileName; + FullPath = fullPath; + ReadContent = readContent; + LastWriteDateTimeUtc = lastWriteDateTimeUtc; + } + public MarkupFile(string fileName, string fullPath, string contents, DateTime lastWriteDateTimeUtc = default) { FileName = fileName; diff --git a/src/Framework/Testing/DotvvmTestHelper.cs b/src/Framework/Testing/DotvvmTestHelper.cs index f265e21006..269b42981f 100644 --- a/src/Framework/Testing/DotvvmTestHelper.cs +++ b/src/Framework/Testing/DotvvmTestHelper.cs @@ -100,11 +100,23 @@ public static void RegisterMockServices(IServiceCollection services) var config = CreateConfiguration(); config.ExperimentalFeatures.UseDotvvmSerializationForStaticCommandArguments.Enable(); config.RouteTable.Add("TestRoute", "TestRoute", "TestView.dothtml"); + config.Diagnostics.Apply(config); config.Freeze(); return config; }); public static DotvvmConfiguration DefaultConfig => _defaultConfig.Value; + private static Lazy _debugConfig = new Lazy(() => { + var config = CreateConfiguration(); + config.ExperimentalFeatures.UseDotvvmSerializationForStaticCommandArguments.Enable(); + config.RouteTable.Add("TestRoute", "TestRoute", "TestView.dothtml"); + config.Debug = true; + config.Diagnostics.Apply(config); + config.Freeze(); + return config; + }); + public static DotvvmConfiguration DebugConfig => _debugConfig.Value; + public static DotvvmConfiguration CreateConfiguration(Action? customServices = null) => DotvvmConfiguration.CreateDefault(s => { s.AddSingleton(); diff --git a/src/Tests/Runtime/MarkupLoaderTests.cs b/src/Tests/Runtime/MarkupLoaderTests.cs new file mode 100644 index 0000000000..1c8eaf1ee4 --- /dev/null +++ b/src/Tests/Runtime/MarkupLoaderTests.cs @@ -0,0 +1,112 @@ +using System; +using System.Collections.Generic; +using System.IO; +using System.Linq; +using DotVVM.Framework.Compilation; +using DotVVM.Framework.Configuration; +using DotVVM.Framework.Controls; +using DotVVM.Framework.Hosting; +using DotVVM.Framework.Runtime; +using DotVVM.Framework.Testing; +using Microsoft.Extensions.DependencyInjection; +using Microsoft.VisualStudio.TestTools.UnitTesting; + +namespace DotVVM.Framework.Tests.Runtime +{ + [TestClass] + public class MarkupLoaderTests: IDisposable + { + readonly string compilationPageResource = DotvvmTestHelper.DebugConfig.RouteTable[DotvvmCompilationPageConfiguration.DefaultRouteName].VirtualPath; + + readonly List tempFiles = []; + + [TestMethod] + public void EmbeddedResource() + { + var loader = new EmbeddedMarkupFileLoader(); + var file = loader.GetMarkup(DotvvmTestHelper.DebugConfig, compilationPageResource); + + XAssert.StartsWith("@viewModel DotVVM.Framework.Diagnostics.CompilationPageViewModel", file.ReadContent()); + } + +#if DotNetCore + [TestMethod] + public void EmbeddedResourceLazyAllocation() + { + var loader = new EmbeddedMarkupFileLoader(); + // warmup for assembly loading and such + loader.GetMarkup(DotvvmTestHelper.DebugConfig, compilationPageResource); + + // GetMarkup allocates constant memory, as it is being called repeatedly if file reloading is enabled + var a = GC.GetAllocatedBytesForCurrentThread(); + var file = loader.GetMarkup(DotvvmTestHelper.DebugConfig, compilationPageResource); + var b = GC.GetAllocatedBytesForCurrentThread(); + XAssert.InRange(b - a, 0, 1000); + + // ReadContent actually reads the file and allocates the string + a = GC.GetAllocatedBytesForCurrentThread(); + var content = file.ReadContent(); + b = GC.GetAllocatedBytesForCurrentThread(); + XAssert.InRange(content.Length, 1000, int.MaxValue); + XAssert.InRange(b - a, content.Length * 2, content.Length * 5); + } +#endif + + [TestMethod] + [DataRow(true)] + [DataRow(false)] + public void FileReloading(bool debug) + { + var directory = MakeTempDir(); + var file = Path.Combine(directory, "test.dotcontrol"); + File.WriteAllText(file, "@viewModel string\n\n"); + + var config = debug ? DotvvmTestHelper.DebugConfig : DotvvmTestHelper.DefaultConfig; + + var controlBuilder = config.ServiceProvider.GetRequiredService(); + var builder0 = controlBuilder.GetControlBuilder(file); + Assert.AreEqual(typeof(string), builder0.descriptor.DataContextType); + + var builderUnchanged = controlBuilder.GetControlBuilder(file); + + Assert.AreSame(builder0.builder, builderUnchanged.builder); // same Lazy instance + + File.WriteAllText(file, "@viewModel int\n\n"); + + var builderChanged = controlBuilder.GetControlBuilder(file); + var control = builderChanged.builder.Value.BuildControl(config.ServiceProvider.GetRequiredService(), config.ServiceProvider); + if (debug) + { + Assert.AreEqual(typeof(int), builderChanged.descriptor.DataContextType); + Assert.AreNotSame(builder0.builder, builderChanged.builder); // different Lazy instance + XAssert.Equal(["Changed"], control.GetThisAndAllDescendants().OfType().Select(c => c.Text)); + } + else + { + // not reloaded in Release mode by default + Assert.AreEqual(typeof(string), builderChanged.descriptor.DataContextType); + Assert.AreSame(builder0.builder, builderChanged.builder); // different Lazy instance + XAssert.Equal(["Initial"], control.GetThisAndAllDescendants().OfType().Select(c => c.Text)); + } + } + + public string MakeTempDir() + { + var path = Path.Combine(Path.GetTempPath(), "dotvvm-tests-tmp-" + Path.GetRandomFileName()); + Directory.CreateDirectory(path); + tempFiles.Add(path); + return path; + } + + public void Dispose() + { + foreach (var file in tempFiles) + { + if (Directory.Exists(file)) + Directory.Delete(file, true); + else if (File.Exists(file)) + File.Delete(file); + } + } + } +} From 79bd52c9da9ab3ec26bf95fbdcf122ab16be8a80 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Standa=20Luke=C5=A1?= Date: Sat, 23 Nov 2024 20:15:11 +0100 Subject: [PATCH 04/14] Remove Obsolete attribute from RegisterScriptModule this was most likely added by accided to both overload instead of only the following one. fixes #1887 --- .../ResourceManagement/ResourceRepositoryExtensions.cs | 3 +-- 1 file changed, 1 insertion(+), 2 deletions(-) diff --git a/src/Framework/Framework/ResourceManagement/ResourceRepositoryExtensions.cs b/src/Framework/Framework/ResourceManagement/ResourceRepositoryExtensions.cs index 0abb6a4ff9..5026850c01 100644 --- a/src/Framework/Framework/ResourceManagement/ResourceRepositoryExtensions.cs +++ b/src/Framework/Framework/ResourceManagement/ResourceRepositoryExtensions.cs @@ -77,7 +77,6 @@ public static LinkResourceBase RegisterScriptFile( /// Registers a from the specified file. /// The file can be anywhere in the filesystem, it does not have to be in the wwwroot folder. /// DotVVM will handle its serving, caching, ... automatically - [Obsolete("