Open Source Repository

Home /spring/spring-web-portlet-3.0.5 | Repository Home



org/springframework/web/portlet/mvc/PortletWrappingController.java
/*
 * Copyright 2002-2010 the original author or authors.
 *
 * 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 org.springframework.web.portlet.mvc;

import java.util.Collections;
import java.util.Enumeration;
import java.util.HashSet;
import java.util.LinkedHashMap;
import java.util.Locale;
import java.util.Map;
import java.util.ResourceBundle;
import javax.portlet.ActionRequest;
import javax.portlet.ActionResponse;
import javax.portlet.EventPortlet;
import javax.portlet.EventRequest;
import javax.portlet.EventResponse;
import javax.portlet.Portlet;
import javax.portlet.PortletConfig;
import javax.portlet.PortletContext;
import javax.portlet.PortletSession;
import javax.portlet.RenderRequest;
import javax.portlet.RenderResponse;
import javax.portlet.ResourceRequest;
import javax.portlet.ResourceResponse;
import javax.portlet.ResourceServingPortlet;
import javax.xml.XMLConstants;
import javax.xml.namespace.QName;

import org.springframework.beans.factory.BeanNameAware;
import org.springframework.beans.factory.DisposableBean;
import org.springframework.beans.factory.InitializingBean;
import org.springframework.web.portlet.ModelAndView;
import org.springframework.web.portlet.NoHandlerFoundException;
import org.springframework.web.portlet.context.PortletConfigAware;
import org.springframework.web.portlet.context.PortletContextAware;
import org.springframework.web.portlet.util.PortletUtils;

/**
 {@link Controller} implementation that wraps a portlet instance which it manages
 * internally. Such a wrapped portlet is not known outside of this controller;
 * its entire lifecycle is covered here.
 *
 <p>Useful to invoke an existing portlet via Spring's dispatching infrastructure,
 * for example to apply Spring
 {@link org.springframework.web.portlet.HandlerInterceptor HandlerInterceptors}
 * to its requests.
 *
 <p><b>Example:</b>
 *
 <pre>&lt;bean id="wrappingController" class="org.springframework.web.portlet.mvc.PortletWrappingController"&gt;
 *   &lt;property name="portletClass"&gt;
 *     &lt;value&gt;org.springframework.web.portlet.sample.HelloWorldPortlet&lt;/value&gt;
 *   &lt;/property&gt;
 *   &lt;property name="portletName"&gt;
 *     &lt;value&gt;hello-world&lt;/value&gt;
 *   &lt;/property&gt;
 *   &lt;property name="initParameters"&gt;
 *     &lt;props&gt;
 *       &lt;prop key="config"&gt;/WEB-INF/hello-world-portlet-config.xml&lt;/prop&gt;
 *     &lt;/props&gt;
 *   &lt;/property&gt;
 * &lt;/bean&gt;</pre>
 *
 @author Juergen Hoeller
 @author John A. Lewis
 @since 2.0
 */
public class PortletWrappingController extends AbstractController
    implements ResourceAwareController, EventAwareController,
    BeanNameAware, InitializingBean, DisposableBean, PortletContextAware, PortletConfigAware {

  private boolean useSharedPortletConfig = true;

  private PortletContext portletContext;

  private PortletConfig portletConfig;

  private Class portletClass;

  private String portletName;

  private Map<String, String> initParameters = new LinkedHashMap<String, String>();

  private String beanName;

  private Portlet portletInstance;


  /**
   * Set whether to use the shared PortletConfig object passed in
   * through <code>setPortletConfig</code>, if available.
   <p>Default is "true". Turn this setting to "false" to pass in
   * a mock PortletConfig object with the bean name as portlet name,
   * holding the current PortletContext.
   @see #setPortletConfig
   */
  public void setUseSharedPortletConfig(boolean useSharedPortletConfig) {
    this.useSharedPortletConfig = useSharedPortletConfig;
  }

  @Override
  public void setPortletContext(PortletContext portletContext) {
    this.portletContext = portletContext;
  }

  public void setPortletConfig(PortletConfig portletConfig) {
    this.portletConfig = portletConfig;
  }

  /**
   * Set the class of the Portlet to wrap.
   * Needs to implement <code>javax.portlet.Portlet</code>.
   @see javax.portlet.Portlet
   */
  public void setPortletClass(Class portletClass) {
    this.portletClass = portletClass;
  }

  /**
   * Set the name of the Portlet to wrap.
   * Default is the bean name of this controller.
   */
  public void setPortletName(String portletName) {
    this.portletName = portletName;
  }

  /**
   * Specify init parameters for the portlet to wrap,
   * as name-value pairs.
   */
  public void setInitParameters(Map<String, String> initParameters) {
    this.initParameters = initParameters;
  }

  public void setBeanName(String name) {
    this.beanName = name;
  }


  public void afterPropertiesSet() throws Exception {
    if (this.portletClass == null) {
      throw new IllegalArgumentException("portletClass is required");
    }
    if (!Portlet.class.isAssignableFrom(this.portletClass)) {
      throw new IllegalArgumentException("portletClass [" this.portletClass.getName() +
        "] needs to implement interface [javax.portlet.Portlet]");
    }
    if (this.portletName == null) {
      this.portletName = this.beanName;
    }
    PortletConfig config = this.portletConfig;
    if (config == null || !this.useSharedPortletConfig) {
      config = new DelegatingPortletConfig();
    }
    this.portletInstance = (Portletthis.portletClass.newInstance();
    this.portletInstance.init(config);
  }


  @Override
  protected void handleActionRequestInternal(
      ActionRequest request, ActionResponse responsethrows Exception {

    this.portletInstance.processAction(request, response);
  }

  @Override
  protected ModelAndView handleRenderRequestInternal(
      RenderRequest request, RenderResponse responsethrows Exception {

    this.portletInstance.render(request, response);
    return null;
  }

  public ModelAndView handleResourceRequest(
      ResourceRequest request, ResourceResponse responsethrows Exception {

    if (!(this.portletInstance instanceof ResourceServingPortlet)) {
      throw new NoHandlerFoundException("Cannot handle resource request - target portlet [" +
          this.portletInstance.getClass() " does not implement ResourceServingPortlet");
    }
    ResourceServingPortlet resourcePortlet = (ResourceServingPortletthis.portletInstance;

    // Delegate to PortletContentGenerator for checking and preparing.
    checkAndPrepare(request, response);

    // Execute in synchronized block if required.
    if (isSynchronizeOnSession()) {
      PortletSession session = request.getPortletSession(false);
      if (session != null) {
        Object mutex = PortletUtils.getSessionMutex(session);
        synchronized (mutex) {
          resourcePortlet.serveResource(request, response);
          return null;
        }
      }
    }

    resourcePortlet.serveResource(request, response);
    return null;
  }

  public void handleEventRequest(
      EventRequest request, EventResponse responsethrows Exception {

    if (!(this.portletInstance instanceof EventPortlet)) {
      logger.debug("Ignoring event request for non-event target portlet: " this.portletInstance.getClass());
      return;
    }
    EventPortlet eventPortlet = (EventPortletthis.portletInstance;

    // Delegate to PortletContentGenerator for checking and preparing.
    check(request, response);

    // Execute in synchronized block if required.
    if (isSynchronizeOnSession()) {
      PortletSession session = request.getPortletSession(false);
      if (session != null) {
        Object mutex = PortletUtils.getSessionMutex(session);
        synchronized (mutex) {
          eventPortlet.processEvent(request, response);
          return;
        }
      }
    }

    eventPortlet.processEvent(request, response);
  }


  public void destroy() {
    this.portletInstance.destroy();
  }


  /**
   * Internal implementation of the PortletConfig interface, to be passed
   * to the wrapped portlet.
   <p>Delegates to {@link PortletWrappingController} fields
   * and methods to provide init parameters and other environment info.
   */
  private class DelegatingPortletConfig implements PortletConfig {

    public String getPortletName() {
      return portletName;
    }

    public PortletContext getPortletContext() {
      return portletContext;
    }

    public String getInitParameter(String paramName) {
      return initParameters.get(paramName);
    }

    public Enumeration<String> getInitParameterNames() {
      return Collections.enumeration(initParameters.keySet());
    }

    public ResourceBundle getResourceBundle(Locale locale) {
      return (portletConfig != null ? portletConfig.getResourceBundle(localenull);
    }

    public Enumeration<String> getPublicRenderParameterNames() {
      return Collections.enumeration(new HashSet<String>());
    }

    public String getDefaultNamespace() {
      return XMLConstants.NULL_NS_URI;
    }

    public Enumeration<QName> getPublishingEventQNames() {
      return Collections.enumeration(new HashSet<QName>());
    }

    public Enumeration<QName> getProcessingEventQNames() {
      return Collections.enumeration(new HashSet<QName>());
    }

    public Enumeration<Locale> getSupportedLocales() {
      return Collections.enumeration(new HashSet<Locale>());
    }

    public Map<String, String[]> getContainerRuntimeOptions() {
      return (portletConfig != null ? portletConfig.getContainerRuntimeOptions() null);
    }
  }

}