org.firebirdsql.gds.ng.FbBlob Maven / Gradle / Ivy
/*
* 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.ng;
import org.firebirdsql.gds.ISCConstants;
import org.firebirdsql.gds.ng.listeners.ExceptionListenable;
import java.sql.SQLException;
/**
* Interface for blob operations.
*
* All methods defined in this interface are required to notify all {@code SQLException} thrown from the methods
* defined in this interface.
*
*
* @author Mark Rotteveel
* @since 3.0
*/
public interface FbBlob extends ExceptionListenable, AutoCloseable {
long NO_BLOB_ID = 0;
/**
* @return The Firebird blob id
*/
long getBlobId();
/**
* @return The Firebird blob handle identifier
*/
int getHandle();
/**
* @return The database connection that created this blob
*/
FbDatabase getDatabase();
/**
* Opens an existing input blob, or creates an output blob.
*
* @throws SQLException
* If the blob is already open, this is a (closed) output blob and it already has a blobId, the
* transaction is not active, or a database connection error occurred
*/
void open() throws SQLException;
/**
* @return true
if this blob is currently open.
*/
boolean isOpen();
/**
* @return true
if this blob has reached the end or has been closed, always true
for an
* open output blob.
*/
boolean isEof();
/**
* Closes the blob.
*
* Closing an already closed blob is a no-op.
*
*
* @throws SQLException
* If the transaction is not active, or a database connection error occurred
*/
void close() throws SQLException;
/**
* Cancels an output blob (which means its contents will be thrown away).
*
* Calling cancel on an input blob will close it. Contrary to {@link #close()}, calling cancel on an
* already closed (or cancelled) blob will throw an {@link java.sql.SQLException}.
*
*
* @throws SQLException
* If the blob has already been closed, the transaction is not active, or a database connection error
* occurred.
*/
void cancel() throws SQLException;
/**
* @return true
if this is an output blob (write only), false
if this is an
* input blob (read only)
*/
boolean isOutput();
/**
* Get synchronization object.
*
* @return object, cannot be null
.
*/
Object getSynchronizationObject();
// TODO Consider different blob api, eg more like InputStream / OutputStream methods taking a buffer with offset and length?
/**
* Gets a segment of blob data.
*
* When sizeRequested
exceeds {@link #getMaximumSegmentSize()} it is silently reduced to the maximum
* segment size.
*
* TODO: Consider allowing this and have the implementation handle longer segments by sending multiple (batched?) requests.
*
* @param sizeRequested
* Requested segment size (> 0).
* @return Retrieved segment (size may be less than requested)
* @throws SQLException
* If this is an output blob, the blob is closed, the transaction is not active, or a database connection
* error occurred.
*/
byte[] getSegment(int sizeRequested) throws SQLException;
/**
* Writes a segment of blob data.
*
* Implementation must handle segment length exceeding {@link #getMaximumSegmentSize()} by batching. TODO: reconsider and let caller handle that?
*
*
* Passing a section that is length 0 will throw an SQLException
.
*
*
* @param segment
* Segment to write
* @throws SQLException
* If this is an input blob, the blob is closed, the transaction is not active, the segment is length 0 or
* longer than the maximum segment size, or a database connection error occurred.
*/
void putSegment(byte[] segment) throws SQLException;
/**
* Performs a seek on a blob with the specified seekMode
and offset
.
*
* Firebird only supports seek on stream blobs.
*
*
* @param offset
* Offset of the seek, effect depends on value of seekMode
* @param seekMode
* Value of {@link org.firebirdsql.gds.ng.FbBlob.SeekMode}
* @throws SQLException
* If the blob is closed, the transaction is not active, or a database error occurred.
*/
void seek(int offset, SeekMode seekMode) throws SQLException;
/**
* The maximum segment size allowed by the protocol for {@link #getSegment(int)} and {@link #putSegment(byte[])}.
*
* This value is not the segment size (optionally) defined for the column.
*
*
* @return The maximum segment size allowed for get or put.
*/
int getMaximumSegmentSize();
/**
* Request blob info.
*
* @param requestItems
* Array of info items to request
* @param bufferLength
* Response buffer length to use
* @param infoProcessor
* Implementation of {@link org.firebirdsql.gds.ng.InfoProcessor} to transform
* the info response
* @return Transformed info response of type T
* @throws SQLException
* For errors retrieving or transforming the response.
*/
T getBlobInfo(byte[] requestItems, int bufferLength, InfoProcessor infoProcessor)
throws SQLException;
/**
* Requests the blob length from the server.
*
* @return Length of the blob.
* @throws SQLException
* For Errors retrieving the length, or if the blob is not associated with a blob id, or the database is not
* attached.
*/
long length() throws SQLException;
/**
* Request blob info.
*
* @param requestItems
* Array of info items to request
* @param bufferLength
* Response buffer length to use
* @return Response buffer
* @throws SQLException
*/
byte[] getBlobInfo(byte[] requestItems, int bufferLength) throws SQLException;
/**
* Seek mode for {@link FbBlob#seek(int, org.firebirdsql.gds.ng.FbBlob.SeekMode)}.
*/
enum SeekMode {
/**
* Absolute seek from start of blob.
*/
ABSOLUTE(ISCConstants.blb_seek_from_head),
/**
* Relative seek from current position in blob.
*/
RELATIVE(ISCConstants.blb_seek_relative),
/**
* Absolute seek from end of blob.
*/
ABSOLUTE_FROM_END(ISCConstants.blb_seek_from_tail);
final int seekModeId;
SeekMode(int seekModeId) {
this.seekModeId = seekModeId;
}
/**
* @return Seek mode value used within Firebird (and the native API, wire protocol)
*/
public int getSeekModeId() {
return seekModeId;
}
/**
* Get the SeekMode instance by the (Firebird) seekmode id.
*
* Valid values are:
*
* 0
- for {@link SeekMode#ABSOLUTE}
* 1
- for {@link SeekMode#RELATIVE}
* 2
- for {@link SeekMode#ABSOLUTE_FROM_END}
*
*
*
* @param seekModeId
* The seekmode id.
* @return Instance of SeekMode matching the id.
* @throws java.lang.IllegalArgumentException
* For values that do not have a SeekMode instance.
*/
public static SeekMode getById(int seekModeId) {
for (SeekMode seekMode : values()) {
if (seekMode.getSeekModeId() == seekModeId) return seekMode;
}
throw new IllegalArgumentException(String.format("No SeekMode with id %d", seekModeId));
}
}
}