com.github.tomakehurst.wiremock.servlet.WireMockHandlerDispatchingServlet Maven / Gradle / Ivy
/*
* Copyright (C) 2011 Thomas Akehurst
*
* 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.github.tomakehurst.wiremock.servlet;
import com.github.tomakehurst.wiremock.common.LocalNotifier;
import com.github.tomakehurst.wiremock.common.Notifier;
import com.github.tomakehurst.wiremock.core.FaultInjector;
import com.github.tomakehurst.wiremock.core.WireMockApp;
import com.github.tomakehurst.wiremock.http.*;
import javax.servlet.*;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.IOException;
import static com.github.tomakehurst.wiremock.common.Exceptions.throwUnchecked;
import static com.github.tomakehurst.wiremock.http.RequestMethod.GET;
import static com.google.common.base.Charsets.UTF_8;
import static java.net.HttpURLConnection.HTTP_NOT_FOUND;
import static java.net.URLDecoder.decode;
public class WireMockHandlerDispatchingServlet extends HttpServlet {
public static final String SHOULD_FORWARD_TO_FILES_CONTEXT = "shouldForwardToFilesContext";
public static final String MAPPED_UNDER_KEY = "mappedUnder";
private static final long serialVersionUID = -6602042274260495538L;
private RequestHandler requestHandler;
private FaultInjectorFactory faultHandlerFactory;
private String mappedUnder;
private Notifier notifier;
private String wiremockFileSourceRoot = "/";
private boolean shouldForwardToFilesContext;
@Override
public void init(ServletConfig config) {
ServletContext context = config.getServletContext();
shouldForwardToFilesContext = getFileContextForwardingFlagFrom(config);
if (context.getInitParameter("WireMockFileSourceRoot") != null) {
wiremockFileSourceRoot = context.getInitParameter("WireMockFileSourceRoot");
}
String handlerClassName = config.getInitParameter(RequestHandler.HANDLER_CLASS_KEY);
String faultInjectorFactoryClassName = config.getInitParameter(FaultInjectorFactory.INJECTOR_CLASS_KEY);
mappedUnder = getNormalizedMappedUnder(config);
context.log(RequestHandler.HANDLER_CLASS_KEY + " from context returned " + handlerClassName +
". Normalized mapped under returned '" + mappedUnder + "'");
requestHandler = (RequestHandler) context.getAttribute(handlerClassName);
faultHandlerFactory = faultInjectorFactoryClassName != null ?
(FaultInjectorFactory) context.getAttribute(faultInjectorFactoryClassName) :
new NoFaultInjectorFactory();
notifier = (Notifier) context.getAttribute(Notifier.KEY);
}
/**
* @param config Servlet configuration to read
* @return Normalized mappedUnder attribute without trailing slash
*/
private String getNormalizedMappedUnder(ServletConfig config) {
String mappedUnder = config.getInitParameter(MAPPED_UNDER_KEY);
if(mappedUnder == null) {
return null;
}
if (mappedUnder.endsWith("/")) {
mappedUnder = mappedUnder.substring(0, mappedUnder.length() - 1);
}
return mappedUnder;
}
private boolean getFileContextForwardingFlagFrom(ServletConfig config) {
String flagValue = config.getInitParameter(SHOULD_FORWARD_TO_FILES_CONTEXT);
return Boolean.valueOf(flagValue);
}
@Override
protected void service(HttpServletRequest httpServletRequest, HttpServletResponse httpServletResponse) throws ServletException, IOException {
LocalNotifier.set(notifier);
Request request = new WireMockHttpServletRequestAdapter(httpServletRequest, mappedUnder);
Response response = requestHandler.handle(request);
if (Thread.currentThread().isInterrupted()) {
return;
}
if (response.wasConfigured()) {
applyResponse(response, httpServletRequest, httpServletResponse);
} else if (request.getMethod().equals(GET) && shouldForwardToFilesContext) {
forwardToFilesContext(httpServletRequest, httpServletResponse, request);
} else {
httpServletResponse.sendError(HTTP_NOT_FOUND);
}
}
public void applyResponse(Response response, HttpServletRequest httpServletRequest, HttpServletResponse httpServletResponse) {
Fault fault = response.getFault();
if (fault != null) {
FaultInjector faultInjector = buildFaultInjector(httpServletRequest, httpServletResponse);
fault.apply(faultInjector);
httpServletResponse.addHeader(Fault.class.getName(), fault.name());
return;
}
if (response.getStatusMessage() == null) {
httpServletResponse.setStatus(response.getStatus());
} else {
httpServletResponse.setStatus(response.getStatus(), response.getStatusMessage());
}
for (HttpHeader header: response.getHeaders().all()) {
for (String value: header.values()) {
httpServletResponse.addHeader(header.key(), value);
}
}
writeAndTranslateExceptions(httpServletResponse, response.getBody());
}
private FaultInjector buildFaultInjector(HttpServletRequest httpServletRequest, HttpServletResponse httpServletResponse) {
return faultHandlerFactory.buildFaultInjector(httpServletRequest, httpServletResponse);
}
private static void writeAndTranslateExceptions(HttpServletResponse httpServletResponse, byte[] content) {
try {
ServletOutputStream out = httpServletResponse.getOutputStream();
out.write(content);
out.flush();
out.close();
} catch (IOException e) {
throwUnchecked(e);
}
}
private void forwardToFilesContext(HttpServletRequest httpServletRequest,
HttpServletResponse httpServletResponse, Request request) throws ServletException, IOException {
String forwardUrl = wiremockFileSourceRoot + WireMockApp.FILES_ROOT + request.getUrl();
RequestDispatcher dispatcher = httpServletRequest.getRequestDispatcher(decode(forwardUrl, UTF_8.name()));
dispatcher.forward(httpServletRequest, httpServletResponse);
}
}