org.firebirdsql.management.FBServiceManager 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
/*
* Firebird Open Source JDBC Driver
*
* Distributable under LGPL license.
* You may obtain a copy of the License at http://www.gnu.org/copyleft/lgpl.html
*
* 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
* LGPL License for more details.
*
* This file was created by members of the firebird development team.
* All individual contributions remain the Copyright (C) of those
* individuals. Contributors to this file are either listed here or
* can be obtained from a source control history command.
*
* All rights reserved.
*/
package org.firebirdsql.management;
import org.firebirdsql.gds.ServiceRequestBuffer;
import org.firebirdsql.gds.impl.DbAttachInfo;
import org.firebirdsql.gds.impl.GDSFactory;
import org.firebirdsql.gds.impl.GDSServerVersion;
import org.firebirdsql.gds.impl.GDSType;
import org.firebirdsql.gds.ng.*;
import org.firebirdsql.jaybird.props.PropertyConstants;
import org.firebirdsql.jaybird.props.PropertyNames;
import org.firebirdsql.jaybird.props.def.ConnectionProperty;
import java.io.IOException;
import java.io.OutputStream;
import java.sql.SQLException;
import java.util.Map;
import static org.firebirdsql.gds.ISCConstants.isc_info_end;
import static org.firebirdsql.gds.ISCConstants.isc_info_svc_to_eof;
import static org.firebirdsql.gds.ISCConstants.isc_info_truncated;
import static org.firebirdsql.gds.ISCConstants.isc_infunk;
import static org.firebirdsql.jaybird.fb.constants.SpbItems.isc_spb_dbname;
import static org.firebirdsql.jaybird.fb.constants.SpbItems.isc_spb_options;
import static org.firebirdsql.gds.VaxEncoding.iscVaxInteger2;
/**
* An implementation of the basic Firebird Service API functionality.
*
* @author Roman Rokytskyy
* @author Mark Rotteveel
*/
public class FBServiceManager implements ServiceManager {
private final IServiceProperties serviceProperties = new FbServiceProperties();
private final FbDatabaseFactory dbFactory;
private String database;
private OutputStream logger;
public static final int BUFFER_SIZE = 1024; //1K
/**
* Create a new instance of FBServiceManager
based on
* the default GDSType.
*/
public FBServiceManager() {
this(GDSFactory.getDefaultGDSType());
}
/**
* Create a new instance of FBServiceManager
based on
* a given GDSType.
*
* @param gdsType
* type must be PURE_JAVA, EMBEDDED, or NATIVE
*/
public FBServiceManager(String gdsType) {
this(GDSType.getType(gdsType));
}
/**
* Create a new instance of FBServiceManager
based on
* a given GDSType.
*
* @param gdsType
* The GDS implementation type to use
*/
public FBServiceManager(GDSType gdsType) {
serviceProperties.setType(gdsType.toString());
dbFactory = GDSFactory.getDatabaseFactoryForType(gdsType);
}
@Override
public final void setType(String type) {
throw new IllegalStateException("Type must be specified on construction");
}
// NOTE: we're redirecting the default implementations of the interface here to ensure the
// service manager can be introspected as a JavaBean (default methods are not returned by the introspector)
@Override
public void setCharSet(String charSet) {
ServiceManager.super.setCharSet(charSet);
}
@Override
public String getCharSet() {
return ServiceManager.super.getCharSet();
}
@Override
public void setUser(String user) {
ServiceManager.super.setUser(user);
}
@Override
public String getUser() {
return ServiceManager.super.getUser();
}
@Override
public void setPassword(String password) {
ServiceManager.super.setPassword(password);
}
@Override
public String getPassword() {
return ServiceManager.super.getPassword();
}
@Override
public String getServerName() {
return ServiceManager.super.getServerName();
}
@Override
public void setServerName(String serverName) {
ServiceManager.super.setServerName(serverName);
}
@Override
public int getPortNumber() {
return ServiceManager.super.getPortNumber();
}
@Override
public void setPortNumber(int portNumber) {
ServiceManager.super.setPortNumber(portNumber);
}
@Override
public String getServiceName() {
return ServiceManager.super.getServiceName();
}
@Override
public void setServiceName(String serviceName) {
ServiceManager.super.setServiceName(serviceName);
}
/**
* {@inheritDoc}
*
* NOTE: The {@link #setDatabase(String)} property will also set this property, so in general this property doesn't
* need to be set explicitly.
*
*/
@Override
public void setExpectedDb(String expectedDb) {
ServiceManager.super.setExpectedDb(expectedDb);
}
@Override
public String getExpectedDb() {
return ServiceManager.super.getExpectedDb();
}
@Override
public void setDatabase(String database) {
this.database = database;
setExpectedDb(database);
}
@Override
public String getDatabase() {
return database;
}
@SuppressWarnings("removal")
@Override
@Deprecated(since = "5", forRemoval = true)
public String getHost() {
return getServerName();
}
@SuppressWarnings("removal")
@Override
@Deprecated(since = "5", forRemoval = true)
public void setHost(String host) {
setServerName(host);
}
@SuppressWarnings("removal")
@Override
@Deprecated(since = "5", forRemoval = true)
public int getPort() {
return getPortNumber();
}
@SuppressWarnings("removal")
@Override
@Deprecated(since = "5", forRemoval = true)
public void setPort(int port) {
setPortNumber(port);
}
@Override
public String getWireCrypt() {
return ServiceManager.super.getWireCrypt();
}
@Override
public WireCrypt getWireCryptAsEnum() {
return serviceProperties.getWireCryptAsEnum();
}
@Override
public void setWireCryptAsEnum(WireCrypt wireCrypt) {
serviceProperties.setWireCryptAsEnum(wireCrypt);
}
@Override
public String getDbCryptConfig() {
return ServiceManager.super.getDbCryptConfig();
}
@Override
public void setDbCryptConfig(String dbCryptConfig) {
ServiceManager.super.setDbCryptConfig(dbCryptConfig);
}
@Override
public String getAuthPlugins() {
return ServiceManager.super.getDbCryptConfig();
}
@Override
public void setAuthPlugins(String authPlugins) {
ServiceManager.super.setAuthPlugins(authPlugins);
}
@Override
public boolean isWireCompression() {
return ServiceManager.super.isWireCompression();
}
@Override
public void setWireCompression(boolean wireCompression) {
ServiceManager.super.setWireCompression(wireCompression);
}
@Override
public String getProperty(String name) {
return serviceProperties.getProperty(name);
}
@Override
public void setProperty(String name, String value) {
if (PropertyNames.type.equals(name)) {
// Triggers exception
setType(value);
}
serviceProperties.setProperty(name, value);
}
@Override
public Integer getIntProperty(String name) {
return serviceProperties.getIntProperty(name);
}
@Override
public void setIntProperty(String name, Integer value) {
serviceProperties.setIntProperty(name, value);
}
@Override
public Boolean getBooleanProperty(String name) {
return serviceProperties.getBooleanProperty(name);
}
@Override
public void setBooleanProperty(String name, Boolean value) {
serviceProperties.setBooleanProperty(name, value);
}
@Override
public Map connectionPropertyValues() {
return serviceProperties.connectionPropertyValues();
}
/**
* @return Returns the out.
*/
public synchronized OutputStream getLogger() {
return logger;
}
/**
* @param logger
* The out to set.
*/
public synchronized void setLogger(OutputStream logger) {
this.logger = logger;
}
public FbService attachServiceManager() throws SQLException {
FbService fbService = dbFactory.serviceConnect(serviceProperties);
fbService.attach();
return fbService;
}
protected FbDatabase attachDatabase() throws SQLException {
if (database == null) {
throw new SQLException("Property database needs to be set.");
}
FbConnectionProperties connectionProperties = new FbConnectionProperties();
createDatabaseAttachInfo().copyTo(connectionProperties);
connectionProperties.setUser(serviceProperties.getUser());
connectionProperties.setPassword(serviceProperties.getPassword());
connectionProperties.setRoleName(serviceProperties.getRoleName());
connectionProperties.setEnableProtocol(serviceProperties.getEnableProtocol());
connectionProperties.setAuthPlugins(serviceProperties.getAuthPlugins());
connectionProperties.setWireCrypt(serviceProperties.getWireCrypt());
FbDatabase fbDatabase = dbFactory.connect(connectionProperties);
fbDatabase.attach();
return fbDatabase;
}
private DbAttachInfo createDatabaseAttachInfo() {
// NOTE: If it turns out we need to tweak this for specific protocol implementations, this may need to move to
// FbDatabaseFactory (e.g. as a default method to be overridden by implementations)
final String serverName = serviceProperties.getServerName();
final String serviceAttachObjectName = serviceProperties.getAttachObjectName();
if (serverName != null || serviceAttachObjectName == null || serviceAttachObjectName.isEmpty()) {
return new DbAttachInfo(serverName, serviceProperties.getPortNumber(), database);
}
String databaseAttachObjectName;
if (serviceAttachObjectName.equals(PropertyConstants.DEFAULT_SERVICE_NAME)) {
databaseAttachObjectName = database;
} else if (serviceAttachObjectName.endsWith(PropertyConstants.DEFAULT_SERVICE_NAME)) {
databaseAttachObjectName = serviceAttachObjectName
.substring(0, serviceAttachObjectName.length() - 11 /* service_mgr */) + database;
} else if (serviceAttachObjectName.endsWith("/") || serviceAttachObjectName.endsWith(":")) {
// e.g. //localhost/ or inet://localhost/ or localhost:
databaseAttachObjectName = serviceAttachObjectName + database;
} else {
// e.g. //localhost or inet://localhost
// NOTE: This is probably the most error-prone conversion, but making this more robust increases complexity
// significantly, so instead we'll just let Firebird fail if this guess is wrong
databaseAttachObjectName = serviceAttachObjectName + '/' + database;
}
return new DbAttachInfo(null, serviceProperties.getPortNumber(), databaseAttachObjectName);
}
public void queueService(FbService service) throws SQLException, IOException {
OutputStream currentLogger = getLogger();
ServiceRequestBuffer infoSRB = service.createServiceRequestBuffer();
infoSRB.addArgument(isc_info_svc_to_eof);
int bufferSize = BUFFER_SIZE;
boolean processing = true;
while (processing) {
byte[] buffer = service.getServiceInfo(null, infoSRB, bufferSize);
switch (buffer[0]) {
case isc_info_svc_to_eof -> {
int dataLength = iscVaxInteger2(buffer, 1);
if (dataLength == 0) {
if (buffer[3] != isc_info_end) {
throw new SQLException("Unexpected end of stream reached.");
}
processing = false;
} else if (currentLogger != null) {
currentLogger.write(buffer, 3, dataLength);
}
}
case isc_info_truncated -> bufferSize = bufferSize * 2;
case isc_info_end -> processing = false;
default -> throw FbExceptionBuilder.toException(isc_infunk);
}
}
}
/**
* Execute a Services API operation in the database. All output from the operation is sent to
* this {@code ServiceManager}'s logger.
*
* @param service
* service instance to execute on
* @param srb
* The buffer containing the task request
* @throws SQLException
* if a database access error occurs or incorrect parameters are supplied
*/
protected final void executeServicesOperation(FbService service, ServiceRequestBuffer srb) throws SQLException {
try {
service.startServiceAction(srb);
queueService(service);
} catch (IOException ioe) {
throw new SQLException(ioe);
}
}
protected ServiceRequestBuffer createRequestBuffer(FbService service, int operation, int options) {
ServiceRequestBuffer srb = service.createServiceRequestBuffer();
srb.addArgument(operation);
if (getDatabase() != null) {
srb.addArgument(isc_spb_dbname, getDatabase());
}
srb.addArgument(isc_spb_options, options);
return srb;
}
@Override
public GDSServerVersion getServerVersion() throws SQLException {
try (FbService service = attachServiceManager()) {
return service.getServerVersion();
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy