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

org.mariadb.jdbc.MariaDbBlob Maven / Gradle / Ivy

/*
MariaDB Client for Java

Copyright (c) 2012-2014 Monty Program 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.ExceptionMapper;
import org.mariadb.jdbc.internal.util.Utils;

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(java.io.ObjectOutputStream out)
            throws IOException {
        out.writeInt(actualSize);
        if (actualSize > 0) {
            out.write(blobContent, 0, actualSize);
        }
    }

    private void readObject(java.io.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 java.sql.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 java.sql.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 java.sql.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 java.sql.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 java.sql.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 java.sql.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; } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy