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

org.modeshape.jdbc.LocalJcrDriver Maven / Gradle / Ivy

/*
 * ModeShape (http://www.modeshape.org)
 *
 * 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.modeshape.jdbc;

import java.sql.Connection;
import java.sql.DriverManager;
import java.sql.DriverPropertyInfo;
import java.sql.SQLException;
import java.sql.SQLFeatureNotSupportedException;
import java.util.Properties;
import java.util.Set;
import javax.jcr.Repository;
import javax.naming.Context;
import javax.naming.NamingException;
import org.modeshape.common.collection.Collections;
import org.modeshape.common.logging.Logger;
import org.modeshape.jdbc.delegate.ConnectionInfo;
import org.modeshape.jdbc.delegate.LocalRepositoryDelegate;
import org.modeshape.jdbc.delegate.RepositoryDelegate;
import org.modeshape.jdbc.delegate.RepositoryDelegateFactory;

/**
 * A JDBC driver implementation that is able to access a JCR repository to query its contents using JCR-SQL2. 

Connection URLs

*

* The driver accepts the following URL format for accessing a local JCR repository using JNDI: * *

 *     jdbc:jcr:jndi:{jndiName}
 * 
* * or * *
 *     jdbc:jcr:jndi:{jndiName}?{firstProperty}&{secondProperty}&...
 * 
* * where *
    *
  • {jndiName} is the JNDI name where the {@link Repository} or {@literal org.modeshape.jcr.api.Repositories} * instance can be found;
  • *
  • {firstProperty} consists of the first property name followed by '=' followed by the property's value;
  • *
  • {secondProperty} consists of the second property name followed by '=' followed by the property's value;
  • *
* Note that any use of URL encoding ('%' followed by a two-digit hexadecimal value) will be decoded before being used. *

*

* Here's an example of a URL that defines a {@link Repository} instance located at "jcr/local" with a repository * name of "repository" and a user, password of "secret", and workspace name of "My Workspace": * *

 *     jdbc:jcr:jndi:jcr/local?repositoryName=repository&user=jsmith&password=secret&workspace=My%20Workspace
 * 
* * The "repository" property is required only if the object in JNDI is a {@literal org.modeshape.jcr.api.Repositories} object. *

*

* Note that any use of URL encoding ('%' followed by a two-digit hexadecimal value) will be decoded before being used. *

*/ public class LocalJcrDriver implements java.sql.Driver { protected static Logger logger = Logger.getLogger("org.modeshape.jdbc"); //$NON-NLS-1$ public static final String WORKSPACE_PROPERTY_NAME = "workspace"; public static final String REPOSITORY_PROPERTY_NAME = "repositoryName"; public static final String USERNAME_PROPERTY_NAME = "user"; public static final String PASSWORD_PROPERTY_NAME = "password"; public static final String TEIID_SUPPORT_PROPERTY_NAME = "teiidsupport"; protected static final Set ALL_PROPERTY_NAMES = Collections.unmodifiableSet(WORKSPACE_PROPERTY_NAME, REPOSITORY_PROPERTY_NAME, USERNAME_PROPERTY_NAME, PASSWORD_PROPERTY_NAME, TEIID_SUPPORT_PROPERTY_NAME); /* URL Prefix used for JNDI access */ public static final String JNDI_URL_PREFIX = "jdbc:jcr:jndi:"; private static LocalJcrDriver INSTANCE = new LocalJcrDriver(); static { try { DriverManager.registerDriver(INSTANCE); } catch (SQLException e) { // Logging logger.error(JdbcLocalI18n.driverErrorRegistering, e.getMessage()); } } private final JcrContextFactory contextFactory; private final RepositoryDelegateFactory delegateFactory; private final DriverInfo driverInfo; /** * No-arg constructor, required by the {@link DriverManager}. */ public LocalJcrDriver() { this(null); } /** * Create an instance of this driver using the supplied JNDI naming context factory. This is useful for testing, but is * otherwise not generally recommended. * * @param namingContextFactory the naming context factory; may be null if one should be created automatically */ protected LocalJcrDriver( JcrContextFactory namingContextFactory ) { this(LocalRepositoryDelegate.FACTORY, new DriverInfo(JdbcLocalI18n.driverName.text(), JdbcLocalI18n.driverVendor.text(), JdbcLocalI18n.driverVendorUrl.text(), JdbcLocalI18n.driverVersion.text()), namingContextFactory); } /** * Constructor for subclasses, that should be called by subclasses no-arg constructor. * * @param delegateFactory the factory that should be used to create {@link RepositoryDelegate} instances; may not be null * @param driverInfo the information about the driver; may not be null * @param namingContextFactory the naming context factory; may be null if one should be created automatically */ protected LocalJcrDriver( RepositoryDelegateFactory delegateFactory, DriverInfo driverInfo, JcrContextFactory namingContextFactory ) { assert delegateFactory != null; assert driverInfo != null; this.delegateFactory = delegateFactory; this.driverInfo = driverInfo; this.contextFactory = namingContextFactory; } @Override public boolean acceptsURL( String url ) { return delegateFactory.acceptUrl(url); } @Override public DriverPropertyInfo[] getPropertyInfo( String url, Properties info ) throws SQLException { // Get the connection information ... RepositoryDelegate repositoryDelegate = delegateFactory.createRepositoryDelegate(url, info, this.contextFactory); ConnectionInfo connectionInfo = repositoryDelegate.getConnectionInfo(); return connectionInfo.getPropertyInfos(); } /** * Get the information describing the connection. This method can be overridden to return a subclass of {@link ConnectionInfo} * that implements {@link ConnectionInfo#getCredentials()} for a specific JCR implementation. * * @param url the JDBC URL * @param info the JDBC connection properties * @return the connection information, or null if the URL is null or not of the proper format * @throws SQLException */ protected ConnectionInfo createConnectionInfo( String url, Properties info ) throws SQLException { RepositoryDelegate repositoryDelegate = delegateFactory.createRepositoryDelegate(url, info, this.contextFactory); return repositoryDelegate.getConnectionInfo(); } /** * {@inheritDoc} *

* Note that if the supplied properties and URL contain properties with the same name, the value from the supplied Properties * object will take precedence. *

* * @see java.sql.Driver#connect(java.lang.String, java.util.Properties) */ @Override public Connection connect( String url, Properties info ) throws SQLException { if (!acceptsURL(url)) { return null; } RepositoryDelegate repositoryDelegate = delegateFactory.createRepositoryDelegate(url, info, this.contextFactory); return repositoryDelegate.createConnection(getDriverInfo()); } @Override public int getMajorVersion() { return getDriverInfo().getMajorVersion(); } @Override public int getMinorVersion() { return getDriverInfo().getMinorVersion(); } public String getVendorName() { return getDriverInfo().getVendorName(); } public String getVendorUrl() { return getDriverInfo().getVendorUrl(); } public String getVersion() { return getDriverInfo().getVersion(); } @Override public boolean jdbcCompliant() { return false; } protected final DriverInfo getDriverInfo() { return driverInfo; } /** * This method always throws {@link SQLFeatureNotSupportedException}. *

* Note: This method is part of the JDBC API in JDK 1.7. *

* * @return the parent logger * @throws SQLFeatureNotSupportedException */ @Override public java.util.logging.Logger getParentLogger() throws SQLFeatureNotSupportedException { throw new SQLFeatureNotSupportedException(); } /** * And interface that can be passed to this driver's constructor to create the JNDI naming context given the set of connection * properties. */ public interface JcrContextFactory { /** * Create a JNDI naming context from the supplied connection properties. * * @param properties the connection properties; may be null or empty * @return the naming context; may not be null * @throws NamingException if there is a problem creating or obtaining the naming context */ Context createContext( Properties properties ) throws NamingException; } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy