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

com.ibm.as400.access.SQLVarcharBase Maven / Gradle / Ivy

There is a newer version: 20.0.8
Show newest version
///////////////////////////////////////////////////////////////////////////////
//                                                                             
// JTOpen (IBM Toolbox for Java - OSS version)                                 
//                                                                             
// Filename: SQLVarchar.java
//                                                                             
// The source code contained herein is licensed under the IBM Public License   
// Version 1.0, which has been approved by the Open Source Initiative.         
// Copyright (C) 1997-2014 International Business Machines Corporation and     
// others. All rights reserved.                                                
//                                                                             
///////////////////////////////////////////////////////////////////////////////

package com.ibm.as400.access;

import java.io.InputStream;
import java.io.Reader;
import java.io.StringReader;
import java.sql.Blob;
import java.sql.Clob;
/* ifdef JDBC40 */
import java.sql.NClob;
import java.sql.RowId;
/* endif */ 
import java.sql.SQLException;
/* ifdef JDBC40 */
import java.sql.SQLXML;
/* endif */ 
import java.sql.Time;
import java.sql.Timestamp;
import java.util.Calendar;
/* ifdef JDBC40 */
import java.util.GregorianCalendar; 
/* endif */ 

/* ifdef JDBC42
import java.time.LocalTime;
import java.time.LocalDate;
import java.time.LocalDateTime;
endif */ 

import java.net.URL;

abstract class SQLVarcharBase
extends SQLDataBase  implements SQLVariableCompressible
{
    static final String copyright = "Copyright (C) 1997-2013 International Business Machines Corporation and others.";

    // Private data.
    protected int                     length_;
    protected int                     maxLength_;
    protected String                  value_;
    // We need the untruncated value for UTF-8 conversions @X4A
    protected String                  untruncatedValue_; 
    protected int                     bytesPerCharacter_; 
    protected int                     sizeAfterTruncation_ = 0; 

    // Note: maxLength is in bytes not counting 2 for LL.
    //
    SQLVarcharBase(SQLConversionSettings settings, int length, int maxLength, String value)
    {
        super(settings); 
        length_         = length;
        maxLength_      = maxLength;
        value_          = value;
        untruncatedValue_ = value; 
        bytesPerCharacter_ = 1; 
    }

    SQLVarcharBase(SQLConversionSettings settings, int length, int maxLength, String value, int bytesPerCharacter)
    {
        super(settings); 
        length_         = length;
        maxLength_      = maxLength;
        value_          = value;
        untruncatedValue_ = value; 
        bytesPerCharacter_ = bytesPerCharacter;  
    }

    // @A2A
    // Added method trim() to trim the string.
    public void trim()                                // @A2A
    {                                                 // @A2A
        value_ = value_.trim();                       // @A2A
    }                                                 // @A2A

    //---------------------------------------------------------//
    //                                                         //
    // CONVERSION TO AND FROM RAW BYTES                        //
    //                                                         //
    //---------------------------------------------------------//

    public void convertFromRawBytes(byte[] rawBytes, int offset, ConvTable ccsidConverter, boolean ignoreConversionErrors)
    throws SQLException
    {
        length_ = BinaryConverter.byteArrayToUnsignedShort(rawBytes, offset);

        int bidiStringType = settings_.getBidiStringType();
        // if bidiStringType is not set by user, use ccsid to get value
        if(bidiStringType == -1)
            bidiStringType = ccsidConverter.bidiStringType_;
            
        BidiConversionProperties bidiConversionProperties = new BidiConversionProperties(bidiStringType);  //@KBA
        bidiConversionProperties.setBidiImplicitReordering(settings_.getBidiImplicitReordering());         //@KBA
        bidiConversionProperties.setBidiNumericOrderingRoundTrip(settings_.getBidiNumericOrdering());      //@KBA

        try{
            // If the field is VARGRAPHIC, length_ contains the number
            // of characters in the string, while the converter is expecting
            // the number of bytes. Thus, we need to multiply length_ by bytesPerCharacter.
            sizeAfterTruncation_ = length_*bytesPerCharacter_; 
            value_ = ccsidConverter.byteArrayToString(rawBytes, offset+2, length_*bytesPerCharacter_, bidiConversionProperties);   //@KBC changed to use bidiConversionProperties instead of bidiStringType
            untruncatedValue_ = value_; 
        }catch(Exception e){
            JDError.throwSQLException(JDError.EXC_CHAR_CONVERSION_INVALID, e);
        }
    }

    public void convertToRawBytes(byte[] rawBytes, int offset, ConvTable ccsidConverter)
    throws SQLException
    {
        try
        {
            int bidiStringType = settings_.getBidiStringType();
            // if bidiStringType is not set by user, use ccsid to get value
            if(bidiStringType == -1)
                bidiStringType = ccsidConverter.bidiStringType_;
                
            BidiConversionProperties bidiConversionProperties = new BidiConversionProperties(bidiStringType);  //@KBA
            bidiConversionProperties.setBidiImplicitReordering(settings_.getBidiImplicitReordering());         //@KBA
            bidiConversionProperties.setBidiNumericOrderingRoundTrip(settings_.getBidiNumericOrdering());      //@KBA
            truncated_ = 0; outOfBounds_ = false ; 
            
            String value = value_; 
            // For CCSID 1208 we must use the untruncated value to avoid having half a UTF-16 character
            if (ccsidConverter.getCcsid() == 1208) {
              value = untruncatedValue_; 
            }
            // The length in the first 2 bytes is actually the length in characters.
            byte[] temp = ccsidConverter.stringToByteArray(value, bidiConversionProperties);   //@KBC changed to used bidiConversionProperties instead of bidiStringType
            
            BinaryConverter.unsignedShortToByteArray(temp.length/bytesPerCharacter_, rawBytes, offset);
            sizeAfterTruncation_ = temp.length; 
            if(temp.length > maxLength_)
            {
              // Normally truncation is detected before setting the string.  In the case of UTF-8
              // truncation may be detected here. 
              // Make sure the length sent to the server is the truncated length -- otherwise the string
              // would include garbage. @V6A 
              BinaryConverter.unsignedShortToByteArray(maxLength_/bytesPerCharacter_, rawBytes, offset);
              
              sizeAfterTruncation_ = maxLength_; 
                truncated_ = temp.length - maxLength_;  /*@H2C*/
                // Not sure why maxLength_ was changed. 
                // maxLength_ = temp.length;
                // We now set the truncated information and let the truncated data through
                // JDError.throwSQLException(this, JDError.EXC_INTERNAL, "Change Descriptor");
                // the testDataTruncation method will be called to see if
                // the truncation message should be thrown. 
                System.arraycopy(temp, 0, rawBytes, offset+2, maxLength_);
            } else { 
              System.arraycopy(temp, 0, rawBytes, offset+2, temp.length);
            }

            // The buffer we are filling with data is big enough to hold the entire field.
            // For varchar fields the actual data is often smaller than the field width.
            // That means whatever is in the buffer from the previous send is sent to the
            // system.  The data stream includes actual data length so the old bytes are not 
            // written to the database, but the junk left over may decrease the effectiveness 
            // of compression.  The following code will write hex 0s to the buffer when
            // actual length is less that field length.  Note the 0s are written only if 
            // the field length is pretty big.  The data stream code (DBBaseRequestDS)
            // does not compress anything smaller than 1K.
            if( (maxLength_ - temp.length > 16))  //@rle
            {
                int stopHere = offset + 2 + maxLength_;
                for(int i=offset + 2 + temp.length; i maxLength_) {
        truncated_ = temp.length - maxLength_; /* @H2C */
        // maxLength_ = temp.length;
        // JDError.throwSQLException(this, JDError.EXC_INTERNAL,
        // "Change Descriptor");
        // @X6D.
        // Complete @H2 fix and report truncation
        System.arraycopy(temp, 0, rawBytes, offset + 2, maxLength_);
        bytesWritten += maxLength_;
      } else {
        if (temp.length > 0) {
          System.arraycopy(temp, 0, rawBytes, offset + 2, temp.length);
          bytesWritten += temp.length;
        }
      }
    } catch (Exception e)        {
            JDError.throwSQLException(this, JDError.EXC_INTERNAL, e);
        }
        return bytesWritten; 
    }


    
    
    //---------------------------------------------------------//
    //                                                         //
    // SET METHODS                                             //
    //                                                         //
    //---------------------------------------------------------//

    public void set(Object object, Calendar calendar, int scale)
    throws SQLException
    {
        String value = null;                                                        // @C1A

        if(object instanceof String)
            value = (String) object;                                                // @C1C

        else if(object instanceof Number)
            value = object.toString();                                              // @C1C

        else if(object instanceof Boolean)
        { 
            // @PDC
            // if "translate boolean" == false, then use "0" and "1" values to match native driver
            if(settings_.getTranslateBoolean() == true)
                value = object.toString();  //"true" or "false"     
            else
                value = ((Boolean)object).booleanValue() == true ? "1" : "0";
        }
        else if(object instanceof Time)
            value = SQLTime.timeToString((Time) object, settings_, calendar);      // @C1C

        else if(object instanceof Timestamp)
            value = SQLTimestamp.timestampToStringTrimTrailingZeros((Timestamp) object, calendar, settings_);  // @C1C

        else if(object instanceof java.util.Date)                                  // @F5M @F5C
            value = SQLDate.dateToString((java.util.Date) object, settings_, calendar); // @C1C @F5C

        else if(object instanceof URL)
            value = object.toString();

        else if( object instanceof Clob)
        {                                                                          // @C1C
            Clob clob = (Clob)object;                                              // @C1C
            value = clob.getSubString(1, (int)clob.length());                      // @C1C  @D1
        }                                                                     // @C1C
        else if(object instanceof Reader)
        {
          value = getStringFromReader((Reader) object, ALL_READER_BYTES, this);
        }
            
/* ifdef JDBC40 */
        else if(object instanceof SQLXML) //@PDA jdbc40 
        {    
            SQLXML xml = (SQLXML)object;
            value = xml.getString();
        }   
/* endif */ 
        /* ifdef JDBC42

        else if(object instanceof java.time.LocalDate) 
        {    
             value = SQLDate.localDateToString((java.time.LocalDate)object, settings_, calendar); 
        }
        
        else if(object instanceof LocalTime)
            value = SQLTime.localTimeToString((LocalTime) object, settings_, calendar);   

        else if(object instanceof LocalDateTime)
            value = SQLTimestamp.localDateTimeToStringTrimTrailingZeros((LocalDateTime) object, calendar, settings_); 

             
        endif */ 

        if(value == null)           {                                                // @C1C
          if (JDTrace.isTraceOn()) {
              if (object == null) { 
                  JDTrace.logInformation(this, "Unable to assign null object");
                } else { 
                    JDTrace.logInformation(this, "Unable to assign object("+object+") of class("+object.getClass().toString()+")");
                }
          }

          JDError.throwSQLException(this, JDError.EXC_DATA_TYPE_MISMATCH);
        }
        value_ = value;                                                            // @C1A
        untruncatedValue_ = value_; 


        // Truncate if necessary.
        int valueLength = value_.length();

        int truncLimit = maxLength_ / bytesPerCharacter_;              // @F2a

        if(valueLength > truncLimit)             // @F2c
        {
            value_ = value_.substring(0, truncLimit); // @F2c
            truncated_ = valueLength - truncLimit;     // @F2c
            outOfBounds_ = false;
        }
        else
            truncated_ = 0; outOfBounds_ = false; 

        length_ = value_.length();
    }

    //---------------------------------------------------------//
    //                                                         //
    // DESCRIPTION OF SQL TYPE                                 //
    //                                                         //
    //---------------------------------------------------------//

    public int getSQLType()
    {
        return SQLData.VARCHAR;
    }

    public String getCreateParameters()
    {
        return AS400JDBCDriver.getResource("MAXLENGTH",null);
    }

    public int getDisplaySize()
    {
        return maxLength_;
    }

    //@F1A JDBC 3.0
    public String getJavaClassName()
    {
        return "java.lang.String";   
    }

    public String getLiteralPrefix()
    {
        return "\'";
    }

    public String getLiteralSuffix()
    {
        return "\'";
    }

    public String getLocalName()
    {
        return "VARCHAR";
    }

    public int getMaximumPrecision()
    {
        return 32739;
    }

    public int getMaximumScale()
    {
        return 0;
    }

    public int getMinimumScale()
    {
        return 0;
    }

    public int getNativeType()
    {
        return 448;
    }

    public int getPrecision()
    {
        return maxLength_ / bytesPerCharacter_;
    }

    public int getRadix()
    {
        return 0;
    }

    public int getScale()
    {
        return 0;
    }

    public int getType()
    {
        return java.sql.Types.VARCHAR;
    }

    public String getTypeName()
    {
        return "VARCHAR";
    }

    public boolean isSigned()
    {
        return false;
    }

    public boolean isText()
    {
        return true;
    }

    // Returns the size after truncation has occurred 
    public int getActualSize()
    {
      if (sizeAfterTruncation_ != 0) {
        return sizeAfterTruncation_; 
      } else { 
        return value_.length();
      }
    }

    public int getTruncated()
    {
        return truncated_;
    }
    public boolean getOutOfBounds() {
      return outOfBounds_; 
    }

    //---------------------------------------------------------//
    //                                                         //
    // CONVERSIONS TO JAVA TYPES                               //
    //                                                         //
    //---------------------------------------------------------//


    public InputStream getBinaryStream()
    throws SQLException
    {
        truncated_ = 0; outOfBounds_ = false; 
        return new HexReaderInputStream(new StringReader(getString()));
    }

    public Blob getBlob()
    throws SQLException
    {
        truncated_ = 0; outOfBounds_ = false; 
        try
        {
            byte[] bytes = BinaryConverter.stringToBytes(getString());
            return new AS400JDBCBlob(bytes, bytes.length);
        }
        catch(NumberFormatException nfe)
        {
            // this field contains non-hex characters
            JDError.throwSQLException(this, JDError.EXC_DATA_TYPE_MISMATCH, nfe);
            return null;
        }
    }


    public byte[] getBytes()
    throws SQLException
    {
        truncated_ = 0; outOfBounds_ = false; 
        try
        {
            return BinaryConverter.stringToBytes(getString());
        }
        catch(NumberFormatException nfe)
        {
            // this field contains non-hex characters
            JDError.throwSQLException(this, JDError.EXC_DATA_TYPE_MISMATCH, nfe);
            return null;
        }
    }


    public Object getObject()
    throws SQLException
    {
        truncated_ = 0; outOfBounds_ = false; 
        // This is written in terms of getString(), since it will
        // handle truncating to the max field size if needed.
        return getString();
    }


    public String getString() throws SQLException
    {
        truncated_ = 0; outOfBounds_ = false; 
        // Truncate to the max field size if needed.
        // Do not signal a DataTruncation per the spec. @B1A
        int maxFieldSize = settings_.getMaxFieldSize();
        if((value_.length() > maxFieldSize) && (maxFieldSize > 0))
        {
            // @B1D truncated_ = value_.length() - maxFieldSize;
            return value_.substring(0, maxFieldSize);
        }
        else
        {
            // @B1D truncated_ = 0; outOfBounds_ = false; 
            return value_;
        }
    }

    
    //@pda jdbc40
    public String getNString() throws SQLException
    {
        // Truncate to the max field size if needed.
        // Do not signal a DataTruncation per the spec.
        int maxFieldSize = settings_.getMaxFieldSize();
        if((value_.length() > maxFieldSize) && (maxFieldSize > 0))
        {
            return value_.substring(0, maxFieldSize);
        }
        else
        {
            return value_;
        } 
    }

/* ifdef JDBC40 */
    //@pda jdbc40
    public RowId getRowId() throws SQLException
    {
        JDError.throwSQLException(this, JDError.EXC_DATA_TYPE_MISMATCH);
        return null;
    }

    //@pda jdbc40
    public SQLXML getSQLXML() throws SQLException
    {
        //This is written in terms of getString(), since it will
        // handle truncating to the max field size if needed.
        truncated_ = 0; outOfBounds_ = false; 
        return new AS400JDBCSQLXML(getString());     
    }
/* endif */ 
    
    public void saveValue() {
       savedValue_ = untruncatedValue_; 
    }
    
}





© 2015 - 2025 Weber Informatics LLC | Privacy Policy