org.modeshape.jcr.JcrRepositoriesContainer Maven / Gradle / Ivy
/*
* 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