org.firebirdsql.gds.ng.wire.WireConnection 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.gds.ng.wire;
import org.firebirdsql.encodings.EncodingFactory;
import org.firebirdsql.encodings.IEncodingFactory;
import org.firebirdsql.gds.ClumpletReader;
import org.firebirdsql.gds.ISCConstants;
import org.firebirdsql.gds.JaybirdErrorCodes;
import org.firebirdsql.gds.VaxEncoding;
import org.firebirdsql.gds.impl.DbAttachInfo;
import org.firebirdsql.gds.impl.wire.WireProtocolConstants;
import org.firebirdsql.gds.impl.wire.XdrInputStream;
import org.firebirdsql.gds.impl.wire.XdrOutputStream;
import org.firebirdsql.gds.ng.AbstractConnection;
import org.firebirdsql.gds.ng.FbExceptionBuilder;
import org.firebirdsql.gds.ng.IAttachProperties;
import org.firebirdsql.gds.ng.LockCloseable;
import org.firebirdsql.gds.ng.WarningMessageCallback;
import org.firebirdsql.gds.ng.dbcrypt.DbCryptCallback;
import org.firebirdsql.gds.ng.wire.auth.ClientAuthBlock;
import org.firebirdsql.gds.ng.wire.crypt.KnownServerKey;
import org.firebirdsql.jaybird.props.def.ConnectionProperty;
import org.firebirdsql.jaybird.util.ByteArrayHelper;
import javax.net.SocketFactory;
import java.io.ByteArrayOutputStream;
import java.io.Closeable;
import java.io.IOException;
import java.lang.reflect.Constructor;
import java.net.*;
import java.nio.charset.StandardCharsets;
import java.security.AccessController;
import java.security.PrivilegedAction;
import java.sql.SQLException;
import java.sql.SQLTimeoutException;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Collections;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.Optional;
import java.util.Properties;
import java.util.concurrent.TimeUnit;
import static java.lang.System.Logger.Level.DEBUG;
import static org.firebirdsql.gds.impl.wire.WireProtocolConstants.*;
/**
* Class managing the TCP/IP connection and initial handshaking with the Firebird server.
*
* @param
* Type of attach properties
* @param
* Type of connection handle
* @author Mark Rotteveel
* @since 3.0
*/
public abstract class WireConnection, C extends FbWireAttachment>
extends AbstractConnection implements Closeable {
// TODO Check if methods currently throwing IOException should throw SQLException instead
private static final System.Logger log = System.getLogger(WireConnection.class.getName());
private static final String REJECTION_POSSIBLE_REASON =
"The server and client could not agree on connection options. A possible reason is attempting to connect "
+ "to an unsupported Firebird version; see the documentation of connection property 'enableProtocol' for "
+ "a workaround.";
private static final WarningMessageCallback NOOP_WARNING_MESSAGE_CALLBACK = warning -> {};
// Micro-optimization: we usually expect at most 3 (Firebird 5)
private final List knownServerKeys = new ArrayList<>(3);
private final DbAttachInfo dbAttachInfo;
private ClientAuthBlock clientAuthBlock;
private Socket socket;
private ProtocolCollection protocols;
private int protocolVersion;
private int protocolArchitecture;
private int protocolMinimumType;
private XdrOutputStream xdrOut;
private XdrInputStream xdrIn;
private final XdrStreamAccess streamAccess = new XdrStreamAccess() {
@Override
public XdrInputStream getXdrIn() throws SQLException {
if (isConnected() && xdrIn != null) {
return xdrIn;
} else {
throw FbExceptionBuilder.connectionClosed();
}
}
@Override
public XdrOutputStream getXdrOut() throws SQLException {
if (isConnected() && xdrOut != null) {
return xdrOut;
} else {
throw FbExceptionBuilder.connectionClosed();
}
}
};
/**
* Creates a WireConnection (without establishing a connection to the
* server) with the default protocol collection.
*
* @param attachProperties
* Attach properties
*/
protected WireConnection(T attachProperties) throws SQLException {
this(attachProperties, EncodingFactory.getPlatformDefault(),
ProtocolCollection.getProtocols(attachProperties.getEnableProtocol()));
}
/**
* Creates a WireConnection (without establishing a connection to the
* server).
*
* @param attachProperties
* Attach properties
* @param encodingFactory
* Factory for encoding definitions
* @param protocols
* The collection of protocols to use for this connection.
*/
protected WireConnection(T attachProperties, IEncodingFactory encodingFactory,
ProtocolCollection protocols) throws SQLException {
super(attachProperties, encodingFactory);
this.protocols = protocols;
clientAuthBlock = new ClientAuthBlock(this.attachProperties);
dbAttachInfo = toDbAttachInfo(attachProperties);
}
// Allow access to withLock() at package level, without making it public in parent class
final LockCloseable withLockProxy() {
return withLock();
}
public final String getServerName() {
return dbAttachInfo.serverName();
}
public final int getPortNumber() {
return dbAttachInfo.portNumber();
}
/**
* @return The file name to use in the p_cnct_file of the op_connect request
*/
protected String getCnctFile() {
return getAttachObjectName();
}
public final String getAttachObjectName() {
return dbAttachInfo.attachObjectName();
}
protected abstract DbAttachInfo toDbAttachInfo(T attachProperties) throws SQLException;
public final boolean isConnected() {
return !(socket == null || socket.isClosed());
}
public final int getProtocolVersion() {
return protocolVersion;
}
public final int getProtocolArchitecture() {
return protocolArchitecture;
}
public final int getProtocolMinimumType() {
return protocolMinimumType;
}
public final ClientAuthBlock getClientAuthBlock() {
return clientAuthBlock;
}
/**
* Sets the socket blocking timeout (SO_TIMEOUT) of the socket.
*
* This method can also be called if a connection is established
*
*
* @param socketTimeout
* Value of the socket timeout (in milliseconds)
* @throws SQLException
* If the timeout value cannot be changed
*/
public final void setSoTimeout(int socketTimeout) throws SQLException {
attachProperties.setSoTimeout(socketTimeout);
resetSocketTimeout();
}
/**
* Resets the socket timeout to the configured socketTimeout. Does nothing
* if currently not connected.
*
* @throws SQLException
* If the timeout value cannot be changed
*/
public final void resetSocketTimeout() throws SQLException {
if (isConnected()) {
try {
final int soTimeout = attachProperties.getSoTimeout();
final int desiredTimeout = soTimeout != -1 ? soTimeout : 0;
if (socket.getSoTimeout() != desiredTimeout) {
socket.setSoTimeout(desiredTimeout);
}
} catch (SocketException e) {
throw FbExceptionBuilder.forException(JaybirdErrorCodes.jb_couldNotChangeSoTimeout).cause(e)
.toSQLException();
}
}
}
/**
* Establishes the TCP/IP connection to serverName and portNumber of this connection.
*
* @throws SQLTimeoutException
* if the connection cannot be established within the connect timeout (either explicitly set or implied by
* the OS timeout of the socket)
* @throws SQLException
* if the connection cannot be established.
*/
public final void socketConnect() throws SQLException {
try {
socket = createSocket();
socket.setTcpNoDelay(true);
final int connectTimeout = attachProperties.getConnectTimeout();
// connectTimeout is in seconds, need milliseconds, lower bound 0 (indefinite, for overflow or not set)
final int socketConnectTimeout = Math.max(0, (int) TimeUnit.SECONDS.toMillis(connectTimeout));
if (socketConnectTimeout != 0) {
// Blocking timeout initially identical to connect timeout
socket.setSoTimeout(socketConnectTimeout);
} else {
// Blocking timeout to normal socket timeout, 0 if not set
socket.setSoTimeout(Math.max(attachProperties.getSoTimeout(), 0));
}
final int socketBufferSize = attachProperties.getSocketBufferSize();
if (socketBufferSize != IAttachProperties.DEFAULT_SOCKET_BUFFER_SIZE) {
socket.setReceiveBufferSize(socketBufferSize);
socket.setSendBufferSize(socketBufferSize);
}
socket.connect(new InetSocketAddress(getServerName(), getPortNumber()), socketConnectTimeout);
} catch (SocketTimeoutException ste) {
throw FbExceptionBuilder.forTimeoutException(ISCConstants.isc_network_error)
.messageParameter(getServerName())
.cause(ste)
.toSQLException();
} catch (IOException ioex) {
throw FbExceptionBuilder.forNonTransientConnectionException(ISCConstants.isc_network_error)
.messageParameter(getServerName())
.cause(ioex)
.toSQLException();
}
}
private Socket createSocket() throws IOException, SQLException {
try {
return createSocketFactory().createSocket();
} catch (RuntimeException e) {
throw FbExceptionBuilder
.forNonTransientConnectionException(JaybirdErrorCodes.jb_socketFactoryFailedToCreateSocket)
.messageParameter(attachProperties.getSocketFactory())
.cause(e)
.toSQLException();
}
}
private SocketFactory createSocketFactory() throws SQLException {
String socketFactoryName = attachProperties.getSocketFactory();
if (socketFactoryName == null) {
return SocketFactory.getDefault();
}
return createSocketFactory0(socketFactoryName);
}
private SocketFactory createSocketFactory0(String socketFactoryName) throws SQLException {
log.log(DEBUG, "Attempting to create custom socket factory {0}", socketFactoryName);
try {
Class extends SocketFactory> socketFactoryClass =
Class.forName(socketFactoryName).asSubclass(SocketFactory.class);
try {
Constructor extends SocketFactory> propsConstructor =
socketFactoryClass.getConstructor(Properties.class);
return propsConstructor.newInstance(getSocketFactoryProperties());
} catch (ReflectiveOperationException e) {
log.log(DEBUG, socketFactoryName
+ "has no Properties constructor, or constructor execution resulted in an exception", e);
}
try {
Constructor extends SocketFactory> noArgConstructor = socketFactoryClass.getConstructor();
return noArgConstructor.newInstance();
} catch (ReflectiveOperationException e) {
log.log(DEBUG, socketFactoryName
+ "has no no-arg constructor, or constructor execution resulted in an exception", e);
}
throw FbExceptionBuilder
.forNonTransientConnectionException(JaybirdErrorCodes.jb_socketFactoryConstructorNotFound)
.messageParameter(socketFactoryName)
.toSQLException();
} catch (ClassNotFoundException | ClassCastException e) {
throw FbExceptionBuilder.forNonTransientConnectionException(JaybirdErrorCodes.jb_socketFactoryClassNotFound)
.messageParameter(socketFactoryName)
.cause(e)
.toSQLException();
}
}
private Properties getSocketFactoryProperties() {
var props = new Properties();
attachProperties.connectionPropertyValues().entrySet().stream()
.filter(e ->
e.getValue() != null && e.getKey().name().endsWith("@socketFactory"))
.forEach(e -> {
ConnectionProperty connectionProperty = e.getKey();
props.setProperty(connectionProperty.name(), connectionProperty.type().asString(e.getValue()));
});
return props;
}
public final XdrStreamAccess getXdrStreamAccess() {
return streamAccess;
}
/**
* Performs the connection identification phase of the Wire protocol and
* returns the FbWireDatabase implementation for the agreed protocol.
*
* @return FbWireDatabase
*/
@Override
public final C identify() throws SQLException {
try {
xdrIn = new XdrInputStream(socket.getInputStream());
xdrOut = new XdrOutputStream(socket.getOutputStream());
sendConnectAttach(xdrOut);
int operation = handleCryptKeyCallbackBeforeAttachResponse();
if (operation == op_accept || operation == op_cond_accept || operation == op_accept_data) {
return handleConnectAttachAccept(xdrIn, operation);
} else {
throw handleConnectAttachReject(operation);
}
} catch (SocketTimeoutException ste) {
throw FbExceptionBuilder.forTimeoutException(ISCConstants.isc_network_error)
.messageParameter(getServerName()).cause(ste).toSQLException();
} catch (IOException ioex) {
throw FbExceptionBuilder.forException(ISCConstants.isc_network_error)
.messageParameter(getServerName()).cause(ioex).toSQLException();
}
}
private void sendConnectAttach(XdrOutputStream xdrOut) throws IOException, SQLException {
xdrOut.writeInt(op_connect);
xdrOut.writeInt(op_attach); // p_cnct_operation
xdrOut.writeInt(CONNECT_VERSION3); // p_cnct_cversion
xdrOut.writeInt(arch_generic); // p_cnct_client
xdrOut.writeString(getCnctFile(), getEncoding()); // p_cnct_file
xdrOut.writeInt(protocols.getProtocolCount()); // Count of protocols understood
xdrOut.writeBuffer(createUserIdentificationBlock());
for (ProtocolDescriptor protocol : protocols) {
writeProtocolDescriptor(xdrOut, protocol);
}
xdrOut.flush();
}
private void writeProtocolDescriptor(XdrOutputStream xdrOut, ProtocolDescriptor protocol) throws IOException {
xdrOut.writeInt(protocol.getVersion()); // Protocol version
xdrOut.writeInt(protocol.getArchitecture()); // Architecture of client
xdrOut.writeInt(protocol.getMinimumType()); // Minimum type
if (protocol.supportsWireCompression() && attachProperties.isWireCompression()) {
xdrOut.writeInt(protocol.getMaximumType() | pflag_compress);
} else {
xdrOut.writeInt(protocol.getMaximumType()); // Maximum type
}
xdrOut.writeInt(protocol.getWeight()); // Preference weight
}
private int handleCryptKeyCallbackBeforeAttachResponse() throws IOException, SQLException {
int operation = readNextOperation();
FbWireOperations cryptKeyCallbackWireOperations = null;
DbCryptCallback dbCryptCallback = null;
while (operation == op_crypt_key_callback) {
if (cryptKeyCallbackWireOperations == null) {
cryptKeyCallbackWireOperations = getCryptKeyCallbackWireOperations();
}
if (dbCryptCallback == null) {
dbCryptCallback = createDbCryptCallback();
}
cryptKeyCallbackWireOperations.handleCryptKeyCallback(dbCryptCallback);
operation = readNextOperation();
}
return operation;
}
private C handleConnectAttachAccept(XdrInputStream xdrIn, int operation) throws IOException, SQLException {
var acceptPacket = new FbWireAttachment.AcceptPacket();
acceptPacket.operation = operation;
protocolVersion = xdrIn.readInt(); // p_acpt_version - Protocol version
protocolArchitecture = xdrIn.readInt(); // p_acpt_architecture - Architecture for protocol
int acceptType = xdrIn.readInt(); // p_acpt_type - Minimum type
protocolMinimumType = acceptType & ptype_MASK;
final boolean compress = (acceptType & pflag_compress) != 0;
if (protocolVersion < 0) {
protocolVersion = (protocolVersion & FB_PROTOCOL_MASK) | FB_PROTOCOL_FLAG;
}
if (operation == op_cond_accept || operation == op_accept_data) {
byte[] data = acceptPacket.p_acpt_data = xdrIn.readBuffer();
acceptPacket.p_acpt_plugin = xdrIn.readString(getEncoding());
final boolean authComplete = xdrIn.readInt() == 1;
byte[] serverKeys = acceptPacket.p_acpt_keys = xdrIn.readBuffer();
clientAuthBlock.setServerData(data);
clientAuthBlock.setAuthComplete(authComplete);
addServerKeys(serverKeys);
clientAuthBlock.resetClient(serverKeys);
clientAuthBlock.switchPlugin(acceptPacket.p_acpt_plugin);
} else {
clientAuthBlock.resetClient(null);
}
if (compress) {
xdrOut.enableCompression();
xdrIn.enableDecompression();
}
ProtocolDescriptor descriptor = protocols.getProtocolDescriptor(protocolVersion);
if (descriptor == null) {
throw new SQLException(String.format(
"Unsupported or unexpected protocol version %d connecting to database %s. Supported version(s): %s",
protocolVersion, getServerName(), protocols.getProtocolVersions()));
}
C connectionHandle = createConnectionHandle(descriptor);
if (operation == op_cond_accept) {
connectionHandle.authReceiveResponse(acceptPacket);
}
return connectionHandle;
}
private SQLException handleConnectAttachReject(int operation) throws IOException {
try {
if (operation == op_response) {
// Handle exception from response
AbstractWireOperations wireOperations = getDefaultWireOperations();
Response response = wireOperations.processOperation(operation);
if (response instanceof GenericResponse genericResponse && genericResponse.exception() != null) {
return genericResponse.exception();
}
} else if (operation == op_reject) {
return FbExceptionBuilder.forException(ISCConstants.isc_connect_reject)
.messageParameter(REJECTION_POSSIBLE_REASON).toSQLException();
}
log.log(DEBUG, "Reached end of identify without error or connection, last operation: {0}", operation);
// If we reach here, authentication failed (or never authenticated for lack of username and password)
return FbExceptionBuilder.toException(ISCConstants.isc_login);
} catch (SQLException e) {
return e;
} finally {
try {
close();
} catch (Exception ex) {
log.log(DEBUG, "Ignoring exception on disconnect in connect phase of protocol", ex);
}
}
}
/**
* Clear authentication data.
*/
public final void clearAuthData() {
clientAuthBlock = null;
clearServerKeys();
}
private byte[] createUserIdentificationBlock() throws IOException, SQLException {
// Here we identify the user to the engine.
// This may or may not be used as login info to a database.
final byte[] userBytes = getSystemUserName().getBytes(StandardCharsets.UTF_8);
final byte[] hostBytes = getSystemHostName().getBytes(StandardCharsets.UTF_8);
ByteArrayOutputStream userId = new ByteArrayOutputStream();
clientAuthBlock.authenticateStep0();
clientAuthBlock.writePluginDataTo(userId);
userId.write(CNCT_client_crypt);
VaxEncoding.encodeVaxInteger(userId, attachProperties.getWireCryptAsEnum().getWireProtocolCryptLevel());
userId.write(CNCT_user);
int userLength = Math.min(userBytes.length, 255);
userId.write(userLength);
userId.write(userBytes, 0, userLength);
userId.write(CNCT_host);
int hostLength = Math.min(hostBytes.length, 255);
userId.write(hostLength);
userId.write(hostBytes, 0, hostLength);
userId.write(CNCT_user_verification);
userId.write(0);
return userId.toByteArray();
}
void addServerKeys(byte[] serverKeys) throws SQLException {
final var newKeys = new ClumpletReader(ClumpletReader.Kind.UnTagged, serverKeys);
for (newKeys.rewind(); !newKeys.isEof(); newKeys.moveNext()) {
addServerKey(newKeys);
}
}
private void addServerKey(ClumpletReader newKeys) throws SQLException {
int currentTag = newKeys.getClumpTag();
switch (currentTag) {
case TAG_KNOWN_PLUGINS -> {
// Nothing to do (yet)
}
case TAG_PLUGIN_SPECIFIC ->
// Nothing to do (yet)
log.log(DEBUG, "Possible implementation problem, found TAG_PLUGIN_SPECIFIC without TAG_KEY_TYPE");
case TAG_KEY_TYPE -> extractServerKey(newKeys).ifPresent(knownServerKeys::add);
default -> log.log(DEBUG, "Ignored unexpected tag type: {0}", currentTag);
}
}
private static Optional extractServerKey(ClumpletReader newKeys) throws SQLException {
String keyType = newKeys.getString(StandardCharsets.ISO_8859_1);
newKeys.moveNext();
if (newKeys.isEof()) return Optional.empty();
int currentTag = newKeys.getClumpTag();
if (currentTag != TAG_KEY_PLUGINS) {
throw new SQLException("Unexpected tag type: " + currentTag);
}
String keyPlugins = newKeys.getString(StandardCharsets.ISO_8859_1);
Map pluginSpecificData = null;
while (newKeys.directNext(TAG_PLUGIN_SPECIFIC)) {
byte[] data = newKeys.getBytes();
int sepIdx = ByteArrayHelper.indexOf(data, (byte) 0);
if (sepIdx > 0) {
String plugin = new String(data, 0, sepIdx, StandardCharsets.ISO_8859_1);
byte[] specificData = Arrays.copyOfRange(data, sepIdx + 1, data.length);
if (pluginSpecificData == null) {
pluginSpecificData = new HashMap<>();
}
pluginSpecificData.put(plugin, specificData);
}
}
return Optional.of(new KnownServerKey(keyType, keyPlugins, pluginSpecificData));
}
void clearServerKeys() {
knownServerKeys.forEach(KnownServerKey::clear);
knownServerKeys.clear();
}
private AbstractWireOperations getDefaultWireOperations() {
ProtocolDescriptor protocolDescriptor = protocols
.getProtocolDescriptor(WireProtocolConstants.PROTOCOL_VERSION10);
return (AbstractWireOperations) protocolDescriptor.createWireOperations(this, NOOP_WARNING_MESSAGE_CALLBACK);
}
/**
* @return Instance of FbWireOperations that can read crypt key callbacks (in practice: v15).
*/
private FbWireOperations getCryptKeyCallbackWireOperations() {
ProtocolDescriptor protocolDescriptor = protocols
.getProtocolDescriptor(WireProtocolConstants.PROTOCOL_VERSION15);
return protocolDescriptor.createWireOperations(this, NOOP_WARNING_MESSAGE_CALLBACK);
}
/**
* Creates the connection handle for this type of connection.
*
* @param protocolDescriptor
* The protocol descriptor selected by the identify phase
* @return Connection handle
*/
protected abstract C createConnectionHandle(ProtocolDescriptor protocolDescriptor);
/**
* Reads the next operation code. Skips all {@link org.firebirdsql.gds.impl.wire.WireProtocolConstants#op_dummy}
* codes received.
*
* @return Operation code
* @throws IOException
* if an error occurs while reading from the underlying InputStream
*/
public final int readNextOperation() throws IOException {
int op;
do {
op = xdrIn.readInt();
} while (op == op_dummy);
return op;
}
/**
* Closes the TCP/IP connection. This is not a normal detach operation.
*
* @throws IOException
* if closing fails
*/
@SuppressWarnings("EmptyTryBlock")
public final void close() throws IOException {
try (var ignored1 = socket;
var ignored2 = xdrIn;
var ignored3 = xdrOut) {
// Ignored: Use try-with-resources to close
} finally {
xdrOut = null;
xdrIn = null;
socket = null;
protocols = null;
}
}
private static String getSystemUserName() {
try {
return getSystemPropertyPrivileged("user.name");
} catch (SecurityException ex) {
log.log(DEBUG, "Unable to retrieve user.name property", ex);
return "jaybird";
}
}
private static String getSystemHostName() {
try {
return InetAddress.getLocalHost().getHostName();
} catch (UnknownHostException ex) {
try {
return InetAddress.getLocalHost().getHostAddress();
} catch (UnknownHostException ex1) {
return "127.0.0.1";
}
}
}
@SuppressWarnings("SameParameterValue")
private static String getSystemPropertyPrivileged(final String propertyName) {
return AccessController.doPrivileged((PrivilegedAction) () -> System.getProperty(propertyName));
}
/**
* Writes directly to the {@code OutputStream} of the underlying socket.
*
* @param data
* Data to write
* @throws IOException
* If there is no socket, the socket is closed, or for errors writing to the socket.
*/
public final void writeDirect(byte[] data) throws IOException {
xdrOut.writeDirect(data);
}
final List getPluginSpecificData() {
if (knownServerKeys.isEmpty()) {
return Collections.emptyList();
}
List pluginSpecificData = new ArrayList<>();
for (KnownServerKey knownServerKey : knownServerKeys) {
pluginSpecificData.addAll(knownServerKey.getPluginSpecificData());
}
return pluginSpecificData;
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy