Merge commit from fork

This commit is contained in:
Stef Heyenrath
2025-03-04 17:58:38 +01:00
committed by GitHub
parent be55022a2a
commit ff36c1ee6f
8 changed files with 86 additions and 20 deletions
+5
View File
@@ -64,6 +64,11 @@ A breaking change is introduced which is related to System.Linq.Dynamic.Core Dyn
- The `LinqMatcher` is not allowed. - The `LinqMatcher` is not allowed.
- The [Handlebars.Net.Helpers.DynamicLinq](https://www.nuget.org/packages/Handlebars.Net.Helpers.DynamicLinq) package is not included anymore. - The [Handlebars.Net.Helpers.DynamicLinq](https://www.nuget.org/packages/Handlebars.Net.Helpers.DynamicLinq) package is not included anymore.
### 1.8.0
A breaking change is introduced which is related to the usage of the custom Handlebars.Net `File`-helper.
By default this is not allowed anymore because of security reasons (insecure server-side template injection).
To still enable this feature, you need to set the `AllowedCustomHandlebarHelpers` property to `File` in the `WireMockServerSettings` class.
--- ---
## :memo: Development ## :memo: Development
@@ -0,0 +1,16 @@
using System;
namespace WireMock.Types;
/// <summary>
/// A enum defining the supported Handlebar helpers.
/// </summary>
[Flags]
public enum CustomHandlebarHelpers
{
None = 0,
File = 1,
All = File
}
@@ -329,4 +329,13 @@ public class WireMockServerSettings
/// </summary> /// </summary>
[PublicAPI] [PublicAPI]
public string? AdminPath { get; set; } public string? AdminPath { get; set; }
/// <summary>
/// Defines the allowed custom HandlebarHelpers which can be used. Possible values are:
/// - <see cref="CustomHandlebarHelpers.None"/> (Default)
/// - <see cref="CustomHandlebarHelpers.File"/>
/// - <see cref="CustomHandlebarHelpers.All"/>
/// </summary>
[PublicAPI]
public CustomHandlebarHelpers AllowedCustomHandlebarHelpers { get; set; } = CustomHandlebarHelpers.None;
} }
@@ -50,6 +50,7 @@ public static class WireMockServerSettingsParser
AdminPath = parser.GetStringValue(nameof(WireMockServerSettings.AdminPath), "/__admin"), AdminPath = parser.GetStringValue(nameof(WireMockServerSettings.AdminPath), "/__admin"),
AllowBodyForAllHttpMethods = parser.GetBoolValue(nameof(WireMockServerSettings.AllowBodyForAllHttpMethods)), AllowBodyForAllHttpMethods = parser.GetBoolValue(nameof(WireMockServerSettings.AllowBodyForAllHttpMethods)),
AllowCSharpCodeMatcher = parser.GetBoolValue(nameof(WireMockServerSettings.AllowCSharpCodeMatcher)), AllowCSharpCodeMatcher = parser.GetBoolValue(nameof(WireMockServerSettings.AllowCSharpCodeMatcher)),
AllowedCustomHandlebarHelpers = parser.GetEnumValue(nameof(WireMockServerSettings.AllowedCustomHandlebarHelpers), CustomHandlebarHelpers.None),
AllowOnlyDefinedHttpStatusCodeInResponse = parser.GetBoolValue(nameof(WireMockServerSettings.AllowOnlyDefinedHttpStatusCodeInResponse)), AllowOnlyDefinedHttpStatusCodeInResponse = parser.GetBoolValue(nameof(WireMockServerSettings.AllowOnlyDefinedHttpStatusCodeInResponse)),
AllowPartialMapping = parser.GetBoolValue(nameof(WireMockServerSettings.AllowPartialMapping)), AllowPartialMapping = parser.GetBoolValue(nameof(WireMockServerSettings.AllowPartialMapping)),
Culture = parser.GetValue(nameof(WireMockServerSettings.Culture), strings => CultureInfoUtils.Parse(strings.FirstOrDefault()), CultureInfo.CurrentCulture), Culture = parser.GetValue(nameof(WireMockServerSettings.Culture), strings => CultureInfoUtils.Parse(strings.FirstOrDefault()), CultureInfo.CurrentCulture),
@@ -11,6 +11,8 @@ namespace WireMock.Transformers.Handlebars;
internal class FileHelpers : BaseHelpers, IHelpers internal class FileHelpers : BaseHelpers, IHelpers
{ {
internal const string Name = "File";
private readonly IFileSystemHandler _fileSystemHandler; private readonly IFileSystemHandler _fileSystemHandler;
public FileHelpers(IHandlebars context, IFileSystemHandler fileSystemHandler) : base(context) public FileHelpers(IHandlebars context, IFileSystemHandler fileSystemHandler) : base(context)
@@ -18,12 +20,12 @@ internal class FileHelpers : BaseHelpers, IHelpers
_fileSystemHandler = Guard.NotNull(fileSystemHandler); _fileSystemHandler = Guard.NotNull(fileSystemHandler);
} }
[HandlebarsWriter(WriterType.String, usage: HelperUsage.Both, passContext: true, name: "File")] [HandlebarsWriter(WriterType.String, usage: HelperUsage.Both, passContext: true, name: Name)]
public string Read(Context context, string path) public string Read(Context context, string path)
{ {
var templateFunc = Context.Compile(path); var templateFunc = Context.Compile(path);
var transformed = templateFunc(context.Value); var transformedPath = templateFunc(context.Value);
return _fileSystemHandler.ReadResponseBodyAsString(transformed); return _fileSystemHandler.ReadResponseBodyAsString(transformedPath);
} }
public Category Category => Category.Custom; public Category Category => Category.Custom;
@@ -23,7 +23,7 @@ internal class HandlebarsContextFactory : ITransformerContextFactory
}; };
var handlebars = HandlebarsDotNet.Handlebars.Create(config); var handlebars = HandlebarsDotNet.Handlebars.Create(config);
WireMockHandlebarsHelpers.Register(handlebars, _settings.FileSystemHandler); WireMockHandlebarsHelpers.Register(handlebars, _settings);
_settings.HandlebarsRegistrationCallback?.Invoke(handlebars, _settings.FileSystemHandler); _settings.HandlebarsRegistrationCallback?.Invoke(handlebars, _settings.FileSystemHandler);
@@ -2,21 +2,20 @@
using System; using System;
using System.Collections.Generic; using System.Collections.Generic;
using System.Diagnostics;
using System.IO; using System.IO;
using System.Reflection;
using HandlebarsDotNet; using HandlebarsDotNet;
using HandlebarsDotNet.Helpers; using HandlebarsDotNet.Helpers;
using HandlebarsDotNet.Helpers.Helpers; using HandlebarsDotNet.Helpers.Helpers;
using WireMock.Handlers; using WireMock.Settings;
using WireMock.Types;
namespace WireMock.Transformers.Handlebars; namespace WireMock.Transformers.Handlebars;
internal static class WireMockHandlebarsHelpers internal static class WireMockHandlebarsHelpers
{ {
public static void Register(IHandlebars handlebarsContext, IFileSystemHandler fileSystemHandler) internal static void Register(IHandlebars handlebarsContext, WireMockServerSettings settings)
{ {
// Register https://github.com/StefH/Handlebars.Net.Helpers // Register https://github.com/Handlebars.Net/Handlebars.Net.Helpers
HandlebarsHelpers.Register(handlebarsContext, o => HandlebarsHelpers.Register(handlebarsContext, o =>
{ {
var paths = new List<string> var paths = new List<string>
@@ -33,17 +32,18 @@ internal static class WireMockHandlebarsHelpers
customHelperPaths.Add(path!); customHelperPaths.Add(path!);
} }
} }
Add(Path.GetDirectoryName(Assembly.GetEntryAssembly()?.Location), paths); Add(Path.GetDirectoryName(System.Reflection.Assembly.GetEntryAssembly()?.Location), paths);
Add(Path.GetDirectoryName(Assembly.GetCallingAssembly().Location), paths); Add(Path.GetDirectoryName(System.Reflection.Assembly.GetCallingAssembly().Location), paths);
Add(Path.GetDirectoryName(Assembly.GetExecutingAssembly().Location), paths); Add(Path.GetDirectoryName(System.Reflection.Assembly.GetExecutingAssembly().Location), paths);
Add(Path.GetDirectoryName(Process.GetCurrentProcess().MainModule?.FileName), paths); Add(Path.GetDirectoryName(System.Diagnostics.Process.GetCurrentProcess().MainModule?.FileName), paths);
#endif #endif
o.CustomHelperPaths = paths; o.CustomHelperPaths = paths;
o.CustomHelpers = new Dictionary<string, IHelpers> o.CustomHelpers = new Dictionary<string, IHelpers>();
if (settings.AllowedCustomHandlebarHelpers.HasFlag(CustomHandlebarHelpers.File))
{ {
{ "File", new FileHelpers(handlebarsContext, fileSystemHandler) } o.CustomHelpers.Add(FileHelpers.Name, new FileHelpers(handlebarsContext, settings.FileSystemHandler));
}; }
}); });
} }
@@ -1,7 +1,10 @@
// Copyright © WireMock.Net // Copyright © WireMock.Net
using System;
using System.Threading.Tasks; using System.Threading.Tasks;
using FluentAssertions;
using HandlebarsDotNet; using HandlebarsDotNet;
using HandlebarsDotNet.Helpers;
using Moq; using Moq;
using Newtonsoft.Json.Linq; using Newtonsoft.Json.Linq;
using NFluent; using NFluent;
@@ -9,15 +12,17 @@ using WireMock.Handlers;
using WireMock.Models; using WireMock.Models;
using WireMock.ResponseBuilders; using WireMock.ResponseBuilders;
using WireMock.Settings; using WireMock.Settings;
using WireMock.Transformers.Handlebars;
using WireMock.Types;
using Xunit; using Xunit;
namespace WireMock.Net.Tests.ResponseBuilders; namespace WireMock.Net.Tests.ResponseBuilders;
public class ResponseWithHandlebarsFileTests public class ResponseWithHandlebarsFileTests
{ {
private readonly WireMockServerSettings _settings = new();
private const string ClientIp = "::1"; private const string ClientIp = "::1";
private readonly WireMockServerSettings _settings;
private readonly Mock<IMapping> _mappingMock; private readonly Mock<IMapping> _mappingMock;
private readonly Mock<IFileSystemHandler> _filesystemHandlerMock; private readonly Mock<IFileSystemHandler> _filesystemHandlerMock;
@@ -28,7 +33,11 @@ public class ResponseWithHandlebarsFileTests
_filesystemHandlerMock = new Mock<IFileSystemHandler>(MockBehavior.Strict); _filesystemHandlerMock = new Mock<IFileSystemHandler>(MockBehavior.Strict);
_filesystemHandlerMock.Setup(fs => fs.ReadResponseBodyAsString(It.IsAny<string>())).Returns("abc"); _filesystemHandlerMock.Setup(fs => fs.ReadResponseBodyAsString(It.IsAny<string>())).Returns("abc");
_settings.FileSystemHandler = _filesystemHandlerMock.Object; _settings = new()
{
AllowedCustomHandlebarHelpers = CustomHandlebarHelpers.File,
FileSystemHandler = _filesystemHandlerMock.Object
};
} }
[Fact] [Fact]
@@ -48,7 +57,7 @@ public class ResponseWithHandlebarsFileTests
var response = await responseBuilder.ProvideResponseAsync(_mappingMock.Object, request, _settings).ConfigureAwait(false); var response = await responseBuilder.ProvideResponseAsync(_mappingMock.Object, request, _settings).ConfigureAwait(false);
// Assert // Assert
JObject j = JObject.FromObject(response.Message.BodyData.BodyAsJson); var j = JObject.FromObject(response.Message.BodyData.BodyAsJson);
Check.That(j["Data"].Value<string>()).Equals("abc"); Check.That(j["Data"].Value<string>()).Equals("abc");
// Verify // Verify
@@ -73,7 +82,7 @@ public class ResponseWithHandlebarsFileTests
var response = await responseBuilder.ProvideResponseAsync(_mappingMock.Object, request, _settings).ConfigureAwait(false); var response = await responseBuilder.ProvideResponseAsync(_mappingMock.Object, request, _settings).ConfigureAwait(false);
// Assert // Assert
JObject j = JObject.FromObject(response.Message.BodyData.BodyAsJson); var j = JObject.FromObject(response.Message.BodyData.BodyAsJson);
Check.That(j["Data"].Value<string>()).Equals("abc"); Check.That(j["Data"].Value<string>()).Equals("abc");
// Verify // Verify
@@ -101,4 +110,28 @@ public class ResponseWithHandlebarsFileTests
_filesystemHandlerMock.Verify(fs => fs.ReadResponseBodyAsString(It.IsAny<string>()), Times.Never); _filesystemHandlerMock.Verify(fs => fs.ReadResponseBodyAsString(It.IsAny<string>()), Times.Never);
_filesystemHandlerMock.VerifyNoOtherCalls(); _filesystemHandlerMock.VerifyNoOtherCalls();
} }
[Fact]
public void Response_ProvideResponseAsync_Handlebars_File_NotAllowed_Throws_HandlebarsRuntimeException()
{
// Assign
var settings = new WireMockServerSettings
{
AllowedCustomHandlebarHelpers = CustomHandlebarHelpers.None,
FileSystemHandler = _filesystemHandlerMock.Object
};
var request = new RequestMessage(new UrlDetails("http://localhost:1234?id=x"), "GET", ClientIp);
var responseBuilder = Response.Create()
.WithBody("{{File \"{{request.query.id}}.json\"}}")
.WithTransformer();
// Act
Func<Task> action = () => responseBuilder.ProvideResponseAsync(_mappingMock.Object, request, settings);
action.Should().ThrowAsync<HandlebarsRuntimeException>();
// Verify
_filesystemHandlerMock.VerifyNoOtherCalls();
}
} }