Jasig/WebproxyPortlet

View on GitHub
src/main/java/org/jasig/portlet/proxy/mvc/portlet/gateway/GatewayPortletController.java

Summary

Maintainability
A
3 hrs
Test Coverage
/**
 * Licensed to Apereo under one or more contributor license
 * agreements. See the NOTICE file distributed with this work
 * for additional information regarding copyright ownership.
 * Apereo licenses this file to you 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 the following location:
 *
 *   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.jasig.portlet.proxy.mvc.portlet.gateway;

import org.apache.commons.lang.StringUtils;
import org.jasig.portlet.proxy.mvc.IViewSelector;
import org.jasig.portlet.proxy.service.IFormField;
import org.jasig.portlet.proxy.service.web.HttpContentRequestImpl;
import org.jasig.portlet.proxy.service.web.IAuthenticationFormModifier;
import org.jasig.portlet.proxy.service.web.interceptor.IPreInterceptor;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.beans.InvalidPropertyException;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Controller;
import org.springframework.ui.Model;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RequestParam;
import org.springframework.web.portlet.ModelAndView;
import org.springframework.web.portlet.bind.annotation.RenderMapping;
import org.springframework.web.portlet.bind.annotation.ResourceMapping;

import javax.annotation.PostConstruct;
import javax.annotation.Resource;
import javax.portlet.RenderRequest;
import javax.portlet.ResourceRequest;
import javax.portlet.ResourceResponse;
import java.io.IOException;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.HashSet;
import java.util.List;
import java.util.Map;

/**
 * <p>GatewayPortletController class.</p>
 *
 * @author bjagg
 * @version $Id: $Id
 */
@Controller
@RequestMapping("VIEW")
public class GatewayPortletController extends BaseGatewayPortletController {
    private static final String HTTPS = "HTTPS";

    protected final Logger logger = LoggerFactory.getLogger(this.getClass());
    
    @Resource(name="gatewayEntries")
    private List<GatewayEntry> gatewayEntries;

    @Autowired(required=false)
    private String viewName = "gateway";
    
    @Autowired(required=false)
    private String mobileViewName = "mobileGateway";
    
    @Autowired(required=true)
    private IViewSelector viewSelector;

    @PostConstruct
    private void validateGatewayEntries() {
        HashSet<GatewayEntry> set = new HashSet<GatewayEntry>();
        for (GatewayEntry entry : gatewayEntries) {
            if (!set.add(entry)) {
                throw new InvalidPropertyException(GatewayEntry.class, "name",
                        "Error initializing Gateway Entries, multiple entries with name " + entry.getName());
            }
        }
    }

    /**
     * <p>getView.</p>
     *
     * @param request a {@link javax.portlet.RenderRequest} object
     * @return a {@link org.springframework.web.portlet.ModelAndView} object
     */
    @RenderMapping
    public ModelAndView getView(RenderRequest request){
        final ModelAndView mv = new ModelAndView();
        final List<GatewayEntry> entries =  removeInaccessibleEntries(gatewayEntries, request);
        final Map<String, Boolean> validations = new HashMap<String, Boolean>();
        for (GatewayEntry entry : entries) {
            for (Map.Entry<HttpContentRequestImpl, List<IPreInterceptor>> requestEntry : entry.getContentRequests().entrySet()){
    
                // run each content request through any configured preinterceptors to validate each entry
                final HttpContentRequestImpl contentRequest = requestEntry.getKey();
                for (IPreInterceptor interceptor : requestEntry.getValue()) {
                    boolean isValid = interceptor.validate(contentRequest, request);
                    validations.put(entry.getName(), isValid);
                }
            }
        }

        mv.addObject("entries", entries);
        mv.addObject("validations", validations);

        String openInNewPage = request.getPreferences().getValue("openInNewPage", "true");
        mv.addObject("openInNewPage", openInNewPage);
        
        final String view = viewSelector.isMobile(request) ? mobileViewName : viewName;
        mv.setView(view);
        return mv;
    }

    /**
     * <p>showTarget.</p>
     *
     * @param portletRequest a {@link javax.portlet.ResourceRequest} object
     * @param portletResponse a {@link javax.portlet.ResourceResponse} object
     * @param model a {@link org.springframework.ui.Model} object
     * @param beanName a {@link java.lang.String} object
     * @return a {@link java.lang.String} object
     * @throws java.io.IOException if any.
     */
    @ResourceMapping()
    public String showTarget(ResourceRequest portletRequest, ResourceResponse portletResponse, Model model, @RequestParam("index") String beanName) throws IOException {
        prepareGatewayResponse(portletRequest, portletResponse, beanName, model);
        return "json";
    }

    /**
     * <p>showTargetInNewWindow.</p>
     *
     * @param portletRequest a {@link javax.portlet.ResourceRequest} object
     * @param portletResponse a {@link javax.portlet.ResourceResponse} object
     * @param model a {@link org.springframework.ui.Model} object
     * @param beanName a {@link java.lang.String} object
     * @return a {@link java.lang.String} object
     * @throws java.io.IOException if any.
     */
    @ResourceMapping(value = "showTargetInNewWindow")
    public String showTargetInNewWindow(ResourceRequest portletRequest, ResourceResponse portletResponse, Model model,
                                        @RequestParam("index") String beanName) throws IOException {
        model.addAttribute("index", beanName);
        return "gatewayNewPage";
    }

    // Removed because there were concerns that browsers might be more likely to cache html pages vs. Ajax requests
    // even though response requested no caching of pages.
    // For optimal mobile performance, can switch mobileGateway.jsp and possibly gateway.jsp to use this page instead
    // of ajax-requesting page.
//    @ResourceMapping(value = "showTargetInNewWindowNoAjax")
//    public String showTargetInNewWindowNoAjax(ResourceRequest portletRequest, ResourceResponse portletResponse, Model model,
//                                              @RequestParam("index") int index) throws IOException {
//        prepareGatewayResponse(portletRequest, portletResponse, index, model);
//        return "gatewayNewPageNoAjax";
//    }

    private void prepareGatewayResponse(ResourceRequest portletRequest, ResourceResponse portletResponse,
                                        String beanName, Model model) throws IOException {
        // get the requested gateway link entry from the list configured for
        // this portlet
        final List<GatewayEntry> entries =  gatewayEntries;
        final GatewayEntry entry = getAccessibleEntry(entries, portletRequest, beanName);
        if (entry == null) {
            return;
        }

        // build a list of content requests
        final List<HttpContentRequestImpl> contentRequests = new ArrayList<HttpContentRequestImpl>();
        for (Map.Entry<HttpContentRequestImpl, List<IPreInterceptor>> requestEntry : entry.getContentRequests().entrySet()){

            // run each content request through any configured preinterceptors
            // before adding it to the list.  Use a clone so that preinterceptors can change the
            // values without impacting future executions (e.g. need to retain substitution tokens).
            final HttpContentRequestImpl contentRequest = requestEntry.getKey().duplicate();
            for (IPreInterceptor interceptor : requestEntry.getValue()) {
                interceptor.intercept(contentRequest, portletRequest);
            }
            contentRequests.add(contentRequest);
        }

        // add custom form field processing logic to the ModelAndView
        for (IAuthenticationFormModifier authenticationFormModifier :
                (List<IAuthenticationFormModifier>) entry.getAuthenticationFormModifier()) {
            for (HttpContentRequestImpl contentRequest : contentRequests) {
                authenticationFormModifier.modifyHttpContentRequest(contentRequest, portletRequest.getPreferences());
            }
        }

        // Insure the proxiedLocation value is secure (HTTPS) if required.
        for (HttpContentRequestImpl contentRequest : contentRequests) {
            if (entry.isRequireSecure() && StringUtils.isNotBlank(contentRequest.getProxiedLocation())
                    && contentRequest.getProxiedLocation().length() >= HTTPS.length()) {
                if (!HTTPS.equalsIgnoreCase(contentRequest.getProxiedLocation().substring(0, HTTPS.length()))) {
                    logger.error("Proxied location '" + contentRequest.getProxiedLocation() + "' for gateway entry "
                            + entry.getName() + " is not secure - discarding entry!!!");
                    contentRequest.setParameters(new HashMap<String, IFormField>());
                    contentRequest.setProxiedLocation("/HTTPSUrlRequiredButNotSpecified");   // Force a failure that's clear
                }
            }
        }

        model.addAttribute("contentRequests", contentRequests);
        model.addAttribute("javascriptFile", entry.getJavascriptFile());

        // we don't want this response to be cached by the browser since it may
        // include one-time-only authentication tokens
        // See http://stackoverflow.com/questions/49547/making-sure-a-web-page-is-not-cached-across-all-browsers
        portletResponse.setProperty("Cache-Control", "no-cache, no-store, must-revalidate"); // HTTP 1.1.
        portletResponse.setProperty("Pragma", "no-cache"); // HTTP 1.0.
        portletResponse.setProperty("Expires", "0"); // Proxies.
    }

}