All Downloads are FREE. Search and download functionalities are using the official Maven repository.

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