org.apache.jetspeed.modules.actions.portlets.PsmlManagerAction.java Source code

Java tutorial

Introduction

Here is the source code for org.apache.jetspeed.modules.actions.portlets.PsmlManagerAction.java

Source

/*
 * Copyright 2000-2004 The Apache Software Foundation.
 * 
 * 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.apache.jetspeed.modules.actions.portlets;

// Jetspeed
import org.apache.commons.lang.SerializationUtils;
import org.apache.jetspeed.om.profile.Profile;
import org.apache.jetspeed.om.profile.QueryLocator;
import org.apache.jetspeed.portal.Portlet;
import org.apache.jetspeed.portal.portlets.browser.DatabaseBrowserIterator;
import org.apache.jetspeed.services.Profiler;
import org.apache.jetspeed.util.PortletConfigState;
import org.apache.jetspeed.util.PortletUtils;
import org.apache.jetspeed.util.PortletSessionState;
import org.apache.jetspeed.services.security.PortalResource;
import org.apache.jetspeed.om.security.JetspeedUser;
import org.apache.jetspeed.services.JetspeedSecurity;
import org.apache.jetspeed.modules.actions.portlets.security.SecurityConstants;
import org.apache.jetspeed.om.profile.ProfileLocator;
import org.apache.jetspeed.om.profile.PSMLDocument;
import org.apache.jetspeed.om.profile.Portlets;
import org.apache.jetspeed.om.registry.Parameter;
import org.apache.jetspeed.om.registry.PortletEntry;
import org.apache.jetspeed.services.Registry;
import org.apache.jetspeed.services.PsmlManager;
import org.apache.jetspeed.services.logging.JetspeedLogFactoryService;
import org.apache.jetspeed.services.logging.JetspeedLogger;
import org.apache.jetspeed.util.template.JetspeedLink;
import org.apache.jetspeed.util.template.JetspeedLinkFactory;
import org.apache.jetspeed.services.resources.JetspeedResources;
import org.apache.jetspeed.services.psmlmanager.PsmlManagerService;
import org.apache.jetspeed.om.profile.BasePSMLDocument;
import org.apache.jetspeed.om.security.Role;
import org.apache.jetspeed.om.security.Group;
import org.apache.jetspeed.services.rundata.JetspeedRunData;

// Turbine stuff
import org.apache.turbine.util.RunData;
import org.apache.turbine.util.DynamicURI;
import org.apache.turbine.util.StringUtils;
import org.apache.turbine.util.security.EntityExistsException;
import org.apache.turbine.services.TurbineServices;
import org.apache.turbine.services.servlet.TurbineServlet;
import org.apache.turbine.services.resources.ResourceService;

// Velocity Stuff
import org.apache.velocity.context.Context;

// Java
import java.util.ArrayList;
import java.util.Hashtable;
import java.util.Iterator;
import java.io.File;
import java.io.FileWriter;
import java.io.FileReader;
import java.io.IOException;
import java.util.Vector;
import java.util.StringTokenizer;

// castor support
import org.exolab.castor.xml.MarshalException;
import org.exolab.castor.xml.Marshaller;
import org.exolab.castor.xml.Unmarshaller;
import org.exolab.castor.xml.ValidationException;
import org.exolab.castor.mapping.Mapping;
import org.exolab.castor.mapping.MappingException;
import org.xml.sax.InputSource;

// serialization support
import org.apache.xml.serialize.Serializer;
import org.apache.xml.serialize.XMLSerializer;
import org.apache.xml.serialize.OutputFormat;

/**
 * This action enables to manage psml entries
 * within current user's security context
 *
 * @author <a href="mailto:morciuch@apache.org">Mark Orciuch</a>
 * @version $Id: PsmlManagerAction.java,v 1.7 2004/02/23 02:56:58 jford Exp $
 */
public class PsmlManagerAction extends GenericMVCAction {
    public static final String CACHED_PSML = "PsmlManagerAction.cached.psml";
    protected static final String PSML_REFRESH_FLAG = "psmlRefreshFlag";
    protected static final String TRUE = "true";
    protected static final String FALSE = "false";
    protected static final String PROFILE_ITERATOR = "profileIterator";
    protected static final String LAST_SEARCH_TYPE = "lastSearchType";
    protected static final String LAST_SEARCH_VALUE = "lastSearchValue";
    protected static final String PAGE_SIZE = "page-size";
    protected static final String DEFAULT_SEARCH = "default-search";
    protected static final String CUSTOMIZE_TEMPLATE = "customize-template";
    private static final String PEID = "js_peid";

    /** name of the parameter that holds the filter value */
    public static final String FILTER_VALUE = "filter_value";

    /** name of the parameter that holds the regexp flag */
    public static final String FILTER_REGEXP = "filter_regexp";

    /** name of the parameter that holds the filter type */
    public static final String FILTER_TYPE = "filter_type";

    /** value of the filter type parameter for searching by username */
    public static final String FILTER_TYPE_USER = "filter_type_user";

    /** value of the filter type parameter for searching by role */
    public static final String FILTER_TYPE_ROLE = "filter_type_role";

    /** value of the filter type parameter for searching by group */
    public static final String FILTER_TYPE_GROUP = "filter_type_group";

    private static Hashtable queryModes = new Hashtable();
    static {
        queryModes.put("All", String.valueOf(QueryLocator.QUERY_ALL));
        queryModes.put("User", String.valueOf(QueryLocator.QUERY_USER));
        queryModes.put("Role", String.valueOf(QueryLocator.QUERY_ROLE));
        queryModes.put("Group", String.valueOf(QueryLocator.QUERY_GROUP));
    }

    protected static final String CATEGORY_NAME = "categoryName";
    protected static final String CATEGORY_VALUE = "categoryValue";
    protected static final String COPY_FROM = "copyFrom";
    protected static final String COPY_TO = "copyTo";
    protected static final String TEMP_LOCATOR = "tempLocator";
    protected static final String PSML_UPDATE_PANE = "PsmlForm";

    /**
     * Static initialization of the logger for this class
     */
    private static final JetspeedLogger logger = JetspeedLogFactoryService
            .getLogger(PsmlManagerAction.class.getName());

    /**
     * Subclasses should override this method if they wish to
     * provide their own customization behavior.
     * Default is to use Portal base customizer action
     */
    protected void buildConfigureContext(Portlet portlet, Context context, RunData rundata) {
        try {
            super.buildConfigureContext(portlet, context, rundata);
        } catch (Exception ex) {
            logger.error("Exception", ex);
        }
        context.put(PAGE_SIZE, PortletConfigState.getParameter(portlet, rundata, PAGE_SIZE, "20"));
        setTemplate(rundata, PortletConfigState.getParameter(portlet, rundata, CUSTOMIZE_TEMPLATE, null));
    }

    /**
     * Subclasses must override this method to provide default behavior
     * for the portlet action
     * 
     * @param portlet
     * @param context
     * @param rundata
     */
    protected void buildNormalContext(Portlet portlet, Context context, RunData rundata) {

        PortletEntry entry = (PortletEntry) Registry.getEntry(Registry.PORTLET, portlet.getName());
        context.put("can-search", canPerformAction(rundata, entry, entry.getParameter("can-search")));
        context.put("can-clone", canPerformAction(rundata, entry, entry.getParameter("can-clone")));
        context.put("can-import", canPerformAction(rundata, entry, entry.getParameter("can-import")));
        context.put("can-export", canPerformAction(rundata, entry, entry.getParameter("can-export")));
        context.put("can-import-all", canPerformAction(rundata, entry, entry.getParameter("can-import-all")));
        context.put("can-export-all", canPerformAction(rundata, entry, entry.getParameter("can-export-all")));
        context.put("can-add", canPerformAction(rundata, entry, entry.getParameter("can-add")));
        context.put("can-remove", canPerformAction(rundata, entry, entry.getParameter("can-remove")));

        String mode = rundata.getParameters().getString("mode", "browse");
        if (mode.equals("browse")) {
            this.setTemplate(rundata,
                    PortletConfigState.getParameter(portlet, rundata, "template", "browser-psml"));
            buildBrowseNormalContext(portlet, context, rundata);
        } else {
            this.setTemplate(rundata,
                    PortletConfigState.getParameter(portlet, rundata, "detail-template", "psml-form"));
            buildDetailNormalContext(portlet, context, rundata);
        }

    }

    /**
     * Performs building of normal context for browsing mode
     * 
     * @param portlet
     * @param context
     * @param rundata
     */
    private void buildBrowseNormalContext(Portlet portlet, Context context, RunData rundata) {

        context.put("queryModes", queryModes);

        int start = rundata.getParameters().getInt("start", 0);

        if (start < 0) {
            start = 0;
        }

        String pageSize = PortletConfigState.getParameter(portlet, rundata, PAGE_SIZE, "20");
        int size = Integer.parseInt(pageSize);

        int next = start + size + 1;
        int prev = start - size - 1;

        //System.out.println("start="+start+" size="+size+" next="+next+" prev="+prev);

        //check to see if resultset has changed due to PsmlUpdateAction
        //if so reconstruct the iterator and reset the flag

        boolean refreshFlag = getRefreshPsmlFlag(rundata);

        // By default, only return psml pages for the current user
        String defaultSearch = PortletConfigState.getParameter(portlet, rundata, DEFAULT_SEARCH, "true");

        //Get the iterator
        DatabaseBrowserIterator windowIterator = (DatabaseBrowserIterator) PortletSessionState.getAttribute(portlet,
                rundata, PROFILE_ITERATOR);
        if (refreshFlag) {
            setRefreshPsmlFlag(rundata, FALSE);
            PortletSessionState.clearAttribute(portlet, rundata, PROFILE_ITERATOR);

            // Use last used search value to refresh
            Integer type = (Integer) PortletSessionState.getAttribute(portlet, rundata, LAST_SEARCH_TYPE);
            String value = (String) PortletSessionState.getAttribute(portlet, rundata, LAST_SEARCH_VALUE);

            if (type != null && value != null) {
                windowIterator = performSearch(rundata, portlet, type.intValue(), value);
            }

        } else if (windowIterator != null) {
            windowIterator.setTop(start);
        } else if (defaultSearch != null && defaultSearch.equals("true")) {
            windowIterator = performSearch(rundata, portlet, QueryLocator.QUERY_USER,
                    rundata.getUser().getUserName());
        }

        if (windowIterator != null) {
            context.put("psml", windowIterator);
            if (start > 0) {
                context.put("prev", String.valueOf(prev + 1));
            }
            if (next <= windowIterator.getResultSetSize()) {
                context.put("next", String.valueOf(next - 1));
            }

        }

    }

    /**
     * Returns true if current user can peform specific profile action
     * 
     * @param data
     * @param profile
     * @return 
     */
    private String canPerformAction(RunData rundata, PortletEntry entry, Parameter actionParam) {
        String result = "true";

        if (actionParam != null && entry != null) {
            PortalResource portalResource = new PortalResource(entry, actionParam);

            result = new Boolean(JetspeedSecurity.checkPermission((JetspeedUser) rundata.getUser(), portalResource,
                    JetspeedSecurity.PERMISSION_CUSTOMIZE)).toString();

            //System.out.println("parameter=" + actionParam.getName() + ", canAccess = " + result.toString());
        }

        return result;
    }

    /**
     * This method is called when the user configures any of the parameters.
     * 
     * @param rundata
     * @param context The velocity context for this request.
     */
    public void doUpdate(RunData rundata, Context context) {
        String pageSize = null;

        Portlet portlet = (Portlet) context.get("portlet");
        if (portlet != null) {
            String peid = portlet.getID();
            if ((peid != null) && peid.equals(rundata.getParameters().getString(PEID))) {
                pageSize = rundata.getParameters().getString(PAGE_SIZE);
            }
            if (pageSize != null) {
                PortletConfigState.setInstanceParameter(portlet, rundata, PAGE_SIZE, pageSize);
                PortletSessionState.clearAttribute(portlet, rundata, PROFILE_ITERATOR);
            }
        }

        buildNormalContext(portlet, context, rundata);
    }

    /**
     * This method is to refresh psml from disk or database.
     * 
     * @param rundata
     * @param context The velocity context for this request.
     */
    public void doRefresh(RunData rundata, Context context) {
        Portlet portlet = (Portlet) context.get("portlet");
        PortletSessionState.clearAttribute(portlet, rundata, PROFILE_ITERATOR);
        PortletSessionState.clearAttribute(portlet, rundata, LAST_SEARCH_TYPE);
        PortletSessionState.clearAttribute(portlet, rundata, LAST_SEARCH_VALUE);
        rundata.getParameters().remove(FILTER_VALUE);
        buildNormalContext(portlet, context, rundata);
    }

    /**
     * This method is to enter filtering mode.
     * 
     * @param rundata
     * @param context The velocity context for this request.
     */
    public void doFilter(RunData rundata, Context context) {
        // Is filtering requested?
        int filterType = rundata.getParameters().getInt(FILTER_TYPE, QueryLocator.QUERY_ALL);
        String filterValue = rundata.getParameters().getString(FILTER_VALUE, null);

        Portlet portlet = (Portlet) context.get("portlet");

        performSearch(rundata, portlet, filterType, filterValue);

        buildNormalContext(portlet, context, rundata);
    }

    /**
     * 
     * @param rundata
     * @param portlet
     * @param type
     * @param value
     * @return 
     */
    private DatabaseBrowserIterator performSearch(RunData rundata, Portlet portlet, int type, String value) {
        // Initialize the query locator
        QueryLocator ql = new QueryLocator(type);
        if (value != null) {
            switch (type) {
            case QueryLocator.QUERY_USER: {
                try {
                    ql.setUser(JetspeedSecurity.getUser(value));
                } catch (Exception e) {
                }
            }
            case QueryLocator.QUERY_ROLE: {
                try {
                    ql.setRole(JetspeedSecurity.getRole(value));
                } catch (Exception e) {
                }
            }
            case QueryLocator.QUERY_GROUP: {
                try {
                    ql.setGroup(JetspeedSecurity.getGroup(value));
                } catch (Exception e) {
                }
            }
            }
        }
        //ql.setQueryString(value);

        ArrayList entries = new ArrayList();
        Iterator i = Profiler.query(ql);

        try {
            while (i.hasNext()) {
                Profile profile = (Profile) i.next();

                if (PortletUtils.canAccessProfile(rundata, profile)) {
                    entries.add(profile);
                }
            }
        } catch (Exception e) {
            logger.error("Exception", e);
        }

        ArrayList entryType = new ArrayList();
        entryType.add("Profile");

        int size = Integer.parseInt(PortletConfigState.getParameter(portlet, rundata, PAGE_SIZE, "20"));
        DatabaseBrowserIterator windowIterator = new DatabaseBrowserIterator(entries, entryType, entryType, size);
        PortletSessionState.clearAttribute(portlet, rundata, PROFILE_ITERATOR);
        PortletSessionState.setAttribute(portlet, rundata, PROFILE_ITERATOR, windowIterator);
        PortletSessionState.setAttribute(portlet, rundata, LAST_SEARCH_TYPE, new Integer(type));
        PortletSessionState.setAttribute(portlet, rundata, LAST_SEARCH_VALUE, value);

        return windowIterator;

    }

    /**
     * Build the normal state context for detail modes.
     *
     * @param portlet The template-based portlet that is being built.
     * @param context The velocity context for this request.
     * @param rundata The turbine rundata context for this request.
     */
    private void buildDetailNormalContext(Portlet portlet, Context context, RunData rundata) {
        try {
            //
            // if there was an error, display the message
            //
            String msgid = rundata.getParameters().getString(SecurityConstants.PARAM_MSGID);
            if (msgid != null) {
                int id = Integer.parseInt(msgid);
                if (id < SecurityConstants.MESSAGES.length) {
                    context.put(SecurityConstants.PARAM_MSG, SecurityConstants.MESSAGES[id]);
                }

                // get the bad entered data and put it back for convenient update
                ProfileLocator locator = (ProfileLocator) rundata.getUser().getTemp(TEMP_LOCATOR);
                if (locator != null) {
                    context.put("profile", Profiler.createProfile(locator));
                }
            }

            String mode = rundata.getParameters().getString(SecurityConstants.PARAM_MODE);
            context.put(SecurityConstants.PARAM_MODE, mode);
            String path = rundata.getParameters().getString(SecurityConstants.PARAM_ENTITY_ID);

            if (mode != null && mode.equals(SecurityConstants.PARAM_MODE_DELETE)) {
                ProfileLocator locator = Profiler.createLocator();
                locator.createFromPath(path);
                Profile profile = Profiler.getProfile(locator);
                if (profile != null) {
                    rundata.getUser().setTemp(TEMP_LOCATOR, locator);
                    context.put("profile", profile);
                } else {
                    logger.error("Profile for Path:" + path + " Not Found!");
                }
            }

            if (mode != null && mode.equals(SecurityConstants.PARAM_MODE_INSERT)) {
                org.apache.jetspeed.om.registry.Registry mediaTypes = Registry.get(Registry.MEDIA_TYPE);
                context.put("mediaTypes", mediaTypes.listEntryNames());
                JetspeedRunData jdata = (JetspeedRunData) rundata;
                context.put("defMediaType", jdata.getCapability().getPreferredMediaType());
                if (msgid == null) {
                    if (((String) context.get("can-clone")).equals("false")) {
                        context.put(CATEGORY_NAME, "user");
                        context.put("categoryValue", rundata.getUser().getUserName());
                        context.put("copyFrom", "none");
                        context.put("title", "My Page");
                    } else if (path == null) {
                        context.put(CATEGORY_NAME, "user");
                        context.put("categoryValue", "anon");
                        context.put("copyFrom", "user/anon/media-type/html/page/default.psml");
                    } else {
                        ProfileLocator tmpLocator = Profiler.createLocator();
                        tmpLocator.createFromPath(path);
                        Profile profile = Profiler.getProfile(tmpLocator);
                        if (profile != null) {
                            rundata.getUser().setTemp(TEMP_LOCATOR, tmpLocator);
                            context.put("profile", profile);
                            context.put("title", profile.getRootSet().getTitle());
                        }
                        String categoryName = "group";
                        String categoryValue = tmpLocator.getGroupName();
                        if (categoryValue == null) {
                            categoryName = "role";
                            categoryValue = tmpLocator.getRoleName();
                            if (categoryValue == null) {
                                categoryName = "user";
                                categoryValue = tmpLocator.getUserName();
                                if (categoryValue == null) {
                                    categoryName = "user";
                                    categoryValue = "anon";
                                }
                            }

                        }
                        context.put(CATEGORY_NAME, categoryName);
                        context.put("categoryValue", categoryValue);
                        context.put("copyFrom", path);
                    }
                } else {
                    context.put(CATEGORY_NAME, rundata.getUser().getTemp(CATEGORY_NAME));
                    context.put(CATEGORY_VALUE, rundata.getUser().getTemp(CATEGORY_VALUE));
                    context.put(COPY_FROM, rundata.getUser().getTemp(COPY_FROM));
                }
            }

            if (mode != null && mode.equals("export")) {
                if (msgid == null) {
                    String tmpPath = JetspeedResources.getString(JetspeedResources.TEMP_DIRECTORY_KEY, "/tmp");
                    String exportPath = JetspeedResources.getString("psml.export.default.path",
                            TurbineServlet.getRealPath(tmpPath));
                    if (path == null) {
                        context.put(COPY_TO, exportPath);
                        context.put(COPY_FROM,
                                Profiler.PARAM_USER + File.separator + Profiler.PARAM_ANON + File.separator
                                        + Profiler.PARAM_MEDIA_TYPE + File.separator + "html" + File.separator
                                        + Profiler.PARAM_PAGE + File.separator + Profiler.FULL_DEFAULT_PROFILE);
                    } else {
                        ProfileLocator tmpLocator = Profiler.createLocator();
                        tmpLocator.createFromPath(path);
                        Profile profile = Profiler.getProfile(tmpLocator);
                        if (profile != null) {
                            rundata.getUser().setTemp(TEMP_LOCATOR, tmpLocator);
                            context.put("profile", profile);
                        }

                        String categoryName = Profiler.PARAM_GROUP;
                        String categoryValue = tmpLocator.getGroupName();
                        if (categoryValue == null) {
                            categoryName = Profiler.PARAM_ROLE;
                            categoryValue = tmpLocator.getRoleName();
                            if (categoryValue == null) {
                                categoryName = Profiler.PARAM_USER;
                                categoryValue = tmpLocator.getUserName();
                                if (categoryValue == null) {
                                    categoryName = Profiler.PARAM_USER;
                                    categoryValue = Profiler.PARAM_ANON;
                                }
                            }

                        }

                        context.put(COPY_TO, exportPath + File.separator + tmpLocator.getName());
                        context.put(COPY_FROM, path);
                    }
                } else {
                    context.put(COPY_TO, rundata.getUser().getTemp(COPY_TO));
                    context.put(COPY_FROM, rundata.getUser().getTemp(COPY_FROM));
                }
            }

            if (mode != null && mode.equals("export_all")) {
                if (msgid == null) {
                    // get the PSML Root Directory
                    ResourceService serviceConf = ((TurbineServices) TurbineServices.getInstance())
                            .getResources(PsmlManagerService.SERVICE_NAME);
                    String root = serviceConf.getString("root", "/WEB-INF/psml");
                    context.put(COPY_TO, TurbineServlet.getRealPath(root));
                } else {
                    context.put(COPY_TO, rundata.getUser().getTemp(COPY_TO));
                }
            }

            if (mode != null && mode.equals("import")) {
                org.apache.jetspeed.om.registry.Registry mediaTypes = Registry.get(Registry.MEDIA_TYPE);
                context.put("mediaTypes", mediaTypes.listEntryNames());
                if (msgid == null) {
                    // get the PSML Root Directory
                    ResourceService serviceConf = ((TurbineServices) TurbineServices.getInstance())
                            .getResources(PsmlManagerService.SERVICE_NAME);
                    String root = serviceConf.getString("root", "/WEB-INF/psml");
                    root = TurbineServlet.getRealPath(root);

                    if (path == null) {
                        context.put(CATEGORY_NAME, Profiler.PARAM_USER);
                        context.put("categoryValue", Profiler.PARAM_ANON);
                        context.put("copyFrom",
                                root + File.separator + Profiler.PARAM_USER + File.separator + Profiler.PARAM_ANON
                                        + File.separator + Profiler.PARAM_MEDIA_TYPE + File.separator + "html"
                                        + File.separator + Profiler.PARAM_PAGE + File.separator
                                        + Profiler.FULL_DEFAULT_PROFILE);
                    } else {
                        ProfileLocator tmpLocator = Profiler.createLocator();
                        tmpLocator.createFromPath(path);
                        Profile profile = Profiler.getProfile(tmpLocator);
                        if (profile != null) {
                            rundata.getUser().setTemp(TEMP_LOCATOR, tmpLocator);
                            context.put("profile", profile);
                        }
                        String categoryName = Profiler.PARAM_GROUP;
                        String categoryValue = tmpLocator.getGroupName();
                        if (categoryValue == null) {
                            categoryName = Profiler.PARAM_ROLE;
                            categoryValue = tmpLocator.getRoleName();
                            if (categoryValue == null) {
                                categoryName = Profiler.PARAM_USER;
                                categoryValue = tmpLocator.getUserName();
                                if (categoryValue == null) {
                                    categoryName = Profiler.PARAM_USER;
                                    categoryValue = Profiler.PARAM_ANON;
                                }
                            }

                        }
                        context.put(CATEGORY_NAME, categoryName);
                        context.put("categoryValue", categoryValue);
                        String filePath = this.mapLocatorToFile(tmpLocator);
                        context.put("copyFrom", root + File.separator + filePath.toString());
                    }
                } else {
                    context.put(CATEGORY_NAME, rundata.getUser().getTemp(CATEGORY_NAME));
                    context.put(CATEGORY_VALUE, rundata.getUser().getTemp(CATEGORY_VALUE));
                    context.put(COPY_FROM, rundata.getUser().getTemp(COPY_FROM));
                }
            }

            if (mode != null && mode.equals("import_all")) {
                if (msgid == null) {
                    // get the PSML Root Directory
                    ResourceService serviceConf = ((TurbineServices) TurbineServices.getInstance())
                            .getResources(PsmlManagerService.SERVICE_NAME);
                    String root = serviceConf.getString("root", "/WEB-INF/psml");
                    context.put(COPY_FROM, TurbineServlet.getRealPath(root));
                } else {
                    context.put(COPY_FROM, rundata.getUser().getTemp(COPY_FROM));
                }
            }

        } catch (Exception e) {
            logger.error("Exception", e);
            rundata.setMessage("Error in PsmlUpdateAction: " + e.toString());
            rundata.setStackTrace(StringUtils.stackTrace(e), e);
            rundata.setScreenTemplate(JetspeedResources.getString("template.error", "Error"));
        }
    }

    /**
     * Database Insert Action for Psml.
     *
     * @param rundata The turbine rundata context for this request.
     * @param context The velocity context for this request.
     */
    public void doInsert(RunData rundata, Context context) throws Exception {
        Profile profile = null;
        ProfileLocator locator = null;
        String categoryName = null;
        String categoryValue = null;
        String copyFrom = null;
        String name = null;
        String title = null;

        try {
            categoryName = rundata.getParameters().getString("CategoryName");
            categoryValue = rundata.getParameters().getString("CategoryValue");
            copyFrom = rundata.getParameters().getString("CopyFrom");
            name = rundata.getParameters().getString("name");
            title = rundata.getParameters().getString("title");
            //
            //create a new locator and set its values according to users input
            //
            locator = Profiler.createLocator();
            if (categoryName.equalsIgnoreCase(Profiler.PARAM_GROUP)) {
                locator.setGroupByName(categoryValue);
            } else if (categoryName.equalsIgnoreCase(Profiler.PARAM_ROLE)) {
                locator.setRoleByName(categoryValue);
            } else if (categoryName.equalsIgnoreCase(Profiler.PARAM_USER)) {
                locator.setUser(JetspeedSecurity.getUser(categoryValue));
            } else {
                locator.setAnonymous(true);
            }

            String tempVar = rundata.getParameters().getString("MediaType");
            if (tempVar != null && tempVar.trim().length() > 0) {
                locator.setMediaType(tempVar);
            }

            tempVar = rundata.getParameters().getString("Language");
            if (tempVar != null && tempVar.trim().length() > 0) {
                locator.setLanguage(tempVar);
            }

            tempVar = rundata.getParameters().getString("Country");
            if (tempVar != null && tempVar.trim().length() > 0) {
                locator.setCountry(tempVar);
            }

            if (!name.endsWith(Profiler.DEFAULT_EXTENSION)) {
                name = name + Profiler.DEFAULT_EXTENSION;
            }
            locator.setName(name);

            //check if profile to be created already exists
            if (PsmlManager.getDocument(locator) != null) {
                throw new EntityExistsException("Profile:" + locator.getPath() + " Already Exists!");
            }

            //
            // validate that its not an 'blank' profile -- not allowed
            //
            if (name == null || name.trim().length() == 0) {
                JetspeedLink link = JetspeedLinkFactory.getInstance(rundata);
                DynamicURI duri = link
                        .addPathInfo(SecurityConstants.PARAM_MODE, SecurityConstants.PARAM_MODE_INSERT)
                        .addPathInfo(SecurityConstants.PARAM_MSGID, SecurityConstants.MID_INVALID_ENTITY_NAME);
                JetspeedLinkFactory.putInstance(link);
                rundata.setRedirectURI(duri.toString());

                //save user entered values
                if (locator != null)
                    rundata.getUser().setTemp(TEMP_LOCATOR, locator);
                if (categoryName != null)
                    rundata.getUser().setTemp(CATEGORY_NAME, categoryName);
                if (categoryValue != null)
                    rundata.getUser().setTemp(CATEGORY_VALUE, categoryValue);
                if (copyFrom != null)
                    rundata.getUser().setTemp(COPY_FROM, copyFrom);
                return;
            }

            //
            // retrieve the profile to clone
            //
            Profile baseProfile = null;
            if (copyFrom != null && !copyFrom.equals("none")) {
                ProfileLocator baseLocator = Profiler.createLocator();
                baseLocator.createFromPath(copyFrom);
                baseProfile = Profiler.getProfile(baseLocator);
            }

            //
            // create a new profile
            //

            // AT THIS POINT "portlet" is null ????
            //Portlet portlet = (Portlet) context.get("portlet");
            //PortletEntry entry = (PortletEntry) Registry.getEntry(Registry.PORTLET, portlet.getName());

            if (copyFrom.equals("none")) {
                profile = Profiler.createProfile(locator, null);
                profile.getDocument().getPortlets().setTitle(title);
                profile.store();
                //System.out.println("Profile title: " + profile.getDocument().getPortlets().getTitle() + " for " + profile.getDocument().getPortlets());
                setRefreshPsmlFlag(rundata, TRUE);
            } else if (baseProfile != null) {
                PSMLDocument doc = baseProfile.getDocument();
                if (doc != null) {
                    Portlets portlets = doc.getPortlets();

                    Portlets clonedPortlets = (Portlets) SerializationUtils.clone(portlets);
                    org.apache.jetspeed.util.PortletUtils.regenerateIds(clonedPortlets);
                    profile = Profiler.createProfile(locator, clonedPortlets);
                } else {
                    profile = Profiler.createProfile(locator, null);
                }
                setRefreshPsmlFlag(rundata, TRUE);
            } else {
                logger.error("Profile listed in Copy From Not Found!");
            }

            goBackToBrowser(rundata);
        } catch (EntityExistsException e) {
            // log the error msg
            logger.error("Exception", e);

            //
            // dup key found - display error message - bring back to same screen
            //
            JetspeedLink link = JetspeedLinkFactory.getInstance(rundata);
            DynamicURI duri = link.addPathInfo(SecurityConstants.PARAM_MODE, SecurityConstants.PARAM_MODE_INSERT)
                    .addPathInfo(SecurityConstants.PARAM_MSGID, SecurityConstants.MID_ENTITY_ALREADY_EXISTS);
            JetspeedLinkFactory.putInstance(link);
            rundata.setRedirectURI(duri.toString());
        } catch (Exception e) {
            // log the error msg
            logger.error("Exception", e);

            //
            // dup key found - display error message - bring back to same screen
            //
            JetspeedLink link = JetspeedLinkFactory.getInstance(rundata);
            DynamicURI duri = link.addPathInfo(SecurityConstants.PARAM_MODE, SecurityConstants.PARAM_MODE_INSERT)
                    .addPathInfo(SecurityConstants.PARAM_MSGID, SecurityConstants.MID_UPDATE_FAILED);
            JetspeedLinkFactory.putInstance(link);
            rundata.setRedirectURI(duri.toString());
        }
        // save values that user just entered so they don't have to re-enter
        if (locator != null)
            rundata.getUser().setTemp(TEMP_LOCATOR, locator);
        if (categoryName != null)
            rundata.getUser().setTemp(CATEGORY_NAME, categoryName);
        if (categoryValue != null)
            rundata.getUser().setTemp(CATEGORY_VALUE, categoryValue);
        if (copyFrom != null)
            rundata.getUser().setTemp(COPY_FROM, copyFrom);

    }

    /**
     * Delete Psml entry
     * 
     * @param rundata
     * @param context
     * @exception Exception
     */
    public void doDelete(RunData rundata, Context context) throws Exception {
        try {
            ProfileLocator locator = (ProfileLocator) rundata.getUser().getTemp(TEMP_LOCATOR);
            if (locator != null) {
                Profiler.removeProfile(locator);
                setRefreshPsmlFlag(rundata, TRUE);
            } else {
                logger.error("ProfileLocator not found!");
            }
        } catch (Exception e) {
            // log the error msg
            logger.error("Exception", e);

            //
            // dup key found - display error message - bring back to same screen
            //
            JetspeedLink link = JetspeedLinkFactory.getInstance(rundata);
            DynamicURI duri = link.addPathInfo(SecurityConstants.PARAM_MODE, SecurityConstants.PARAM_MODE_DELETE)
                    .addPathInfo(SecurityConstants.PARAM_MSGID, SecurityConstants.MID_DELETE_FAILED);
            JetspeedLinkFactory.putInstance(link);
            rundata.setRedirectURI(duri.toString());
        }

    }

    /**
     * 
     * @param rundata
     * @param value
     */
    private void setRefreshPsmlFlag(RunData rundata, String value) {
        rundata.getUser().setTemp(PSML_REFRESH_FLAG, value);
        if (value.equals(TRUE)) {
            rundata.getUser().removeTemp(CACHED_PSML);
        }
    }

    /**
     * 
     * @param rundata
     * @return 
     */
    private boolean getRefreshPsmlFlag(RunData rundata) {
        return (rundata.getUser().getTemp(PSML_REFRESH_FLAG, FALSE)).equals(TRUE);
    }

    /**
     * File Export Action for Psml.
     *
     * @param rundata The turbine rundata context for this request.
     * @param context The velocity context for this request.
     */
    public void doExport(RunData rundata, Context context) throws Exception {
        Profile profile = null;
        ProfileLocator locator = null;
        String copyTo = null;
        String copyFrom = null;

        try {
            copyFrom = rundata.getParameters().getString("CopyFrom");
            copyTo = rundata.getParameters().getString("CopyTo");

            //
            // retrieve the profile to clone
            //
            ProfileLocator baseLocator = Profiler.createLocator();
            baseLocator.createFromPath(copyFrom);
            Profile baseProfile = Profiler.getProfile(baseLocator);

            //
            // Export profile
            //
            if (baseProfile != null) {
                PSMLDocument doc = baseProfile.getDocument();
                if (doc != null) {
                    if (!this.saveDocument(copyTo, doc))
                        throw new Exception("Failed to save PSML document");
                    rundata.addMessage("Profile [" + copyFrom + "] has been saved to disk in [" + copyTo + "]<br>");
                }
            } else {
                logger.error("Profile listed in Copy From Not Found!");
            }
        } catch (Exception e) {
            // log the error msg
            logger.error("Exception", e);

            //
            // dup key found - display error message - bring back to same screen
            //
            JetspeedLink link = JetspeedLinkFactory.getInstance(rundata);
            DynamicURI duri = link.addPathInfo(SecurityConstants.PARAM_MODE, "export")
                    .addPathInfo(SecurityConstants.PARAM_MSGID, SecurityConstants.MID_UPDATE_FAILED);
            JetspeedLinkFactory.putInstance(link);
            rundata.setRedirectURI(duri.toString());
        }
        // save values that user just entered so they don't have to re-enter
        if (copyTo != null)
            rundata.getUser().setTemp(COPY_TO, copyTo);
        if (copyFrom != null)
            rundata.getUser().setTemp(COPY_FROM, copyFrom);

    }

    /**
     * File Export All Action for Psml.
     *
     * @param rundata The turbine rundata context for this request.
     * @param context The velocity context for this request.
     */
    public void doExportall(RunData rundata, Context context) throws Exception {
        String copyTo = null;

        logger.info("PsmlUpdateAction: Starting export all operation");

        try {
            copyTo = rundata.getParameters().getString("CopyTo");

            //
            // retrieve the profiles to export
            //
            Iterator i = Profiler.query(new QueryLocator(QueryLocator.QUERY_ALL));
            while (i.hasNext()) {
                Profile profile = (Profile) i.next();
                PSMLDocument doc = profile.getDocument();
                if (doc != null) {
                    // Build the fully qualified file name
                    StringBuffer copyToFile = new StringBuffer(copyTo);
                    copyToFile.append(File.separator);
                    if (profile.getGroupName() != null) {
                        copyToFile.append("group");
                        copyToFile.append(File.separator);
                        copyToFile.append(profile.getGroupName());
                        copyToFile.append(File.separator);
                    } else if (profile.getRoleName() != null) {
                        copyToFile.append("role");
                        copyToFile.append(File.separator);
                        copyToFile.append(profile.getRoleName());
                        copyToFile.append(File.separator);
                    } else if (profile.getUserName() != null) {
                        copyToFile.append("user");
                        copyToFile.append(File.separator);
                        copyToFile.append(profile.getUserName());
                        copyToFile.append(File.separator);
                    }
                    if (profile.getMediaType() != null) {
                        copyToFile.append(profile.getMediaType());
                        copyToFile.append(File.separator);
                    }
                    if (profile.getLanguage() != null) {
                        copyToFile.append(profile.getLanguage());
                        copyToFile.append(File.separator);
                    }
                    if (profile.getCountry() != null) {
                        copyToFile.append(profile.getCountry());
                        copyToFile.append(File.separator);
                    }
                    copyToFile.append(profile.getName());

                    if (!this.saveDocument(copyToFile.toString(), doc)) {
                        logger.error("Failed to save PSML document for [" + profile.getPath());
                    } else {
                        String msg = "Profile [" + profile.getPath() + "] has been saved to disk in ["
                                + copyToFile.toString() + "]<br>";
                        logger.info(msg);
                        rundata.addMessage(msg);
                    }
                }
            }

        } catch (Exception e) {
            // log the error msg
            logger.error("Exception", e);

            //
            // dup key found - display error message - bring back to same screen
            //
            JetspeedLink link = JetspeedLinkFactory.getInstance(rundata);
            DynamicURI duri = link.addPathInfo(SecurityConstants.PARAM_MODE, "export_all")
                    .addPathInfo(SecurityConstants.PARAM_MSGID, SecurityConstants.MID_UPDATE_FAILED);
            JetspeedLinkFactory.putInstance(link);
            rundata.setRedirectURI(duri.toString());
        }
        // save values that user just entered so they don't have to re-enter
        if (copyTo != null) {
            rundata.getUser().setTemp(COPY_TO, copyTo);
        }

        logger.info("PsmlUpdateAction: Ending export all operation");
    }

    /**
     * Save the PSML document on disk to the specififed fileOrUrl
     * 
     * @param fileOrUrl a String representing either an absolute URL
     *                  or an absolute filepath
     * @param doc       the document to save
     * @return 
     */
    private boolean saveDocument(String fileOrUrl, PSMLDocument doc) {
        boolean success = false;

        if (doc == null)
            return false;
        File f = new File(fileOrUrl);
        File d = new File(f.getParent());
        d.mkdirs();

        FileWriter writer = null;

        try {
            writer = new FileWriter(f);
            // create the serializer output format
            OutputFormat format = new OutputFormat();
            format.setIndenting(true);
            format.setIndent(4);
            Serializer serializer = new XMLSerializer(writer, format);
            Marshaller marshaller = new Marshaller(serializer.asDocumentHandler());
            marshaller.setMapping(this.loadMapping());
            marshaller.marshal(doc.getPortlets());

            success = true;
        } catch (MarshalException e) {
            logger.error("PsmlManagerAction: Could not marshal the file " + f.getAbsolutePath(), e);
        } catch (MappingException e) {
            logger.error("PsmlManagerAction: Could not marshal the file " + f.getAbsolutePath(), e);
        } catch (ValidationException e) {
            logger.error("PsmlManagerAction: document " + f.getAbsolutePath() + " is not valid", e);
        } catch (IOException e) {
            logger.error("PsmlManagerAction: Could not save the file " + f.getAbsolutePath(), e);
        } catch (Exception e) {
            logger.error("PsmlManagerAction: Error while saving  " + f.getAbsolutePath(), e);
        } finally {
            try {
                writer.close();
            } catch (IOException e) {
            }
        }

        return success;
    }

    /**
     * Loads psml mapping file
     *
     * @exception Exception
     */
    private Mapping loadMapping() throws Exception {
        // get configuration parameters from Jetspeed Resources
        ResourceService serviceConf = ((TurbineServices) TurbineServices.getInstance())
                .getResources(PsmlManagerService.SERVICE_NAME);

        // test the mapping file and create the mapping object
        Mapping mapping = null;
        String mapFile = serviceConf.getString("mapping", "${webappRoot}/WEB-INF/conf/psml-mapping.xml");
        mapFile = TurbineServlet.getRealPath(mapFile);
        if (mapFile != null) {
            File map = new File(mapFile);
            if (logger.isDebugEnabled()) {
                logger.debug("Loading psml mapping file " + mapFile);
            }
            if (map.exists() && map.isFile() && map.canRead()) {
                try {
                    mapping = new Mapping();
                    InputSource is = new InputSource(new FileReader(map));
                    is.setSystemId(mapFile);
                    mapping.loadMapping(is);
                } catch (Exception e) {
                    logger.error("Error in psml mapping creation", e);
                    throw new Exception("Error in mapping");
                }
            } else {
                throw new Exception("PSML Mapping not found or not a file or unreadable: " + mapFile);
            }
        }

        return mapping;
    }

    /**
     * File Import Action for Psml.
     *
     * TODO: Implement file upload.
     *
     * @param rundata The turbine rundata context for this request.
     * @param context The velocity context for this request.
     */
    public void doImport(RunData rundata, Context context) throws Exception {
        Profile profile = null;
        ProfileLocator locator = null;
        String categoryName = null;
        String categoryValue = null;
        String copyFrom = null;
        String name = null;

        try {
            categoryName = rundata.getParameters().getString("CategoryName");
            categoryValue = rundata.getParameters().getString("CategoryValue");
            copyFrom = rundata.getParameters().getString("CopyFrom");
            name = rundata.getParameters().getString("name");
            //
            //create a new locator and set its values according to users input
            //
            locator = Profiler.createLocator();
            if (categoryName.equalsIgnoreCase(Profiler.PARAM_GROUP)) {
                locator.setGroupByName(categoryValue);
            } else if (categoryName.equalsIgnoreCase(Profiler.PARAM_ROLE)) {
                locator.setRoleByName(categoryValue);
            } else if (categoryName.equalsIgnoreCase(Profiler.PARAM_USER)) {
                locator.setUser(JetspeedSecurity.getUser(categoryValue));
            } else {
                locator.setAnonymous(true);
            }

            String tempVar = rundata.getParameters().getString("MediaType");
            if (tempVar != null && tempVar.trim().length() > 0) {
                locator.setMediaType(tempVar);
            }

            tempVar = rundata.getParameters().getString("Language");
            if (tempVar != null && tempVar.trim().length() > 0) {
                locator.setLanguage(tempVar);
            }

            tempVar = rundata.getParameters().getString("Country");
            if (tempVar != null && tempVar.trim().length() > 0) {
                locator.setCountry(tempVar);
            }

            locator.setName(name);

            //
            // validate that its not an 'blank' profile -- not allowed
            //
            if (name == null || name.trim().length() == 0) {
                JetspeedLink link = JetspeedLinkFactory.getInstance(rundata);
                DynamicURI duri = link.addPathInfo(SecurityConstants.PARAM_MODE, "import")
                        .addPathInfo(SecurityConstants.PARAM_MSGID, SecurityConstants.MID_INVALID_ENTITY_NAME);
                JetspeedLinkFactory.putInstance(link);
                rundata.setRedirectURI(duri.toString());

                //save user entered values
                if (locator != null) {
                    rundata.getUser().setTemp(TEMP_LOCATOR, locator);
                }
                if (categoryName != null) {
                    rundata.getUser().setTemp(CATEGORY_NAME, categoryName);
                }
                if (categoryValue != null) {
                    rundata.getUser().setTemp(CATEGORY_VALUE, categoryValue);
                }
                if (copyFrom != null) {
                    rundata.getUser().setTemp(COPY_FROM, copyFrom);
                }
                return;
            }

            //
            // Retrieve the document to import
            //
            PSMLDocument doc = this.loadDocument(copyFrom);

            //
            // create a new profile
            //
            if (doc != null) {
                Portlets portlets = doc.getPortlets();
                //
                // Profiler does not provide update capability - must remove before replacing
                //
                if (PsmlManager.getDocument(locator) != null) {
                    Profiler.removeProfile(locator);
                }
                profile = Profiler.createProfile(locator, portlets);
            } else {
                throw new Exception("Failed to load PSML document from disk");
            }
            rundata.addMessage(
                    "Profile for [" + locator.getPath() + "] has been imported from file [" + copyFrom + "]<br>");
            setRefreshPsmlFlag(rundata, TRUE);

            goBackToBrowser(rundata);

        } catch (Exception e) {
            // log the error msg
            logger.error("Exception", e);

            //
            // dup key found - display error message - bring back to same screen
            //
            JetspeedLink link = JetspeedLinkFactory.getInstance(rundata);
            DynamicURI duri = link.addPathInfo(SecurityConstants.PARAM_MODE, "import")
                    .addPathInfo(SecurityConstants.PARAM_MSGID, SecurityConstants.MID_UPDATE_FAILED);
            JetspeedLinkFactory.putInstance(link);
            rundata.setRedirectURI(duri.toString());
        }
        // save values that user just entered so they don't have to re-enter
        if (locator != null) {
            rundata.getUser().setTemp(TEMP_LOCATOR, locator);
        }
        if (categoryName != null) {
            rundata.getUser().setTemp(CATEGORY_NAME, categoryName);
        }
        if (categoryValue != null) {
            rundata.getUser().setTemp(CATEGORY_VALUE, categoryValue);
        }
        if (copyFrom != null) {
            rundata.getUser().setTemp(COPY_FROM, copyFrom);
        }

    }

    /**
     * File Import All Action for Psml.
     * 
     * @param rundata The turbine rundata context for this request.
     * @param context The velocity context for this request.
     * @exception Exception
     */
    public void doImportall(RunData rundata, Context context) throws Exception {
        String copyFrom = null;

        try {
            copyFrom = rundata.getParameters().getString("CopyFrom");

            //
            // Collect all .psml files from the root specified
            //
            Vector files = new Vector();
            this.collectPsml(files, copyFrom);

            //
            // Process each file
            //
            for (Iterator it = files.iterator(); it.hasNext();) {
                // If error occurs processing one entry, continue on with the others
                String path = null;
                try {
                    String psml = ((File) it.next()).getPath();
                    path = psml.substring(copyFrom.length() + 1);
                    ProfileLocator locator = this.mapFileToLocator(path);

                    PSMLDocument doc = this.loadDocument(psml);

                    //
                    // create a new profile
                    //
                    if (doc != null) {
                        Portlets portlets = doc.getPortlets();
                        //
                        // Profiler does not provide update capability - must remove before replacing
                        //
                        if (PsmlManager.getDocument(locator) != null) {
                            Profiler.removeProfile(locator);
                        }

                        Portlets clonedPortlets = (Portlets) SerializationUtils.clone(portlets);
                        org.apache.jetspeed.util.PortletUtils.regenerateIds(clonedPortlets);
                        Profiler.createProfile(locator, clonedPortlets);
                    } else {
                        throw new Exception("Failed to load PSML document [" + psml + "] from disk");
                    }
                    rundata.addMessage("Profile for [" + locator.getPath() + "] has been imported from file ["
                            + psml + "]<br>");
                    setRefreshPsmlFlag(rundata, TRUE);
                } catch (Exception ouch) {
                    logger.error("Exception", ouch);
                    rundata.addMessage("ERROR importing file [" + path + "]: " + ouch.toString() + "<br>");
                }
            }

        } catch (Exception e) {
            // log the error msg
            logger.error("Exception", e);

            //
            // dup key found - display error message - bring back to same screen
            //
            JetspeedLink link = JetspeedLinkFactory.getInstance(rundata);
            DynamicURI duri = link.addPathInfo(SecurityConstants.PARAM_MODE, "import_all")
                    .addPathInfo(SecurityConstants.PARAM_MSGID, SecurityConstants.MID_UPDATE_FAILED);
            JetspeedLinkFactory.putInstance(link);
            rundata.setRedirectURI(duri.toString());
        }
        // save values that user just entered so they don't have to re-enter
        if (copyFrom != null) {
            rundata.getUser().setTemp(COPY_FROM, copyFrom);
        }

    }

    /**
     * This method recursively collect all .psml documents starting at the given root
     *
     * @param v      Vector to put the file into
     * @param root   Root directory for import
     */
    private void collectPsml(Vector v, String root) {

        File dir = new File(root);
        File[] files = dir.listFiles();
        for (int i = 0; i < files.length; i++) {
            if (files[i].isDirectory()) {
                collectPsml(v, files[i].getPath());
            } else if (files[i].isFile() && files[i].getPath().endsWith(".psml")) {
                v.add(files[i]);
            }
        }

    }

    /**
     * Creates profile locator from a given path in the format:
     *
     *   user/<name>/<mediaType>/<language>/<country>/<page>/
     *
     *   group/ ""
     *   role/  ""
     *
     * @param path The formatted profiler path string.
     * @param path   fully qualified .psml file name
     * @return profile locator
     */
    private ProfileLocator mapFileToLocator(String path) throws Exception {
        if (logger.isDebugEnabled()) {
            logger.debug("PsmlUpdateAction.createFromPath: processing path = " + path);
        }
        ProfileLocator result = Profiler.createLocator();

        // Tokenize the file path into elements
        StringTokenizer tok = new StringTokenizer(path, File.separator);

        // Load path elements into a vector for random access
        Vector tokens = new Vector();
        while (tok.hasMoreTokens()) {
            tokens.add(tok.nextToken());
        }

        // Assume that 1st element is the profile type (user|role|group) and 2nd is the name
        if (tokens.size() > 1) {
            String type = (String) tokens.elementAt(0);
            String name = (String) tokens.elementAt(1);
            if (type.equals(Profiler.PARAM_USER)) {
                result.setUser(JetspeedSecurity.getUser(name));
            } else if (type.equals(Profiler.PARAM_GROUP)) {
                result.setGroup(JetspeedSecurity.getGroup(name));
            } else if (type.equals(Profiler.PARAM_ROLE)) {
                result.setRole(JetspeedSecurity.getRole(name));
            }
        }

        // Assume that the last element is the page name
        if (tokens.size() > 0) {
            result.setName((String) tokens.lastElement());
        }

        // Based on the number of path elements set the other profile attributes
        switch (tokens.size()) {
        case 3: // user|role|group/name/page.psml
            break;
        case 4: // user|role|group/name/media-type/page.psml
            result.setMediaType((String) tokens.elementAt(2));
            break;
        case 5: // user|role|group/name/media-type/language/page.psml
            result.setMediaType((String) tokens.elementAt(2));
            result.setLanguage((String) tokens.elementAt(3));
            break;
        case 6: // user|role|group/name/media-type/language/country/page.psml
            result.setMediaType((String) tokens.elementAt(2));
            result.setLanguage((String) tokens.elementAt(3));
            result.setCountry((String) tokens.elementAt(4));
            break;
        default:
            throw new Exception(
                    "Path must contain 3 to 6 elements: [" + path + "], and the size was: " + tokens.size());
        }

        return result;
    }

    /**
     * Maps a ProfileLocator to a file.
     *
     * @param locator The profile locator describing the PSML resource to be found.
     * @return the String path of the file.
     */
    private String mapLocatorToFile(ProfileLocator locator) {
        StringBuffer path = new StringBuffer();

        // move the base dir is either user or role is specified
        Role role = locator.getRole();
        Group group = locator.getGroup();
        JetspeedUser user = locator.getUser();

        if (user != null) {
            path.append(Profiler.PARAM_USER);
            String name = user.getUserName();
            if (null != name && name.length() > 0) {
                path.append(File.separator).append(name);
            }
        } else if (group != null) {
            path.append(Profiler.PARAM_GROUP);
            String name = group.getName();
            if (null != name && name.length() > 0) {
                path.append(File.separator).append(name);
            }
        } else if (null != role) {
            path.append(Profiler.PARAM_ROLE);
            String name = role.getName();
            if (null != name && name.length() > 0) {
                path.append(File.separator).append(name);
            }
        }

        // Media
        if (null != locator.getMediaType()) {
            path.append(File.separator).append(locator.getMediaType());
        }
        // Language
        if (null != locator.getLanguage()) {
            path.append(File.separator).append(locator.getLanguage());
        }
        // Country
        if (null != locator.getCountry()) {
            path.append(File.separator).append(locator.getCountry());
        }
        // Resource Name
        if (null != locator.getName()) {
            if (!(locator.getName().endsWith(Profiler.DEFAULT_EXTENSION))) {
                path.append(File.separator).append(locator.getName()).append(Profiler.DEFAULT_EXTENSION);
            } else {
                path.append(File.separator).append(locator.getName());
            }
        } else {
            path.append(File.separator).append(Profiler.FULL_DEFAULT_PROFILE);
        }

        return path.toString();
    }

    /**
     * Load a PSMLDOcument from disk
     *
     * @param fileOrUrl a String representing either an absolute URL or an
     * absolute filepath
     */
    private PSMLDocument loadDocument(String fileOrUrl) {
        PSMLDocument doc = null;

        if (fileOrUrl != null) {

            // we'll assume the name is the the location of the file
            File f = null;

            f = new File(fileOrUrl);

            if (!f.exists()) {
                return null;
            }

            doc = new BasePSMLDocument();
            doc.setName(fileOrUrl);

            // now that we have a file reference, try to load the serialized PSML
            Portlets portlets = null;
            FileReader reader = null;
            try {
                reader = new FileReader(f);

                Unmarshaller unmarshaller = new Unmarshaller(this.loadMapping());
                portlets = (Portlets) unmarshaller.unmarshal(reader);

                doc.setPortlets(portlets);

            } catch (IOException e) {
                logger.error("PsmlManagerAction: Could not load the file " + f.getAbsolutePath(), e);
            } catch (MarshalException e) {
                logger.error("PsmlManagerAction: Could not unmarshal the file " + f.getAbsolutePath(), e);
            } catch (MappingException e) {
                logger.error("PsmlManagerAction: Could not unmarshal the file " + f.getAbsolutePath(), e);
            } catch (ValidationException e) {
                logger.error("PsmlManagerAction: document " + f.getAbsolutePath() + " is not valid", e);
            } catch (Exception e) {
                logger.error("PsmlManagerAction: Error while loading  " + f.getAbsolutePath(), e);
            } finally {
                try {
                    reader.close();
                } catch (IOException e) {
                }
            }
        }

        return doc;
    }

    /**
     * 
     * @param rundata
     */
    private void goBackToBrowser(RunData rundata) throws Exception {
        JetspeedLink link = JetspeedLinkFactory.getInstance(rundata);
        rundata.setRedirectURI(link.toString());
        JetspeedLinkFactory.putInstance(link);
    }
}