mirror of
https://github.com/wiremock/WireMock.Net.git
synced 2026-03-24 10:01:00 +01:00
Handlebars #4
This commit is contained in:
57
src/WireMock/Extensions/DictionaryExtensions.cs
Normal file
57
src/WireMock/Extensions/DictionaryExtensions.cs
Normal file
@@ -0,0 +1,57 @@
|
||||
using System.Collections;
|
||||
using System.Collections.Generic;
|
||||
using System.Dynamic;
|
||||
|
||||
namespace WireMock.Extensions
|
||||
{
|
||||
/// <summary>
|
||||
/// Dictionary Extensions
|
||||
/// </summary>
|
||||
public static class DictionaryExtensions
|
||||
{
|
||||
public static dynamic ToExpandoObject<T>(this IDictionary<string, T> dictionary)
|
||||
{
|
||||
dynamic expando = new ExpandoObject();
|
||||
var expandoDic = (IDictionary<string, object>)expando;
|
||||
|
||||
// go through the items in the dictionary and copy over the key value pairs)
|
||||
foreach (var kvp in dictionary)
|
||||
{
|
||||
// if the value can also be turned into an ExpandoObject, then do it!
|
||||
var value = kvp.Value as IDictionary<string, object>;
|
||||
if (value != null)
|
||||
{
|
||||
var expandoValue = value.ToExpandoObject();
|
||||
expandoDic.Add(kvp.Key, expandoValue);
|
||||
}
|
||||
else if (kvp.Value is ICollection)
|
||||
{
|
||||
// iterate through the collection and convert any strin-object dictionaries
|
||||
// along the way into expando objects
|
||||
var itemList = new List<object>();
|
||||
foreach (var item in (ICollection)kvp.Value)
|
||||
{
|
||||
var objects = item as IDictionary<string, object>;
|
||||
if (objects != null)
|
||||
{
|
||||
var expandoItem = objects.ToExpandoObject();
|
||||
itemList.Add(expandoItem);
|
||||
}
|
||||
else
|
||||
{
|
||||
itemList.Add(item);
|
||||
}
|
||||
}
|
||||
|
||||
expandoDic.Add(kvp.Key, itemList);
|
||||
}
|
||||
else
|
||||
{
|
||||
expandoDic.Add(kvp.Key, kvp.Value);
|
||||
}
|
||||
}
|
||||
|
||||
return expando;
|
||||
}
|
||||
}
|
||||
}
|
||||
@@ -31,10 +31,10 @@ namespace WireMock
|
||||
/// </returns>
|
||||
public RequestMessage Map(HttpListenerRequest listenerRequest)
|
||||
{
|
||||
var path = listenerRequest.Url.AbsolutePath;
|
||||
var query = listenerRequest.Url.Query;
|
||||
var verb = listenerRequest.HttpMethod;
|
||||
var body = GetRequestBody(listenerRequest);
|
||||
string path = listenerRequest.Url.AbsolutePath;
|
||||
string query = listenerRequest.Url.Query;
|
||||
string verb = listenerRequest.HttpMethod;
|
||||
string body = GetRequestBody(listenerRequest);
|
||||
var listenerHeaders = listenerRequest.Headers;
|
||||
var headers = listenerHeaders.AllKeys.ToDictionary(k => k, k => listenerHeaders[k]);
|
||||
|
||||
|
||||
@@ -27,7 +27,7 @@ namespace WireMock.RequestBuilders
|
||||
/// <summary>
|
||||
/// The requests.
|
||||
/// </summary>
|
||||
public class Request : CompositeRequestSpec, IVerbRequestBuilder, IHeadersRequestBuilder, IParamsRequestBuilder
|
||||
public class Request : CompositeRequestSpec, IVerbRequestBuilder
|
||||
{
|
||||
/// <summary>
|
||||
/// The _request specs.
|
||||
|
||||
@@ -1,6 +1,8 @@
|
||||
using System.Collections.Generic;
|
||||
using System.Collections.Concurrent;
|
||||
using System.Collections.Generic;
|
||||
using System.Diagnostics.CodeAnalysis;
|
||||
using System.Linq;
|
||||
using WireMock.Extensions;
|
||||
|
||||
[module:
|
||||
SuppressMessage("StyleCop.CSharp.ReadabilityRules",
|
||||
@@ -26,11 +28,6 @@ namespace WireMock
|
||||
/// </summary>
|
||||
public class RequestMessage
|
||||
{
|
||||
/// <summary>
|
||||
/// The _params.
|
||||
/// </summary>
|
||||
private readonly IDictionary<string, List<string>> _params = new Dictionary<string, List<string>>();
|
||||
|
||||
/// <summary>
|
||||
/// Initializes a new instance of the <see cref="RequestMessage"/> class.
|
||||
/// </summary>
|
||||
@@ -58,7 +55,7 @@ namespace WireMock
|
||||
query = query.Substring(1);
|
||||
}
|
||||
|
||||
_params = query.Split('&').Aggregate(
|
||||
Parameters = query.Split('&').Aggregate(
|
||||
new Dictionary<string, List<string>>(),
|
||||
(dict, term) =>
|
||||
{
|
||||
@@ -72,7 +69,19 @@ namespace WireMock
|
||||
return dict;
|
||||
});
|
||||
|
||||
Parameters = _params;
|
||||
var tmpDictionary = new Dictionary<string, object>();
|
||||
foreach (var parameter in Parameters.Where(p => p.Value.Any()))
|
||||
{
|
||||
if (parameter.Value.Count == 1)
|
||||
{
|
||||
tmpDictionary.Add(parameter.Key, parameter.Value.First());
|
||||
}
|
||||
else
|
||||
{
|
||||
tmpDictionary.Add(parameter.Key, parameter.Value);
|
||||
}
|
||||
}
|
||||
Query = tmpDictionary.ToExpandoObject();
|
||||
}
|
||||
|
||||
Path = path;
|
||||
@@ -88,12 +97,12 @@ namespace WireMock
|
||||
{
|
||||
get
|
||||
{
|
||||
if (!_params.Any())
|
||||
if (!Parameters.Any())
|
||||
{
|
||||
return Path;
|
||||
}
|
||||
|
||||
return Path + "?" + string.Join("&", _params.SelectMany(kv => kv.Value.Select(value => kv.Key + "=" + value)));
|
||||
return Path + "?" + string.Join("&", Parameters.SelectMany(kv => kv.Value.Select(value => kv.Key + "=" + value)));
|
||||
}
|
||||
}
|
||||
|
||||
@@ -113,9 +122,14 @@ namespace WireMock
|
||||
public IDictionary<string, string> Headers { get; }
|
||||
|
||||
/// <summary>
|
||||
/// Gets the parameters.
|
||||
/// Gets the query parameters.
|
||||
/// </summary>
|
||||
public IDictionary<string, List<string>> Parameters { get; }
|
||||
public IDictionary<string, List<string>> Parameters { get; } = new Dictionary<string, List<string>>();
|
||||
|
||||
/// <summary>
|
||||
/// Gets the query as object.
|
||||
/// </summary>
|
||||
public dynamic Query { get; }
|
||||
|
||||
/// <summary>
|
||||
/// Gets the body.
|
||||
@@ -133,7 +147,7 @@ namespace WireMock
|
||||
/// </returns>
|
||||
public List<string> GetParameter(string key)
|
||||
{
|
||||
return _params.ContainsKey(key) ? _params[key] : new List<string>();
|
||||
return Parameters.ContainsKey(key) ? Parameters[key] : new List<string>();
|
||||
}
|
||||
}
|
||||
}
|
||||
@@ -12,8 +12,8 @@
|
||||
/// The body.
|
||||
/// </param>
|
||||
/// <returns>
|
||||
/// The <see cref="IDelayResponseBuilder"/>.
|
||||
/// The <see cref="IResponseBuilder"/>.
|
||||
/// </returns>
|
||||
IDelayResponseBuilder WithBody(string body);
|
||||
IResponseBuilder WithBody(string body);
|
||||
}
|
||||
}
|
||||
@@ -14,8 +14,8 @@ namespace WireMock.ResponseBuilders
|
||||
/// The delay.
|
||||
/// </param>
|
||||
/// <returns>
|
||||
/// The <see cref="IProvideResponses"/>.
|
||||
/// The <see cref="IResponseBuilder"/>.
|
||||
/// </returns>
|
||||
IProvideResponses AfterDelay(TimeSpan delay);
|
||||
IResponseBuilder AfterDelay(TimeSpan delay);
|
||||
}
|
||||
}
|
||||
@@ -15,8 +15,8 @@
|
||||
/// The value.
|
||||
/// </param>
|
||||
/// <returns>
|
||||
/// The <see cref="IHeadersResponseBuilder"/>.
|
||||
/// The <see cref="IResponseBuilder"/>.
|
||||
/// </returns>
|
||||
IHeadersResponseBuilder WithHeader(string name, string value);
|
||||
IResponseBuilder WithHeader(string name, string value);
|
||||
}
|
||||
}
|
||||
9
src/WireMock/ResponseBuilders/IResponseBuilder.cs
Normal file
9
src/WireMock/ResponseBuilders/IResponseBuilder.cs
Normal file
@@ -0,0 +1,9 @@
|
||||
namespace WireMock.ResponseBuilders
|
||||
{
|
||||
/// <summary>
|
||||
/// The ResponseBuilder interface.
|
||||
/// </summary>
|
||||
public interface IResponseBuilder : ITransformResponseBuilder
|
||||
{
|
||||
}
|
||||
}
|
||||
16
src/WireMock/ResponseBuilders/ITransformResponseBuilder.cs
Normal file
16
src/WireMock/ResponseBuilders/ITransformResponseBuilder.cs
Normal file
@@ -0,0 +1,16 @@
|
||||
namespace WireMock.ResponseBuilders
|
||||
{
|
||||
/// <summary>
|
||||
/// The BodyResponseBuilder interface.
|
||||
/// </summary>
|
||||
public interface ITransformResponseBuilder : IHeadersResponseBuilder
|
||||
{
|
||||
/// <summary>
|
||||
/// The with transformer.
|
||||
/// </summary>
|
||||
/// <returns>
|
||||
/// The <see cref="IResponseBuilder"/>.
|
||||
/// </returns>
|
||||
IResponseBuilder WithTransformer();
|
||||
}
|
||||
}
|
||||
@@ -1,6 +1,8 @@
|
||||
using System;
|
||||
using System.Collections.Generic;
|
||||
using System.Diagnostics.CodeAnalysis;
|
||||
using System.Threading.Tasks;
|
||||
using HandlebarsDotNet;
|
||||
|
||||
[module:
|
||||
SuppressMessage("StyleCop.CSharp.ReadabilityRules",
|
||||
@@ -21,7 +23,7 @@ namespace WireMock.ResponseBuilders
|
||||
/// <summary>
|
||||
/// The responses.
|
||||
/// </summary>
|
||||
public class Response : IHeadersResponseBuilder
|
||||
public class Response : IResponseBuilder
|
||||
{
|
||||
/// <summary>
|
||||
/// The _response.
|
||||
@@ -33,6 +35,8 @@ namespace WireMock.ResponseBuilders
|
||||
/// </summary>
|
||||
private TimeSpan _delay = TimeSpan.Zero;
|
||||
|
||||
private bool _useTransformer;
|
||||
|
||||
/// <summary>
|
||||
/// Initializes a new instance of the <see cref="Response"/> class.
|
||||
/// </summary>
|
||||
@@ -47,8 +51,8 @@ namespace WireMock.ResponseBuilders
|
||||
/// <summary>
|
||||
/// The with Success status code.
|
||||
/// </summary>
|
||||
/// <returns>The <see cref="IHeadersResponseBuilder"/>.</returns>
|
||||
public static IHeadersResponseBuilder WithSuccess()
|
||||
/// <returns>The <see cref="IResponseBuilder"/>.</returns>
|
||||
public static IResponseBuilder WithSuccess()
|
||||
{
|
||||
return WithStatusCode(200);
|
||||
}
|
||||
@@ -56,8 +60,8 @@ namespace WireMock.ResponseBuilders
|
||||
/// <summary>
|
||||
/// The with NotFound status code.
|
||||
/// </summary>
|
||||
/// <returns>The <see cref="IHeadersResponseBuilder"/>.</returns>
|
||||
public static IHeadersResponseBuilder WithNotFound()
|
||||
/// <returns>The <see cref="IResponseBuilder"/>.</returns>
|
||||
public static IResponseBuilder WithNotFound()
|
||||
{
|
||||
return WithStatusCode(404);
|
||||
}
|
||||
@@ -69,9 +73,9 @@ namespace WireMock.ResponseBuilders
|
||||
/// The code.
|
||||
/// </param>
|
||||
/// <returns>
|
||||
/// The <see cref="IHeadersResponseBuilder"/>.
|
||||
/// The <see cref="IResponseBuilder"/>.
|
||||
/// </returns>
|
||||
public static IHeadersResponseBuilder WithStatusCode(int code)
|
||||
public static IResponseBuilder WithStatusCode(int code)
|
||||
{
|
||||
var response = new ResponseMessage { StatusCode = code };
|
||||
return new Response(response);
|
||||
@@ -88,7 +92,28 @@ namespace WireMock.ResponseBuilders
|
||||
/// </returns>
|
||||
public async Task<ResponseMessage> ProvideResponse(RequestMessage requestMessage)
|
||||
{
|
||||
if (_useTransformer)
|
||||
{
|
||||
var template = new { request = requestMessage };
|
||||
|
||||
// Body
|
||||
var templateBody = Handlebars.Compile(_responseMessage.Body);
|
||||
_responseMessage.Body = templateBody(template);
|
||||
|
||||
// Headers
|
||||
var newHeaders = new Dictionary<string, string>();
|
||||
foreach (var header in _responseMessage.Headers)
|
||||
{
|
||||
var templateHeaderKey = Handlebars.Compile(header.Key);
|
||||
var templateHeaderValue = Handlebars.Compile(header.Value);
|
||||
|
||||
newHeaders.Add(templateHeaderKey(template), templateHeaderValue(template));
|
||||
}
|
||||
_responseMessage.Headers = newHeaders;
|
||||
}
|
||||
|
||||
await Task.Delay(_delay);
|
||||
|
||||
return _responseMessage;
|
||||
}
|
||||
|
||||
@@ -102,9 +127,9 @@ namespace WireMock.ResponseBuilders
|
||||
/// The value.
|
||||
/// </param>
|
||||
/// <returns>
|
||||
/// The <see cref="IHeadersResponseBuilder"/>.
|
||||
/// The <see cref="IResponseBuilder"/>.
|
||||
/// </returns>
|
||||
public IHeadersResponseBuilder WithHeader(string name, string value)
|
||||
public IResponseBuilder WithHeader(string name, string value)
|
||||
{
|
||||
_responseMessage.AddHeader(name, value);
|
||||
return this;
|
||||
@@ -117,14 +142,26 @@ namespace WireMock.ResponseBuilders
|
||||
/// The body.
|
||||
/// </param>
|
||||
/// <returns>
|
||||
/// The <see cref="IDelayResponseBuilder"/>.
|
||||
/// The <see cref="IResponseBuilder"/>.
|
||||
/// </returns>
|
||||
public IDelayResponseBuilder WithBody(string body)
|
||||
public IResponseBuilder WithBody(string body)
|
||||
{
|
||||
_responseMessage.Body = body;
|
||||
return this;
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// The with transformer.
|
||||
/// </summary>
|
||||
/// <returns>
|
||||
/// The <see cref="IResponseBuilder"/>.
|
||||
/// </returns>
|
||||
public IResponseBuilder WithTransformer()
|
||||
{
|
||||
_useTransformer = true;
|
||||
return this;
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// The after delay.
|
||||
/// </summary>
|
||||
@@ -134,7 +171,7 @@ namespace WireMock.ResponseBuilders
|
||||
/// <returns>
|
||||
/// The <see cref="IProvideResponses"/>.
|
||||
/// </returns>
|
||||
public IProvideResponses AfterDelay(TimeSpan delay)
|
||||
public IResponseBuilder AfterDelay(TimeSpan delay)
|
||||
{
|
||||
_delay = delay;
|
||||
return this;
|
||||
|
||||
@@ -23,15 +23,10 @@ namespace WireMock
|
||||
/// </summary>
|
||||
public class ResponseMessage
|
||||
{
|
||||
/// <summary>
|
||||
/// The _headers.
|
||||
/// </summary>
|
||||
private readonly IDictionary<string, string> _headers = new ConcurrentDictionary<string, string>();
|
||||
|
||||
/// <summary>
|
||||
/// Gets the headers.
|
||||
/// </summary>
|
||||
public IDictionary<string, string> Headers => _headers;
|
||||
public IDictionary<string, string> Headers { get; set; } = new ConcurrentDictionary<string, string>();
|
||||
|
||||
/// <summary>
|
||||
/// Gets or sets the status code.
|
||||
@@ -54,7 +49,7 @@ namespace WireMock
|
||||
/// </param>
|
||||
public void AddHeader(string name, string value)
|
||||
{
|
||||
_headers.Add(name, value);
|
||||
Headers.Add(name, value);
|
||||
}
|
||||
}
|
||||
}
|
||||
@@ -30,6 +30,9 @@
|
||||
|
||||
"frameworks": {
|
||||
"net45": {
|
||||
"dependencies": {
|
||||
"Handlebars.Net": "1.8.0"
|
||||
},
|
||||
"frameworkAssemblies": {
|
||||
}
|
||||
}
|
||||
|
||||
Reference in New Issue
Block a user