![JAR search and dependency download from the Maven repository](/logo.png)
com.opentable.extension.BodyTransformer Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of wiremock-body-transformer Show documentation
Show all versions of wiremock-body-transformer Show documentation
A Wiremock extension to transform the response body.
The newest version!
/*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package com.opentable.extension;
import com.fasterxml.jackson.databind.ObjectMapper;
import com.fasterxml.jackson.dataformat.xml.JacksonXmlModule;
import com.fasterxml.jackson.dataformat.xml.XmlMapper;
import com.github.tomakehurst.wiremock.client.ResponseDefinitionBuilder;
import com.github.tomakehurst.wiremock.common.BinaryFile;
import com.github.tomakehurst.wiremock.common.FileSource;
import com.github.tomakehurst.wiremock.extension.Parameters;
import com.github.tomakehurst.wiremock.extension.ResponseDefinitionTransformer;
import com.github.tomakehurst.wiremock.http.Request;
import com.github.tomakehurst.wiremock.http.ResponseDefinition;
import org.apache.commons.lang3.StringUtils;
import java.io.IOException;
import java.io.UnsupportedEncodingException;
import java.net.URLDecoder;
import java.nio.charset.StandardCharsets;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.Random;
import java.util.regex.Matcher;
import java.util.regex.Pattern;
public class BodyTransformer extends ResponseDefinitionTransformer {
private static final Pattern interpolationPattern = Pattern.compile("\\$\\(.*?\\)");
private static final Pattern randomIntegerPattern = Pattern.compile("!RandomInteger");
private static ObjectMapper jsonMapper = initJsonMapper();
private static ObjectMapper xmlMapper = initXmlMapper();
private static ObjectMapper initJsonMapper() {
return new ObjectMapper();
}
private static ObjectMapper initXmlMapper() {
JacksonXmlModule configuration = new JacksonXmlModule();
configuration.setXMLTextElementName("value");
return new XmlMapper(configuration);
}
@Override
public boolean applyGlobally() {
return false;
}
private String transformResponse(Map requestObject, String response) {
String modifiedResponse = response;
Matcher matcher = interpolationPattern.matcher(response);
while (matcher.find()) {
String group = matcher.group();
modifiedResponse = modifiedResponse.replace(group, getValue(group, requestObject));
}
return modifiedResponse;
}
private CharSequence getValue(String group, Map requestObject) {
if (randomIntegerPattern.matcher(group).find()) {
return String.valueOf(new Random().nextInt(2147483647));
}
return getValueFromRequestObject(group, requestObject);
}
private CharSequence getValueFromRequestObject(String group, Map requestObject) {
String fieldName = group.substring(2, group.length() - 1);
String[] fieldNames = fieldName.split("\\.");
Object tempObject = requestObject;
for (String field : fieldNames) {
if (tempObject instanceof Map) {
tempObject = ((Map) tempObject).get(field);
}
}
return String.valueOf(tempObject);
}
private boolean hasEmptyResponseBody(ResponseDefinition responseDefinition) {
return responseDefinition.getBody() == null && responseDefinition.getBodyFileName() == null;
}
private String getResponseBody(ResponseDefinition responseDefinition, FileSource fileSource) {
String body;
if (responseDefinition.getBody() != null) {
body = responseDefinition.getBody();
} else {
BinaryFile binaryFile = fileSource.getBinaryFileNamed(responseDefinition.getBodyFileName());
body = new String(binaryFile.readContents(), StandardCharsets.UTF_8);
}
return body;
}
@Override
public ResponseDefinition transform(Request request, ResponseDefinition responseDefinition, FileSource fileSource, Parameters parameters) {
if (hasEmptyResponseBody(responseDefinition)) {
return responseDefinition;
}
Map object = null;
String requestBody = request.getBodyAsString();
try {
object = jsonMapper.readValue(requestBody, Map.class);
} catch (IOException e) {
try {
object = xmlMapper.readValue(requestBody, Map.class);
} catch (IOException ex) {
// Validate is a body has the 'name=value' parameters
if (StringUtils.isNotEmpty(requestBody) && (requestBody.contains("&") || requestBody.contains("="))) {
object = new HashMap();
String[] pairedValues = requestBody.split("&");
for (String pair : pairedValues) {
String[] values = pair.split("=");
object.put(values[0], values.length > 1 ? decodeUTF8Value(values[1]) : "");
}
} else if (request.getAbsoluteUrl().split("\\?").length == 2) { // Validate query string parameters
object = new HashMap();
String absoluteUrl = request.getAbsoluteUrl();
String[] pairedValues = absoluteUrl.split("\\?")[1].split("&");
for (String pair : pairedValues) {
String[] values = pair.split("=");
object.put(values[0], values.length > 1 ? decodeUTF8Value(values[1]) : "");
}
} else {
System.err.println("[Body parse error] The body doesn't match any of 3 possible formats (JSON, XML, key=value).");
}
}
}
// Update the map with query parameters if any
if (parameters != null) {
String urlRegex = parameters.getString("urlRegex");
if (urlRegex != null) {
Pattern p = Pattern.compile(urlRegex);
Matcher m = p.matcher(request.getUrl());
// There may be more groups in the regex than the number of named capturing groups
List groups = getNamedGroupCandidates(urlRegex);
if (m.matches() &&
groups.size() > 0 &&
groups.size() <= m.groupCount()) {
for (int i = 0; i < groups.size(); i++) {
if (object == null) {
object = new HashMap();
}
object.put(groups.get(i), m.group(i + 1));
}
}
}
}
String responseBody = getResponseBody(responseDefinition, fileSource);
return ResponseDefinitionBuilder
.like(responseDefinition).but()
.withBodyFile(null)
.withBody(transformResponse(object, responseBody))
.build();
}
private static List getNamedGroupCandidates(String regex) {
List namedGroups = new ArrayList<>();
Matcher m = Pattern.compile("\\(\\?<([a-zA-Z][a-zA-Z0-9]*?)>").matcher(regex);
while (m.find()) {
namedGroups.add(m.group(1));
}
return namedGroups;
}
private String decodeUTF8Value(String value) {
String decodedValue = "";
try {
decodedValue = URLDecoder.decode(value, StandardCharsets.UTF_8.name());
} catch (UnsupportedEncodingException e) {
System.err.println("[Body parse error] Can't decode one of the request parameter. It should be UTF-8 charset.");
}
return decodedValue;
}
@Override
public String getName() {
return "body-transformer";
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy