com.wavemaker.runtime.prefab.web.PrefabControllerServlet Maven / Gradle / Ivy
/*******************************************************************************
* Copyright (C) 2022-2023 WaveMaker, Inc.
*
* 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.wavemaker.runtime.prefab.web;
import java.util.Map;
import jakarta.servlet.ServletConfig;
import jakarta.servlet.ServletException;
import jakarta.servlet.http.HttpServletRequest;
import jakarta.servlet.http.HttpServletResponse;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.context.ApplicationContext;
import org.springframework.stereotype.Controller;
import org.springframework.web.context.support.SpringBeanAutowiringSupport;
import org.springframework.web.servlet.DispatcherServlet;
import org.springframework.web.servlet.HandlerAdapter;
import org.springframework.web.servlet.HandlerExceptionResolver;
import org.springframework.web.servlet.HandlerExecutionChain;
import org.springframework.web.servlet.HandlerMapping;
import org.springframework.web.servlet.ModelAndView;
import org.springframework.web.util.UrlPathHelper;
import org.springframework.web.util.WebUtils;
import com.wavemaker.commons.MessageResource;
import com.wavemaker.commons.WMRuntimeException;
import com.wavemaker.runtime.prefab.context.PrefabThreadLocalContextManager;
import com.wavemaker.runtime.prefab.core.Prefab;
import com.wavemaker.runtime.prefab.core.PrefabManager;
import com.wavemaker.runtime.prefab.core.PrefabRegistry;
import com.wavemaker.runtime.prefab.util.PrefabConstants;
/**
* Front controller to handle service requests directed at Spring prefabs. This servlet
* receives service requests for prefabs, matches it with one of the registered contexts
* and forwards it to the appropriate {@link Controller}.
*
* To integrate prefab library with a web application include the following snippet in
* web.xml.
*
*
*
* <servlet>
*
* <servlet-name>prefabs</servlet-name>
*
* <servlet-class>com.wavemaker.runtime.prefab.web.PrefabControllerServlet</servlet-class>
*
* <init-param>
*
* <param-name>contextClass</param-name>
*
* <param-value>org.springframework.web.context.support.AnnotationConfigWebApplicationContext</param-value>
*
* </init-param>
*
* <init-param>
*
* <param-name>contextConfigLocation</param-name>
*
* <param-value>com.wavemaker.runtime.prefab.PrefabServletConfig</param-value>
*
* </init-param>
*
* <load-on-startup>1</load-on-startup>
*
* </servlet>
*
* <servlet-mapping>
*
* <servlet-name>prefabs</servlet-name>
*
* <url-pattern>/prefabs/*</url-pattern>
* </servlet-mapping>
*
*
* @author Dilip Kumar
*/
@SuppressWarnings("serial")
public class PrefabControllerServlet extends DispatcherServlet {
private final UrlPathHelper urlPathHelper = new UrlPathHelper();
@Autowired
private PrefabThreadLocalContextManager prefabThreadLocalContextManager;
@Autowired
private PrefabRegistry prefabRegistry;
@Autowired
private PrefabManager prefabManager;
private static final Logger logger = LoggerFactory.getLogger(PrefabControllerServlet.class);
@Override
public void init(ServletConfig config) throws ServletException {
super.init(config);
SpringBeanAutowiringSupport.processInjectionBasedOnServletContext(this,
config.getServletContext());
}
/**
* Creates a new PrefabControllerServlet
.
*/
public PrefabControllerServlet() {
setDetectAllHandlerMappings(false);
setDetectAllHandlerAdapters(false);
setDetectAllHandlerExceptionResolvers(false);
}
@Override
protected void doDispatch(final HttpServletRequest request, final HttpServletResponse response) throws Exception {
ApplicationContext context = null;
ClassLoader previous = null;
try {
context = lookupContext(request);
prefabThreadLocalContextManager.setContext(context);
previous = Thread.currentThread().getContextClassLoader();
Thread.currentThread().setContextClassLoader(context.getClassLoader());
HttpServletRequest updatedRequest = new PrefabAwareHttpRequestWrapper(request, context.getId());
super.doDispatch(updatedRequest, response);
} finally {
if (context != null) {
Thread.currentThread().setContextClassLoader(previous);
}
prefabThreadLocalContextManager.clearContext();
}
}
/**
* Returns the qualified servlet path with prefab name.
*/
private String getServletPathWithPrefabName(String servletPath, String prefabName) {
servletPath = (servletPath.endsWith("/") ? servletPath.substring(0, (servletPath.length() - 1)) : servletPath);
prefabName = (prefabName.startsWith("/") ? servletPath.substring(1) : prefabName);
return servletPath + "/" + prefabName;
}
@Override
protected HandlerExecutionChain getHandler(final HttpServletRequest request) throws Exception {
ApplicationContext context = lookupContext(request);
Map handlerMappingMap = context.getBeansOfType(HandlerMapping.class);
for (HandlerMapping hm : handlerMappingMap.values()) {
if (logger.isTraceEnabled()) {
logger.trace("Testing handler map [{}] in DispatcherServlet with name '{}'", hm, getServletName());
}
HandlerExecutionChain handler = hm.getHandler(request);
if (handler != null) {
return handler;
}
}
return null;
}
/**
* Returns the prefab name from the lookupPath i.e uri path.
*
* @return prefab name or null, if no prefab name specified.
*/
private String extractPrefabName(final String lookupPath) {
if (lookupPath != null && !(lookupPath.isEmpty() || lookupPath.equals("/"))) {
int startIndex = (lookupPath.startsWith("/") ? 1 : 0);
int endIndex = lookupPath.indexOf('/', startIndex);
if (endIndex != -1) {
return lookupPath.substring(startIndex, lookupPath.indexOf('/', startIndex));
} else {
return lookupPath.substring(startIndex);
}
}
return null;
}
/**
* Looks up applicable prefab context for the given request.
*
* @param request HTTP request object
*
* @return prefab context or null, if no context is registered for the URL path
*/
private ApplicationContext lookupContext(final HttpServletRequest request) {
// checking if any already initialized prefab context is there.
ApplicationContext prefabContext = (ApplicationContext) request
.getAttribute(PrefabConstants.REQUEST_PREFAB_CONTEXT);
if (prefabContext == null) {
String urlPath = urlPathHelper.getLookupPathForRequest(request);
String prefabName = extractPrefabName(urlPath);
if (prefabName == null) {
throw new WMRuntimeException(MessageResource.create("com.wavemaker.runtime.invalid.url.for.accessing.prefab"), urlPath);
}
Prefab prefab = prefabManager.getPrefab(prefabName);
if (prefab == null) {
throw new WMRuntimeException(MessageResource.create("com.wavemaker.runtime.prefab.not.found"), prefabName);
}
prefabContext = prefabRegistry.getPrefabContext(prefab.getName());
request.setAttribute(PrefabConstants.REQUEST_PREFAB_CONTEXT, prefabContext);
}
return prefabContext;
}
/**
* Returns the {@link HandlerAdapter} from the prefab context.
*/
@Override
protected HandlerAdapter getHandlerAdapter(final Object handler) throws ServletException {
Map handlerAdapterMap = prefabThreadLocalContextManager.getContext()
.getBeansOfType(HandlerAdapter.class);
for (HandlerAdapter ha : handlerAdapterMap.values()) {
if (logger.isTraceEnabled()) {
logger.trace("Testing handler adapter [{}]", ha);
}
if (ha.supports(handler)) {
return ha;
}
}
throw new ServletException("No adapter for handler [" + handler +
"]: Does your handler implement a supported interface like Controller?");
}
@Override
protected ModelAndView processHandlerException(
final HttpServletRequest request, final HttpServletResponse response,
final Object handler, final Exception ex) throws Exception {
ApplicationContext context = lookupContext(request);
ModelAndView exMv = null;
Map handlerExceptionResolversMap = context
.getBeansOfType(HandlerExceptionResolver.class);
for (HandlerExceptionResolver handlerExceptionResolver : handlerExceptionResolversMap.values()) {
exMv = handlerExceptionResolver.resolveException(request, response, handler, ex);
if (exMv != null) {
break;
}
}
if (exMv != null) {
if (exMv.isEmpty()) {
return null;
}
if (!exMv.hasView()) {
exMv.setViewName(getDefaultViewName(request));
}
WebUtils.exposeErrorRequestAttributes(request, ex, getServletName());
return exMv;
}
throw ex;
}
@Override
protected void doService(HttpServletRequest request, HttpServletResponse response) throws Exception {
ApplicationContext applicationContext = lookupContext(request);
// setting prefab name in request object
request.setAttribute(WebUtils.INCLUDE_SERVLET_PATH_ATTRIBUTE,
getServletPathWithPrefabName(request.getServletPath(), applicationContext.getId()));
super.doService(request, response);
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy