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

org.firebirdsql.gds.ng.wire.version10.V10Database Maven / Gradle / Ivy

There is a newer version: 6.0.0-beta-1
Show newest version
/*
 * Firebird Open Source JavaEE Connector - 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.version10;

import org.firebirdsql.encodings.Encoding;
import org.firebirdsql.gds.*;
import org.firebirdsql.gds.impl.DatabaseParameterBufferExtension;
import org.firebirdsql.gds.impl.wire.XdrOutputStream;
import org.firebirdsql.gds.ng.FbExceptionBuilder;
import org.firebirdsql.gds.ng.FbStatement;
import org.firebirdsql.gds.ng.FbTransaction;
import org.firebirdsql.gds.ng.TransactionState;
import org.firebirdsql.gds.ng.dbcrypt.DbCryptCallback;
import org.firebirdsql.gds.ng.fields.BlrCalculator;
import org.firebirdsql.gds.ng.wire.*;
import org.firebirdsql.jdbc.SQLStateConstants;
import org.firebirdsql.logging.Logger;
import org.firebirdsql.logging.LoggerFactory;

import java.io.ByteArrayOutputStream;
import java.io.IOException;
import java.nio.charset.Charset;
import java.sql.SQLException;
import java.sql.SQLFeatureNotSupportedException;
import java.sql.SQLNonTransientConnectionException;

import static org.firebirdsql.gds.impl.wire.WireProtocolConstants.*;
import static org.firebirdsql.gds.ng.TransactionHelper.checkTransactionActive;

/**
 * {@link FbWireDatabase} implementation for the version 10 wire protocol.
 *
 * @author Mark Rotteveel
 * @since 3.0
 */
public class V10Database extends AbstractFbWireDatabase implements FbWireDatabase {

    private static final Logger log = LoggerFactory.getLogger(V10Database.class);

    private int handle;
    private BlrCalculator blrCalculator;

    /**
     * Creates a V10Database instance.
     *
     * @param connection
     *         A WireConnection with an established connection to the server.
     * @param descriptor
     *         The ProtocolDescriptor that created this connection (this is
     *         used for creating further dependent objects).
     */
    protected V10Database(WireDatabaseConnection connection, ProtocolDescriptor descriptor) {
        super(connection, descriptor);
    }

    @Override
    public final int getHandle() {
        return handle;
    }

    @Override
    public final void attach() throws SQLException {
        try {
            final DatabaseParameterBuffer dpb = protocolDescriptor.createDatabaseParameterBuffer(connection);
            attachOrCreate(dpb, false);
        } catch (SQLException ex) {
            exceptionListenerDispatcher.errorOccurred(ex);
            throw ex;
        }
    }

    /**
     * @param dpb
     *         Database parameter buffer
     * @param create
     *         true create database, false only
     *         attach
     * @throws SQLException
     *         For errors during attach or create
     */
    protected final void attachOrCreate(DatabaseParameterBuffer dpb, boolean create) throws SQLException {
        checkConnected();
        if (isAttached()) {
            throw new SQLException("Already attached to a database");
        }
        synchronized (getSynchronizationObject()) {
            try {
                try {
                    sendAttachOrCreateToBuffer(dpb, create);
                    getXdrOut().flush();
                } catch (IOException e) {
                    throw new FbExceptionBuilder().exception(ISCConstants.isc_net_write_err).cause(e).toSQLException();
                }
                try {
                    authReceiveResponse(null);
                } catch (IOException e) {
                    throw new FbExceptionBuilder().exception(ISCConstants.isc_net_read_err).cause(e).toSQLException();
                }
            } catch (SQLException e) {
                safelyDetach();
                throw e;
            }
            setAttached();
            afterAttachActions();
        }
    }

    /**
     * Sends the buffer for op_attach or op_create
     *
     * @param dpb
     *         Database parameter buffer
     * @param create
     *         true create database, false only
     *         attach
     * @throws SQLException
     *         If the connection is not open
     * @throws IOException
     *         For errors writing to the connection
     */
    protected final void sendAttachOrCreateToBuffer(DatabaseParameterBuffer dpb, boolean create)
            throws SQLException, IOException {
        final int operation = create ? op_create : op_attach;
        final XdrOutputStream xdrOut = getXdrOut();

        final Encoding filenameEncoding = getFilenameEncoding(dpb);

        xdrOut.writeInt(operation);
        xdrOut.writeInt(0); // Database object ID
        xdrOut.writeString(connection.getAttachObjectName(), filenameEncoding);

        dpb = ((DatabaseParameterBufferExtension) dpb).removeExtensionParams();

        xdrOut.writeTyped(dpb);
    }

    /**
     * Gets the {@code Encoding} to use for the database filename.
     *
     * @param dpb
     *         Database parameter buffer
     * @return Encoding
     */
    protected Encoding getFilenameEncoding(DatabaseParameterBuffer dpb) {
        final String filenameCharset = dpb.getArgumentAsString(DatabaseParameterBufferExtension.FILENAME_CHARSET);
        if (filenameCharset != null) {
            return getEncodingFactory().getOrCreateEncodingForCharset(Charset.forName(filenameCharset));
        }
        return getEncoding();
    }

    /**
     * Processes the response from the server to the attach or create operation.
     *
     * @param genericResponse
     *         GenericResponse received from the server.
     */
    protected final void processAttachOrCreateResponse(GenericResponse genericResponse) {
        handle = genericResponse.getObjectHandle();
    }

    /**
     * Additional tasks to execute directly after attach operation.
     * 

* Implementation retrieves database information like dialect ODS and server version. *

* * @throws SQLException * For errors reading or writing database information. */ protected final void afterAttachActions() throws SQLException { connection.clearAuthData(); getDatabaseInfo(getDescribeDatabaseInfoBlock(), 1024, getDatabaseInformationProcessor()); // During connect and attach the socketTimeout might be set to the connectTimeout, now reset to 'normal' socketTimeout connection.resetSocketTimeout(); } @Override protected final void internalDetach() throws SQLException { // TODO Move to wire operations as it is almost identical to service detach? synchronized (getSynchronizationObject()) { try { try { final XdrOutputStream xdrOut = getXdrOut(); if (isAttached()) { xdrOut.writeInt(op_detach); xdrOut.writeInt(getHandle()); } xdrOut.writeInt(op_disconnect); xdrOut.flush(); } catch (IOException ex) { throw new FbExceptionBuilder().exception(ISCConstants.isc_net_write_err).cause(ex).toSQLException(); } if (isAttached()) { try { // Consume op_detach response wireOperations.readResponse(null); } catch (IOException ex) { throw new FbExceptionBuilder().exception(ISCConstants.isc_net_read_err).cause(ex).toSQLException(); } } try { closeConnection(); } catch (IOException ex) { throw new FbExceptionBuilder().exception(ISCConstants.isc_net_write_err).cause(ex).toSQLException(); } } catch (SQLException ex) { try { closeConnection(); } catch (Exception ex2) { // ignore } throw ex; } finally { setDetached(); } } } @Override public final void createDatabase() throws SQLException { try { final DatabaseParameterBuffer dpb = protocolDescriptor.createDatabaseParameterBuffer(connection); attachOrCreate(dpb, true); } catch (SQLException ex) { exceptionListenerDispatcher.errorOccurred(ex); throw ex; } } @Override public final void dropDatabase() throws SQLException { try { checkAttached(); synchronized (getSynchronizationObject()) { try { try { final XdrOutputStream xdrOut = getXdrOut(); xdrOut.writeInt(op_drop_database); xdrOut.writeInt(getHandle()); xdrOut.flush(); } catch (IOException ioex) { throw new FbExceptionBuilder().exception(ISCConstants.isc_net_write_err).cause(ioex) .toSQLException(); } try { readResponse(null); } catch (IOException ioex) { throw new FbExceptionBuilder().exception(ISCConstants.isc_net_read_err).cause(ioex) .toSQLException(); } } finally { try { closeConnection(); } catch (IOException e) { log.debug("Ignored exception on connection close in dropDatabase()", e); } } } } catch (SQLException ex) { exceptionListenerDispatcher.errorOccurred(ex); throw ex; } } @Override public final FbWireTransaction startTransaction(TransactionParameterBuffer tpb) throws SQLException { try { checkAttached(); synchronized (getSynchronizationObject()) { try { final XdrOutputStream xdrOut = getXdrOut(); xdrOut.writeInt(op_transaction); xdrOut.writeInt(getHandle()); xdrOut.writeTyped(tpb); xdrOut.flush(); } catch (IOException ioex) { throw new FbExceptionBuilder().exception(ISCConstants.isc_net_write_err).cause(ioex) .toSQLException(); } try { final GenericResponse response = readGenericResponse(null); final FbWireTransaction transaction = protocolDescriptor.createTransaction(this, response.getObjectHandle(), TransactionState.ACTIVE); transactionAdded(transaction); return transaction; } catch (IOException ioex) { throw new FbExceptionBuilder().exception(ISCConstants.isc_net_read_err).cause(ioex) .toSQLException(); } } } catch (SQLException ex) { exceptionListenerDispatcher.errorOccurred(ex); throw ex; } } @Override public final FbTransaction reconnectTransaction(long transactionId) throws SQLException { try { checkAttached(); synchronized (getSynchronizationObject()) { try { final XdrOutputStream xdrOut = getXdrOut(); xdrOut.writeInt(op_reconnect); xdrOut.writeInt(getHandle()); final byte[] transactionIdBuffer = getTransactionIdBuffer(transactionId); xdrOut.writeBuffer(transactionIdBuffer); xdrOut.flush(); } catch (IOException ioex) { throw new FbExceptionBuilder().exception(ISCConstants.isc_net_write_err).cause(ioex) .toSQLException(); } try { final GenericResponse response = readGenericResponse(null); final FbWireTransaction transaction = protocolDescriptor.createTransaction(this, response.getObjectHandle(), TransactionState.PREPARED); transactionAdded(transaction); return transaction; } catch (IOException ioex) { throw new FbExceptionBuilder().exception(ISCConstants.isc_net_read_err).cause(ioex) .toSQLException(); } } } catch (SQLException ex) { exceptionListenerDispatcher.errorOccurred(ex); throw ex; } } protected byte[] getTransactionIdBuffer(long transactionId) { ByteArrayOutputStream bos = new ByteArrayOutputStream(4); try { VaxEncoding.encodeVaxIntegerWithoutLength(bos, (int) transactionId); } catch (IOException e) { // ignored: won't happen with a ByteArrayOutputStream } return bos.toByteArray(); } @Override public final FbStatement createStatement(FbTransaction transaction) throws SQLException { try { checkAttached(); final FbStatement stmt = protocolDescriptor.createStatement(this); stmt.addExceptionListener(exceptionListenerDispatcher); stmt.setTransaction(transaction); return stmt; } catch (SQLException ex) { exceptionListenerDispatcher.errorOccurred(ex); throw ex; } } @Override public void cancelOperation(int kind) throws SQLException { try { if (kind == ISCConstants.fb_cancel_abort) { try { // In case of abort we forcibly close the connection // TODO We may need to do additional cleanup (eg notify statements so they can close etc) closeConnection(); } catch (IOException ioe) { throw new SQLNonTransientConnectionException("Connection abort failed", ioe); } } else { throw new SQLFeatureNotSupportedException( String.format("Cancel Operation isn't supported in this version of the wire protocol (%d).", protocolDescriptor.getVersion()), SQLStateConstants.SQL_STATE_FEATURE_NOT_SUPPORTED); } } catch (SQLException ex) { exceptionListenerDispatcher.errorOccurred(ex); throw ex; } } @Override public final byte[] getDatabaseInfo(byte[] requestItems, int maxBufferLength) throws SQLException { // TODO Write common info request implementation shared for db, sql, transaction and blob? try { checkAttached(); synchronized (getSynchronizationObject()) { try { final XdrOutputStream xdrOut = getXdrOut(); xdrOut.writeInt(op_info_database); xdrOut.writeInt(getHandle()); xdrOut.writeInt(0); // incarnation xdrOut.writeBuffer(requestItems); xdrOut.writeInt(maxBufferLength); xdrOut.flush(); } catch (IOException ex) { throw new FbExceptionBuilder().exception(ISCConstants.isc_net_write_err).cause(ex).toSQLException(); } try { final GenericResponse genericResponse = readGenericResponse(null); return genericResponse.getData(); } catch (IOException ex) { throw new FbExceptionBuilder().exception(ISCConstants.isc_net_read_err).cause(ex).toSQLException(); } } } catch (SQLException e) { exceptionListenerDispatcher.errorOccurred(e); throw e; } } @Override public final void executeImmediate(String statementText, FbTransaction transaction) throws SQLException { // TODO also implement op_exec_immediate2 try { if (isAttached()) { if (transaction == null) { throw FbExceptionBuilder .forException(JaybirdErrorCodes.jb_executeImmediateRequiresTransactionAttached) .toFlatSQLException(); } checkTransactionActive(transaction); } else if (transaction != null) { throw FbExceptionBuilder .forException(JaybirdErrorCodes.jb_executeImmediateRequiresNoTransactionDetached) .toFlatSQLException(); } synchronized (getSynchronizationObject()) { try { final XdrOutputStream xdrOut = getXdrOut(); xdrOut.writeInt(op_exec_immediate); xdrOut.writeInt(transaction != null ? transaction.getHandle() : 0); xdrOut.writeInt(getHandle()); xdrOut.writeInt(getConnectionDialect()); xdrOut.writeString(statementText, getEncoding()); // information request items xdrOut.writeBuffer(null); xdrOut.writeInt(0); getXdrOut().flush(); } catch (IOException ex) { throw new FbExceptionBuilder().exception(ISCConstants.isc_net_write_err).cause(ex).toSQLException(); } try { if (!isAttached()) { processAttachOrCreateResponse(readGenericResponse(null)); } readGenericResponse(null); } catch (IOException ex) { throw new FbExceptionBuilder().exception(ISCConstants.isc_net_read_err).cause(ex).toSQLException(); } } } catch (SQLException e) { exceptionListenerDispatcher.errorOccurred(e); throw e; } } @Override public void releaseObject(int operation, int objectId) throws SQLException { checkAttached(); synchronized (getSynchronizationObject()) { try { doReleaseObjectPacket(operation, objectId); getXdrOut().flush(); } catch (IOException ex) { throw new FbExceptionBuilder().exception(ISCConstants.isc_net_write_err).cause(ex).toSQLException(); } try { processReleaseObjectResponse(readResponse(null)); } catch (IOException ex) { throw new FbExceptionBuilder().exception(ISCConstants.isc_net_read_err).cause(ex).toSQLException(); } } } @Override public final FbWireAsynchronousChannel initAsynchronousChannel() throws SQLException { checkAttached(); final int auxHandle; final int port; synchronized (getSynchronizationObject()) { try { final XdrOutputStream xdrOut = getXdrOut(); xdrOut.writeInt(op_connect_request); xdrOut.writeInt(P_REQ_async); // Connection type (p_req_type) xdrOut.writeInt(getHandle()); // p_req_object xdrOut.writeInt(0); // p_req_partner xdrOut.flush(); } catch (IOException ex) { throw new FbExceptionBuilder().exception(ISCConstants.isc_net_write_err).cause(ex).toSQLException(); } try { final GenericResponse response = readGenericResponse(null); auxHandle = response.getObjectHandle(); final byte[] data = response.getData(); // bytes 0 - 1: sin family (ignore) // bytes 2 - 3: sin port (port to connect to) port = ((data[2] & 0xFF) << 8) + (data[3] & 0xFF); // remaining bytes: IP address + other info(?) (ignore, can't be trusted, and invalid in FB3 and higher; always use original hostname) } catch (IOException ex) { throw new FbExceptionBuilder().exception(ISCConstants.isc_net_read_err).cause(ex).toSQLException(); } } final FbWireAsynchronousChannel channel = protocolDescriptor.createAsynchronousChannel(this); channel.connect(connection.getServerName(), port, auxHandle); return channel; } /** * Sends - without flushing - the (release) operation and objectId. * * @param operation * Operation * @param objectId * Id of the object to release * @throws IOException * For errors writing to the connection * @throws SQLException * If the database connection is not available */ protected final void doReleaseObjectPacket(int operation, int objectId) throws IOException, SQLException { final XdrOutputStream xdrOut = getXdrOut(); xdrOut.writeInt(operation); xdrOut.writeInt(objectId); } /** * Process the release object response * * @param response * The response object */ protected final void processReleaseObjectResponse(@SuppressWarnings("UnusedParameters") Response response) { // Do nothing } @Override public final BlrCalculator getBlrCalculator() { if (blrCalculator == null) { blrCalculator = protocolDescriptor.createBlrCalculator(this); } return blrCalculator; } @Override public void enqueueDeferredAction(DeferredAction deferredAction) { throw new UnsupportedOperationException("enqueueDeferredAction is not supported in the V10 protocol"); } @Override public final void authReceiveResponse(AcceptPacket acceptPacket) throws IOException, SQLException { final DbCryptCallback dbCryptCallback = connection.createDbCryptCallback(); wireOperations.authReceiveResponse(acceptPacket, dbCryptCallback, new FbWireOperations.ProcessAttachCallback() { @Override public void processAttachResponse(GenericResponse response) { processAttachOrCreateResponse(response); } }); } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy