All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.modeshape.jcr.JcrRepositoriesContainer Maven / Gradle / Ivy

There is a newer version: 5.4.1.Final
Show newest version
/*
 * ModeShape (http://www.modeshape.org)
 * See the COPYRIGHT.txt file distributed with this work for information
 * regarding copyright ownership.  Some portions may be licensed
 * to Red Hat, Inc. under one or more contributor license agreements.
 * See the AUTHORS.txt file in the distribution for a full listing of 
 * individual contributors. 
 *
 * ModeShape is free software. Unless otherwise indicated, all code in ModeShape
 * is licensed to you under the terms of the GNU Lesser General Public License as
 * published by the Free Software Foundation; either version 2.1 of
 * the License, or (at your option) any later version.
 *
 * ModeShape 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.
 *
 * You should have received a copy of the GNU Lesser General Public
 * License along with this software; if not, write to the Free
 * Software Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA
 * 02110-1301 USA, or see the FSF site: http://www.fsf.org.
 */
package org.modeshape.jcr;

import java.net.MalformedURLException;
import java.net.URL;
import java.util.Collections;
import java.util.HashMap;
import java.util.Hashtable;
import java.util.Map;
import java.util.Set;
import java.util.concurrent.ExecutionException;
import java.util.concurrent.Future;
import java.util.concurrent.TimeUnit;
import java.util.concurrent.TimeoutException;
import javax.jcr.RepositoryException;
import javax.naming.InitialContext;
import javax.naming.NamingException;
import org.infinispan.schematic.document.ParsingException;
import org.modeshape.common.annotation.ThreadSafe;
import org.modeshape.common.logging.Logger;
import org.modeshape.common.util.StringUtil;
import org.modeshape.jcr.api.NamedRepository;
import org.modeshape.jcr.api.Repositories;
import org.modeshape.jcr.api.RepositoriesContainer;
import org.modeshape.jcr.api.RepositoryFactory;

/**
 * Service provider implementation of the {@link RepositoriesContainer} interface.
 * 
 * @author Horia Chiorean ([email protected])
 */
@ThreadSafe
public final class JcrRepositoriesContainer implements RepositoriesContainer {

    protected static final Logger LOG = Logger.getLogger(JcrRepositoriesContainer.class);
    protected static final String REPOSITORY_NAME_URL_PARAM = "repositoryName";
    protected static final String FILE_PROTOCOL = "file";
    protected static final String JNDI_PROTOCOL = "jndi";

    /**
     * The engine that hosts the deployed repository instances.
     */
    protected static final ModeShapeEngine ENGINE = new ModeShapeEngine();

    @Override
    @SuppressWarnings( {"unchecked", "rawtypes"} )
    public JcrRepository getRepository( String repositoryName,
                                        Map parameters ) throws RepositoryException {
        if (!StringUtil.isBlank(repositoryName)) {
            try {
                JcrRepository repository = engine().getRepository(repositoryName);
                if (repository.getState() == ModeShapeEngine.State.STARTING
                    || repository.getState() == ModeShapeEngine.State.RUNNING) {
                    return repository;
                }
            } catch (NoSuchRepositoryException e) {
                // there is no such repository, so try to create & initialize it
            }
        }

        URL repositoryURL = repositoryURLFromParams(parameters);
        if (repositoryURL == null) {
            return null;
        }

        Map configParams = parameters;
        if (!StringUtil.isBlank(repositoryName)) {
            configParams = new HashMap(parameters);
            configParams.put(RepositoryFactory.REPOSITORY_NAME, repositoryName);
        }

        String protocol = repositoryURL.getProtocol();
        if (JNDI_PROTOCOL.equalsIgnoreCase(protocol)) {
            return new JNDIRepositoryLookup().repository(configParams, repositoryURL);
        }
        return new FileRepositoryLookup().repository(configParams, repositoryURL);
    }

    @Override
    public Set getRepositoryNames( Map parameters ) throws RepositoryException {
        Set repositoryNames = engine().getRepositoryNames();
        URL repositoryURL = repositoryURLFromParams(parameters);
        if (repositoryURL == null) {
            return repositoryNames;
        }

        String protocol = repositoryURL.getProtocol();
        if (JNDI_PROTOCOL.equalsIgnoreCase(protocol)) {
            Set jndiRepositories = new JNDIRepositoryLookup().repositoryNames(parameters, repositoryURL);
            repositoryNames.addAll(jndiRepositories);
        } else {
            // based on the parameters try to lookup a repository (note that this may actually start it)
            JcrRepository repository = new FileRepositoryLookup().repository(parameters, repositoryURL);
            if (repository != null) {
                repositoryNames.add(repository.getName());
            }
        }
        return repositoryNames;
    }

    @Override
    public Future shutdown() {
        return ENGINE.shutdown();
    }

    @Override
    public boolean shutdown( long timeout,
                             TimeUnit unit ) throws InterruptedException {
        try {
            return ENGINE.shutdown().get(timeout, unit);
        } catch (ExecutionException e) {
            LOG.error(e, JcrI18n.errorShuttingDownJcrRepositoryFactory);
            return false;
        } catch (TimeoutException e) {
            LOG.warn(e, JcrI18n.timeoutWhileShuttingRepositoryDown);
            return false;
        }
    }

    protected ModeShapeEngine engine() {
        // Make sure the engine is started ...
        switch (ENGINE.getState()) {
            case NOT_RUNNING:
                ENGINE.start();
                break;
            case STOPPING:
                // Wait until it's shutdown ...
                try {
                    ENGINE.shutdown().get(10, TimeUnit.SECONDS);
                } catch (Exception e) {
                    // ignore and let if fail ...
                }
                break;
            case RESTORING:
            case RUNNING:
            case STARTING:
                // do nothing ...
        }
        return ENGINE;
    }

    private URL repositoryURLFromParams( Map parameters ) {
        if (parameters == null) {
            LOG.debug("The supplied parameters are null");
            return null;
        }

        Object rawUrl = parameters.get(RepositoryFactory.URL);
        if (rawUrl == null) {
            LOG.debug("No parameter found with key: " + RepositoryFactory.URL);
            return null;
        }

        // Get the URL ...
        URL url = null;
        if (rawUrl instanceof URL) {
            url = (URL)rawUrl;
        } else {
            try {
                url = new URL(rawUrl.toString());
            } catch (MalformedURLException e) {
                LOG.warn(e, JcrI18n.invalidUrl, rawUrl);
                return null;
            }
        }
        return url;
    }

    protected String repositoryNameFrom( URL url,
                                         Map parameters ) {
        // First look in the parameters ...
        Object repoName = parameters.get(RepositoryFactory.REPOSITORY_NAME);
        if (repoName != null) {
            return repoName.toString();
        }
        // Then look for a query parameter in the URL ...
        String query = url.getQuery();
        if (query != null) {
            for (String keyValuePair : query.split("&")) {
                String[] splitPair = keyValuePair.split("=");

                if (splitPair.length == 2 && REPOSITORY_NAME_URL_PARAM.equals(splitPair[0])) {
                    return splitPair[1];
                }
            }
        }
        return null;
    }

    protected class FileRepositoryLookup {
        protected JcrRepository repository( Map parameters,
                                            URL repositoryURL ) throws RepositoryException {
            String repositoryName = repositoryNameFrom(repositoryURL, parameters);
            repositoryURL = postProcessRepositoryURL(repositoryURL);

            JcrRepository repository = lookForAlreadyDeployedRepositoryKey(repositoryURL.toString());
            if (repository != null) {
                return repository;
            }

            RepositoryConfiguration configuration = loadRepositoryConfigurationFrom(repositoryURL);
            repository = lookForAlreadyDeployedRepositoryName(configuration.getName());
            if (repository != null) {
                return StringUtil.isBlank(repositoryName) || configuration.getName().equalsIgnoreCase(repositoryName) ? repository : null;
            }

            // Now deploy the new repository ...
            try {
                repository = ENGINE.deploy(configuration);
                repository.start();

                // The name might not match the configuration's name ...
                if (StringUtil.isBlank(repositoryName) || repository.getName().equals(repositoryName)) {
                    return repository;
                }
                LOG.warn(JcrI18n.repositoryNotFound, repositoryName, repositoryURL, parameters);
                return null;
            } catch (RepositoryException re) {
                throw re;
            } catch (Exception e) {
                throw new RepositoryException(e);
            }
        }

        private JcrRepository lookForAlreadyDeployedRepositoryKey( String key ) {
            if (engine().getRepositoryKeys().contains(key)) {
                try {
                    return engine().getRepository(key);
                } catch (NoSuchRepositoryException e) {
                    // Must have been removed since we checked, so just continue on to redeploy it ...
                }
            }
            return null;
        }

        private JcrRepository lookForAlreadyDeployedRepositoryName( String name ) {
            if (engine().getRepositoryNames().contains(name)) {
                try {
                    return engine().getRepository(name);
                } catch (NoSuchRepositoryException e) {
                    // Must have been removed since we checked, so just continue on to redeploy it ...
                }
            }
            return null;
        }

        private URL postProcessRepositoryURL( URL repositoryURL ) {
            if (FILE_PROTOCOL.equalsIgnoreCase(repositoryURL.getProtocol())) {
                // Strip any query parameters from the incoming file URLs by creating a new URL with the same protocol, host,
                // and port, but using the URL path as returned by URL#getPath() instead of the URL path and query parameters as
                // returned by URL#getFile()
                try {
                    return new URL(repositoryURL.getProtocol(), repositoryURL.getHost(), repositoryURL.getPort(),
                                   repositoryURL.getPath());
                } catch (MalformedURLException e) {
                    // should not happen
                    throw new IllegalArgumentException(e);
                }
            }
            return repositoryURL;
        }

        private RepositoryConfiguration loadRepositoryConfigurationFrom( URL repositoryURL ) throws RepositoryException {
            try {
                if (FILE_PROTOCOL.equalsIgnoreCase(repositoryURL.getProtocol())) {
                    try {
                        return RepositoryConfiguration.read(repositoryURL);
                    } catch (ParsingException pe) {
                        // Try reading from the classpath ...
                        String path = classpathResource(repositoryURL);
                        return RepositoryConfiguration.read(path);
                    }
                }
                // Just try resolving the URL ...
                return RepositoryConfiguration.read(repositoryURL);
            } catch (Exception e) {
                throw new RepositoryException(e);
            }
        }

        private String classpathResource( URL url ) {
            String path = url.getPath();
            while (path.startsWith("/") && path.length() > 1) {
                path = path.substring(1);
            }
            return path.length() != 0 ? path : null;
        }
    }

    protected class JNDIRepositoryLookup {

        private Object doJNDILookup( String jndiName,
                                     Map parameters ) throws RepositoryException {
            if (jndiName == null) {
                return null;
            }
            try {
                InitialContext ic = new InitialContext(hashtable(parameters));
                return ic.lookup(jndiName);
            } catch (NamingException e) {
                throw new RepositoryException(e);
            }
        }

        protected JcrRepository repository( Map parameters,
                                            URL repositoryURL ) throws RepositoryException {
            String jndiName = repositoryURL.getPath();
            String repositoryName = repositoryNameFrom(repositoryURL, parameters);
            Object jndiObject = doJNDILookup(jndiName, parameters);

            if (jndiObject instanceof ModeShapeEngine) {
                return repositoryFromEngine((ModeShapeEngine)jndiObject, jndiName, repositoryName);
            } else if (jndiObject instanceof JcrRepository) {
                // Just return the repository instance ...
                return (JcrRepository)jndiObject;
            } else {
                return null;
            }
        }

        protected Set repositoryNames( Map parameters,
                                               URL repositoryURL ) throws RepositoryException {
            String jndiName = repositoryURL.getPath();
            Object jndiObject = doJNDILookup(jndiName, parameters);

            if (jndiObject instanceof NamedRepository) {
                return Collections.singleton(((NamedRepository)jndiObject).getName());
            } else if (jndiObject instanceof Repositories) {
                return ((Repositories)jndiObject).getRepositoryNames();
            } else {
                return Collections.emptySet();
            }
        }

        private JcrRepository repositoryFromEngine( ModeShapeEngine engine,
                                                    String jndiName,
                                                    String repositoryName ) {
            switch (engine.getState()) {
                case NOT_RUNNING:
                case STOPPING:
                    LOG.error(JcrI18n.engineAtJndiLocationIsNotRunning, jndiName);
                    return null;
                case RUNNING:
                case RESTORING:
                case STARTING:
                    break; // continue
            }

            if (repositoryName == null && engine.getRepositories().size() == 1) {
                repositoryName = engine.getRepositories().keySet().iterator().next();
            }

            if (StringUtil.isBlank(repositoryName)) {
                return null;
            }

            if (repositoryName != null) {
                repositoryName = repositoryName.trim();
            }
            // Look for a repository with the supplied name ...
            try {
                JcrRepository repository = engine.getRepository(repositoryName);
                switch (repository.getState()) {
                    case STARTING:
                    case RUNNING:
                        return repository;
                    default:
                        LOG.debug("The '{0}' repository in JNDI at '{1}' is not (yet) running, but may be (re)started when needed.",
                                  repositoryName,
                                  jndiName);
                        return repository;
                }
            } catch (NoSuchRepositoryException e) {
                LOG.warn(JcrI18n.repositoryNotFoundInEngineAtJndiLocation, repositoryName, jndiName);
                return null;
            }
        }

        private Hashtable hashtable( Map map ) {
            if (map == null) {
                return new Hashtable();
            }
            Hashtable hash = new Hashtable(map.size());
            for (Map.Entry entry : map.entrySet()) {
                Object value = entry.getValue();
                hash.put(entry.getKey().toString(), value != null ? value.toString() : null);
            }

            return hash;
        }
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy