org.firebirdsql.gds.impl.GDSFactory Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of jaybird Show documentation
Show all versions of jaybird Show documentation
JDBC Driver for the Firebird RDBMS
/*
* Public Firebird Java API.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions are met:
* 1. Redistributions of source code must retain the above copyright notice,
* this list of conditions and the following disclaimer.
* 2. Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in the
* documentation and/or other materials provided with the distribution.
* 3. The name of the author may not be used to endorse or promote products
* derived from this software without specific prior written permission.
*
* THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR IMPLIED
* WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
* MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO
* EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
* SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
* PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS;
* OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY,
* WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR
* OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF
* ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*/
package org.firebirdsql.gds.impl;
import org.firebirdsql.gds.impl.wire.WireGDSFactoryPlugin;
import org.firebirdsql.gds.ng.FbDatabaseFactory;
import org.firebirdsql.jaybird.props.DatabaseConnectionProperties;
import org.firebirdsql.jaybird.util.PluginLoader;
import org.firebirdsql.util.InternalApi;
import java.sql.SQLException;
import java.util.*;
import java.util.Map.Entry;
/**
* The class {@code GDSFactory} exists to provide a way to obtain objects
* implementing GDS and Clumplet.
*
* @author David Jencks
* @author Mark Rotteveel
*/
public final class GDSFactory {
private static final Set registeredPlugins = new HashSet<>();
private static final Map typeToPluginMap = new HashMap<>();
/**
* This sorting effectively puts the shortest JDBC URLs at the end of the map,
* so the correct default protocol handling can be implemented.
*/
private static final TreeMap jdbcUrlToPluginMap = new TreeMap<>(Comparator.reverseOrder());
private static GDSType defaultType;
static {
PluginLoader.findPlugins(GDSFactoryPlugin.class, List.of("org.firebirdsql.gds.impl.wire.WireGDSFactoryPlugin"))
.forEach(GDSFactory::registerPlugin);
GDSType pureJavaType = GDSType.getType(WireGDSFactoryPlugin.PURE_JAVA_TYPE_NAME);
if (pureJavaType != null && defaultType != pureJavaType && typeToPluginMap.containsKey(pureJavaType)) {
// ensure defaultType is PURE_JAVA if that plugin was registered
defaultType = pureJavaType;
}
}
private GDSFactory() {
// no instances
}
/**
* Register plugin for this factory. Usually there is no need to register
* plugins, since this happens automatically during initialization of this
* class. However, there might be a situation when automatic plugin
* registration does not work.
*
* @param plugin
* instance of {@link GDSFactoryPlugin} to register.
*/
public static void registerPlugin(GDSFactoryPlugin plugin) {
boolean newPlugin = registeredPlugins.add(plugin);
if (!newPlugin)
return;
GDSType type = GDSType.registerType(plugin.getTypeName());
typeToPluginMap.put(type, plugin);
// set the default type (see also the static initializer which ensures PURE_JAVA will be default if available)
if (defaultType == null) defaultType = type;
// register aliases
for (String alias : plugin.getTypeAliasList()) {
GDSType aliasType = GDSType.registerType(alias);
typeToPluginMap.put(aliasType, plugin);
}
for (String jdbcUrl : plugin.getSupportedProtocolList()) {
GDSFactoryPlugin otherPlugin = jdbcUrlToPluginMap.put(jdbcUrl, plugin);
if (otherPlugin != null && !otherPlugin.equals(plugin))
throw new IllegalArgumentException(
"Duplicate JDBC URL pattern detected: URL %s, plugin %s, other plugin %s".formatted(
jdbcUrl, plugin.getTypeName(), otherPlugin.getTypeName()));
}
}
/**
* Get default GDS type.
*
* @return instance of {@link GDSType}.
*/
public static GDSType getDefaultGDSType() {
return defaultType;
}
public static FbDatabaseFactory getDatabaseFactoryForType(GDSType gdsType) {
if (gdsType == null) gdsType = defaultType;
return getPlugin(gdsType).getDatabaseFactory();
}
/**
* Get connection string for the specified server name, port and database
* name/path. This method delegates call to the factory plugin corresponding
* to the specified type.
*
* @param gdsType
* instance of {@link GDSType} for which connection string should be returned.
* @param server
* name or IP address of the database server, applies only to IPC and TCP connection modes, in other cases
* should be {@code null}.
* @param port
* port on which database server opened listening socket, applies to TCP connection mode only, may be
* {@code null}.
* @param path
* database name or path to the database
* @return full connection string
* @throws SQLException
* if connection string cannot be obtained.
*/
public static String getDatabasePath(GDSType gdsType, String server, Integer port, String path)
throws SQLException {
return getPlugin(gdsType).getDatabasePath(server, port, path);
}
/**
* Get path to the database from the specified JDBC URL. This method finds
* the appropriate plugin and delegates the call to it. Plugin is
* responsible for the call execution.
*
* @param gdsType
* type of the plugin, to which operation will be delegated to.
* @param jdbcUrl
* JDBC url from which the database path must be extracted.
* @return path to the database specified in the JDBC URL.
* @throws SQLException
* error when database path cannot be extracted.
*/
public static String getDatabasePath(GDSType gdsType, String jdbcUrl) throws SQLException {
return getPlugin(gdsType).getDatabasePath(jdbcUrl);
}
/**
* Get collection of the supported JDBC protocols.
*
* @return set of the supported protocols.
*/
public static Set getSupportedProtocols() {
return Collections.unmodifiableSet(jdbcUrlToPluginMap.keySet());
}
/**
* Create JDBC URL for the specified GDS type and database connection properties.
*
* @param gdsType
* type of the plugin, to which operation will be delegated to.
* @param dbConnectionProperties
* Database connection properties
* @return newly created JDBC URL
* @throws SQLException When required information is missing to build the URL
*/
public static String getJdbcUrl(GDSType gdsType, DatabaseConnectionProperties dbConnectionProperties)
throws SQLException {
DbAttachInfo dbAttachInfo = DbAttachInfo.of(dbConnectionProperties);
GDSFactoryPlugin plugin = getPlugin(gdsType);
return plugin.getDefaultProtocol() + plugin.getDatabasePath(dbAttachInfo);
}
/**
* Get GDS type for the specified JDBC URL. This method finds the plugin
* corresponding to the specified type and delegates the call to it.
*
* @param jdbcUrl
* JDBC URL for which GDS type should be obtained.
* @return instance of {@link GDSType}.
*/
public static GDSType getTypeForProtocol(String jdbcUrl) {
for (Entry entry : jdbcUrlToPluginMap.entrySet()) {
String jdbcProtocol = entry.getKey();
if (jdbcUrl.startsWith(jdbcProtocol)) {
return GDSType.getType(entry.getValue().getTypeName());
}
}
return null;
}
/**
* Get class extending the {@link org.firebirdsql.jdbc.FBConnection}
* that will be instantiated when new connection is created. This method
* finds the plugin for the specified type and delegates the call to it.
*
* @param gdsType
* instance of {@link GDSType}
* @return class to instantiate for the database connection.
*/
public static Class> getConnectionClass(GDSType gdsType) {
return getPlugin(gdsType).getConnectionClass();
}
/**
* Get plugin for the specified GDS type.
*
* @param gdsType
* GDS type.
* @return instance of {@link GDSFactoryPlugin}
* @throws IllegalArgumentException
* if specified type is not known.
*/
@InternalApi
public static GDSFactoryPlugin getPlugin(GDSType gdsType) {
GDSFactoryPlugin gdsPlugin = typeToPluginMap.get(gdsType);
if (gdsPlugin == null) {
throw new IllegalArgumentException("Specified GDS type " + gdsType + " is unknown.");
}
return gdsPlugin;
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy