Package org.apache.cocoon.acting

Source Code of org.apache.cocoon.acting.AbstractControllerAction

/*

============================================================================
                   The Apache Software License, Version 1.1
============================================================================

Copyright (C) 1999-2003 The Apache Software Foundation. All rights reserved.

Redistribution and use in source and binary forms, with or without modifica-
tion, are permitted provided that the following conditions are met:

1. Redistributions of  source code must  retain the above copyright  notice,
    this list of conditions and the following disclaimer.

2. Redistributions in binary form must reproduce the above copyright notice,
    this list of conditions and the following disclaimer in the documentation
    and/or other materials provided with the distribution.

3. The end-user documentation included with the redistribution, if any, must
    include  the following  acknowledgment:  "This product includes  software
    developed  by the  Apache Software Foundation  (http://www.apache.org/)."
    Alternately, this  acknowledgment may  appear in the software itself,  if
    and wherever such third-party acknowledgments normally appear.

4. The names "Apache Cocoon" and  "Apache Software Foundation" must  not  be
    used to  endorse or promote  products derived from  this software without
    prior written permission. For written permission, please contact
    apache@apache.org.

5. Products  derived from this software may not  be called "Apache", nor may
    "Apache" appear  in their name,  without prior written permission  of the
    Apache Software Foundation.

THIS SOFTWARE IS PROVIDED ``AS IS'' AND ANY EXPRESSED OR IMPLIED WARRANTIES,
INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND
FITNESS  FOR A PARTICULAR  PURPOSE ARE  DISCLAIMED.  IN NO  EVENT SHALL  THE
APACHE SOFTWARE  FOUNDATION  OR ITS CONTRIBUTORS  BE LIABLE FOR  ANY DIRECT,
INDIRECT, INCIDENTAL, SPECIAL,  EXEMPLARY, OR CONSEQUENTIAL  DAMAGES (INCLU-
DING, BUT NOT LIMITED TO, PROCUREMENT  OF SUBSTITUTE GOODS OR SERVICES; LOSS
OF USE, DATA, OR  PROFITS; OR BUSINESS  INTERRUPTION)  HOWEVER CAUSED AND ON
ANY  THEORY OF LIABILITY,  WHETHER  IN CONTRACT,  STRICT LIABILITY,  OR TORT
(INCLUDING  NEGLIGENCE OR  OTHERWISE) ARISING IN  ANY WAY OUT OF THE  USE OF
THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.

This software  consists of voluntary contributions made  by many individuals
on  behalf of the Apache Software  Foundation and was  originally created by
Stefano Mazzocchi  <stefano@apache.org>. For more  information on the Apache
Software Foundation, please see <http://www.apache.org/>.

*/
package org.apache.cocoon.acting;

import org.apache.avalon.framework.CascadingRuntimeException;
import org.apache.avalon.framework.configuration.Configuration;
import org.apache.avalon.framework.configuration.ConfigurationException;
import org.apache.avalon.framework.parameters.Parameters;
import org.apache.avalon.framework.thread.ThreadSafe;
import org.apache.cocoon.Constants;
import org.apache.cocoon.components.validation.Validator;
import org.apache.cocoon.components.xmlform.Form;
import org.apache.cocoon.environment.ObjectModelHelper;
import org.apache.cocoon.environment.Redirector;
import org.apache.cocoon.environment.Request;
import org.apache.cocoon.environment.SourceResolver;

import java.lang.reflect.Method;
import java.util.Enumeration;
import java.util.HashMap;
import java.util.Map;

/**
* This action extends the idea of the AbstractXMLFormAction. It
* uses a configuration file similar to Jakarta Struts, and uses
* the concept of the MultiAction.
*
* Warning! This classes is an experimental one.
*
* @author <a href="mailto:stephan@apache.org">Stephan Michels</a>
* @version CVS $Id: AbstractControllerAction.java,v 1.2 2003/05/15 10:35:14 bruno Exp $
*/
public abstract class AbstractControllerAction
  extends AbstractComplementaryConfigurableAction implements ThreadSafe {

    private static final String ACTION_METHOD_PREFIX = "do";
    private static final String ACTION_METHOD_PARAMETER = "method";

    private HashMap methodIndex;

    private static final String removePrefix(String name) {
        int prefixLen = ACTION_METHOD_PREFIX.length();

        return name.substring(prefixLen, prefixLen+1).toLowerCase()+
               name.substring(prefixLen+1);
    }

    /**
     * Configure the action.
     */
    public void configure(Configuration conf) throws ConfigurationException {
        super.configure(conf);

        try {
            Method[] methods = this.getClass().getMethods();

            methodIndex = new HashMap();

            for (int i = 0; i<methods.length; i++) {
                String methodName = methods[i].getName();

                if (methodName.startsWith(ACTION_METHOD_PREFIX)) {
                    String actionName = removePrefix(methodName);

                    methodIndex.put(actionName, methods[i]);
                    if (getLogger().isDebugEnabled()) {
                        getLogger().debug("registered method \""+methodName+
                                          "\" as action \""+actionName+"\"");
                    }
                }
            }
        } catch (Exception e) {
            throw new ConfigurationException("cannot get methods by reflection",
                                             e);
        }
    }

    /**
     * Perform the action.
     */
    public Map act(Redirector redirector, SourceResolver resolver,
                   Map objectModel, String src,
                   Parameters params) throws Exception {

        Request request = (Request) (objectModel.get(ObjectModelHelper.REQUEST_OBJECT));

        // load controller configuration
        Configuration config = getConfiguration(src, resolver, true);

        // current path
        String path = request.getSitemapURI();

        String base = request.getRequestURI().substring(0,
                          request.getRequestURI().length()-path.length());

        // ensure that there is a form available
        // through the rest of the flow
        Form form = getForm(config, path, objectModel);

        if (form==null) {
            throw new IllegalStateException("Action could not obtain the Form");
        }

        // populate form with request parameters
        // population is automatically followed by validation by default.
        // If this is not the desired behaviour, the Form class can be subclassed
        form.populate(objectModel);

        // If the currect form have incorrect values, then process the same page
        if (form.getViolations()!=null) {
            return EMPTY_MAP;
        }

        // find and save the action command
        String command = getCommand(request);

        if ((command==null) || (command.length()==0)) {
            // if no command send, process current page
            return EMPTY_MAP;
        }

        // find the action name for the method, which should be performed.
        String action = getAction(config, path);

        // perform action method
        String forward = command;

        if ((action!=null) && (action.length()>0)) {
            Method method = (Method) methodIndex.get(action);

            if (method!=null) {
                forward = ((String) method.invoke(this, new Object[]{ command,
                                                                      form }));
                if (forward==null) {
                    forward = command;
                }
            }
        }
        if ((command==null) || (command.length()==0)) {
            // if no command send, process current page
            return EMPTY_MAP;
        }

        // if action returns violation, process current form
        if ((forward==null) && (form.getViolations()!=null)) {
            return EMPTY_MAP;
        }

        // process forward
        String newpath = getForward(config, path, forward);

        if (path.equals(newpath)) {
            return EMPTY_MAP;
        } else {
            redirector.redirect(true, base+newpath);
            return null;
        }
    }

    /**
     * Returns the entry for a given page.
     */
    private int getActionEntry(Configuration config, String path) {

        Configuration[] actions = config.getChild("action-mappings").getChildren("action");

        int entry = -1;

        for (int i = 0; i<actions.length; i++)
            if ((entry<0) &&
                (actions[i].getAttribute("path", null).equals(path))) {
                entry = i;
            }

        if (entry<0) {
            throw new RuntimeException("Could not find entry for '"+path+"'");
        }

        return entry;
    }

    /**
     * Returns name of the action for a given page.
     */
    private String getAction(Configuration config, String path) {

        Configuration[] actions = config.getChild("action-mappings").getChildren("action");

        String action = null;

        for (int i = 0; i<actions.length; i++)
            if (actions[i].getAttribute("path", null).equals(path)) {
                action = actions[i].getAttribute("name", null);
            }

        return (action!=null) ? action.toLowerCase() : null;
    }

    /**
     * Get the command which was submitted with the form.
     * It is extracted from the standard cocoon-action-* request parameter
     */
    private String getCommand(Request request) {
        Enumeration enum = request.getParameterNames();

        while (enum.hasMoreElements()) {
            String paramName = (String) enum.nextElement();

            // search for the command
            if (paramName.startsWith(Constants.ACTION_PARAM_PREFIX)) {
                return paramName.substring(Constants.ACTION_PARAM_PREFIX.length(),
                                           paramName.length());
            }
        }
        return null;
    }

    /**
     * Return the page for a given forward.
     */
    private String getForward(Configuration config, String path,
                              String name) {

        Configuration[] actions = config.getChild("action-mappings").getChildren("action");

        for (int i = 0; i<actions.length; i++)
            if (actions[i].getAttribute("path", null).equals(path)) {
                Configuration[] forwards = actions[i].getChildren("forward");

                for (int j = 0; j<forwards.length; j++)
                    if (forwards[j].getAttribute("name", null).equals(name)) {
                        return forwards[j].getAttribute("path", null);
                    }
            }

        Configuration[] forwards = config.getChild("global-forwards").getChildren("forward");

        for (int i = 0; i<forwards.length; i++)
            if (forwards[i].getAttribute("name", null).equals(name)) {
                return forwards[i].getAttribute("path", null);
            }

        return null;
    }

    /**
     * Invoked during the form population process
     *
     * Provides default implementation, which
     * can be extended or replaced by subclasses
     *
     * Implementations of this method are responsible
     * for creating and
     * returning the Form object which the action
     * is working on.
     *
     * @return Form the form object this action works with
     */
    private Form getForm(Configuration config, String path, Map objectModel) {
        Form form = Form.lookup(objectModel, getFormId(config, path));

        if (form!=null) {
            return form;
        } else {
            // create new form
            form = new Form(getFormId(config, path),
                            getFormModel(config, path));
            Validator v = null; // getFormValidator();

            form.setValidator(v);
            form.save(objectModel, getFormScope(config, path));
            return form;
        }
    }

    /**
     * Extract xmlform-model action parameter and
     * instantiate a new form model it.
     *
     * Subclasses may override this method
     * to use custom model instantiation technique
     *
     * @return Form model
     */
    private Object getFormModel(Configuration config, String path) {

        String formId = getFormId(config, path);

        Configuration[] beans = config.getChild("form-beans").getChildren("form-bean");

        String modelClassName = null;

        for (int i = 0; i<beans.length; i++)
            if (beans[i].getAttribute("name", null).equals(formId)) {
                modelClassName = beans[i].getAttribute("type", null);
            }

        try {
            Class modelClass = Class.forName(modelClassName);
            Object o = modelClass.newInstance();

            return o;
        } catch (Exception e) {
            throw new CascadingRuntimeException(" Failed instantiating form model ",
                                                e);
        }
    }

    /**
     * Returns the form id for a given page.
     */
    private String getFormId(Configuration config, String path) {
        Configuration[] actions = config.getChild("action-mappings").getChildren("action");

        String formId = null;

        for (int i = 0; i<actions.length; i++)
            if (actions[i].getAttribute("path", null).equals(path)) {
                formId = actions[i].getAttribute("form", null);
            }

        if (formId==null) {
            throw new RuntimeException(" xmlform-id not specified ");
        } else {
            return formId;
        }
    }

    /**
     * Returns the scope of the form for a given page.
     */
    private String getFormScope(Configuration config, String path) {

        Configuration[] actions = config.getChild("action-mappings").getChildren("action");

        String formScope = null;

        for (int i = 0; i<actions.length; i++)
            if (actions[i].getAttribute("path", null).equals(path)) {
                formScope = actions[i].getAttribute("scope", null);
            }

        if (formScope==null) {
            // default to request scope
            formScope = Form.SCOPE_REQUEST;
        }

        return formScope;
    }
}
TOP

Related Classes of org.apache.cocoon.acting.AbstractControllerAction

TOP
Copyright © 2018 www.massapi.com. All rights reserved.
All source code are property of their respective owners. Java is a trademark of Sun Microsystems, Inc and owned by ORACLE Inc. Contact coftware#gmail.com.