mirror of
https://github.com/wiremock/WireMock.Net.git
synced 2026-03-21 00:28:59 +01:00
265 lines
10 KiB
C#
265 lines
10 KiB
C#
using System;
|
|
using System.Collections.Generic;
|
|
using System.Linq;
|
|
using JetBrains.Annotations;
|
|
using Newtonsoft.Json;
|
|
using Newtonsoft.Json.Linq;
|
|
using WireMock.Types;
|
|
using WireMock.Util;
|
|
|
|
namespace WireMock.Transformers
|
|
{
|
|
internal class Transformer : ITransformer
|
|
{
|
|
private readonly ITransformerContextFactory _factory;
|
|
|
|
public Transformer([NotNull] ITransformerContextFactory factory)
|
|
{
|
|
_factory = factory ?? throw new ArgumentNullException(nameof(factory));
|
|
}
|
|
|
|
public (IBodyData BodyData, IDictionary<string, WireMockList<string>> Headers) Transform(RequestMessage originalRequestMessage, ResponseMessage originalResponseMessage, IBodyData bodyData, IDictionary<string, WireMockList<string>> headers, ReplaceNodeOptions options)
|
|
{
|
|
var transformerContext = _factory.Create();
|
|
|
|
var model = new
|
|
{
|
|
request = originalRequestMessage,
|
|
response = originalResponseMessage
|
|
};
|
|
|
|
IBodyData newBodyData = null;
|
|
if (bodyData?.DetectedBodyType != null)
|
|
{
|
|
newBodyData = TransformBodyData(transformerContext, options, model, bodyData, false);
|
|
}
|
|
|
|
return (newBodyData, TransformHeaders(transformerContext, model, headers));
|
|
}
|
|
|
|
public ResponseMessage Transform(RequestMessage requestMessage, ResponseMessage original, bool useTransformerForBodyAsFile, ReplaceNodeOptions options)
|
|
{
|
|
var transformerContext = _factory.Create();
|
|
|
|
var responseMessage = new ResponseMessage();
|
|
|
|
var model = new
|
|
{
|
|
request = requestMessage
|
|
};
|
|
|
|
if (original.BodyData?.DetectedBodyType != null)
|
|
{
|
|
responseMessage.BodyData = TransformBodyData(transformerContext, options, model, original.BodyData, useTransformerForBodyAsFile);
|
|
|
|
if (original.BodyData.DetectedBodyType == BodyType.String)
|
|
{
|
|
responseMessage.BodyOriginal = original.BodyData.BodyAsString;
|
|
}
|
|
}
|
|
|
|
responseMessage.FaultType = original.FaultType;
|
|
responseMessage.FaultPercentage = original.FaultPercentage;
|
|
|
|
responseMessage.Headers = TransformHeaders(transformerContext, model, original.Headers);
|
|
|
|
switch (original.StatusCode)
|
|
{
|
|
case int statusCodeAsInteger:
|
|
responseMessage.StatusCode = statusCodeAsInteger;
|
|
break;
|
|
|
|
case string statusCodeAsString:
|
|
responseMessage.StatusCode = transformerContext.ParseAndRender(statusCodeAsString, model);
|
|
break;
|
|
}
|
|
|
|
return responseMessage;
|
|
}
|
|
|
|
private static IBodyData TransformBodyData(ITransformerContext transformerContext, ReplaceNodeOptions options, object model, IBodyData original, bool useTransformerForBodyAsFile)
|
|
{
|
|
switch (original?.DetectedBodyType)
|
|
{
|
|
case BodyType.Json:
|
|
return TransformBodyAsJson(transformerContext, options, model, original);
|
|
|
|
case BodyType.File:
|
|
return TransformBodyAsFile(transformerContext, model, original, useTransformerForBodyAsFile);
|
|
|
|
case BodyType.String:
|
|
return TransformBodyAsString(transformerContext, model, original);
|
|
|
|
default:
|
|
return null;
|
|
}
|
|
}
|
|
|
|
private static IDictionary<string, WireMockList<string>> TransformHeaders(ITransformerContext transformerContext, object model, IDictionary<string, WireMockList<string>> original)
|
|
{
|
|
if (original == null)
|
|
{
|
|
return new Dictionary<string, WireMockList<string>>();
|
|
}
|
|
|
|
var newHeaders = new Dictionary<string, WireMockList<string>>();
|
|
foreach (var header in original)
|
|
{
|
|
var headerKey = transformerContext.ParseAndRender(header.Key, model);
|
|
var templateHeaderValues = header.Value.Select(text => transformerContext.ParseAndRender(text, model)).ToArray();
|
|
|
|
newHeaders.Add(headerKey, new WireMockList<string>(templateHeaderValues));
|
|
}
|
|
|
|
return newHeaders;
|
|
}
|
|
|
|
private static IBodyData TransformBodyAsJson(ITransformerContext handlebarsContext, ReplaceNodeOptions options, object model, IBodyData original)
|
|
{
|
|
JToken jToken;
|
|
switch (original.BodyAsJson)
|
|
{
|
|
case JObject bodyAsJObject:
|
|
jToken = bodyAsJObject.DeepClone();
|
|
WalkNode(handlebarsContext, options, jToken, model);
|
|
break;
|
|
|
|
case Array bodyAsArray:
|
|
jToken = JArray.FromObject(bodyAsArray);
|
|
WalkNode(handlebarsContext, options, jToken, model);
|
|
break;
|
|
|
|
case string bodyAsString:
|
|
jToken = ReplaceSingleNode(handlebarsContext, options, bodyAsString, model);
|
|
break;
|
|
|
|
default:
|
|
jToken = JObject.FromObject(original.BodyAsJson);
|
|
WalkNode(handlebarsContext, options, jToken, model);
|
|
break;
|
|
}
|
|
|
|
return new BodyData
|
|
{
|
|
Encoding = original.Encoding,
|
|
DetectedBodyType = original.DetectedBodyType,
|
|
DetectedBodyTypeFromContentType = original.DetectedBodyTypeFromContentType,
|
|
BodyAsJson = jToken
|
|
};
|
|
}
|
|
|
|
private static JToken ReplaceSingleNode(ITransformerContext handlebarsContext, ReplaceNodeOptions options, string stringValue, object model)
|
|
{
|
|
string transformedString = handlebarsContext.ParseAndRender(stringValue, model);
|
|
|
|
if (!string.Equals(stringValue, transformedString))
|
|
{
|
|
const string property = "_";
|
|
JObject dummy = JObject.Parse($"{{ \"{property}\": null }}");
|
|
JToken node = dummy[property];
|
|
|
|
ReplaceNodeValue(options, node, transformedString);
|
|
|
|
return dummy[property];
|
|
}
|
|
|
|
return stringValue;
|
|
}
|
|
|
|
private static void WalkNode(ITransformerContext handlebarsContext, ReplaceNodeOptions options, JToken node, object model)
|
|
{
|
|
switch (node.Type)
|
|
{
|
|
case JTokenType.Object:
|
|
// In case of Object, loop all children. Do a ToArray() to avoid `Collection was modified` exceptions.
|
|
foreach (var child in node.Children<JProperty>().ToArray())
|
|
{
|
|
WalkNode(handlebarsContext, options, child.Value, model);
|
|
}
|
|
break;
|
|
|
|
case JTokenType.Array:
|
|
// In case of Array, loop all items. Do a ToArray() to avoid `Collection was modified` exceptions.
|
|
foreach (var child in node.Children().ToArray())
|
|
{
|
|
WalkNode(handlebarsContext, options, child, model);
|
|
}
|
|
break;
|
|
|
|
case JTokenType.String:
|
|
// In case of string, try to transform the value.
|
|
string stringValue = node.Value<string>();
|
|
if (string.IsNullOrEmpty(stringValue))
|
|
{
|
|
return;
|
|
}
|
|
|
|
string transformed = handlebarsContext.ParseAndRender(stringValue, model);
|
|
if (!string.Equals(stringValue, transformed))
|
|
{
|
|
ReplaceNodeValue(options, node, transformed);
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
|
|
private static void ReplaceNodeValue(ReplaceNodeOptions options, JToken node, string transformedString)
|
|
{
|
|
StringUtils.TryParseQuotedString(transformedString, out var result, out _);
|
|
if (bool.TryParse(result, out var valueAsBoolean) || bool.TryParse(transformedString, out valueAsBoolean))
|
|
{
|
|
node.Replace(valueAsBoolean);
|
|
return;
|
|
}
|
|
|
|
JToken value;
|
|
try
|
|
{
|
|
// Try to convert this string into a JsonObject
|
|
value = JToken.Parse(transformedString);
|
|
}
|
|
catch (JsonException)
|
|
{
|
|
// Ignore JsonException and just keep string value and convert to JToken
|
|
value = transformedString;
|
|
}
|
|
|
|
node.Replace(value);
|
|
}
|
|
|
|
private static IBodyData TransformBodyAsString(ITransformerContext handlebarsContext, object model, IBodyData original)
|
|
{
|
|
return new BodyData
|
|
{
|
|
Encoding = original.Encoding,
|
|
DetectedBodyType = original.DetectedBodyType,
|
|
DetectedBodyTypeFromContentType = original.DetectedBodyTypeFromContentType,
|
|
BodyAsString = handlebarsContext.ParseAndRender(original.BodyAsString, model)
|
|
};
|
|
}
|
|
|
|
private static IBodyData TransformBodyAsFile(ITransformerContext handlebarsContext, object model, IBodyData original, bool useTransformerForBodyAsFile)
|
|
{
|
|
string transformedBodyAsFilename = handlebarsContext.ParseAndRender(original.BodyAsFile, model);
|
|
|
|
if (!useTransformerForBodyAsFile)
|
|
{
|
|
return new BodyData
|
|
{
|
|
DetectedBodyType = original.DetectedBodyType,
|
|
DetectedBodyTypeFromContentType = original.DetectedBodyTypeFromContentType,
|
|
BodyAsFile = transformedBodyAsFilename
|
|
};
|
|
}
|
|
|
|
string text = handlebarsContext.FileSystemHandler.ReadResponseBodyAsString(transformedBodyAsFilename);
|
|
return new BodyData
|
|
{
|
|
DetectedBodyType = BodyType.String,
|
|
DetectedBodyTypeFromContentType = original.DetectedBodyTypeFromContentType,
|
|
BodyAsString = handlebarsContext.ParseAndRender(text, model),
|
|
BodyAsFile = transformedBodyAsFilename
|
|
};
|
|
}
|
|
}
|
|
} |