-
Notifications
You must be signed in to change notification settings - Fork 97
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
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.
- Loading branch information
Showing
4 changed files
with
144 additions
and
7 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -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<string> 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<dot:TextBox Text=Initial />"); | ||
|
||
var config = debug ? DotvvmTestHelper.DebugConfig : DotvvmTestHelper.DefaultConfig; | ||
|
||
var controlBuilder = config.ServiceProvider.GetRequiredService<IControlBuilderFactory>(); | ||
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<dot:TextBox Text=Changed />"); | ||
|
||
var builderChanged = controlBuilder.GetControlBuilder(file); | ||
var control = builderChanged.builder.Value.BuildControl(config.ServiceProvider.GetRequiredService<IControlBuilderFactory>(), config.ServiceProvider); | ||
if (debug) | ||
{ | ||
Assert.AreEqual(typeof(int), builderChanged.descriptor.DataContextType); | ||
Check failure on line 80 in src/Tests/Runtime/MarkupLoaderTests.cs GitHub Actions / .NET unit tests (windows-2022)FileReloading (True)
|
||
Assert.AreNotSame(builder0.builder, builderChanged.builder); // different Lazy instance | ||
XAssert.Equal(["Changed"], control.GetThisAndAllDescendants().OfType<TextBox>().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<TextBox>().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); | ||
} | ||
} | ||
} | ||
} |