mirror of
https://github.com/wiremock/WireMock.Net.git
synced 2026-03-31 14:43:40 +02:00
Fix ProtoBuf mapping.json (#1236)
* Fix ProtoBuf Mappings * [Fact(Skip = "#1233")] * fix? * PortUtils
This commit is contained in:
@@ -215,14 +215,16 @@ public interface IWireMockServer : IDisposable
|
|||||||
/// This can be used if you have 1 or more <see cref="MappingModel"/> defined and want to register these in WireMock.Net directly instead of using the fluent syntax.
|
/// This can be used if you have 1 or more <see cref="MappingModel"/> defined and want to register these in WireMock.Net directly instead of using the fluent syntax.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
/// <param name="mappings">The MappingModels</param>
|
/// <param name="mappings">The MappingModels</param>
|
||||||
|
/// <returns><see cref="IWireMockServer"/></returns>
|
||||||
IWireMockServer WithMapping(params MappingModel[] mappings);
|
IWireMockServer WithMapping(params MappingModel[] mappings);
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Register the mappings (via json string).
|
/// Register the mappings (via json string).
|
||||||
///
|
///
|
||||||
/// This can be used if you the mappings as json string defined and want to register these in WireMock.Net directly instead of using the fluent syntax.
|
/// This can be used if you've the mappings as json string defined and want to register these in WireMock.Net directly instead of using the fluent syntax.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
/// <param name="mappings">The mapping(s) as json string.</param>
|
/// <param name="mappings">The mapping(s) as json string.</param>
|
||||||
|
/// <returns><see cref="IWireMockServer"/></returns>
|
||||||
IWireMockServer WithMapping(string mappings);
|
IWireMockServer WithMapping(string mappings);
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
@@ -238,5 +240,5 @@ public interface IWireMockServer : IDisposable
|
|||||||
/// </summary>
|
/// </summary>
|
||||||
/// <param name="converterType">The <see cref="MappingConverterType"/></param>
|
/// <param name="converterType">The <see cref="MappingConverterType"/></param>
|
||||||
/// <returns>C# code</returns>
|
/// <returns>C# code</returns>
|
||||||
public string MappingsToCSharpCode(MappingConverterType converterType);
|
string MappingsToCSharpCode(MappingConverterType converterType);
|
||||||
}
|
}
|
||||||
@@ -130,7 +130,7 @@ public interface IWireMockAdminApi
|
|||||||
Task<StatusModel> ReloadStaticMappingsAsync(CancellationToken cancellationToken = default);
|
Task<StatusModel> ReloadStaticMappingsAsync(CancellationToken cancellationToken = default);
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Get a mapping based on the guid
|
/// Get a mapping based on the guid.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
/// <param name="guid">The Guid</param>
|
/// <param name="guid">The Guid</param>
|
||||||
/// <returns>MappingModel</returns>
|
/// <returns>MappingModel</returns>
|
||||||
@@ -138,6 +138,15 @@ public interface IWireMockAdminApi
|
|||||||
[Get("mappings/{guid}")]
|
[Get("mappings/{guid}")]
|
||||||
Task<MappingModel> GetMappingAsync([Path] Guid guid, CancellationToken cancellationToken = default);
|
Task<MappingModel> GetMappingAsync([Path] Guid guid, CancellationToken cancellationToken = default);
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Get a mapping based on the guid.
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="guid">The Guid</param>
|
||||||
|
/// <returns>MappingModel</returns>
|
||||||
|
/// <param name="cancellationToken">The optional cancellationToken.</param>
|
||||||
|
[Get("mappings/{guid}")]
|
||||||
|
Task<MappingModel> GetMappingAsync([Path] string guid, CancellationToken cancellationToken = default);
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Get the C# code from a mapping based on the guid
|
/// Get the C# code from a mapping based on the guid
|
||||||
/// </summary>
|
/// </summary>
|
||||||
|
|||||||
@@ -66,6 +66,12 @@ internal static class BodyDataMatchScoreCalculator
|
|||||||
return stringMatcher.IsMatch(requestMessage.BodyAsString);
|
return stringMatcher.IsMatch(requestMessage.BodyAsString);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// In case the matcher is a IProtoBufMatcher, use the BodyAsBytes to match on.
|
||||||
|
if (matcher is IProtoBufMatcher protoBufMatcher)
|
||||||
|
{
|
||||||
|
return protoBufMatcher.IsMatchAsync(requestMessage.BodyAsBytes).GetAwaiter().GetResult();
|
||||||
|
}
|
||||||
|
|
||||||
return default;
|
return default;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -2,7 +2,6 @@
|
|||||||
|
|
||||||
#if PROTOBUF
|
#if PROTOBUF
|
||||||
using System;
|
using System;
|
||||||
using System.Linq;
|
|
||||||
using System.Threading;
|
using System.Threading;
|
||||||
using System.Threading.Tasks;
|
using System.Threading.Tasks;
|
||||||
using ProtoBufJsonConverter;
|
using ProtoBufJsonConverter;
|
||||||
@@ -28,7 +27,7 @@ public class ProtoBufMatcher : IProtoBufMatcher
|
|||||||
/// <summary>
|
/// <summary>
|
||||||
/// The Func to define the proto definition as id or texts.
|
/// The Func to define the proto definition as id or texts.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public Func<IdOrTexts> ProtoDefinition { get; }
|
public Func<IdOrTexts> ProtoDefinition { get; internal set; }
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// The full type of the protobuf (request/response) message object. Format is "{package-name}.{type-name}".
|
/// The full type of the protobuf (request/response) message object. Format is "{package-name}.{type-name}".
|
||||||
|
|||||||
@@ -2,7 +2,6 @@
|
|||||||
|
|
||||||
using System;
|
using System;
|
||||||
using System.Collections.Generic;
|
using System.Collections.Generic;
|
||||||
using JsonConverter.Abstractions;
|
|
||||||
using WireMock.Matchers;
|
using WireMock.Matchers;
|
||||||
using WireMock.Util;
|
using WireMock.Util;
|
||||||
|
|
||||||
|
|||||||
@@ -5,8 +5,10 @@
|
|||||||
using System;
|
using System;
|
||||||
using System.Collections.Generic;
|
using System.Collections.Generic;
|
||||||
using System.Collections.ObjectModel;
|
using System.Collections.ObjectModel;
|
||||||
|
using System.Diagnostics.CodeAnalysis;
|
||||||
using System.Linq;
|
using System.Linq;
|
||||||
using Stef.Validation;
|
using Stef.Validation;
|
||||||
|
using WireMock.Matchers;
|
||||||
using WireMock.Matchers.Request;
|
using WireMock.Matchers.Request;
|
||||||
|
|
||||||
namespace WireMock.RequestBuilders;
|
namespace WireMock.RequestBuilders;
|
||||||
@@ -71,6 +73,19 @@ public partial class Request : RequestMessageCompositeMatcher, IRequestBuilder
|
|||||||
return _requestMatchers.OfType<T>().FirstOrDefault(func);
|
return _requestMatchers.OfType<T>().FirstOrDefault(func);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
internal bool TryGetProtoBufMatcher([NotNullWhen(true)] out IProtoBufMatcher? protoBufMatcher)
|
||||||
|
{
|
||||||
|
protoBufMatcher = GetRequestMessageMatcher<RequestMessageProtoBufMatcher>()?.Matcher;
|
||||||
|
if (protoBufMatcher != null)
|
||||||
|
{
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
var bodyMatcher = GetRequestMessageMatcher<RequestMessageBodyMatcher>();
|
||||||
|
protoBufMatcher = bodyMatcher?.Matchers?.OfType<IProtoBufMatcher>().FirstOrDefault();
|
||||||
|
return protoBufMatcher != null;
|
||||||
|
}
|
||||||
|
|
||||||
private IRequestBuilder Add<T>(T requestMatcher) where T : IRequestMatcher
|
private IRequestBuilder Add<T>(T requestMatcher) where T : IRequestMatcher
|
||||||
{
|
{
|
||||||
foreach (var existing in _requestMatchers.OfType<T>().ToArray())
|
foreach (var existing in _requestMatchers.OfType<T>().ToArray())
|
||||||
|
|||||||
@@ -49,7 +49,7 @@ public partial class Response
|
|||||||
public IResponseBuilder WithTrailingHeader(string name, params string[] values)
|
public IResponseBuilder WithTrailingHeader(string name, params string[] values)
|
||||||
{
|
{
|
||||||
#if !TRAILINGHEADERS
|
#if !TRAILINGHEADERS
|
||||||
throw new System.NotSupportedException("The WithBodyAsProtoBuf method can not be used for .NETStandard1.3 or .NET Framework 4.6.1 or lower.");
|
throw new System.NotSupportedException("The WithTrailingHeader method can not be used for .NETStandard1.3 or .NET Framework 4.6.1 or lower.");
|
||||||
#else
|
#else
|
||||||
|
|
||||||
Guard.NotNull(name);
|
Guard.NotNull(name);
|
||||||
@@ -63,7 +63,7 @@ public partial class Response
|
|||||||
public IResponseBuilder WithTrailingHeaders(IDictionary<string, string> headers)
|
public IResponseBuilder WithTrailingHeaders(IDictionary<string, string> headers)
|
||||||
{
|
{
|
||||||
#if !TRAILINGHEADERS
|
#if !TRAILINGHEADERS
|
||||||
throw new System.NotSupportedException("The WithBodyAsProtoBuf method can not be used for .NETStandard1.3 or .NET Framework 4.6.1 or lower.");
|
throw new System.NotSupportedException("The WithTrailingHeaders method can not be used for .NETStandard1.3 or .NET Framework 4.6.1 or lower.");
|
||||||
#else
|
#else
|
||||||
|
|
||||||
Guard.NotNull(headers);
|
Guard.NotNull(headers);
|
||||||
@@ -77,7 +77,7 @@ public partial class Response
|
|||||||
public IResponseBuilder WithTrailingHeaders(IDictionary<string, string[]> headers)
|
public IResponseBuilder WithTrailingHeaders(IDictionary<string, string[]> headers)
|
||||||
{
|
{
|
||||||
#if !TRAILINGHEADERS
|
#if !TRAILINGHEADERS
|
||||||
throw new System.NotSupportedException("The WithBodyAsProtoBuf method can not be used for .NETStandard1.3 or .NET Framework 4.6.1 or lower.");
|
throw new System.NotSupportedException("The WithTrailingHeaders method can not be used for .NETStandard1.3 or .NET Framework 4.6.1 or lower.");
|
||||||
#else
|
#else
|
||||||
|
|
||||||
Guard.NotNull(headers);
|
Guard.NotNull(headers);
|
||||||
|
|||||||
@@ -8,7 +8,6 @@ using System.Threading;
|
|||||||
using System.Threading.Tasks;
|
using System.Threading.Tasks;
|
||||||
using JetBrains.Annotations;
|
using JetBrains.Annotations;
|
||||||
using Stef.Validation;
|
using Stef.Validation;
|
||||||
using WireMock.Matchers.Request;
|
|
||||||
using WireMock.Proxy;
|
using WireMock.Proxy;
|
||||||
using WireMock.RequestBuilders;
|
using WireMock.RequestBuilders;
|
||||||
using WireMock.Settings;
|
using WireMock.Settings;
|
||||||
@@ -264,16 +263,15 @@ public partial class Response : IResponseBuilder
|
|||||||
|
|
||||||
if (UseTransformer)
|
if (UseTransformer)
|
||||||
{
|
{
|
||||||
// Check if the body matcher is a RequestMessageProtoBufMatcher and try to decode the byte-array to a BodyAsJson.
|
// If the body matcher is a RequestMessageProtoBufMatcher or BodyMatcher with a ProtoBufMatcher then try to decode the byte-array to a BodyAsJson.
|
||||||
if (mapping.RequestMatcher is Request requestMatcher && requestMessage is RequestMessage request)
|
if (mapping.RequestMatcher is Request request && requestMessage is RequestMessage requestMessageImplementation)
|
||||||
{
|
{
|
||||||
var protoBufMatcher = requestMatcher.GetRequestMessageMatcher<RequestMessageProtoBufMatcher>()?.Matcher;
|
if (request.TryGetProtoBufMatcher(out var protoBufMatcher))
|
||||||
if (protoBufMatcher != null)
|
|
||||||
{
|
{
|
||||||
var decoded = await protoBufMatcher.DecodeAsync(request.BodyData?.BodyAsBytes).ConfigureAwait(false);
|
var decoded = await protoBufMatcher.DecodeAsync(requestMessage.BodyData?.BodyAsBytes).ConfigureAwait(false);
|
||||||
if (decoded != null)
|
if (decoded != null)
|
||||||
{
|
{
|
||||||
request.BodyAsJson = JsonUtils.ConvertValueToJToken(decoded);
|
requestMessageImplementation.BodyAsJson = JsonUtils.ConvertValueToJToken(decoded);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -379,7 +379,7 @@ internal class MappingConverter(MatcherMapper mapper)
|
|||||||
}
|
}
|
||||||
|
|
||||||
var bodyMatchers =
|
var bodyMatchers =
|
||||||
protoBufMatcher?.Matcher != null ? new[] { protoBufMatcher.Matcher } : null ??
|
protoBufMatcher?.Matcher != null ? [protoBufMatcher.Matcher] : null ??
|
||||||
multiPartMatcher?.Matchers ??
|
multiPartMatcher?.Matchers ??
|
||||||
graphQLMatcher?.Matchers ??
|
graphQLMatcher?.Matchers ??
|
||||||
bodyMatcher?.Matchers;
|
bodyMatcher?.Matchers;
|
||||||
|
|||||||
@@ -220,7 +220,7 @@ internal class MatcherMapper
|
|||||||
{
|
{
|
||||||
model.Pattern = texts[0];
|
model.Pattern = texts[0];
|
||||||
}
|
}
|
||||||
else
|
else if (texts.Count > 1)
|
||||||
{
|
{
|
||||||
model.Patterns = texts.Cast<object>().ToArray();
|
model.Patterns = texts.Cast<object>().ToArray();
|
||||||
}
|
}
|
||||||
@@ -296,27 +296,9 @@ internal class MatcherMapper
|
|||||||
{
|
{
|
||||||
var objectMatcher = Map(matcher.ContentMatcher) as IObjectMatcher;
|
var objectMatcher = Map(matcher.ContentMatcher) as IObjectMatcher;
|
||||||
|
|
||||||
IdOrTexts protoDefinitionAsIdOrTexts;
|
|
||||||
if (protoDefinitions.Count == 1)
|
|
||||||
{
|
|
||||||
var idOrText = protoDefinitions[0];
|
|
||||||
if (_settings.ProtoDefinitions?.TryGetValue(idOrText, out var protoDefinitionFromSettings) == true)
|
|
||||||
{
|
|
||||||
protoDefinitionAsIdOrTexts = new(idOrText, protoDefinitionFromSettings);
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
protoDefinitionAsIdOrTexts = new(null, protoDefinitions);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
protoDefinitionAsIdOrTexts = new(null, protoDefinitions);
|
|
||||||
}
|
|
||||||
|
|
||||||
return new ProtoBufMatcher(
|
return new ProtoBufMatcher(
|
||||||
() => protoDefinitionAsIdOrTexts,
|
() => ProtoDefinitionHelper.GetIdOrTexts(_settings, protoDefinitions.ToArray()),
|
||||||
matcher!.ProtoBufMessageType!,
|
matcher.ProtoBufMessageType!,
|
||||||
matchBehaviour ?? MatchBehaviour.AcceptOnMatch,
|
matchBehaviour ?? MatchBehaviour.AcceptOnMatch,
|
||||||
objectMatcher
|
objectMatcher
|
||||||
);
|
);
|
||||||
|
|||||||
@@ -123,14 +123,14 @@ public interface IRespondWithAProvider
|
|||||||
void ThenRespondWithStatusCode(HttpStatusCode code);
|
void ThenRespondWithStatusCode(HttpStatusCode code);
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Sets the the scenario.
|
/// Sets the scenario.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
/// <param name="scenario">The scenario.</param>
|
/// <param name="scenario">The scenario.</param>
|
||||||
/// <returns>The <see cref="IRespondWithAProvider"/>.</returns>
|
/// <returns>The <see cref="IRespondWithAProvider"/>.</returns>
|
||||||
IRespondWithAProvider InScenario(string scenario);
|
IRespondWithAProvider InScenario(string scenario);
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Sets the the scenario with an integer value.
|
/// Sets the scenario with an integer value.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
/// <param name="scenario">The scenario.</param>
|
/// <param name="scenario">The scenario.</param>
|
||||||
/// <returns>The <see cref="IRespondWithAProvider"/>.</returns>
|
/// <returns>The <see cref="IRespondWithAProvider"/>.</returns>
|
||||||
@@ -220,7 +220,7 @@ public interface IRespondWithAProvider
|
|||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Data Object which can be used when WithTransformer is used.
|
/// Data Object which can be used when WithTransformer is used.
|
||||||
/// e.g. lookup an path in this object using
|
/// e.g. lookup a path in this object using
|
||||||
/// <param name="data">The data dictionary object.</param>
|
/// <param name="data">The data dictionary object.</param>
|
||||||
/// <example>
|
/// <example>
|
||||||
/// lookup data "1"
|
/// lookup data "1"
|
||||||
|
|||||||
@@ -17,7 +17,7 @@ using WireMock.Util;
|
|||||||
namespace WireMock.Server;
|
namespace WireMock.Server;
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// The respond with a provider.
|
/// The RespondWithAProvider.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
internal class RespondWithAProvider : IRespondWithAProvider
|
internal class RespondWithAProvider : IRespondWithAProvider
|
||||||
{
|
{
|
||||||
@@ -37,7 +37,6 @@ internal class RespondWithAProvider : IRespondWithAProvider
|
|||||||
private int _timesInSameState = 1;
|
private int _timesInSameState = 1;
|
||||||
private bool? _useWebhookFireAndForget;
|
private bool? _useWebhookFireAndForget;
|
||||||
private double? _probability;
|
private double? _probability;
|
||||||
private IdOrTexts? _protoDefinition;
|
|
||||||
private GraphQLSchemaDetails? _graphQLSchemaDetails;
|
private GraphQLSchemaDetails? _graphQLSchemaDetails;
|
||||||
|
|
||||||
public Guid Guid { get; private set; }
|
public Guid Guid { get; private set; }
|
||||||
@@ -48,6 +47,8 @@ internal class RespondWithAProvider : IRespondWithAProvider
|
|||||||
|
|
||||||
public object? Data { get; private set; }
|
public object? Data { get; private set; }
|
||||||
|
|
||||||
|
public IdOrTexts? ProtoDefinition { get; private set; }
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Initializes a new instance of the <see cref="RespondWithAProvider"/> class.
|
/// Initializes a new instance of the <see cref="RespondWithAProvider"/> class.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
@@ -104,9 +105,9 @@ internal class RespondWithAProvider : IRespondWithAProvider
|
|||||||
mapping.WithProbability(_probability.Value);
|
mapping.WithProbability(_probability.Value);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (_protoDefinition != null)
|
if (ProtoDefinition != null)
|
||||||
{
|
{
|
||||||
mapping.WithProtoDefinition(_protoDefinition.Value);
|
mapping.WithProtoDefinition(ProtoDefinition.Value);
|
||||||
}
|
}
|
||||||
|
|
||||||
_registrationCallback(mapping, _saveToFile);
|
_registrationCallback(mapping, _saveToFile);
|
||||||
@@ -296,7 +297,7 @@ internal class RespondWithAProvider : IRespondWithAProvider
|
|||||||
Guard.NotNull(url);
|
Guard.NotNull(url);
|
||||||
Guard.NotNull(method);
|
Guard.NotNull(method);
|
||||||
|
|
||||||
Webhooks = new[] { InitWebhook(url, method, headers, useTransformer, transformerType) };
|
Webhooks = [InitWebhook(url, method, headers, useTransformer, transformerType)];
|
||||||
|
|
||||||
if (body != null)
|
if (body != null)
|
||||||
{
|
{
|
||||||
@@ -323,7 +324,7 @@ internal class RespondWithAProvider : IRespondWithAProvider
|
|||||||
Guard.NotNull(url);
|
Guard.NotNull(url);
|
||||||
Guard.NotNull(method);
|
Guard.NotNull(method);
|
||||||
|
|
||||||
Webhooks = new[] { InitWebhook(url, method, headers, useTransformer, transformerType) };
|
Webhooks = [InitWebhook(url, method, headers, useTransformer, transformerType)];
|
||||||
|
|
||||||
if (body != null)
|
if (body != null)
|
||||||
{
|
{
|
||||||
@@ -355,23 +356,7 @@ internal class RespondWithAProvider : IRespondWithAProvider
|
|||||||
{
|
{
|
||||||
Guard.NotNull(protoDefinitionOrId);
|
Guard.NotNull(protoDefinitionOrId);
|
||||||
|
|
||||||
if (protoDefinitionOrId.Length == 1)
|
ProtoDefinition = ProtoDefinitionHelper.GetIdOrTexts(_settings, protoDefinitionOrId);
|
||||||
{
|
|
||||||
var idOrText = protoDefinitionOrId[0];
|
|
||||||
if (_settings.ProtoDefinitions?.TryGetValue(idOrText, out var protoDefinitions) == true)
|
|
||||||
{
|
|
||||||
_protoDefinition = new(idOrText, protoDefinitions);
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
_protoDefinition = new(null, protoDefinitionOrId);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
_protoDefinition = new(null, protoDefinitionOrId);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
return this;
|
return this;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -42,9 +42,9 @@ public partial class WireMockServer
|
|||||||
Guard.NotNull(mappingModel.Request);
|
Guard.NotNull(mappingModel.Request);
|
||||||
Guard.NotNull(mappingModel.Response);
|
Guard.NotNull(mappingModel.Response);
|
||||||
|
|
||||||
var requestBuilder = InitRequestBuilder(mappingModel.Request);
|
var request = (Request)InitRequestBuilder(mappingModel.Request, mappingModel);
|
||||||
|
|
||||||
var respondProvider = Given(requestBuilder, mappingModel.SaveToFile == true);
|
var respondProvider = Given(request, mappingModel.SaveToFile == true);
|
||||||
|
|
||||||
if (guid != null)
|
if (guid != null)
|
||||||
{
|
{
|
||||||
@@ -116,13 +116,23 @@ public partial class WireMockServer
|
|||||||
respondProvider.WithProbability(mappingModel.Probability.Value);
|
respondProvider.WithProbability(mappingModel.Probability.Value);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// ProtoDefinition is defined at Mapping level
|
||||||
|
if (mappingModel.ProtoDefinition != null)
|
||||||
|
{
|
||||||
|
respondProvider.WithProtoDefinition(mappingModel.ProtoDefinition);
|
||||||
|
}
|
||||||
|
else if (mappingModel.ProtoDefinitions != null)
|
||||||
|
{
|
||||||
|
respondProvider.WithProtoDefinition(mappingModel.ProtoDefinitions);
|
||||||
|
}
|
||||||
|
|
||||||
var responseBuilder = InitResponseBuilder(mappingModel.Response);
|
var responseBuilder = InitResponseBuilder(mappingModel.Response);
|
||||||
respondProvider.RespondWith(responseBuilder);
|
respondProvider.RespondWith(responseBuilder);
|
||||||
|
|
||||||
return respondProvider.Guid;
|
return respondProvider.Guid;
|
||||||
}
|
}
|
||||||
|
|
||||||
private IRequestBuilder InitRequestBuilder(RequestModel requestModel)
|
private IRequestBuilder InitRequestBuilder(RequestModel requestModel, MappingModel? mappingModel = null)
|
||||||
{
|
{
|
||||||
var requestBuilder = Request.Create();
|
var requestBuilder = Request.Create();
|
||||||
|
|
||||||
@@ -216,7 +226,7 @@ public partial class WireMockServer
|
|||||||
|
|
||||||
if (requestModel.Params != null)
|
if (requestModel.Params != null)
|
||||||
{
|
{
|
||||||
foreach (var paramModel in requestModel.Params.Where(p => p is { Matchers: { } }))
|
foreach (var paramModel in requestModel.Params.Where(p => p is { Matchers: not null }))
|
||||||
{
|
{
|
||||||
var ignoreCase = paramModel.IgnoreCase == true;
|
var ignoreCase = paramModel.IgnoreCase == true;
|
||||||
requestBuilder = requestBuilder.WithParam(paramModel.Name, ignoreCase, paramModel.Matchers!.Select(_matcherMapper.Map).OfType<IStringMatcher>().ToArray());
|
requestBuilder = requestBuilder.WithParam(paramModel.Name, ignoreCase, paramModel.Matchers!.Select(_matcherMapper.Map).OfType<IStringMatcher>().ToArray());
|
||||||
@@ -225,7 +235,15 @@ public partial class WireMockServer
|
|||||||
|
|
||||||
if (requestModel.Body?.Matcher != null)
|
if (requestModel.Body?.Matcher != null)
|
||||||
{
|
{
|
||||||
requestBuilder = requestBuilder.WithBody(_matcherMapper.Map(requestModel.Body.Matcher)!);
|
var bodyMatcher = _matcherMapper.Map(requestModel.Body.Matcher)!;
|
||||||
|
#if PROTOBUF
|
||||||
|
// If the BodyMatcher is a ProtoBufMatcher, and if ProtoDefinition is defined on Mapping-level, set the ProtoDefinition from that Mapping.
|
||||||
|
if (bodyMatcher is ProtoBufMatcher protoBufMatcher && mappingModel?.ProtoDefinition != null)
|
||||||
|
{
|
||||||
|
protoBufMatcher.ProtoDefinition = () => ProtoDefinitionHelper.GetIdOrTexts(_settings, mappingModel.ProtoDefinition);
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
requestBuilder = requestBuilder.WithBody(bodyMatcher);
|
||||||
}
|
}
|
||||||
else if (requestModel.Body?.Matchers != null)
|
else if (requestModel.Body?.Matchers != null)
|
||||||
{
|
{
|
||||||
@@ -308,7 +326,7 @@ public partial class WireMockServer
|
|||||||
}
|
}
|
||||||
else if (responseModel.HeadersRaw != null)
|
else if (responseModel.HeadersRaw != null)
|
||||||
{
|
{
|
||||||
foreach (string headerLine in responseModel.HeadersRaw.Split(["\n", "\r\n"], StringSplitOptions.RemoveEmptyEntries))
|
foreach (var headerLine in responseModel.HeadersRaw.Split(["\n", "\r\n"], StringSplitOptions.RemoveEmptyEntries))
|
||||||
{
|
{
|
||||||
int indexColon = headerLine.IndexOf(":", StringComparison.Ordinal);
|
int indexColon = headerLine.IndexOf(":", StringComparison.Ordinal);
|
||||||
string key = headerLine.Substring(0, indexColon).TrimStart(' ', '\t');
|
string key = headerLine.Substring(0, indexColon).TrimStart(' ', '\t');
|
||||||
@@ -317,6 +335,22 @@ public partial class WireMockServer
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (responseModel.TrailingHeaders != null)
|
||||||
|
{
|
||||||
|
foreach (var entry in responseModel.TrailingHeaders)
|
||||||
|
{
|
||||||
|
if (entry.Value is string value)
|
||||||
|
{
|
||||||
|
responseBuilder.WithTrailingHeader(entry.Key, value);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
var headers = JsonUtils.ParseJTokenToObject<string[]>(entry.Value);
|
||||||
|
responseBuilder.WithTrailingHeader(entry.Key, headers);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
if (responseModel.BodyAsBytes != null)
|
if (responseModel.BodyAsBytes != null)
|
||||||
{
|
{
|
||||||
responseBuilder = responseBuilder.WithBody(responseModel.BodyAsBytes, responseModel.BodyDestination, ToEncoding(responseModel.BodyEncoding));
|
responseBuilder = responseBuilder.WithBody(responseModel.BodyAsBytes, responseModel.BodyDestination, ToEncoding(responseModel.BodyEncoding));
|
||||||
@@ -327,7 +361,26 @@ public partial class WireMockServer
|
|||||||
}
|
}
|
||||||
else if (responseModel.BodyAsJson != null)
|
else if (responseModel.BodyAsJson != null)
|
||||||
{
|
{
|
||||||
responseBuilder = responseBuilder.WithBodyAsJson(responseModel.BodyAsJson, ToEncoding(responseModel.BodyEncoding), responseModel.BodyAsJsonIndented == true);
|
if (responseModel.ProtoBufMessageType != null)
|
||||||
|
{
|
||||||
|
if (responseModel.ProtoDefinition != null)
|
||||||
|
{
|
||||||
|
responseBuilder = responseBuilder.WithBodyAsProtoBuf(responseModel.ProtoDefinition, responseModel.ProtoBufMessageType, responseModel.BodyAsJson);
|
||||||
|
}
|
||||||
|
else if (responseModel.ProtoDefinitions != null)
|
||||||
|
{
|
||||||
|
responseBuilder = responseBuilder.WithBodyAsProtoBuf(responseModel.ProtoDefinitions, responseModel.ProtoBufMessageType, responseModel.BodyAsJson);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
// ProtoDefinition(s) is/are defined at Mapping/Server level
|
||||||
|
responseBuilder = responseBuilder.WithBodyAsProtoBuf(responseModel.ProtoBufMessageType, responseModel.BodyAsJson);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
responseBuilder = responseBuilder.WithBodyAsJson(responseModel.BodyAsJson, ToEncoding(responseModel.BodyEncoding), responseModel.BodyAsJsonIndented == true);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
else if (responseModel.BodyAsFile != null)
|
else if (responseModel.BodyAsFile != null)
|
||||||
{
|
{
|
||||||
|
|||||||
@@ -1,7 +1,9 @@
|
|||||||
// Copyright © WireMock.Net
|
// Copyright © WireMock.Net
|
||||||
|
|
||||||
using System;
|
using System;
|
||||||
|
using System.Collections.Generic;
|
||||||
using System.Diagnostics.CodeAnalysis;
|
using System.Diagnostics.CodeAnalysis;
|
||||||
|
using System.Linq;
|
||||||
using System.Net;
|
using System.Net;
|
||||||
using System.Net.Sockets;
|
using System.Net.Sockets;
|
||||||
using System.Text.RegularExpressions;
|
using System.Text.RegularExpressions;
|
||||||
@@ -17,25 +19,96 @@ internal static class PortUtils
|
|||||||
private static readonly Regex UrlDetailsRegex = new(@"^((?<proto>\w+)://)(?<host>[^/]+?):(?<port>\d+)\/?$", RegexOptions.Compiled, WireMockConstants.DefaultRegexTimeout);
|
private static readonly Regex UrlDetailsRegex = new(@"^((?<proto>\w+)://)(?<host>[^/]+?):(?<port>\d+)\/?$", RegexOptions.Compiled, WireMockConstants.DefaultRegexTimeout);
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Finds a free TCP port.
|
/// Finds a random, free port to be listened on.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
/// <remarks>see http://stackoverflow.com/questions/138043/find-the-next-tcp-port-in-net.</remarks>
|
/// <returns>A random, free port to be listened on.</returns>
|
||||||
|
/// <remarks>https://github.com/SeleniumHQ/selenium/blob/trunk/dotnet/src/webdriver/Internal/PortUtilities.cs</remarks>
|
||||||
public static int FindFreeTcpPort()
|
public static int FindFreeTcpPort()
|
||||||
{
|
{
|
||||||
TcpListener? tcpListener = null;
|
// Locate a free port on the local machine by binding a socket to an IPEndPoint using IPAddress.Any and port 0.
|
||||||
|
// The socket will select a free port.
|
||||||
|
var portSocket = new Socket(AddressFamily.InterNetwork, SocketType.Stream, ProtocolType.Tcp);
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
tcpListener = new TcpListener(IPAddress.Loopback, 0);
|
var socketEndPoint = new IPEndPoint(IPAddress.Any, 0);
|
||||||
tcpListener.Start();
|
portSocket.Bind(socketEndPoint);
|
||||||
|
socketEndPoint = (IPEndPoint)portSocket.LocalEndPoint!;
|
||||||
return ((IPEndPoint)tcpListener.LocalEndpoint).Port;
|
return socketEndPoint.Port;
|
||||||
}
|
}
|
||||||
finally
|
finally
|
||||||
{
|
{
|
||||||
tcpListener?.Stop();
|
#if !NETSTANDARD1_3
|
||||||
|
portSocket.Close();
|
||||||
|
#endif
|
||||||
|
portSocket.Dispose();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Finds a specified number of random, free ports to be listened on.
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="count">The number of free ports to find.</param>
|
||||||
|
/// <returns>A list of random, free ports to be listened on.</returns>
|
||||||
|
public static IReadOnlyList<int> FindFreeTcpPorts(int count)
|
||||||
|
{
|
||||||
|
var sockets = Enumerable
|
||||||
|
.Range(0, count)
|
||||||
|
.Select(_ => new Socket(AddressFamily.InterNetwork, SocketType.Stream, ProtocolType.Tcp))
|
||||||
|
.ToArray();
|
||||||
|
|
||||||
|
var freePorts = new List<int>();
|
||||||
|
|
||||||
|
try
|
||||||
|
{
|
||||||
|
foreach (var socket in sockets)
|
||||||
|
{
|
||||||
|
var socketEndPoint = new IPEndPoint(IPAddress.Any, 0);
|
||||||
|
socket.Bind(socketEndPoint);
|
||||||
|
socketEndPoint = (IPEndPoint)socket.LocalEndPoint!;
|
||||||
|
|
||||||
|
freePorts.Add(socketEndPoint.Port);
|
||||||
|
}
|
||||||
|
|
||||||
|
return freePorts;
|
||||||
|
}
|
||||||
|
finally
|
||||||
|
{
|
||||||
|
foreach (var socket in sockets)
|
||||||
|
{
|
||||||
|
#if !NETSTANDARD1_3
|
||||||
|
socket.Close();
|
||||||
|
#endif
|
||||||
|
socket.Dispose();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
///// <summary>
|
||||||
|
///// Finds free TCP ports.
|
||||||
|
///// </summary>
|
||||||
|
//public static IReadOnlyList<int> FindFreeTcpPorts(int numPorts)
|
||||||
|
//{
|
||||||
|
// var freePorts = new List<int>();
|
||||||
|
|
||||||
|
// TcpListener? tcpListener = null;
|
||||||
|
// try
|
||||||
|
// {
|
||||||
|
// for (var i = 0; i < numPorts; i++)
|
||||||
|
// {
|
||||||
|
// tcpListener = new TcpListener(IPAddress.Loopback, 0);
|
||||||
|
// tcpListener.Start();
|
||||||
|
|
||||||
|
// freePorts.Add(((IPEndPoint)tcpListener.LocalEndpoint).Port);
|
||||||
|
// }
|
||||||
|
// }
|
||||||
|
// finally
|
||||||
|
// {
|
||||||
|
// tcpListener?.Stop();
|
||||||
|
// }
|
||||||
|
|
||||||
|
// return freePorts;
|
||||||
|
//}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Extract the isHttps, isHttp2, protocol, host and port from a URL.
|
/// Extract the isHttps, isHttp2, protocol, host and port from a URL.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
|
|||||||
27
src/WireMock.Net/Util/ProtoDefinitionHelper.cs
Normal file
27
src/WireMock.Net/Util/ProtoDefinitionHelper.cs
Normal file
@@ -0,0 +1,27 @@
|
|||||||
|
// Copyright © WireMock.Net
|
||||||
|
|
||||||
|
using WireMock.Models;
|
||||||
|
using WireMock.Settings;
|
||||||
|
|
||||||
|
namespace WireMock.Util;
|
||||||
|
|
||||||
|
internal static class ProtoDefinitionHelper
|
||||||
|
{
|
||||||
|
internal static IdOrTexts GetIdOrTexts(WireMockServerSettings settings, params string[] protoDefinitionOrId)
|
||||||
|
{
|
||||||
|
switch (protoDefinitionOrId.Length)
|
||||||
|
{
|
||||||
|
case 1:
|
||||||
|
var idOrText = protoDefinitionOrId[0];
|
||||||
|
if (settings.ProtoDefinitions?.TryGetValue(idOrText, out var protoDefinitions) == true)
|
||||||
|
{
|
||||||
|
return new(idOrText, protoDefinitions);
|
||||||
|
}
|
||||||
|
|
||||||
|
return new(null, protoDefinitionOrId);
|
||||||
|
|
||||||
|
default:
|
||||||
|
return new(null, protoDefinitionOrId);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -37,7 +37,32 @@ message HelloReply {
|
|||||||
public async Task IWireMockAdminApi_GetMappingsAsync_WithBodyAsProtoBuf_ShouldReturnCorrectMappingModels()
|
public async Task IWireMockAdminApi_GetMappingsAsync_WithBodyAsProtoBuf_ShouldReturnCorrectMappingModels()
|
||||||
{
|
{
|
||||||
// Arrange
|
// Arrange
|
||||||
using var server = WireMockServer.StartWithAdminInterface();
|
using var server = Given_WithBodyAsProtoBuf_AddedToServer();
|
||||||
|
|
||||||
|
// Act
|
||||||
|
var api = RestClient.For<IWireMockAdminApi>(server.Url);
|
||||||
|
var getMappingsResult = await api.GetMappingsAsync().ConfigureAwait(false);
|
||||||
|
|
||||||
|
await Verifier.Verify(getMappingsResult, VerifySettings);
|
||||||
|
}
|
||||||
|
|
||||||
|
[Fact]
|
||||||
|
public async Task HttpClient_GetMappingsAsync_WithBodyAsProtoBuf_ShouldReturnCorrectMappingModels()
|
||||||
|
{
|
||||||
|
// Arrange
|
||||||
|
using var server = Given_WithBodyAsProtoBuf_AddedToServer();
|
||||||
|
|
||||||
|
// Act
|
||||||
|
var client = server.CreateClient();
|
||||||
|
var getMappingsResult = await client.GetStringAsync("/__admin/mappings").ConfigureAwait(false);
|
||||||
|
|
||||||
|
await Verifier.VerifyJson(getMappingsResult, VerifySettings);
|
||||||
|
}
|
||||||
|
|
||||||
|
public WireMockServer Given_WithBodyAsProtoBuf_AddedToServer()
|
||||||
|
{
|
||||||
|
// Arrange
|
||||||
|
var server = WireMockServer.StartWithAdminInterface();
|
||||||
|
|
||||||
var protoBufJsonMatcher = new JsonPartialWildcardMatcher(new { name = "*" });
|
var protoBufJsonMatcher = new JsonPartialWildcardMatcher(new { name = "*" });
|
||||||
|
|
||||||
@@ -122,13 +147,7 @@ message HelloReply {
|
|||||||
.WithTransformer()
|
.WithTransformer()
|
||||||
);
|
);
|
||||||
|
|
||||||
// Act
|
return server;
|
||||||
var api = RestClient.For<IWireMockAdminApi>(server.Url);
|
|
||||||
var getMappingsResult = await api.GetMappingsAsync().ConfigureAwait(false);
|
|
||||||
|
|
||||||
await Verifier.Verify(getMappingsResult, VerifySettings);
|
|
||||||
|
|
||||||
server.Stop();
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
@@ -0,0 +1,235 @@
|
|||||||
|
[
|
||||||
|
{
|
||||||
|
Guid: Guid_1,
|
||||||
|
UpdatedAt: DateTimeOffset_1,
|
||||||
|
Request: {
|
||||||
|
Path: {
|
||||||
|
Matchers: [
|
||||||
|
{
|
||||||
|
Name: WildcardMatcher,
|
||||||
|
Pattern: /grpc/greet.Greeter/SayHello,
|
||||||
|
IgnoreCase: false
|
||||||
|
}
|
||||||
|
]
|
||||||
|
},
|
||||||
|
Methods: [
|
||||||
|
POST
|
||||||
|
],
|
||||||
|
Body: {
|
||||||
|
Matcher: {
|
||||||
|
Name: ProtoBufMatcher,
|
||||||
|
Pattern:
|
||||||
|
syntax = "proto3";
|
||||||
|
|
||||||
|
package greet;
|
||||||
|
|
||||||
|
service Greeter {
|
||||||
|
rpc SayHello (HelloRequest) returns (HelloReply);
|
||||||
|
}
|
||||||
|
|
||||||
|
message HelloRequest {
|
||||||
|
string name = 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
message HelloReply {
|
||||||
|
string message = 1;
|
||||||
|
}
|
||||||
|
,
|
||||||
|
ContentMatcher: {
|
||||||
|
Name: JsonPartialWildcardMatcher,
|
||||||
|
Pattern: {
|
||||||
|
name: *
|
||||||
|
},
|
||||||
|
IgnoreCase: false,
|
||||||
|
Regex: false
|
||||||
|
},
|
||||||
|
ProtoBufMessageType: greet.HelloRequest
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
Response: {
|
||||||
|
BodyAsJson: {
|
||||||
|
message: hello {{request.BodyAsJson.name}}
|
||||||
|
},
|
||||||
|
UseTransformer: true,
|
||||||
|
TransformerType: Handlebars,
|
||||||
|
TransformerReplaceNodeOptions: EvaluateAndTryToConvert,
|
||||||
|
Headers: {
|
||||||
|
Content-Type: application/grpc
|
||||||
|
},
|
||||||
|
TrailingHeaders: {
|
||||||
|
grpc-status: 0
|
||||||
|
},
|
||||||
|
ProtoDefinition:
|
||||||
|
syntax = "proto3";
|
||||||
|
|
||||||
|
package greet;
|
||||||
|
|
||||||
|
service Greeter {
|
||||||
|
rpc SayHello (HelloRequest) returns (HelloReply);
|
||||||
|
}
|
||||||
|
|
||||||
|
message HelloRequest {
|
||||||
|
string name = 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
message HelloReply {
|
||||||
|
string message = 1;
|
||||||
|
}
|
||||||
|
,
|
||||||
|
ProtoBufMessageType: greet.HelloReply
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
Guid: Guid_2,
|
||||||
|
UpdatedAt: DateTimeOffset_2,
|
||||||
|
Request: {
|
||||||
|
Path: {
|
||||||
|
Matchers: [
|
||||||
|
{
|
||||||
|
Name: WildcardMatcher,
|
||||||
|
Pattern: /grpc2/greet.Greeter/SayHello,
|
||||||
|
IgnoreCase: false
|
||||||
|
}
|
||||||
|
]
|
||||||
|
},
|
||||||
|
Methods: [
|
||||||
|
POST
|
||||||
|
],
|
||||||
|
Body: {
|
||||||
|
Matcher: {
|
||||||
|
Name: ProtoBufMatcher,
|
||||||
|
ContentMatcher: {
|
||||||
|
Name: JsonPartialWildcardMatcher,
|
||||||
|
Pattern: {
|
||||||
|
name: *
|
||||||
|
},
|
||||||
|
IgnoreCase: false,
|
||||||
|
Regex: false
|
||||||
|
},
|
||||||
|
ProtoBufMessageType: greet.HelloRequest
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
Response: {
|
||||||
|
BodyAsJson: {
|
||||||
|
message: hello {{request.BodyAsJson.name}}
|
||||||
|
},
|
||||||
|
UseTransformer: true,
|
||||||
|
TransformerType: Handlebars,
|
||||||
|
TransformerReplaceNodeOptions: EvaluateAndTryToConvert,
|
||||||
|
Headers: {
|
||||||
|
Content-Type: application/grpc
|
||||||
|
},
|
||||||
|
TrailingHeaders: {
|
||||||
|
grpc-status: 0
|
||||||
|
},
|
||||||
|
ProtoBufMessageType: greet.HelloReply
|
||||||
|
},
|
||||||
|
ProtoDefinition:
|
||||||
|
syntax = "proto3";
|
||||||
|
|
||||||
|
package greet;
|
||||||
|
|
||||||
|
service Greeter {
|
||||||
|
rpc SayHello (HelloRequest) returns (HelloReply);
|
||||||
|
}
|
||||||
|
|
||||||
|
message HelloRequest {
|
||||||
|
string name = 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
message HelloReply {
|
||||||
|
string message = 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
},
|
||||||
|
{
|
||||||
|
Guid: Guid_3,
|
||||||
|
UpdatedAt: DateTimeOffset_3,
|
||||||
|
Request: {
|
||||||
|
Path: {
|
||||||
|
Matchers: [
|
||||||
|
{
|
||||||
|
Name: WildcardMatcher,
|
||||||
|
Pattern: /grpc3/greet.Greeter/SayHello,
|
||||||
|
IgnoreCase: false
|
||||||
|
}
|
||||||
|
]
|
||||||
|
},
|
||||||
|
Methods: [
|
||||||
|
POST
|
||||||
|
],
|
||||||
|
Body: {
|
||||||
|
Matcher: {
|
||||||
|
Name: ProtoBufMatcher,
|
||||||
|
ContentMatcher: {
|
||||||
|
Name: JsonPartialWildcardMatcher,
|
||||||
|
Pattern: {
|
||||||
|
name: *
|
||||||
|
},
|
||||||
|
IgnoreCase: false,
|
||||||
|
Regex: false
|
||||||
|
},
|
||||||
|
ProtoBufMessageType: greet.HelloRequest
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
Response: {
|
||||||
|
BodyAsJson: {
|
||||||
|
message: hello {{request.BodyAsJson.name}}
|
||||||
|
},
|
||||||
|
UseTransformer: true,
|
||||||
|
TransformerType: Handlebars,
|
||||||
|
TransformerReplaceNodeOptions: EvaluateAndTryToConvert,
|
||||||
|
Headers: {
|
||||||
|
Content-Type: application/grpc
|
||||||
|
},
|
||||||
|
TrailingHeaders: {
|
||||||
|
grpc-status: 0
|
||||||
|
},
|
||||||
|
ProtoBufMessageType: greet.HelloReply
|
||||||
|
},
|
||||||
|
ProtoDefinition: my-greeter
|
||||||
|
},
|
||||||
|
{
|
||||||
|
Guid: Guid_4,
|
||||||
|
UpdatedAt: DateTimeOffset_4,
|
||||||
|
Request: {
|
||||||
|
Path: {
|
||||||
|
Matchers: [
|
||||||
|
{
|
||||||
|
Name: WildcardMatcher,
|
||||||
|
Pattern: /grpc4/greet.Greeter/SayHello,
|
||||||
|
IgnoreCase: false
|
||||||
|
}
|
||||||
|
]
|
||||||
|
},
|
||||||
|
Methods: [
|
||||||
|
POST
|
||||||
|
],
|
||||||
|
Body: {
|
||||||
|
Matcher: {
|
||||||
|
Name: ProtoBufMatcher,
|
||||||
|
ProtoBufMessageType: greet.HelloRequest
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
Response: {
|
||||||
|
BodyAsJson: {
|
||||||
|
message: hello {{request.BodyAsJson.name}}
|
||||||
|
},
|
||||||
|
UseTransformer: true,
|
||||||
|
TransformerType: Handlebars,
|
||||||
|
TransformerReplaceNodeOptions: EvaluateAndTryToConvert,
|
||||||
|
Headers: {
|
||||||
|
Content-Type: application/grpc
|
||||||
|
},
|
||||||
|
TrailingHeaders: {
|
||||||
|
grpc-status: 0
|
||||||
|
},
|
||||||
|
ProtoBufMessageType: greet.HelloReply
|
||||||
|
},
|
||||||
|
ProtoDefinition: my-greeter
|
||||||
|
}
|
||||||
|
]
|
||||||
@@ -0,0 +1,172 @@
|
|||||||
|
// Copyright © WireMock.Net
|
||||||
|
|
||||||
|
#if !(NET452 || NET461 || NETCOREAPP3_1)
|
||||||
|
using System;
|
||||||
|
using System.Linq;
|
||||||
|
using System.Net.Http;
|
||||||
|
using System.Text;
|
||||||
|
using System.Threading.Tasks;
|
||||||
|
using FluentAssertions;
|
||||||
|
using NFluent;
|
||||||
|
using RestEase;
|
||||||
|
using WireMock.Admin.Mappings;
|
||||||
|
using WireMock.Client;
|
||||||
|
using WireMock.Constants;
|
||||||
|
using WireMock.Models;
|
||||||
|
using WireMock.Server;
|
||||||
|
using Xunit;
|
||||||
|
|
||||||
|
namespace WireMock.Net.Tests.AdminApi;
|
||||||
|
|
||||||
|
public partial class WireMockAdminApiTests
|
||||||
|
{
|
||||||
|
public static string RemoveLineContainingUpdatedAt(string text)
|
||||||
|
{
|
||||||
|
var lines = text.Split([Environment.NewLine], StringSplitOptions.None);
|
||||||
|
var filteredLines = lines.Where(line => !line.Contains("\"UpdatedAt\": "));
|
||||||
|
return string.Join(Environment.NewLine, filteredLines);
|
||||||
|
}
|
||||||
|
|
||||||
|
[Theory]
|
||||||
|
[InlineData("protobuf-mapping-1.json", "351f0240-bba0-4bcb-93c6-1feba0fe0001")]
|
||||||
|
[InlineData("protobuf-mapping-2.json", "351f0240-bba0-4bcb-93c6-1feba0fe0002")]
|
||||||
|
[InlineData("protobuf-mapping-3.json", "351f0240-bba0-4bcb-93c6-1feba0fe0003")]
|
||||||
|
[InlineData("protobuf-mapping-4.json", "351f0240-bba0-4bcb-93c6-1feba0fe0004")]
|
||||||
|
public async Task HttpClient_PostMappingsAsync_ForProtoBufMapping(string mappingFile, string guid)
|
||||||
|
{
|
||||||
|
// Arrange
|
||||||
|
var mappingsJson = ReadMappingFile(mappingFile);
|
||||||
|
|
||||||
|
using var server = WireMockServer.StartWithAdminInterface();
|
||||||
|
var httpClient = server.CreateClient();
|
||||||
|
|
||||||
|
// Act
|
||||||
|
var result = await httpClient.PostAsync("/__admin/mappings", new StringContent(mappingsJson, Encoding.UTF8, WireMockConstants.ContentTypeJson));
|
||||||
|
result.EnsureSuccessStatusCode();
|
||||||
|
|
||||||
|
// Assert
|
||||||
|
var mapping = await httpClient.GetStringAsync($"/__admin/mappings/{guid}");
|
||||||
|
mapping = RemoveLineContainingUpdatedAt(mapping);
|
||||||
|
mapping.Should().Be(mappingsJson);
|
||||||
|
}
|
||||||
|
|
||||||
|
[Fact]
|
||||||
|
public async Task IWireMockAdminApi_PostMappingsAsync()
|
||||||
|
{
|
||||||
|
// Arrange
|
||||||
|
var server = WireMockServer.StartWithAdminInterface();
|
||||||
|
var api = RestClient.For<IWireMockAdminApi>(server.Urls[0]);
|
||||||
|
|
||||||
|
// Act
|
||||||
|
var model1 = new MappingModel
|
||||||
|
{
|
||||||
|
Request = new RequestModel { Path = "/1" },
|
||||||
|
Response = new ResponseModel { Body = "txt 1" },
|
||||||
|
Title = "test 1"
|
||||||
|
};
|
||||||
|
var model2 = new MappingModel
|
||||||
|
{
|
||||||
|
Request = new RequestModel { Path = "/2" },
|
||||||
|
Response = new ResponseModel { Body = "txt 2" },
|
||||||
|
Title = "test 2"
|
||||||
|
};
|
||||||
|
var result = await api.PostMappingsAsync(new[] { model1, model2 }).ConfigureAwait(false);
|
||||||
|
|
||||||
|
// Assert
|
||||||
|
Check.That(result).IsNotNull();
|
||||||
|
Check.That(result.Status).IsNotNull();
|
||||||
|
Check.That(result.Guid).IsNull();
|
||||||
|
Check.That(server.Mappings.Where(m => !m.IsAdminInterface)).HasSize(2);
|
||||||
|
|
||||||
|
server.Stop();
|
||||||
|
}
|
||||||
|
|
||||||
|
[Theory]
|
||||||
|
[InlineData(null, null)]
|
||||||
|
[InlineData(-1, -1)]
|
||||||
|
[InlineData(0, 0)]
|
||||||
|
[InlineData(200, 200)]
|
||||||
|
[InlineData("200", "200")]
|
||||||
|
public async Task IWireMockAdminApi_PostMappingAsync_WithStatusCode(object statusCode, object expectedStatusCode)
|
||||||
|
{
|
||||||
|
// Arrange
|
||||||
|
var server = WireMockServer.StartWithAdminInterface();
|
||||||
|
var api = RestClient.For<IWireMockAdminApi>(server.Urls[0]);
|
||||||
|
|
||||||
|
// Act
|
||||||
|
var model = new MappingModel
|
||||||
|
{
|
||||||
|
Request = new RequestModel { Path = "/1" },
|
||||||
|
Response = new ResponseModel { Body = "txt", StatusCode = statusCode },
|
||||||
|
Priority = 500,
|
||||||
|
Title = "test"
|
||||||
|
};
|
||||||
|
var result = await api.PostMappingAsync(model).ConfigureAwait(false);
|
||||||
|
|
||||||
|
// Assert
|
||||||
|
Check.That(result).IsNotNull();
|
||||||
|
Check.That(result.Status).IsNotNull();
|
||||||
|
Check.That(result.Guid).IsNotNull();
|
||||||
|
|
||||||
|
var mapping = server.Mappings.Single(m => m.Priority == 500);
|
||||||
|
Check.That(mapping).IsNotNull();
|
||||||
|
Check.That(mapping.Title).Equals("test");
|
||||||
|
|
||||||
|
var response = await mapping.ProvideResponseAsync(new RequestMessage(new UrlDetails("http://localhost/1"), "GET", "")).ConfigureAwait(false);
|
||||||
|
Check.That(response.Message.StatusCode).Equals(expectedStatusCode);
|
||||||
|
|
||||||
|
server.Stop();
|
||||||
|
}
|
||||||
|
|
||||||
|
[Fact]
|
||||||
|
public async Task IWireMockAdminApi_PostMappingsAsync_WithDuplicateGuids_Should_Return_400()
|
||||||
|
{
|
||||||
|
// Arrange
|
||||||
|
var guid = Guid.Parse("1b731398-4a5b-457f-a6e3-d65e541c428f");
|
||||||
|
var server = WireMockServer.StartWithAdminInterface();
|
||||||
|
var api = RestClient.For<IWireMockAdminApi>(server.Urls[0]);
|
||||||
|
|
||||||
|
// Act
|
||||||
|
var model1WithGuid = new MappingModel
|
||||||
|
{
|
||||||
|
Guid = guid,
|
||||||
|
Request = new RequestModel { Path = "/1g" },
|
||||||
|
Response = new ResponseModel { Body = "txt 1g" },
|
||||||
|
Title = "test 1g"
|
||||||
|
};
|
||||||
|
var model2WithGuid = new MappingModel
|
||||||
|
{
|
||||||
|
Guid = guid,
|
||||||
|
Request = new RequestModel { Path = "/2g" },
|
||||||
|
Response = new ResponseModel { Body = "txt 2g" },
|
||||||
|
Title = "test 2g"
|
||||||
|
};
|
||||||
|
var model1 = new MappingModel
|
||||||
|
{
|
||||||
|
Request = new RequestModel { Path = "/1" },
|
||||||
|
Response = new ResponseModel { Body = "txt 1" },
|
||||||
|
Title = "test 1"
|
||||||
|
};
|
||||||
|
var model2 = new MappingModel
|
||||||
|
{
|
||||||
|
Request = new RequestModel { Path = "/2" },
|
||||||
|
Response = new ResponseModel { Body = "txt 2" },
|
||||||
|
Title = "test 2"
|
||||||
|
};
|
||||||
|
|
||||||
|
var models = new[]
|
||||||
|
{
|
||||||
|
model1WithGuid,
|
||||||
|
model2WithGuid,
|
||||||
|
model1,
|
||||||
|
model2
|
||||||
|
};
|
||||||
|
|
||||||
|
var sutMethod = async () => await api.PostMappingsAsync(models);
|
||||||
|
var exceptionAssertions = await sutMethod.Should().ThrowAsync<ApiException>();
|
||||||
|
exceptionAssertions.Which.Content.Should().Be(@"{""Status"":""The following Guids are duplicate : '1b731398-4a5b-457f-a6e3-d65e541c428f' (Parameter 'mappingModels')""}");
|
||||||
|
|
||||||
|
server.Stop();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
#endif
|
||||||
@@ -184,124 +184,7 @@ public partial class WireMockAdminApiTests
|
|||||||
server.Stop();
|
server.Stop();
|
||||||
}
|
}
|
||||||
|
|
||||||
[Theory]
|
|
||||||
[InlineData(null, null)]
|
|
||||||
[InlineData(-1, -1)]
|
|
||||||
[InlineData(0, 0)]
|
|
||||||
[InlineData(200, 200)]
|
|
||||||
[InlineData("200", "200")]
|
|
||||||
public async Task IWireMockAdminApi_PostMappingAsync_WithStatusCode(object statusCode, object expectedStatusCode)
|
|
||||||
{
|
|
||||||
// Arrange
|
|
||||||
var server = WireMockServer.StartWithAdminInterface();
|
|
||||||
var api = RestClient.For<IWireMockAdminApi>(server.Urls[0]);
|
|
||||||
|
|
||||||
// Act
|
|
||||||
var model = new MappingModel
|
|
||||||
{
|
|
||||||
Request = new RequestModel { Path = "/1" },
|
|
||||||
Response = new ResponseModel { Body = "txt", StatusCode = statusCode },
|
|
||||||
Priority = 500,
|
|
||||||
Title = "test"
|
|
||||||
};
|
|
||||||
var result = await api.PostMappingAsync(model).ConfigureAwait(false);
|
|
||||||
|
|
||||||
// Assert
|
|
||||||
Check.That(result).IsNotNull();
|
|
||||||
Check.That(result.Status).IsNotNull();
|
|
||||||
Check.That(result.Guid).IsNotNull();
|
|
||||||
|
|
||||||
var mapping = server.Mappings.Single(m => m.Priority == 500);
|
|
||||||
Check.That(mapping).IsNotNull();
|
|
||||||
Check.That(mapping.Title).Equals("test");
|
|
||||||
|
|
||||||
var response = await mapping.ProvideResponseAsync(new RequestMessage(new UrlDetails("http://localhost/1"), "GET", "")).ConfigureAwait(false);
|
|
||||||
Check.That(response.Message.StatusCode).Equals(expectedStatusCode);
|
|
||||||
|
|
||||||
server.Stop();
|
|
||||||
}
|
|
||||||
|
|
||||||
[Fact]
|
|
||||||
public async Task IWireMockAdminApi_PostMappingsAsync()
|
|
||||||
{
|
|
||||||
// Arrange
|
|
||||||
var server = WireMockServer.StartWithAdminInterface();
|
|
||||||
var api = RestClient.For<IWireMockAdminApi>(server.Urls[0]);
|
|
||||||
|
|
||||||
// Act
|
|
||||||
var model1 = new MappingModel
|
|
||||||
{
|
|
||||||
Request = new RequestModel { Path = "/1" },
|
|
||||||
Response = new ResponseModel { Body = "txt 1" },
|
|
||||||
Title = "test 1"
|
|
||||||
};
|
|
||||||
var model2 = new MappingModel
|
|
||||||
{
|
|
||||||
Request = new RequestModel { Path = "/2" },
|
|
||||||
Response = new ResponseModel { Body = "txt 2" },
|
|
||||||
Title = "test 2"
|
|
||||||
};
|
|
||||||
var result = await api.PostMappingsAsync(new[] { model1, model2 }).ConfigureAwait(false);
|
|
||||||
|
|
||||||
// Assert
|
|
||||||
Check.That(result).IsNotNull();
|
|
||||||
Check.That(result.Status).IsNotNull();
|
|
||||||
Check.That(result.Guid).IsNull();
|
|
||||||
Check.That(server.Mappings.Where(m => !m.IsAdminInterface)).HasSize(2);
|
|
||||||
|
|
||||||
server.Stop();
|
|
||||||
}
|
|
||||||
|
|
||||||
[Fact]
|
|
||||||
public async Task IWireMockAdminApi_PostMappingsAsync_WithDuplicateGuids_Should_Return_400()
|
|
||||||
{
|
|
||||||
// Arrange
|
|
||||||
var guid = Guid.Parse("1b731398-4a5b-457f-a6e3-d65e541c428f");
|
|
||||||
var server = WireMockServer.StartWithAdminInterface();
|
|
||||||
var api = RestClient.For<IWireMockAdminApi>(server.Urls[0]);
|
|
||||||
|
|
||||||
// Act
|
|
||||||
var model1WithGuid = new MappingModel
|
|
||||||
{
|
|
||||||
Guid = guid,
|
|
||||||
Request = new RequestModel { Path = "/1g" },
|
|
||||||
Response = new ResponseModel { Body = "txt 1g" },
|
|
||||||
Title = "test 1g"
|
|
||||||
};
|
|
||||||
var model2WithGuid = new MappingModel
|
|
||||||
{
|
|
||||||
Guid = guid,
|
|
||||||
Request = new RequestModel { Path = "/2g" },
|
|
||||||
Response = new ResponseModel { Body = "txt 2g" },
|
|
||||||
Title = "test 2g"
|
|
||||||
};
|
|
||||||
var model1 = new MappingModel
|
|
||||||
{
|
|
||||||
Request = new RequestModel { Path = "/1" },
|
|
||||||
Response = new ResponseModel { Body = "txt 1" },
|
|
||||||
Title = "test 1"
|
|
||||||
};
|
|
||||||
var model2 = new MappingModel
|
|
||||||
{
|
|
||||||
Request = new RequestModel { Path = "/2" },
|
|
||||||
Response = new ResponseModel { Body = "txt 2" },
|
|
||||||
Title = "test 2"
|
|
||||||
};
|
|
||||||
|
|
||||||
var models = new[]
|
|
||||||
{
|
|
||||||
model1WithGuid,
|
|
||||||
model2WithGuid,
|
|
||||||
model1,
|
|
||||||
model2
|
|
||||||
};
|
|
||||||
|
|
||||||
var sutMethod = async () => await api.PostMappingsAsync(models);
|
|
||||||
var exceptionAssertions = await sutMethod.Should().ThrowAsync<ApiException>();
|
|
||||||
exceptionAssertions.Which.Content.Should().Be(@"{""Status"":""The following Guids are duplicate : '1b731398-4a5b-457f-a6e3-d65e541c428f' (Parameter 'mappingModels')""}");
|
|
||||||
|
|
||||||
server.Stop();
|
|
||||||
}
|
|
||||||
|
|
||||||
[Fact]
|
[Fact]
|
||||||
public async Task IWireMockAdminApi_FindRequestsAsync()
|
public async Task IWireMockAdminApi_FindRequestsAsync()
|
||||||
@@ -1140,5 +1023,10 @@ text
|
|||||||
// Assert
|
// Assert
|
||||||
status.Status.Should().Be("Static Mappings reloaded");
|
status.Status.Should().Be("Static Mappings reloaded");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private static string ReadMappingFile(string filename)
|
||||||
|
{
|
||||||
|
return File.ReadAllText(Path.Combine(Directory.GetCurrentDirectory(), "__admin", "mappings", filename));
|
||||||
|
}
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
@@ -2,19 +2,24 @@
|
|||||||
|
|
||||||
#if PROTOBUF
|
#if PROTOBUF
|
||||||
using System;
|
using System;
|
||||||
|
using System.IO;
|
||||||
using System.Net;
|
using System.Net;
|
||||||
using System.Net.Http;
|
using System.Net.Http;
|
||||||
using System.Net.Http.Headers;
|
using System.Net.Http.Headers;
|
||||||
|
using System.Text;
|
||||||
using System.Threading.Tasks;
|
using System.Threading.Tasks;
|
||||||
using FluentAssertions;
|
using FluentAssertions;
|
||||||
using Google.Protobuf.WellKnownTypes;
|
using Google.Protobuf.WellKnownTypes;
|
||||||
using Greet;
|
using Greet;
|
||||||
using Grpc.Net.Client;
|
using Grpc.Net.Client;
|
||||||
using NarrowIntegrationTest.Lookup;
|
using NarrowIntegrationTest.Lookup;
|
||||||
|
using WireMock.Constants;
|
||||||
using WireMock.Matchers;
|
using WireMock.Matchers;
|
||||||
using WireMock.RequestBuilders;
|
using WireMock.RequestBuilders;
|
||||||
using WireMock.ResponseBuilders;
|
using WireMock.ResponseBuilders;
|
||||||
using WireMock.Server;
|
using WireMock.Server;
|
||||||
|
using WireMock.Settings;
|
||||||
|
using WireMock.Util;
|
||||||
using Xunit;
|
using Xunit;
|
||||||
|
|
||||||
// ReSharper disable once CheckNamespace
|
// ReSharper disable once CheckNamespace
|
||||||
@@ -486,20 +491,17 @@ message Other {
|
|||||||
);
|
);
|
||||||
|
|
||||||
// Act
|
// Act
|
||||||
var channel = GrpcChannel.ForAddress(server.Url!);
|
var reply = await When_GrpcClient_Calls_SayHelloAsync(server.Url!);
|
||||||
var client = new Greeter.GreeterClient(channel);
|
|
||||||
|
|
||||||
var reply = await client.SayHelloAsync(new HelloRequest { Name = "stef" });
|
|
||||||
|
|
||||||
// Assert
|
// Assert
|
||||||
reply.Message.Should().Be("hello stef POST");
|
Then_ReplyMessage_Should_BeCorrect(reply);
|
||||||
}
|
}
|
||||||
|
|
||||||
[Fact]
|
[Fact]
|
||||||
public async Task WireMockServer_WithBodyAsProtoBuf_WithWellKnownTypes_Empty_UsingGrpcGeneratedClient()
|
public async Task WireMockServer_WithBodyAsProtoBuf_WithWellKnownTypes_Empty_UsingGrpcGeneratedClient()
|
||||||
{
|
{
|
||||||
// Arrange
|
// Arrange
|
||||||
var definition = await System.IO.File.ReadAllTextAsync("./Grpc/greet.proto");
|
var definition = await File.ReadAllTextAsync("./Grpc/greet.proto");
|
||||||
|
|
||||||
using var server = WireMockServer.Start(useHttp2: true);
|
using var server = WireMockServer.Start(useHttp2: true);
|
||||||
|
|
||||||
@@ -532,7 +534,7 @@ message Other {
|
|||||||
// Arrange
|
// Arrange
|
||||||
const int seconds = 1722301323;
|
const int seconds = 1722301323;
|
||||||
const int nanos = 12300;
|
const int nanos = 12300;
|
||||||
var definition = await System.IO.File.ReadAllTextAsync("./Grpc/greet.proto");
|
var definition = await File.ReadAllTextAsync("./Grpc/greet.proto");
|
||||||
|
|
||||||
using var server = WireMockServer.Start(useHttp2: true);
|
using var server = WireMockServer.Start(useHttp2: true);
|
||||||
|
|
||||||
@@ -573,7 +575,7 @@ message Other {
|
|||||||
// Arrange
|
// Arrange
|
||||||
const int seconds = 1722301323;
|
const int seconds = 1722301323;
|
||||||
const int nanos = 12300;
|
const int nanos = 12300;
|
||||||
var definition = await System.IO.File.ReadAllTextAsync("./Grpc/greet.proto");
|
var definition = await File.ReadAllTextAsync("./Grpc/greet.proto");
|
||||||
|
|
||||||
using var server = WireMockServer.Start(useHttp2: true);
|
using var server = WireMockServer.Start(useHttp2: true);
|
||||||
|
|
||||||
@@ -612,7 +614,7 @@ message Other {
|
|||||||
public async Task WireMockServer_WithBodyAsProtoBuf_Enum_UsingGrpcGeneratedClient()
|
public async Task WireMockServer_WithBodyAsProtoBuf_Enum_UsingGrpcGeneratedClient()
|
||||||
{
|
{
|
||||||
// Arrange
|
// Arrange
|
||||||
var definition = await System.IO.File.ReadAllTextAsync("./Grpc/greet.proto");
|
var definition = await File.ReadAllTextAsync("./Grpc/greet.proto");
|
||||||
|
|
||||||
using var server = WireMockServer.Start(useHttp2: true);
|
using var server = WireMockServer.Start(useHttp2: true);
|
||||||
|
|
||||||
@@ -653,7 +655,7 @@ message Other {
|
|||||||
const int nanos = 12300;
|
const int nanos = 12300;
|
||||||
const string version = "test";
|
const string version = "test";
|
||||||
const string correlationId = "correlation";
|
const string correlationId = "correlation";
|
||||||
var definition = await System.IO.File.ReadAllTextAsync("./Grpc/policy.proto");
|
var definition = await File.ReadAllTextAsync("./Grpc/policy.proto");
|
||||||
|
|
||||||
using var server = WireMockServer.Start(useHttp2: true);
|
using var server = WireMockServer.Start(useHttp2: true);
|
||||||
|
|
||||||
@@ -696,5 +698,68 @@ message Other {
|
|||||||
reply.Client.ClientName.Should().Be(NarrowIntegrationTest.Lookup.Client.Types.Clients.BillingCenter);
|
reply.Client.ClientName.Should().Be(NarrowIntegrationTest.Lookup.Client.Types.Clients.BillingCenter);
|
||||||
reply.Client.CorrelationId.Should().Be(correlationId);
|
reply.Client.CorrelationId.Should().Be(correlationId);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
[Fact]
|
||||||
|
public async Task WireMockServer_WithBodyAsProtoBuf_ServerProtoDefinitionFromJson_UsingGrpcGeneratedClient()
|
||||||
|
{
|
||||||
|
var server = Given_When_ServerStartedUsingHttp2();
|
||||||
|
Given_ProtoDefinition_IsAddedOnServerLevel(server);
|
||||||
|
await Given_When_ProtoBufMappingIsAddedViaAdminInterfaceAsync(server);
|
||||||
|
|
||||||
|
var reply = await When_GrpcClient_Calls_SayHelloAsync(server.Urls[1]);
|
||||||
|
|
||||||
|
Then_ReplyMessage_Should_BeCorrect(reply);
|
||||||
|
}
|
||||||
|
|
||||||
|
private static WireMockServer Given_When_ServerStartedUsingHttp2()
|
||||||
|
{
|
||||||
|
var ports = PortUtils.FindFreeTcpPorts(2);
|
||||||
|
|
||||||
|
var settings = new WireMockServerSettings
|
||||||
|
{
|
||||||
|
Urls = [$"http://*:{ports[0]}/", $"grpc://*:{ports[1]}/"],
|
||||||
|
StartAdminInterface = true
|
||||||
|
};
|
||||||
|
return WireMockServer.Start(settings);
|
||||||
|
}
|
||||||
|
|
||||||
|
private static void Given_ProtoDefinition_IsAddedOnServerLevel(WireMockServer server)
|
||||||
|
{
|
||||||
|
server.AddProtoDefinition("my-greeter", ReadProtoFile("greet.proto"));
|
||||||
|
}
|
||||||
|
|
||||||
|
private static async Task Given_When_ProtoBufMappingIsAddedViaAdminInterfaceAsync(WireMockServer server)
|
||||||
|
{
|
||||||
|
var mappingsJson = ReadMappingFile("protobuf-mapping-3.json");
|
||||||
|
|
||||||
|
using var httpClient = server.CreateClient();
|
||||||
|
|
||||||
|
var result = await httpClient.PostAsync("/__admin/mappings", new StringContent(mappingsJson, Encoding.UTF8, WireMockConstants.ContentTypeJson));
|
||||||
|
result.EnsureSuccessStatusCode();
|
||||||
|
}
|
||||||
|
|
||||||
|
private static async Task<HelloReply> When_GrpcClient_Calls_SayHelloAsync(string address)
|
||||||
|
{
|
||||||
|
var channel = GrpcChannel.ForAddress(address);
|
||||||
|
|
||||||
|
var client = new Greeter.GreeterClient(channel);
|
||||||
|
|
||||||
|
return await client.SayHelloAsync(new HelloRequest { Name = "stef" });
|
||||||
|
}
|
||||||
|
|
||||||
|
private static void Then_ReplyMessage_Should_BeCorrect(HelloReply reply)
|
||||||
|
{
|
||||||
|
reply.Message.Should().Be("hello stef POST");
|
||||||
|
}
|
||||||
|
|
||||||
|
private static string ReadMappingFile(string filename)
|
||||||
|
{
|
||||||
|
return File.ReadAllText(Path.Combine(Directory.GetCurrentDirectory(), "__admin", "mappings", filename));
|
||||||
|
}
|
||||||
|
|
||||||
|
private static string ReadProtoFile(string filename)
|
||||||
|
{
|
||||||
|
return File.ReadAllText(Path.Combine(Directory.GetCurrentDirectory(), "Grpc", filename));
|
||||||
|
}
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
@@ -15,7 +15,6 @@ using RestEase;
|
|||||||
using WireMock.Client;
|
using WireMock.Client;
|
||||||
using WireMock.Handlers;
|
using WireMock.Handlers;
|
||||||
using WireMock.Logging;
|
using WireMock.Logging;
|
||||||
using WireMock.Matchers;
|
|
||||||
using WireMock.Matchers.Request;
|
using WireMock.Matchers.Request;
|
||||||
using WireMock.RequestBuilders;
|
using WireMock.RequestBuilders;
|
||||||
using WireMock.ResponseBuilders;
|
using WireMock.ResponseBuilders;
|
||||||
@@ -27,7 +26,8 @@ namespace WireMock.Net.Tests;
|
|||||||
|
|
||||||
public class WireMockServerAdminTests
|
public class WireMockServerAdminTests
|
||||||
{
|
{
|
||||||
// For for AppVeyor + OpenCover
|
private const int NumStaticMappings = 10;
|
||||||
|
|
||||||
private static string GetCurrentFolder()
|
private static string GetCurrentFolder()
|
||||||
{
|
{
|
||||||
return Directory.GetCurrentDirectory();
|
return Directory.GetCurrentDirectory();
|
||||||
@@ -40,8 +40,8 @@ public class WireMockServerAdminTests
|
|||||||
string folder = Path.Combine(GetCurrentFolder(), "__admin", "mappings");
|
string folder = Path.Combine(GetCurrentFolder(), "__admin", "mappings");
|
||||||
server.ReadStaticMappings(folder);
|
server.ReadStaticMappings(folder);
|
||||||
|
|
||||||
Check.That(server.Mappings).HasSize(6);
|
Check.That(server.Mappings).HasSize(NumStaticMappings);
|
||||||
Check.That(server.MappingModels).HasSize(6);
|
Check.That(server.MappingModels).HasSize(NumStaticMappings);
|
||||||
|
|
||||||
// Act
|
// Act
|
||||||
server.ResetMappings();
|
server.ResetMappings();
|
||||||
@@ -220,7 +220,7 @@ public class WireMockServerAdminTests
|
|||||||
server.ReadStaticMappings(folder);
|
server.ReadStaticMappings(folder);
|
||||||
|
|
||||||
var mappings = server.Mappings.ToArray();
|
var mappings = server.Mappings.ToArray();
|
||||||
Check.That(mappings).HasSize(6);
|
Check.That(mappings).HasSize(NumStaticMappings);
|
||||||
|
|
||||||
server.Stop();
|
server.Stop();
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -0,0 +1,49 @@
|
|||||||
|
{
|
||||||
|
"Guid": "351f0240-bba0-4bcb-93c6-1feba0fe0001",
|
||||||
|
"Title": "ProtoBuf Mapping 1",
|
||||||
|
"Request": {
|
||||||
|
"Path": {
|
||||||
|
"Matchers": [
|
||||||
|
{
|
||||||
|
"Name": "WildcardMatcher",
|
||||||
|
"Pattern": "/greet.Greeter/SayHello",
|
||||||
|
"IgnoreCase": false
|
||||||
|
}
|
||||||
|
]
|
||||||
|
},
|
||||||
|
"Methods": [
|
||||||
|
"POST"
|
||||||
|
],
|
||||||
|
"Body": {
|
||||||
|
"Matcher": {
|
||||||
|
"Name": "ProtoBufMatcher",
|
||||||
|
"Pattern": "\r\nsyntax = \"proto3\";\r\n\r\npackage greet;\r\n\r\nservice Greeter {\r\n rpc SayHello (HelloRequest) returns (HelloReply);\r\n}\r\n\r\nmessage HelloRequest {\r\n string name = 1;\r\n}\r\n\r\nmessage HelloReply {\r\n string message = 1;\r\n}\r\n",
|
||||||
|
"ContentMatcher": {
|
||||||
|
"Name": "JsonPartialWildcardMatcher",
|
||||||
|
"Pattern": {
|
||||||
|
"name": "*"
|
||||||
|
},
|
||||||
|
"IgnoreCase": false,
|
||||||
|
"Regex": false
|
||||||
|
},
|
||||||
|
"ProtoBufMessageType": "greet.HelloRequest"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"Response": {
|
||||||
|
"BodyAsJson": {
|
||||||
|
"message": "hello {{request.BodyAsJson.name}}"
|
||||||
|
},
|
||||||
|
"UseTransformer": true,
|
||||||
|
"TransformerType": "Handlebars",
|
||||||
|
"TransformerReplaceNodeOptions": "EvaluateAndTryToConvert",
|
||||||
|
"Headers": {
|
||||||
|
"Content-Type": "application/grpc"
|
||||||
|
},
|
||||||
|
"TrailingHeaders": {
|
||||||
|
"grpc-status": "0"
|
||||||
|
},
|
||||||
|
"ProtoDefinition": "\r\nsyntax = \"proto3\";\r\n\r\npackage greet;\r\n\r\nservice Greeter {\r\n rpc SayHello (HelloRequest) returns (HelloReply);\r\n}\r\n\r\nmessage HelloRequest {\r\n string name = 1;\r\n}\r\n\r\nmessage HelloReply {\r\n string message = 1;\r\n}\r\n",
|
||||||
|
"ProtoBufMessageType": "greet.HelloReply"
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,48 @@
|
|||||||
|
{
|
||||||
|
"Guid": "351f0240-bba0-4bcb-93c6-1feba0fe0002",
|
||||||
|
"Title": "ProtoBuf Mapping 2",
|
||||||
|
"Request": {
|
||||||
|
"Path": {
|
||||||
|
"Matchers": [
|
||||||
|
{
|
||||||
|
"Name": "WildcardMatcher",
|
||||||
|
"Pattern": "/greet.Greeter/SayHello",
|
||||||
|
"IgnoreCase": false
|
||||||
|
}
|
||||||
|
]
|
||||||
|
},
|
||||||
|
"Methods": [
|
||||||
|
"POST"
|
||||||
|
],
|
||||||
|
"Body": {
|
||||||
|
"Matcher": {
|
||||||
|
"Name": "ProtoBufMatcher",
|
||||||
|
"ContentMatcher": {
|
||||||
|
"Name": "JsonPartialWildcardMatcher",
|
||||||
|
"Pattern": {
|
||||||
|
"name": "*"
|
||||||
|
},
|
||||||
|
"IgnoreCase": false,
|
||||||
|
"Regex": false
|
||||||
|
},
|
||||||
|
"ProtoBufMessageType": "greet.HelloRequest"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"Response": {
|
||||||
|
"BodyAsJson": {
|
||||||
|
"message": "hello {{request.BodyAsJson.name}}"
|
||||||
|
},
|
||||||
|
"UseTransformer": true,
|
||||||
|
"TransformerType": "Handlebars",
|
||||||
|
"TransformerReplaceNodeOptions": "EvaluateAndTryToConvert",
|
||||||
|
"Headers": {
|
||||||
|
"Content-Type": "application/grpc"
|
||||||
|
},
|
||||||
|
"TrailingHeaders": {
|
||||||
|
"grpc-status": "0"
|
||||||
|
},
|
||||||
|
"ProtoBufMessageType": "greet.HelloReply"
|
||||||
|
},
|
||||||
|
"ProtoDefinition": "\r\nsyntax = \"proto3\";\r\n\r\npackage greet;\r\n\r\nservice Greeter {\r\n rpc SayHello (HelloRequest) returns (HelloReply);\r\n}\r\n\r\nmessage HelloRequest {\r\n string name = 1;\r\n}\r\n\r\nmessage HelloReply {\r\n string message = 1;\r\n}\r\n"
|
||||||
|
}
|
||||||
@@ -0,0 +1,48 @@
|
|||||||
|
{
|
||||||
|
"Guid": "351f0240-bba0-4bcb-93c6-1feba0fe0003",
|
||||||
|
"Title": "ProtoBuf Mapping 3",
|
||||||
|
"Request": {
|
||||||
|
"Path": {
|
||||||
|
"Matchers": [
|
||||||
|
{
|
||||||
|
"Name": "WildcardMatcher",
|
||||||
|
"Pattern": "/greet.Greeter/SayHello",
|
||||||
|
"IgnoreCase": false
|
||||||
|
}
|
||||||
|
]
|
||||||
|
},
|
||||||
|
"Methods": [
|
||||||
|
"POST"
|
||||||
|
],
|
||||||
|
"Body": {
|
||||||
|
"Matcher": {
|
||||||
|
"Name": "ProtoBufMatcher",
|
||||||
|
"ContentMatcher": {
|
||||||
|
"Name": "JsonPartialWildcardMatcher",
|
||||||
|
"Pattern": {
|
||||||
|
"name": "*"
|
||||||
|
},
|
||||||
|
"IgnoreCase": true,
|
||||||
|
"Regex": false
|
||||||
|
},
|
||||||
|
"ProtoBufMessageType": "greet.HelloRequest"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"Response": {
|
||||||
|
"BodyAsJson": {
|
||||||
|
"message": "hello {{request.BodyAsJson.name}} {{request.method}}"
|
||||||
|
},
|
||||||
|
"UseTransformer": true,
|
||||||
|
"TransformerType": "Handlebars",
|
||||||
|
"TransformerReplaceNodeOptions": "EvaluateAndTryToConvert",
|
||||||
|
"Headers": {
|
||||||
|
"Content-Type": "application/grpc"
|
||||||
|
},
|
||||||
|
"TrailingHeaders": {
|
||||||
|
"grpc-status": "0"
|
||||||
|
},
|
||||||
|
"ProtoBufMessageType": "greet.HelloReply"
|
||||||
|
},
|
||||||
|
"ProtoDefinition": "my-greeter"
|
||||||
|
}
|
||||||
@@ -0,0 +1,40 @@
|
|||||||
|
{
|
||||||
|
"Guid": "351f0240-bba0-4bcb-93c6-1feba0fe0004",
|
||||||
|
"Title": "ProtoBuf Mapping 4",
|
||||||
|
"Request": {
|
||||||
|
"Path": {
|
||||||
|
"Matchers": [
|
||||||
|
{
|
||||||
|
"Name": "WildcardMatcher",
|
||||||
|
"Pattern": "/greet.Greeter/SayHello",
|
||||||
|
"IgnoreCase": false
|
||||||
|
}
|
||||||
|
]
|
||||||
|
},
|
||||||
|
"Methods": [
|
||||||
|
"POST"
|
||||||
|
],
|
||||||
|
"Body": {
|
||||||
|
"Matcher": {
|
||||||
|
"Name": "ProtoBufMatcher",
|
||||||
|
"ProtoBufMessageType": "greet.HelloRequest"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"Response": {
|
||||||
|
"BodyAsJson": {
|
||||||
|
"message": "hello {{request.BodyAsJson.name}}"
|
||||||
|
},
|
||||||
|
"UseTransformer": true,
|
||||||
|
"TransformerType": "Handlebars",
|
||||||
|
"TransformerReplaceNodeOptions": "EvaluateAndTryToConvert",
|
||||||
|
"Headers": {
|
||||||
|
"Content-Type": "application/grpc"
|
||||||
|
},
|
||||||
|
"TrailingHeaders": {
|
||||||
|
"grpc-status": "0"
|
||||||
|
},
|
||||||
|
"ProtoBufMessageType": "greet.HelloReply"
|
||||||
|
},
|
||||||
|
"ProtoDefinition": "my-greeter"
|
||||||
|
}
|
||||||
Reference in New Issue
Block a user