org.pentaho.platform.web.servlet.PentahoXmlaServlet.java Source code

Java tutorial

Introduction

Here is the source code for org.pentaho.platform.web.servlet.PentahoXmlaServlet.java

Source

/*!
 *
 * This program is free software; you can redistribute it and/or modify it under the
 * terms of the GNU Lesser General Public License, version 2.1 as published by the Free Software
 * Foundation.
 *
 * You should have received a copy of the GNU Lesser General Public License along with this
 * program; if not, you can obtain a copy at http://www.gnu.org/licenses/old-licenses/lgpl-2.1.html
 * or from the Free Software Foundation, Inc.,
 * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
 *
 * This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY;
 * without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
 * See the GNU Lesser General Public License for more details.
 *
 *
 * Copyright (c) 2002-2018 Hitachi Vantara. All rights reserved.
 *
 */

package org.pentaho.platform.web.servlet;

import mondrian.olap.Connection;
import mondrian.olap.DriverManager;
import mondrian.olap.MondrianException;
import mondrian.olap.MondrianServer;
import mondrian.server.DynamicContentFinder;
import mondrian.server.FileRepository;
import mondrian.server.RepositoryContentFinder;
import mondrian.spi.CatalogLocator;
import mondrian.spi.impl.ServletContextCatalogLocator;
import mondrian.xmla.XmlaException;
import mondrian.xmla.XmlaHandler.ConnectionFactory;
import mondrian.xmla.impl.DynamicDatasourceXmlaServlet;
import org.apache.commons.collections.CollectionUtils;
import org.apache.commons.collections.Predicate;
import org.apache.commons.logging.Log;
import org.apache.commons.logging.LogFactory;
import org.apache.commons.vfs2.FileSystemException;
import org.apache.commons.vfs2.FileSystemManager;
import org.apache.commons.vfs2.VFS;
import org.apache.commons.vfs2.impl.DefaultFileSystemManager;
import org.dom4j.Document;
import org.dom4j.Element;
import org.dom4j.Node;
import org.dom4j.tree.DefaultElement;
import org.olap4j.OlapConnection;
import org.pentaho.platform.api.engine.ICacheManager;
import org.pentaho.platform.api.engine.IConnectionUserRoleMapper;
import org.pentaho.platform.api.engine.IPentahoSession;
import org.pentaho.platform.api.engine.PentahoAccessControlException;
import org.pentaho.platform.api.repository2.unified.IUnifiedRepository;
import org.pentaho.platform.api.util.XmlParseException;
import org.pentaho.platform.engine.core.system.PentahoSessionHolder;
import org.pentaho.platform.engine.core.system.PentahoSystem;
import org.pentaho.platform.engine.security.SecurityHelper;
import org.pentaho.platform.engine.services.solution.PentahoEntityResolver;
import org.pentaho.platform.plugin.action.mondrian.catalog.IMondrianCatalogService;
import org.pentaho.platform.plugin.action.mondrian.catalog.MondrianCatalog;
import org.pentaho.platform.plugin.action.mondrian.catalog.MondrianCatalogHelper;
import org.pentaho.platform.plugin.services.connections.mondrian.MDXConnection;
import org.pentaho.platform.repository.solution.filebased.MondrianVfs;
import org.pentaho.platform.repository.solution.filebased.SolutionRepositoryVfsFileObject;
import org.pentaho.platform.util.xml.dom4j.XmlDom4JHelper;
import org.pentaho.platform.web.servlet.messages.Messages;
import org.xml.sax.EntityResolver;

import javax.servlet.ServletConfig;
import javax.servlet.ServletException;
import java.sql.SQLException;
import java.util.List;
import java.util.Map;
import java.util.Properties;
import java.util.Set;
import java.util.concurrent.Callable;

/**
 * Filters out <code>DataSource</code> elements that are not XMLA-related.
 * <p/>
 * Background: Pentaho re-used datasources.xml for non-XMLA purposes. But since <code>DefaultXmlaServlet</code> requires
 * actual XMLA datasources, this servlet extends <code>DefaultXmlaServlet</code> and removes the non-XMLA datasources
 * before continuing normal <code>DefaultXmlaServlet</code> behavior.
 * <p/>
 * The convention here is that any <code>DataSource</code> elements with
 * <code>&lt;ProviderType&gt;None&lt;/ProviderType&gt;</code> are considered non-XMLA and are filtered out.
 *
 * @author mlowery
 */
@SuppressWarnings("unchecked")
public class PentahoXmlaServlet extends DynamicDatasourceXmlaServlet {

    // ~ Static fields/initializers ======================================================================================

    /**
     * A cache of {@link RepositoryContentFinder} implementations.
     * The key is the datasource URL.
     */
    final ICacheManager cacheMgr = PentahoSystem.getCacheManager(null);

    final String CACHE_REGION = "org.pentaho.platform.web.servlet.PentahoXmlaServlet";

    private static final long serialVersionUID = 5801343357261568600L;
    private static final Log logger = LogFactory.getLog(PentahoXmlaServlet.class);

    private final IUnifiedRepository repo;

    private final MondrianCatalogHelper mondrianCatalogService;
    private CatalogLocator catalogLocator;

    // - Constructors ================================

    public PentahoXmlaServlet() {
        super();
        if (!cacheMgr.cacheEnabled(CACHE_REGION)) {
            cacheMgr.addCacheRegion(CACHE_REGION);
        }
        repo = PentahoSystem.get(IUnifiedRepository.class);
        mondrianCatalogService = (MondrianCatalogHelper) PentahoSystem.get(IMondrianCatalogService.class);
        try {
            DefaultFileSystemManager dfsm = (DefaultFileSystemManager) VFS.getManager();
            if (!dfsm.hasProvider("mondrian")) {
                dfsm.addProvider("mondrian", new MondrianVfs());
            }
        } catch (FileSystemException e) {
            logger.error(e.getMessage());
        }
    }

    // ~ Methods =========================================================================================================

    @Override
    protected RepositoryContentFinder makeContentFinder(String dataSourcesUrl) {
        // It is safe to cache these for now because their lambda doesn't
        // do anything with security.
        // BEWARE before making modifications that check security rights or all
        // other kind of stateful things.
        @SuppressWarnings("rawtypes")
        Set keys = cacheMgr.getAllKeysFromRegionCache(CACHE_REGION);

        if (!keys.contains(dataSourcesUrl)) {
            cacheMgr.putInRegionCache(CACHE_REGION, dataSourcesUrl, new DynamicContentFinder(dataSourcesUrl) {
                @Override
                public String getContent() {
                    try {
                        String original = generateInMemoryDatasourcesXml();
                        EntityResolver loader = new PentahoEntityResolver();
                        Document originalDocument = XmlDom4JHelper.getDocFromString(original, loader);
                        if (PentahoXmlaServlet.logger.isDebugEnabled()) {
                            PentahoXmlaServlet.logger.debug(Messages.getInstance()
                                    .getString("PentahoXmlaServlet.DEBUG_ORIG_DOC", originalDocument.asXML())); //$NON-NLS-1$
                        }
                        Document modifiedDocument = (Document) originalDocument.clone();
                        List<Node> nodesToRemove = getNodesToRemove(modifiedDocument);
                        if (PentahoXmlaServlet.logger.isDebugEnabled()) {
                            PentahoXmlaServlet.logger.debug(
                                    Messages.getInstance().getString("PentahoXmlaServlet.DEBUG_NODES_TO_REMOVE", //$NON-NLS-1$
                                            String.valueOf(nodesToRemove.size())));
                        }
                        for (Node node : nodesToRemove) {
                            node.detach();
                        }
                        if (PentahoXmlaServlet.logger.isDebugEnabled()) {
                            PentahoXmlaServlet.logger.debug(Messages.getInstance()
                                    .getString("PentahoXmlaServlet.DEBUG_MOD_DOC", modifiedDocument.asXML())); //$NON-NLS-1$
                        }
                        return modifiedDocument.asXML();
                    } catch (XmlParseException e) {
                        PentahoXmlaServlet.logger.error(Messages.getInstance()
                                .getString("PentahoXmlaServlet.ERROR_0004_UNABLE_TO_GET_DOCUMENT_FROM_STRING"), e); //$NON-NLS-1$
                        return null;
                    }
                }

                private List<Node> getNodesToRemove(Document doc) {
                    List<Node> nodesToRemove = doc.selectNodes("/DataSources/DataSource/Catalogs/Catalog");
                    CollectionUtils.filter(nodesToRemove, new Predicate() {
                        @Override
                        public boolean evaluate(Object o) {
                            Element el = ((DefaultElement) o).element("DataSourceInfo");

                            if (el == null || el.getText() == null || el.getTextTrim().length() == 0) {
                                throw new XmlaException(SERVER_FAULT_FC, UNKNOWN_ERROR_CODE, UNKNOWN_ERROR_FAULT_FS,
                                        new MondrianException("DataSourceInfo not defined for "
                                                + ((DefaultElement) o).attribute("name").getText()));
                            }
                            return el.getText().matches("(?i).*EnableXmla=['\"]?false['\"]?");
                        }
                    });
                    return nodesToRemove;
                }
            });
        }
        return (RepositoryContentFinder) cacheMgr.getFromRegionCache(CACHE_REGION, dataSourcesUrl);
    }

    private String generateInMemoryDatasourcesXml() {
        try {
            return SecurityHelper.getInstance().runAsSystem(new Callable<String>() {
                public String call() throws Exception {
                    return mondrianCatalogService.generateInMemoryDatasourcesXml(repo);
                }
            });
        } catch (Exception e) {
            PentahoXmlaServlet.logger.error(e);
            throw new RuntimeException(e);
        }
    }

    @Override
    protected CatalogLocator makeCatalogLocator(ServletConfig servletConfig) {
        return new ServletContextCatalogLocator(servletConfig.getServletContext()) {
            @Override
            public String locate(String catalogPath) {
                if (catalogPath.startsWith("mondrian:")) { //$NON-NLS-1$
                    try {
                        FileSystemManager fsManager = VFS.getManager();
                        SolutionRepositoryVfsFileObject catalog = (SolutionRepositoryVfsFileObject) fsManager
                                .resolveFile(catalogPath);
                        catalogPath = "solution:" + catalog.getFileRef();
                    } catch (FileSystemException e) {
                        logger.error(e.getMessage());
                    }
                } else {
                    catalogPath = super.locate(catalogPath);
                }
                return catalogPath;
            }
        };
    }

    @Override
    protected String makeDataSourcesUrl(ServletConfig config) {
        return "";
    }

    @Override
    protected ConnectionFactory createConnectionFactory(final ServletConfig servletConfig) throws ServletException {

        final ConnectionFactory delegate = super.createConnectionFactory(servletConfig);

        /*
         * This wrapper for the connection factory allows us to
         * override the list of roles with the ones defined in
         * the IPentahoSession and filter it through the
         * IConnectionUserRoleMapper.
         */
        return new ConnectionFactory() {

            public Map<String, Object> getPreConfiguredDiscoverDatasourcesResponse() {
                return delegate.getPreConfiguredDiscoverDatasourcesResponse();
            }

            public OlapConnection getConnection(String databaseName, String catalogName, String roleName,
                    Properties props) throws SQLException {
                // What we do here is to filter the role names with the mapper.
                // First, get a user role mapper, if one is configured.
                final IPentahoSession session = PentahoSessionHolder.getSession();

                final IConnectionUserRoleMapper mondrianUserRoleMapper = PentahoSystem
                        .get(IConnectionUserRoleMapper.class, MDXConnection.MDX_CONNECTION_MAPPER_KEY, null); // Don't use the user session here yet.

                String[] effectiveRoles = new String[0];

                /*
                 * If Catalog/Schema are null (this happens with high level metadata requests,
                 * like DISCOVER_DATASOURCES) we can't use the role mapper, even if it
                 * is present and configured.
                 */
                if (mondrianUserRoleMapper != null && catalogName != null) {
                    // Use the role mapper.
                    try {
                        effectiveRoles = mondrianUserRoleMapper.mapConnectionRoles(session, catalogName);
                        if (effectiveRoles == null) {
                            effectiveRoles = new String[0];
                        }
                    } catch (PentahoAccessControlException e) {
                        throw new SQLException(e);
                    }
                }

                // Now we tokenize that list.
                boolean addComma = false;
                roleName = ""; //$NON-NLS-1$
                for (String role : effectiveRoles) {
                    if (addComma) {
                        roleName = roleName.concat(","); //$NON-NLS-1$
                    }
                    roleName = roleName.concat(role);
                    addComma = true;
                }

                // Now let the delegate connection factory do its magic.
                if (catalogName == null) {

                    return delegate.getConnection(databaseName, catalogName, roleName.equals("") ? null : roleName,
                            props);

                } else {
                    //We create a connection differently so we can ensure that
                    //the XMLA servlet shares the same MondrianServer instance as the rest
                    //of the platform
                    IMondrianCatalogService mcs = PentahoSystem.get(IMondrianCatalogService.class);

                    MondrianCatalog mc = mcs.getCatalog(catalogName, PentahoSessionHolder.getSession());

                    if (mc == null) {
                        throw new XmlaException(CLIENT_FAULT_FC, HSB_BAD_RESTRICTION_LIST_CODE,
                                HSB_BAD_RESTRICTION_LIST_FAULT_FS,
                                new MondrianException("No such catalog: " + catalogName));
                    }

                    Connection con = DriverManager.getConnection(
                            mc.getDataSourceInfo() + ";Catalog=" + mc.getDefinition(), catalogLocator);

                    try {
                        final MondrianServer server = MondrianServer.forConnection(con);

                        FileRepository fr = new FileRepository(makeContentFinder(makeDataSourcesUrl(servletConfig)),
                                catalogLocator);

                        OlapConnection connection = fr.getConnection(server, databaseName, catalogName, roleName,
                                props);
                        fr.shutdown();
                        return connection;
                    } finally {
                        con.close();
                    }
                }
            }
        };
    }

    @Override
    public void init(ServletConfig servletConfig) throws ServletException {
        super.init(servletConfig);

        catalogLocator = makeCatalogLocator(servletConfig);

    }
}