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

com.pivotal.gemfirexd.internal.client.am.ClobLocatorInputStream Maven / Gradle / Ivy

/*
 
   Derby - Class com.pivotal.gemfirexd.internal.client.am.ClobLocatorInputStream
 
   Licensed to the Apache Software Foundation (ASF) under one or more
   contributor license agreements.  See the NOTICE file distributed with
   this work for additional information regarding copyright ownership.
   The ASF licenses this file to You under the Apache License, Version 2.0
   (the "License"); you may not use this file except in compliance with
   the License.  You may obtain a copy of the License at
 
      http://www.apache.org/licenses/LICENSE-2.0
 
   Unless required by applicable law or agreed to in writing, software
   distributed under the License is distributed on an "AS IS" BASIS,
   WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
   See the License for the specific language governing permissions and
   limitations under the License.
 
 */

/*
 * Changes for GemFireXD distributed data platform (some marked by "GemStone changes")
 *
 * Portions Copyright (c) 2010-2015 Pivotal Software, Inc. All rights reserved.
 *
 * Licensed under the Apache License, Version 2.0 (the "License"); you
 * may not use this file except in compliance with the License. You
 * may obtain a copy of the License at
 *
 * http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or
 * implied. See the License for the specific language governing
 * permissions and limitations under the License. See accompanying
 * LICENSE file.
 */

package com.pivotal.gemfirexd.internal.client.am;

import java.sql.SQLException;

import java.io.IOException;

import com.pivotal.gemfirexd.internal.shared.common.error.ExceptionUtil;
import com.pivotal.gemfirexd.internal.shared.common.reference.SQLState;
import com.pivotal.gemfirexd.internal.shared.common.sanity.SanityManager;

/**
 * An InputStream that will use an locator to fetch the
 * Clob value from the server.
 * 

* Closing a ByteArrayInputStream has no effect. The methods in * this class can be called after the stream has been closed without * generating an IOException. *

* This InputStream implementation is pretty basic. No * buffering of data is done. Hence, for efficieny #read(byte[]) * should be used instead of #read(). Marks are not supported, but it * should be pretty simple to extend the implementation to support * this. A more efficient skip implementation should also be * straight-forward. */ public class ClobLocatorInputStream extends java.io.InputStream { /** * Connection used to read Clob from server. */ // GemStone changes BEGIN /* (original code) private final Connection connection; */ // GemStone changes END /** * The Clob to be accessed. */ private final Clob clob; /** * Current position in the underlying Clob. * Clobs are indexed from 1 */ private long currentPos; /** * Create an InputStream for reading the * Clob value represented by the given locator based * Clob object. * @param connection connection to be used to read the * Clob value from the server * @param clob Clob object that contains locator for * the Clob value on the server. */ public ClobLocatorInputStream(Connection connection, Clob clob) throws SqlException{ if (SanityManager.DEBUG) { SanityManager.ASSERT(clob.isLocator()); } // GemStone changes BEGIN /* (original code) this.connection = connection; */ // GemStone changes END this.clob = clob; this.currentPos = 1; } /** * Create an InputStream for reading the * Clob value represented by the given locator based * Clob object. * @param connection connection to be used to read the * Clob value from the server * @param clob Clob object that contains locator for * the Clob value on the server. * @param pos the position inside the Clob from which * the reading must begin. */ public ClobLocatorInputStream(Connection connection, Clob clob, long pos) throws SqlException{ this(connection, clob); this.currentPos = pos; } /** * @see java.io.InputStream#read() * * This method fetches one byte at a time from the server. For more * efficient retrieval, use #read(byte[]). */ public int read() throws IOException { byte[] bytes = readBytes(1); if (bytes.length == 0) { // EOF return -1; } else { // convert byte in range [-128,127] to int in range [0,255] return bytes[0] & 0xff; } } /** * @see java.io.InputStream#read(byte[], int, int) */ public int read(byte[] b, int off, int len) throws IOException { if (len == 0) return 0; if ((off < 0) || (len < 0) || (len > b.length - off)) { throw new IndexOutOfBoundsException(); } byte[] bytes = readBytes(len); if (bytes.length == 0) { // EOF return -1; } else { System.arraycopy(bytes, 0, b, off, bytes.length); return bytes.length; } } /** * Read the next len bytes of the Clob * value from the server. * * @param len number of bytes to read * @throws java.io.IOException Wrapped SqlException if reading * from server fails. * @return byte[] containing the read bytes */ private byte[] readBytes(int len) throws IOException { try { int actualLength = (int )Math.min(len, clob.sqlLength() - currentPos + 1); // GemStone changes BEGIN String resultStr = clob.locatorProcs_. /* (original code) String resultStr = connection.locatorProcedureCall(). */ // GemStone changes END clobGetSubString(clob.getLocator(), currentPos, actualLength); byte[] result = getBytesFromString(resultStr); currentPos += result.length; return result; } catch (SqlException ex) { IOException ioEx = new IOException(); ioEx.initCause(ex); throw ioEx; } } /** * Returns a Byte array from the * String passed as Input. * * @param str the input String. * @return The Byte corresponding * to the String that was * input. */ private byte[] getBytesFromString(String str) { //The Byte array that will hold the final //converted Byte array that will be returned //to the user byte[] result = new byte[str.length()]; //Iterate through the String to //Convert each character in the //String for (int i = 1; i <= str.length(); i++) { //charAt function accpets a index that //starts from 0 and ranges to length()-1 char oneChar = str.charAt(i-1); if (oneChar <= 0xff) { //Check if the value is lesser //than maximum value that can //be stored in a byte. If it is //lesser store it directly in the //byte array result[i-1] = (byte)oneChar; } else { //The value is greater than the //maximum value that can be //stored. Use the value 0x003f //which corresponds to '?' //signifying an unknown character result[i-1] = 0x3f; } } return result; } }