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

org.firebirdsql.jdbc.FBBlob Maven / Gradle / Ivy

There is a newer version: 6.0.0-beta-1
Show newest version
/*
 * Firebird Open Source J2ee 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 CVS history command.
 *
 * All rights reserved.
 */

package org.firebirdsql.jdbc;


import java.io.EOFException;
import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStream;
import java.io.OutputStreamWriter;
import java.io.Reader;
import java.io.UnsupportedEncodingException;
import java.sql.Blob;
import java.sql.SQLException;
import java.sql.SQLFeatureNotSupportedException;
import java.util.Collection;
import java.util.HashSet;
import java.util.Iterator;

import org.firebirdsql.gds.DatabaseParameterBuffer;
import org.firebirdsql.gds.GDSException;
import org.firebirdsql.gds.ISCConstants;
import org.firebirdsql.gds.IscBlobHandle;
import org.firebirdsql.gds.impl.DatabaseParameterBufferExtension;
import org.firebirdsql.gds.impl.GDSHelper;

/**
 * The representation (mapping) in
 * the JavaTM programming
 * language of an SQL
 * BLOB value.  An SQL BLOB is a built-in type
 * that stores a Binary Large Object as a column value in a row of
 * a database table. The driver implements Blob using
 * an SQL locator(BLOB), which means that a
 * Blob object contains a logical pointer to the
 * SQL BLOB data rather than the data itself.
 * A Blob object is valid for the duration of the
 * transaction in which is was created.
 *
 * 

Methods in the interfaces {@link java.sql.ResultSet}, * {@link java.sql.CallableStatement}, and {@link java.sql.PreparedStatement}, such as * getBlob and setBlob allow a programmer to * access an SQL BLOB value. * The Blob interface provides methods for getting the * length of an SQL BLOB (Binary Large Object) value, * for materializing a BLOB value on the client, and for * determining the position of a pattern of bytes within a * BLOB value. *

* This class is new in the JDBC 2.0 API. * @since 1.2 */ public class FBBlob implements FirebirdBlob, Synchronizable { public static final boolean SEGMENTED = true; public static final int READ_FULLY_BUFFER_SIZE = 16 * 1024; /** * bufferlength is the size of the buffer for blob input and output streams, * also used for the BufferedInputStream/BufferedOutputStream wrappers. * */ private int bufferlength; private boolean isNew; private long blob_id; private GDSHelper gdsHelper; private FBObjectListener.BlobListener blobListener; private Collection inputStreams = new HashSet(); private FBBlobOutputStream blobOut = null; private FBBlob(GDSHelper c, boolean isNew, FBObjectListener.BlobListener blobListener) { this.gdsHelper = c; this.isNew = isNew; this.bufferlength = c.getBlobBufferLength(); this.blobListener = blobListener; } /** * Create new Blob instance. This constructor creates new fresh Blob, only * writing to the Blob is allowed. * * @param c connection that will be used to write data to blob. */ public FBBlob(GDSHelper c, FBObjectListener.BlobListener blobListener) { this(c, true, blobListener); } public FBBlob(GDSHelper c) { this(c, null); } /** * Create instance of this class to access existing Blob. * * @param c connection that will be used to access Blob. * * @param blob_id ID of the Blob. */ public FBBlob(GDSHelper c, long blob_id, FBObjectListener.BlobListener blobListener) { this(c, false, blobListener); this.blob_id = blob_id; } public FBBlob(GDSHelper c, long blob_id) { this(c, blob_id, null); } /** * Get synchronization object that will be used to synchronize multithreaded * access to the database. * * @return object that will be used for synchronization. */ public Object getSynchronizationObject() { return gdsHelper; } /** * Close this Blob object. This method closes all open input streams. * * @throws IOException if at least one of the stream raised an exception * when closed. */ public void close() throws IOException { Object syncObject = getSynchronizationObject(); synchronized(syncObject) { IOException error = null; Iterator i = inputStreams.iterator(); while (i.hasNext()) { try { ((FBBlobInputStream)i.next()).close(); } catch(IOException ex) { error = ex; } } inputStreams.clear(); if (error != null) throw error; } } /** * Get information about this Blob. This method should be considered as * temporary because it provides access to low-level API. More information * on how to use the API can be found in "API Guide". * * @param items items in which we are interested. * @param buffer_length buffer where information will be stored. * * @return array of bytes containing information about this Blob. * * @throws SQLException if something went wrong. */ public byte[] getInfo(byte[] items, int buffer_length) throws SQLException { Object syncObject = getSynchronizationObject(); synchronized(syncObject) { try { if (blobListener != null) blobListener.executionStarted(this); IscBlobHandle blob = gdsHelper.openBlob(blob_id, SEGMENTED); try { return gdsHelper.getBlobInfo(blob, items, buffer_length); } finally { gdsHelper.closeBlob(blob); } } catch(GDSException ex) { throw new FBSQLException(ex); } finally { if (blobListener != null) blobListener.executionCompleted(this); } } } public static final byte[] BLOB_LENGTH_REQUEST = new byte[]{ISCConstants.isc_info_blob_total_length}; /** * Returns the number of bytes in the BLOB value * designated by this Blob object. * @return length of the BLOB in bytes * @exception SQLException if there is an error accessing the * length of the BLOB * @since 1.2 * @see What Is in the JDBC 2.0 API */ public long length() throws SQLException { byte[] info = getInfo(BLOB_LENGTH_REQUEST, 20); return interpretLength(info, 0); } /** * Interpret BLOB length from buffer. * * @param info server response. * @param position where to start interpreting. * * @return length of the blob. * * @throws SQLException if length cannot be interpreted. */ public static long interpretLength(GDSHelper gdsHelper, byte[] info, int position) throws SQLException { if (info[position] != ISCConstants.isc_info_blob_total_length) throw new FBSQLException("Length is not available."); int dataLength = gdsHelper.iscVaxInteger(info, position + 1, 2); return gdsHelper.iscVaxInteger(info, position + 3, dataLength); } /** * Interpret BLOB length from buffer. * * @param info server response. * @param position where to start interpreting. * * @return length of the blob. * * @throws SQLException if length cannot be interpreted. */ private long interpretLength(byte[] info, int position) throws SQLException { return interpretLength(gdsHelper, info, position); } /** * Check if blob is segmented. * * @return true if this blob is segmented, * otherwise false * * @throws SQLException if something went wrong. */ public boolean isSegmented() throws SQLException { byte[] info = getInfo( new byte[] {ISCConstants.isc_info_blob_type}, 20); if (info[0] != ISCConstants.isc_info_blob_type) throw new FBSQLException("Cannot determine BLOB type"); int dataLength = gdsHelper.iscVaxInteger(info, 1, 2); int type = gdsHelper.iscVaxInteger(info, 3, dataLength); return type == ISCConstants.isc_bpb_type_segmented; } /** * Detach this blob. This method creates new instance of the same blob * database object that is not under result set control. When result set * is closed, all associated resources are also released, including open * blob streams. This method creates an new instance of blob object with * the same blob ID that can be used even when result set is closed. *

* Note, detached blob will not remember the stream position of this object. * This means that you cannot start reading data from the blob, then detach * it, and then continue reading. Reading from detached blob will begin at * the blob start. * * @return instance of {@link FBBlob} that is not under result set control. * * @throws SQLException if Blob cannot be detached. */ public FirebirdBlob detach() throws SQLException { return new FBBlob(gdsHelper, blob_id, blobListener); } /** * Returns as an array of bytes, part or all of the BLOB * value that this Blob object designates. The byte * array contains up to length consecutive bytes * starting at position pos. * @param pos the ordinal position of the first byte in the * BLOB value to be extracted; the first byte is at * position 1 * @param length the number of consecutive bytes to be copied * @return a byte array containing up to length * consecutive bytes from the BLOB value designated * by this Blob object, starting with the * byte at position pos * @exception SQLException if there is an error accessing the * BLOB * @since 1.2 * @see What Is in the JDBC 2.0 API */ public byte[] getBytes(long pos, int length) throws SQLException{ if (pos < 1) throw new FBSQLException("Blob position should be >= 1"); if (pos > Integer.MAX_VALUE) throw new FBSQLException("Blob position is limited to 2^31 - 1 " + "due to isc_seek_blob limitations.", FBSQLException.SQL_STATE_INVALID_ARG_VALUE); Object syncObject = getSynchronizationObject(); synchronized(syncObject) { if (blobListener != null) blobListener.executionStarted(this); try { FirebirdBlob.BlobInputStream in = (FirebirdBlob.BlobInputStream)getBinaryStream(); try { byte[] result = new byte[length]; if (pos != 1) in.seek((int)pos - 1); in.readFully(result); return result; } finally { in.close(); } } catch(IOException ex) { throw new FBSQLException(ex); } finally { if (blobListener != null) blobListener.executionCompleted(this); } } } /** * Retrieves the BLOB designated by this * Blob instance as a stream. * @return a stream containing the BLOB data * @exception SQLException if there is an error accessing the * BLOB * @since 1.2 * @see What Is in the JDBC 2.0 API */ public InputStream getBinaryStream () throws SQLException { Object syncObject = getSynchronizationObject(); synchronized(syncObject) { FBBlobInputStream blobstream = new FBBlobInputStream(this); inputStreams.add(blobstream); return blobstream; } } /** * Determines the byte position at which the specified byte * pattern begins within the BLOB * value that this Blob object represents. The * search for pattern begins at position * start. * @param pattern the byte array for which to search * @param start the position at which to begin searching; the * first position is 1 * @return the position at which the pattern appears, else -1 * @exception SQLException if there is an error accessing the * BLOB * @since 1.2 * @see What Is in the JDBC 2.0 API */ public long position(byte pattern[], long start) throws SQLException { throw new FBDriverNotCapableException(); } /** * Determines the byte position in the BLOB value * designated by this Blob object at which * pattern begins. The search begins at position * start. * @param pattern the Blob object designating * the BLOB value for which to search * @param start the position in the BLOB value * at which to begin searching; the first position is 1 * @return the position at which the pattern begins, else -1 * @exception SQLException if there is an error accessing the * BLOB * @since 1.2 * @see What Is in the JDBC 2.0 API */ public long position(Blob pattern, long start) throws SQLException { throw new FBDriverNotCapableException(); } //jdbc 3.0 additions /** * This operation is not currently supported * Truncate this Blob to a given length. * * @param param1 The length to truncate this Blob to * @exception java.sql.SQLException this operation is not supported * */ public void truncate(long param1) throws SQLException { throw new FBDriverNotCapableException(); } /** * This operation is not currently supported * Writes all or part of the given byte array to the BLOB value that this * Blob object represents and returns the number of bytes * written. * * @param param1 The position at which to start writing * @param param2 The array of bytes to be written * @return the number of bytes written * @exception java.sql.SQLException because this operation is not supported */ public int setBytes(long param1, byte[] param2) throws SQLException { throw new FBDriverNotCapableException(); } /** * This operation is not currently supported * Writes all or part of the given byte array to the BLOB value that this * Blob object represents and returns the number of bytes * written. * * @param param1 The position at which to start writing * @param param2 The array of bytes to be written * @param param3 the offset into the byte array at which to start reading * the bytes to be set * @param param4 the number of bytes to be written to the BLOB value from * the byte array * @return the number of bytes written * @exception java.sql.SQLException because this operation is not supported */ public int setBytes(long param1, byte[] param2, int param3, int param4) throws SQLException { throw new FBDriverNotCapableException(); } /** * Retrieves a stream that can be used to write to the BLOB value that * this Blob object represents. The stream begins at position pos. * * @param pos The position in the blob to start writing. * @return OuputStream to write to. * @exception java.sql.SQLException if a database access error occurs */ public OutputStream setBinaryStream(long pos) throws SQLException { if (blobListener != null) blobListener.executionStarted(this); if (blobOut != null) throw new FBSQLException("Only one blob output stream open at a time!"); if (pos < 0) throw new FBSQLException( "You can't start before the beginning of the blob", FBSQLException.SQL_STATE_INVALID_ARG_VALUE); if ((isNew) && (pos > 0)) throw new FBSQLException( "Previous value was null, you must start at position 0", FBSQLException.SQL_STATE_INVALID_ARG_VALUE); blobOut = new FBBlobOutputStream(); if (pos > 0) { //copy pos bytes from input to output //implement this later throw new FBDriverNotCapableException("Non-null positions are not yet supported."); } return blobOut; } //package methods /** * Get the identifier for this Blob * * @return This Blob's identifier * @throws SQLException if a database access error occurs */ public long getBlobId() throws SQLException { if (isNew) throw new FBSQLException("No Blob ID is available in new Blob object."); return blob_id; } void setBlobId(long blob_id) { this.blob_id = blob_id; this.isNew = false; } public void copyBytes(byte[] bytes, int pos, int len) throws SQLException { OutputStream out = setBinaryStream(0); try { try { out.write(bytes, pos, len); } finally { out.close(); } } catch(IOException ex) { throw new FBSQLException(ex); } } /** * Copy the contents of an InputStream into this Blob. * * @param inputStream the stream from which data will be copied * @param length The maximum number of bytes to read from the InputStream * @throws SQLException if a database access error occurs */ public void copyStream(InputStream inputStream, int length) throws SQLException { OutputStream os = setBinaryStream(0); byte[] buffer = new byte[Math.min(bufferlength, length)]; int chunk; try { while (length >0) { chunk =inputStream.read(buffer, 0, ((lengthInputStream into this Blob. Unlike * the {@link #copyStream(InputStream, int)} method, this one copies bytes * until the EOF is reached. * * @param inputStream the stream from which data will be copied * @throws SQLException if a database access error occurs */ public void copyStream(InputStream inputStream) throws SQLException { OutputStream os = setBinaryStream(0); try { int chunk = 0; byte[] buffer = new byte[bufferlength]; while((chunk = inputStream.read(buffer)) != -1) os.write(buffer, 0, chunk); os.flush(); os.close(); } catch (IOException ioe) { throw new FBSQLException(ioe); } } /** * Copy data from a character stream into this Blob. * * @param inputStream the source of data to copy * @param length The maximum number of bytes to copy * @param encoding The encoding used in the character stream */ public void copyCharacterStream(Reader inputStream, int length, String encoding) throws SQLException { OutputStream os = setBinaryStream(0); try { OutputStreamWriter osw; if (encoding != null) osw = new OutputStreamWriter(os, encoding); else osw = new OutputStreamWriter(os); char[] buffer = new char[Math.min(bufferlength, length)]; int chunk; try { while (length >0) { chunk =inputStream.read(buffer, 0, ((length len) {//not expected to happen System.arraycopy(buffer, pos, b, off, len); pos += len; return len; } System.arraycopy(buffer, pos, b, off, result); buffer = null; pos = 0; return result; } public void readFully(byte[] b, int off, int len) throws IOException { int counter = 0; int pos = 0; byte[] buffer = new byte[Math.min(READ_FULLY_BUFFER_SIZE, len)]; int toRead = len; while(toRead > 0 && (counter = read(buffer, 0, toRead)) != -1) { System.arraycopy(buffer, 0, b, pos, counter); pos += counter; toRead -= counter; } if (counter == -1) throw new EOFException(); } public void readFully(byte[] b) throws IOException { readFully(b, 0, b.length); } public void close() throws IOException { Object syncObject = getSynchronizationObject(); synchronized(syncObject) { if (blob != null) { try { gdsHelper.closeBlob(blob); inputStreams.remove(this); } catch (GDSException ge) { throw new IOException("couldn't close blob: " + ge); } blob = null; closed = true; } } } private void checkClosed() throws IOException { if (closed) throw new IOException("Input stream is already closed."); } } public class FBBlobOutputStream extends OutputStream implements FirebirdBlob.BlobOutputStream { private IscBlobHandle blob; private FBBlobOutputStream() throws SQLException { Object syncObject = getSynchronizationObject(); synchronized(syncObject) { try { DatabaseParameterBuffer dpb = gdsHelper.getDatabaseParameterBuffer(); boolean useStreamBlobs = dpb.hasArgument(DatabaseParameterBufferExtension.USE_STREAM_BLOBS); blob = gdsHelper.createBlob(!useStreamBlobs); } catch (GDSException ge) { throw new FBSQLException(ge); } } if (isNew) { setBlobId(blob.getBlobId()); } } public void seek(int position, int seekMode) throws SQLException { try { gdsHelper.seekBlob(blob, position, seekMode); } catch(GDSException ex) { throw new FBSQLException(ex); } } public long length() throws IOException { Object syncObject = getSynchronizationObject(); synchronized(syncObject) { try { byte[] info = gdsHelper.getBlobInfo( blob, new byte[] {ISCConstants.isc_info_blob_total_length}, 20); return interpretLength(info, 0); } catch (GDSException ex) { throw new IOException(ex.getMessage()); } catch (SQLException ex) { throw new IOException(ex.getMessage()); } } } public void write(int b) throws IOException { //This won't be called, don't implement throw new IOException("FBBlobOutputStream.write(int b) not implemented"); } public void writeSegment(byte[] buf) throws GDSException { Object syncObject = getSynchronizationObject(); synchronized(syncObject) { gdsHelper.putBlobSegment(blob, buf); } } public void write(byte[] b, int off, int len) throws IOException { try { if (off == 0 && len == b.length && len < bufferlength) { /* * If we are just writing the entire byte array, we need to * do nothing but just write it over */ writeSegment(b); } else { /* * In this case, we need to chunk it over since putBlobSegment * cannot currently support length and offset. */ int chunk = bufferlength; int lastChunk = 0; byte[] buf = null; while (len > 0) { if (len < chunk) chunk = len; // this allows us to reused the chunk if its size has // not changed if (chunk != lastChunk) { buf = new byte[chunk]; lastChunk = chunk; } System.arraycopy(b, off, buf, 0, chunk); writeSegment(buf); len -= chunk; off += chunk; } } } catch (GDSException ge) { throw new IOException("Problem writing to FBBlobOutputStream: " + ge); } } public void close() throws IOException { if (blob != null) { try { Object syncObject = getSynchronizationObject(); synchronized(syncObject) { gdsHelper.closeBlob(blob); } setBlobId(blob.getBlobId()); } catch (GDSException ge) { throw new IOException("could not close blob: " + ge); } blob = null; } } } public void free() throws SQLException { } /** * Returns an InputStream object that contains a partial Blob value, * starting with the byte specified by pos, which is length bytes in length. * * @param pos the offset to the first byte of the partial value to be retrieved. * The first byte in the Blob is at position 1 * @param length the length in bytes of the partial value to be retrieved * @return InputStream through which the partial Blob value can be read. * @throws SQLException if pos is less than 1 or if pos is greater than the number of bytes * in the Blob or if pos + length is greater than the number of bytes * in the Blob * * @exception SQLFeatureNotSupportedException if the JDBC driver does not support * this method * @since 1.6 */ public InputStream getBinaryStream(long pos, long length) throws SQLException { throw new FBDriverNotCapableException(); } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy