org.mariadb.jdbc.MariaDbBlob Maven / Gradle / Ivy
Show all versions of mariadb-java-client-jre7 Show documentation
/*
*
* MariaDB Client for Java
*
* Copyright (c) 2012-2014 Monty Program Ab.
* Copyright (c) 2015-2017 MariaDB Ab.
*
* This library is free software; you can redistribute it and/or modify it under
* the terms of the GNU Lesser General Public License as published by the Free
* Software Foundation; either version 2.1 of the License, or (at your option)
* any later version.
*
* This library 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 GNU Lesser General Public License
* for more details.
*
* You should have received a copy of the GNU Lesser General Public License along
* with this library; if not, write to Monty Program Ab [email protected].
*
* This particular MariaDB Client for Java file is work
* derived from a Drizzle-JDBC. Drizzle-JDBC file which is covered by subject to
* the following copyright and notice provisions:
*
* Copyright (c) 2009-2011, Marcus Eriksson
*
* Redistribution and use in source and binary forms, with or without modification,
* are permitted provided that the following conditions are met:
* Redistributions of source code must retain the above copyright notice, this list
* of conditions and the following disclaimer.
*
* 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.
*
* Neither the name of the driver nor the names of its contributors may not be
* used to endorse or promote products derived from this software without specific
* prior written permission.
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "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 COPYRIGHT HOLDER OR CONTRIBUTORS 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.mariadb.jdbc;
import org.mariadb.jdbc.internal.util.Utils;
import org.mariadb.jdbc.internal.util.exceptions.ExceptionMapper;
import java.io.*;
import java.sql.Blob;
import java.sql.SQLException;
public class MariaDbBlob implements Blob, Serializable {
private static final long serialVersionUID = 8557003556592493381L;
/**
* the actual blob content.
*/
protected byte[] blobContent;
/**
* the size of the blob.
*/
protected int actualSize;
/**
* creates an empty blob.
*/
public MariaDbBlob() {
blobContent = new byte[0];
}
/**
* creates a blob with content.
*
* @param bytes the content for the blob.
*/
public MariaDbBlob(byte[] bytes) {
if (bytes == null) {
throw new AssertionError("byte array is null");
}
this.blobContent = bytes;
this.actualSize = bytes.length;
}
private void writeObject(ObjectOutputStream out)
throws IOException {
out.writeInt(actualSize);
if (actualSize > 0) {
out.write(blobContent, 0, actualSize);
}
}
private void readObject(ObjectInputStream in)
throws IOException, ClassNotFoundException {
actualSize = in.readInt();
blobContent = new byte[actualSize];
if (actualSize > 0) {
in.readFully(blobContent, 0, actualSize);
}
}
/**
* Returns the number of bytes in the BLOB
value designated by this Blob
object.
*
* @return length of the BLOB
in bytes
* @throws SQLException if there is an error accessing the length of the BLOB
*/
public long length() throws SQLException {
return actualSize;
}
/**
* Retrieves all or part of the BLOB
value that this Blob
object represents, as an array
* of bytes. This 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; the value for length must be 0 or greater
* @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
* @throws SQLException if there is an error accessing the BLOB
value; if pos is less than 1
* or length is less than 0
* @see #setBytes
* @since 1.2
*/
public byte[] getBytes(final long pos, final int length) throws SQLException {
if (pos < 1) {
throw ExceptionMapper.getSqlException("Pos starts at 1");
}
final int arrayPos = (int) (pos - 1);
return Utils.copyRange(blobContent, arrayPos, arrayPos + length);
}
/**
* Retrieves the BLOB
value designated by this Blob
instance as a stream.
*
* @return a stream containing the BLOB
data
* @throws SQLException if something went wrong
* @see #setBinaryStream
*/
public InputStream getBinaryStream() throws SQLException {
return getBinaryStream(1, actualSize);
}
/**
* 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
*/
public InputStream getBinaryStream(final long pos, final long length) throws SQLException {
if (pos < 1) {
throw ExceptionMapper.getSqlException("Out of range (position should be > 0)");
}
if (pos - 1 > actualSize) {
throw ExceptionMapper.getSqlException("Out of range (position > stream size)");
}
if (pos + length - 1 > actualSize) {
throw ExceptionMapper.getSqlException("Out of range (position + length - 1 > streamSize)");
}
return new ByteArrayInputStream(blobContent, (int) pos - 1, (int) length);
}
/**
* Retrieves the byte position at which the specified byte array 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
*/
public long position(final byte[] pattern, final long start) throws SQLException {
if (start < 1) {
throw ExceptionMapper.getSqlException("Start should be > 0, first position is 1.");
}
if (start > actualSize) {
throw ExceptionMapper.getSqlException("Start should be <= " + actualSize);
}
final long actualStart = start - 1;
for (int i = (int) actualStart; i < actualSize; i++) {
if (blobContent[i] == pattern[0]) {
boolean isEqual = true;
for (int j = 1; j < pattern.length; j++) {
if (i + j >= actualSize) {
return -1;
}
if (blobContent[i + j] != pattern[j]) {
isEqual = false;
}
}
if (isEqual) {
return i + 1;
}
}
}
return -1;
}
/**
* Retrieves 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
*/
public long position(final Blob pattern, final long start) throws SQLException {
return position(pattern.getBytes(1, (int) pattern.length()), start);
}
/**
* Writes the given array of bytes to the BLOB
value that this Blob
object represents,
* starting at position pos
, and returns the number of bytes written. The array of bytes will overwrite
* the existing bytes in the Blob
object starting at the position pos
. If the end of the
* Blob
value is reached while writing the array of bytes, then the length of the Blob
* value will be increased to accomodate the extra bytes.
*
* Note: If the value specified for pos
is greater then the length+1 of the BLOB
* value then the behavior is undefined. Some JDBC drivers may throw a SQLException
while other drivers
* may support this operation.
*
* @param pos the position in the BLOB
object at which to start writing; the first position is 1
* @param bytes the array of bytes to be written to the BLOB
value that this Blob
object
* represents
* @return the number of bytes written
* @see #getBytes
* @since 1.4
*/
public int setBytes(final long pos, final byte[] bytes) throws SQLException {
final int arrayPos = (int) pos - 1;
final int bytesWritten;
if (blobContent == null) {
this.blobContent = new byte[arrayPos + bytes.length];
bytesWritten = blobContent.length;
this.actualSize = bytesWritten;
} else if (blobContent.length > arrayPos + bytes.length) {
bytesWritten = bytes.length;
} else {
blobContent = Utils.copyWithLength(blobContent, arrayPos + bytes.length);
actualSize = blobContent.length;
bytesWritten = bytes.length;
}
System.arraycopy(bytes, 0, this.blobContent, arrayPos, bytes.length);
return bytesWritten;
}
/**
* 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. Writing starts at position
* pos
in the BLOB
value; len
bytes from the given byte array are written.
* The array of bytes will overwrite the existing bytes in the Blob
object starting at the position
* pos
. If the end of the Blob
value is reached while writing the array of bytes, then
* the length of the Blob
value will be increased to accomodate the extra bytes.
*
* Note: If the value specified for pos
is greater then the length+1 of the BLOB
* value then the behavior is undefined. Some JDBC drivers may throw a SQLException
while other drivers
* may support this operation.
*
* @param pos the position in the BLOB
object at which to start writing; the first position is 1
* @param bytes the array of bytes to be written to this BLOB
object
* @param offset the offset into the array bytes
at which to start reading the bytes to be set
* @param len the number of bytes to be written to the BLOB
value from the array of bytes
* bytes
* @return the number of bytes written
* @throws SQLException if there is an error accessing the BLOB
value or if pos is less than
* 1
* @see #getBytes
*/
public int setBytes(final long pos,
final byte[] bytes,
final int offset,
final int len) throws SQLException {
int bytesWritten = 0;
if (blobContent == null) {
this.blobContent = new byte[(int) (pos + bytes.length) - (len - offset)];
for (int i = (int) pos + offset; i < len; i++) {
this.blobContent[((int) (pos + i))] = bytes[i];
bytesWritten++;
}
} else if (this.blobContent.length < (pos + bytes.length) - (len - offset)) {
for (int i = (int) pos + offset; i < len; i++) {
this.blobContent[((int) (pos + i))] = bytes[i];
bytesWritten++;
}
}
this.actualSize += bytesWritten;
return bytesWritten;
}
/**
* Retrieves a stream that can be used to write to the BLOB
value that this Blob
object
* represents. The stream begins at position pos
. The bytes written to the stream will overwrite the
* existing bytes in the Blob
object starting at the position pos
. If the end of the
* Blob
value is reached while writing to the stream, then the length of the Blob
value
* will be increased to accomodate the extra bytes.
*
* Note: If the value specified for pos
is greater then the length+1 of the BLOB
* value then the behavior is undefined. Some JDBC drivers may throw a SQLException
while other drivers
* may support this operation.
*
* @param pos the position in the BLOB
value at which to start writing; the first position is 1
* @return a java.io.OutputStream
object to which data can be written
* @throws SQLException if there is an error accessing the BLOB
value or if pos is less than
* 1
* @see #getBinaryStream
* @since 1.4
*/
public OutputStream setBinaryStream(final long pos) throws SQLException {
if (pos < 1) {
throw ExceptionMapper.getSqlException("Invalid position in blob");
}
return new BlobOutputStream(this, (int) (pos - 1));
}
/**
* Truncates the BLOB
value that this Blob
object represents to be len
bytes
* in length.
*
* Note: If the value specified for pos
is greater then the length+1 of the BLOB
* value then the behavior is undefined. Some JDBC drivers may throw a SQLException
while other drivers
* may support this operation.
*
* @param len the length, in bytes, to which the BLOB
value that this Blob
object
* represents should be truncated
* @throws SQLException if there is an error accessing the BLOB
value or if len is less than
* 0
*/
public void truncate(final long len) throws SQLException {
this.blobContent = Utils.copyWithLength(this.blobContent, (int) len);
this.actualSize = (int) len;
}
/**
* This method frees the Blob
object and releases the resources that it holds. The object is invalid
* once the free
method is called.
*
* After free
has been called, any attempt to invoke a method other than free
will result
* in a SQLException
being thrown. If free
is called multiple times, the subsequent calls
* to free
are treated as a no-op.
*/
public void free() {
this.blobContent = null;
this.actualSize = 0;
}
}