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

org.firebirdsql.gds.ng.wire.AbstractFbWireDatabase Maven / Gradle / Ivy

There is a newer version: 6.0.0-beta-1
Show newest version
/*
 * 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.gds.*;
import org.firebirdsql.gds.impl.wire.XdrInputStream;
import org.firebirdsql.gds.impl.wire.XdrOutputStream;
import org.firebirdsql.gds.ng.*;

import java.io.IOException;
import java.sql.SQLException;

import static java.util.Objects.requireNonNull;
import static org.firebirdsql.gds.JaybirdErrorCodes.jb_notAttachedToDatabase;
import static org.firebirdsql.gds.JaybirdErrorCodes.jb_notConnectedToServer;
import static org.firebirdsql.gds.JaybirdErrorCodes.jb_unableToCancelEventReasonNotConnected;
import static org.firebirdsql.gds.impl.wire.WireProtocolConstants.PROTOCOL_VERSION18;
import static org.firebirdsql.gds.impl.wire.WireProtocolConstants.op_info_blob;
import static org.firebirdsql.gds.impl.wire.WireProtocolConstants.op_info_cursor;
import static org.firebirdsql.gds.impl.wire.WireProtocolConstants.op_info_database;
import static org.firebirdsql.gds.impl.wire.WireProtocolConstants.op_info_request;
import static org.firebirdsql.gds.impl.wire.WireProtocolConstants.op_info_sql;
import static org.firebirdsql.gds.impl.wire.WireProtocolConstants.op_info_transaction;

/**
 * Abstract class for operations common to all version of the wire protocol implementation.
 *
 * @author Mark Rotteveel
 * @since 3.0
 */
public abstract class AbstractFbWireDatabase extends AbstractFbDatabase
        implements FbWireDatabase {

    protected final ProtocolDescriptor protocolDescriptor;
    protected final FbWireOperations wireOperations;
    private FbWireAsynchronousChannel asynchronousChannel;

    /**
     * Creates an AbstractFbWireDatabase 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 AbstractFbWireDatabase(WireDatabaseConnection connection, ProtocolDescriptor descriptor) {
        super(connection, DefaultDatatypeCoder.forEncodingFactory(connection.getEncodingFactory()));
        protocolDescriptor = requireNonNull(descriptor, "parameter descriptor should be non-null");
        wireOperations = descriptor.createWireOperations(connection, getDatabaseWarningCallback());
    }

    @Override
    public final int getHandle() {
        // The handle is always 0 for a TCP/IP database
        return 0;
    }

    @Override
    public void forceClose() throws SQLException {
        try {
            if (connection.isConnected()) {
                connection.close();
            }
        } catch (IOException e) {
            throw FbExceptionBuilder.ioWriteError(e);
        } finally {
            databaseListenerDispatcher.detached(this);
            databaseListenerDispatcher.shutdown();
            exceptionListenerDispatcher.shutdown();
        }
    }

    /**
     * Gets the XdrInputStream.
     *
     * @return Instance of XdrInputStream
     * @throws SQLException
     *         If no connection is opened or when exceptions occur
     *         retrieving the InputStream
     */
    protected final XdrInputStream getXdrIn() throws SQLException {
        return getXdrStreamAccess().getXdrIn();
    }

    /**
     * Gets the XdrOutputStream.
     *
     * @return Instance of XdrOutputStream
     * @throws SQLException
     *         If no connection is opened or when exceptions occur
     *         retrieving the OutputStream
     */
    protected final XdrOutputStream getXdrOut() throws SQLException {
        return getXdrStreamAccess().getXdrOut();
    }

    @Override
    public final XdrStreamAccess getXdrStreamAccess() {
        return connection.getXdrStreamAccess();
    }

    @Override
    public final FbWireOperations getWireOperations() {
        return wireOperations;
    }

    @Override
    public final boolean isAttached() {
        return super.isAttached() && connection.isConnected();
    }

    /**
     * Checks if a physical connection to the server is established.
     *
     * @throws SQLException
     *         If not connected.
     */
    @Override
    protected final void checkConnected() throws SQLException {
        if (!connection.isConnected()) {
            throw FbExceptionBuilder.toNonTransientConnectionException(jb_notConnectedToServer);
        }
    }

    /**
     * Checks if a physical connection to the server is established and if the
     * connection is attached to a database.
     * 

* This method calls {@link #checkConnected()}, so it is not necessary to * call both. *

* * @throws SQLException * If the database not connected or attached. */ protected final void checkAttached() throws SQLException { checkConnected(); if (!isAttached()) { throw FbExceptionBuilder.toNonTransientConnectionException(jb_notAttachedToDatabase); } } /** * Closes the WireConnection associated with this connection. * * @throws IOException * For errors closing the connection. */ protected final void closeConnection() throws IOException { if (!connection.isConnected()) return; try (LockCloseable ignored = withLock()) { try { connection.close(); } finally { setDetached(); } } } @Override public void setNetworkTimeout(int milliseconds) throws SQLException { try (LockCloseable ignored = withLock()) { checkConnected(); wireOperations.setNetworkTimeout(milliseconds); } catch (SQLException e) { exceptionListenerDispatcher.errorOccurred(e); throw e; } } @Override public final FbBlob createBlobForOutput(FbTransaction transaction, BlobParameterBuffer blobParameterBuffer) { final FbWireBlob outputBlob = protocolDescriptor.createOutputBlob(this, (FbWireTransaction) transaction, blobParameterBuffer); outputBlob.addExceptionListener(exceptionListenerDispatcher); return outputBlob; } @Override public final FbBlob createBlobForInput(FbTransaction transaction, BlobParameterBuffer blobParameterBuffer, long blobId) { final FbWireBlob inputBlob = protocolDescriptor.createInputBlob(this, (FbWireTransaction) transaction, blobParameterBuffer, blobId); inputBlob.addExceptionListener(exceptionListenerDispatcher); return inputBlob; } @Override public final void consumePackets(int numberOfResponses, WarningMessageCallback warningCallback) { // TODO Should consumePackets notify the exception listener or not? wireOperations.consumePackets(numberOfResponses, warningCallback); } @Override public final GenericResponse readGenericResponse(WarningMessageCallback warningCallback) throws SQLException, IOException { return wireOperations.readGenericResponse(warningCallback); } @Override public final SqlResponse readSqlResponse(WarningMessageCallback warningCallback) throws SQLException, IOException { return wireOperations.readSqlResponse(warningCallback); } @Override public final Response readResponse(WarningMessageCallback warningCallback) throws SQLException, IOException { return wireOperations.readResponse(warningCallback); } @Override public final void enqueueDeferredAction(DeferredAction deferredAction) throws SQLException { wireOperations.enqueueDeferredAction(deferredAction); } @Override public final void completeDeferredActions() throws SQLException { wireOperations.completeDeferredActions(); } @Override public final EventHandle createEventHandle(String eventName, EventHandler eventHandler) { return new WireEventHandle(eventName, eventHandler, getEncoding()); } @Override public final void queueEvent(EventHandle eventHandle) throws SQLException { try (LockCloseable ignored = withLock()) { checkAttached(); FbWireAsynchronousChannel channel = asynchronousChannel; if (channel == null || !channel.isConnected()) { var asynchronousProcessor = AsynchronousProcessor.getInstance(); if (channel != null) { asynchronousProcessor.unregisterAsynchronousChannel(channel); } channel = asynchronousChannel = initAsynchronousChannel(); asynchronousProcessor.registerAsynchronousChannel(channel); } channel.queueEvent(eventHandle); } catch (SQLException e) { exceptionListenerDispatcher.errorOccurred(e); throw e; } } @Override public final void cancelEvent(EventHandle eventHandle) throws SQLException { try (LockCloseable ignored = withLock()) { checkAttached(); FbWireAsynchronousChannel channel = asynchronousChannel; if (channel == null || !channel.isConnected()) { if (channel != null) { AsynchronousProcessor.getInstance().unregisterAsynchronousChannel(channel); } throw FbExceptionBuilder.toNonTransientException(jb_unableToCancelEventReasonNotConnected); } channel.cancelEvent(eventHandle); } catch (SQLException e) { exceptionListenerDispatcher.errorOccurred(e); throw e; } } @Override public final void countEvents(EventHandle eventHandle) throws SQLException { try { if (!(eventHandle instanceof WireEventHandle)) throw new SQLException("Invalid event handle, type: " + eventHandle.getClass().getName()); ((WireEventHandle) eventHandle).calculateCount(); } catch (SQLException e) { exceptionListenerDispatcher.errorOccurred(e); throw e; } } @Override public final byte[] getDatabaseInfo(byte[] requestItems, int maxBufferLength) throws SQLException { try { checkAttached(); return getInfo(op_info_database, 0, requestItems, maxBufferLength, null); } catch (SQLException e) { exceptionListenerDispatcher.errorOccurred(e); throw e; } } @Override @SuppressWarnings("java:S4274") public byte[] getInfo(int operation, int handle, byte[] requestItems, int maxBufferLength, WarningMessageCallback warningMessageCallback) throws SQLException { assert operation == op_info_sql || operation == op_info_blob || operation == op_info_database || operation == op_info_transaction || operation == op_info_request || operation == op_info_cursor && protocolDescriptor.getVersion() >= PROTOCOL_VERSION18 : "Unsupported operation code for info request " + operation; try (LockCloseable ignored = withLock()) { try { final XdrOutputStream xdrOut = getXdrOut(); xdrOut.writeInt(operation); xdrOut.writeInt(handle); xdrOut.writeInt(0); // incarnation xdrOut.writeBuffer(requestItems); xdrOut.writeInt(maxBufferLength); xdrOut.flush(); } catch (IOException e) { throw FbExceptionBuilder.ioWriteError(e); } try { final GenericResponse genericResponse = readGenericResponse(null); return genericResponse.data(); } catch (IOException e) { throw FbExceptionBuilder.ioReadError(e); } } } /** * Initializes the asynchronous channel (for event notification). * * @throws java.sql.SQLException * For errors establishing the channel, or if the channel already exists. */ public abstract FbWireAsynchronousChannel initAsynchronousChannel() throws SQLException; }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy