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

com.ibm.as400.data.PcmlDataValues Maven / Gradle / Ivy

There is a newer version: 20.0.8
Show newest version
///////////////////////////////////////////////////////////////////////////////
//                                                                             
// JTOpen (IBM Toolbox for Java - OSS version)                              
//                                                                             
// Filename: PcmlDataValues.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-2000 International Business Machines Corporation and     
// others. All rights reserved.                                                
//                                                                             
///////////////////////////////////////////////////////////////////////////////

package com.ibm.as400.data;

import com.ibm.as400.access.AS400DataType;
import com.ibm.as400.access.AS400Text;
import com.ibm.as400.access.AS400Date;
import com.ibm.as400.access.AS400Time;
import com.ibm.as400.access.AS400Timestamp;
import com.ibm.as400.access.BidiStringType;
import com.ibm.as400.access.BinaryConverter;
import com.ibm.as400.access.Trace;                                  // @D3A

import java.math.BigDecimal;
import java.math.BigInteger;

import java.io.IOException;                                         // @C1A
import java.io.ObjectInputStream;                                   // @C1A
import java.io.ObjectOutputStream;                                  // @C1A
import java.io.OutputStream;
import java.io.Serializable;                                        // @C1A

import java.util.Hashtable;
import java.util.StringTokenizer;
import java.util.TimeZone;

class PcmlDataValues extends Object implements Serializable         // @C1C
{
    static final long serialVersionUID = -8169008879805188674L;	    // @C1A

    private PcmlData m_owner;      // PcmlData node that owns this object
    
    private PcmlDimensions m_indices; // Indices into owning PcmlNode's vectors of PcmlDataValues

    private Object m_value;        // Java native value: (String, Short, Integer, Long, Float, Double, BigDecimal. byte[])
    private long   m_valueTs;      // Correlation id of Java native value from PcmlDocument.getCorrelationID()
    private byte[] m_bytes;        // IBM i bytes (ebcdic/big-endian)
    private long   m_bytesTs;      // Correlation id of IBM i bytes from PcmlDocument.getCorrelationID()
    
    private int    m_bidiStringType;    // Type of string that is contained in the value @C6A

    // Default constructor
    private PcmlDataValues() 
    {
        m_value = null;
        m_bytes = null;
        m_valueTs = 0;
        m_bytesTs = 0;
        m_bidiStringType = BidiStringType.DEFAULT;                  // @C6A
    }

    // Constructor 
    PcmlDataValues(PcmlData creator, PcmlDimensions indices) 
    {
        this();
        m_owner = creator;
        m_indices = new PcmlDimensions(indices);
    }

    // Custom serialization
    private void writeObject(ObjectOutputStream out) throws IOException // @C1A
    {                                                               // @C1A
        synchronized (this)                                         // @C1A
        {                                                           // @C1A
            // For input values, serialize only the IBM i byte value 
            // if it is current than the Java value
            if ( m_owner.getUsage() == PcmlDocNode.INPUT )          // @C1A
            {                                                       // @C1A
                // If the byte value is more recent than the Java value
                // convert the bytes to a Java value
                if (m_bytesTs > m_valueTs)                          // @C1A
                {                                                   // @C1A
                    m_value = null;                                 // @C1A
                    m_valueTs = 0;                                  // @C1A
                }                                                   // @C1A
            }                                                       // @C1A

            // For ouput values, serialize only the Java value 
            // if it is current than the IBM i byte value
            if ( m_owner.getUsage() == PcmlDocNode.OUTPUT )         // @C1A
            {
                // If the byte value is more recent than the Java value
                // convert the bytes to a Java value
                if (m_valueTs > m_bytesTs)                          // @C1A
                {                                                   // @C1A
                    m_bytes = null;                                 // @C1A
                    m_bytesTs = 0;                                  // @C1A
                }                                                   // @C1A
            }                                                       // @C1A

            // Perform default serialization
            out.defaultWriteObject();                               // @C1A
        } // end of synchronized code                               // @C1A
    }                                                               // @C1A

	// Custom deserialization
	private void readObject(ObjectInputStream in)
		throws IOException, ClassNotFoundException                  // @C1A
	{                                                               // @C1A
		// Default deserialization
		in.defaultReadObject();                                     // @C1A
	}                                                               // @C1A

	// Custom deserialization post-processing
	// This processing cannot be done during readObject() because
	// references to parent objects in the document are not yet set
	// due to the recursive nature of deserialization.
    void readObjectPostprocessing()                                 // @C1A
    {                                                               // @C1A
		if (m_valueTs == m_bytesTs && m_valueTs > 0)                // @C1A
		{                                                           // @C1A
	        m_valueTs = m_owner.getDoc().getDeserializationTimestamp(); // @C1A
	        m_bytesTs = m_owner.getDoc().getDeserializationTimestamp(); // @C1A
		}                                                           // @C1A
		if (m_valueTs > m_bytesTs)                                  // @C1A
		{                                                           // @C1A
	        m_valueTs = m_owner.getDoc().getDeserializationTimestamp(); // @C1A
	        m_bytesTs = 0;                                          // @C1A
		}                                                           // @C1A
		if (m_bytesTs > m_valueTs)                                  // @C1A
		{                                                           // @C1A
	        m_bytesTs = m_owner.getDoc().getDeserializationTimestamp(); // @C1A
	        m_valueTs = 0;                                          // @C1A
		}                                                           // @C1A
    }                                                               // @C1A

    // Flush data values
    // This allows the object to be reused when the
    // m_owner (PcmlData object) redimensions its vectors.
    void flushValues()
    {
        m_value = null;
        m_bytes = null;
        m_valueTs = 0;
        m_bytesTs = 0;
    }

    // Get qualified name of data object
    String getQualifiedName() 
    {
        return m_owner.getQualifiedName();
    }

    // Get name for exceptions
    String getNameForException() 
    {
        return m_owner.getNameForException();
    }


    // Get Timestamp of data
    long getTimestamp() 
    {
        return Math.max(m_valueTs, m_bytesTs);
    }

    // Get Java native value
    public Object getValue() throws PcmlException 
    {
        return bytesToValue();
    }

    // Set Java native value
    public void setValue(Object v) throws PcmlException 
    {
        // Do not allow a null value to be set
        if (v == null)
            throw new PcmlException(DAMRI.NULL_VALUE, new Object[] {getNameForException()} );

        // If the new value matches the Java type for this element
        // store the new value.
        if (getDataType() == PcmlData.STRUCT) {                    // @D0A
            throw new PcmlException(DAMRI.STRUCT_VALUE, new Object[] {getNameForException()} );   // @D0A
        }
        
        if (getDataType() == PcmlData.CHAR && 
            (v instanceof String || v instanceof char[])) {  //@AI6A
        	m_value = v;
        } 
        else if ( v.getClass().equals(getValueClass())) 
        {
            if (v instanceof BigDecimal) {
              m_value = ((BigDecimal)v).setScale(getPrecision(), BigDecimal.ROUND_HALF_EVEN);       // @D0A
            }
            else {
              m_value = v;
            }
        }
        // New value does not match the Java type for this element.
        // Convert to the Java type needed -- errors may occur.
        else 
        {
           // Determine the timezone
          TimeZone timeZone = m_owner.getDoc().getTimeZone(); 
            m_value = convertValue(v, getDataType(), getLength(), getPrecision(), getNameForException(), timeZone);
        }
        // Update the value timestamp
        m_valueTs = m_owner.getDoc().getCorrelationID();
    }

    // Set IBM i bytes
    public void setBytes(byte[] ba) 
    {
        m_bytes = ba;
        m_bytesTs = m_owner.getDoc().getCorrelationID();           // @C7C
    }

    // Set the bidi string type 
    public void setStringType(int type)             // @C6A
    {
        if (m_bidiStringType == type)               // @C6A
        {
            // If new type is same as old, just return
            return;                                 // @C6A
        }
        // If we are setting a new string type, we want to 
        // clear the java value and timestamp.  This will ensure
        // that the correct conversion occurs on subsequent
        // getValue() calls
        m_bidiStringType = type;                    // @C6A
        m_value = null;                             // @C6A
        m_valueTs = 0;                              // @C6A

    }

    // Convert IBM i bytes to Java native value
    private Object bytesToValue() throws PcmlException 
    {
        // If the byte value is more recent than the Java value
        // convert the bytes to a Java value
        if (m_bytesTs > m_valueTs) 
        {
            m_valueTs = m_bytesTs;  // Update value timestamp
            toObject(m_bytes);
        }

        // If value is not set and there is an
        // init= value specified, initialize the value
        if (m_value == null) 
        {
            String initValue = m_owner.getInit();
            if (initValue != null)
                setValue(initValue);
        }
        
        if (m_value == null)    // If value has not been set       @C5A
            return m_value;     // return null                     @C5A
                
        // Return the Java value
        if (getDataType() == PcmlData.BYTE)
        {
            // Make a copy of the byte array, so caller can 
            // modify the value without repercussions
            byte[] srcArray = (byte[]) m_value;
            byte[] cloneArray = new byte[srcArray.length];
            System.arraycopy(srcArray, 0, cloneArray, 0, srcArray.length);
            return cloneArray;
        }
        else
        {
            // No need to make copy for other data type because they
            // are immutable objects.
            return m_value;          
        }
    }

    int getDataType()
    {
        return m_owner.getDataType();
    }

    int getLength() throws PcmlException
    {
        String charType = m_owner.getCharType();                    // @D2A
        if ((charType != null) && (charType.equals("twobyte")))
        {
            return (resolveIntegerValue( m_owner.getLength(),
                                        m_owner.getLengthId() ) * 2);    // @D2A
        }
        else
        {
        return resolveIntegerValue( m_owner.getLength(),
                                    m_owner.getLengthId() );
    }
    }

    int getOffset() throws PcmlException
    {
        return resolveIntegerValue( m_owner.getOffset(),
                                    m_owner.getOffsetId() );
    }

    int getCcsid() throws PcmlException
    {
        int tmpCcsid = resolveIntegerValue( m_owner.getCcsid(),
                                            m_owner.getCcsidId() );
        // If a CCSID is not explicitly defined for this element,
        // use the CCSID from the  element
        if (tmpCcsid == 0)
        {                                                           // @C2A
            PcmlNode node = m_owner.getParent();                    // @C2A
            node = node.getParent();    // get next ancestor           @C7A
            while (node != m_owner.getDoc())                        // @C2A @C7C
                node = node.getParent();                            // @C2A
            // If this element is a descendent of a Program element
            // use the CCSID saved the last time the 
            if (node instanceof PcmlProgram)                        // @C2A
                return ((PcmlProgram) node).getProgramCCSID();      // @C2A
            else if (node instanceof RfmlDocument)                  // @D0A
                return ((RfmlDocument) node).getCcsidInt();         // @D0A
            else                                                    // @C2A
                return m_owner.getDoc().getAs400().getCcsid();      // @C2A
        }                                                           // @C2A
        else
            return tmpCcsid;
    }


    // Returns the precision= attribute
    // The precision is only meaningful when type=int or packed
    // For type=int:
    //   A 2-byte signed integer is:   length=2 precision=15
    //   A 2-byte unsigned integer is: length=2 precision=16
    //   A 4-byte signed integer is:   length=4 precision=31
    //   A 4-byte unsigned integer is: length=4 precision=32
    // For type=packed:
    //   length= species the total number of digits
    //   precision= specifies the number of decimal digits
    int getPrecision()
    {
        return m_owner.getPrecision();
    }


    // Returns the dateformat= attribute
    // The dateformat is only meaningful when type=date
    String getDateFormat()
    {
        return m_owner.getDateFormat();
    }

    // Returns the dateseparator= attribute
    // The dateseparator is only meaningful when type=date
    String getDateSeparator()
    {
        return m_owner.getDateSeparator();
    }


    // Returns the timeformat= attribute
    // The timeformat is only meaningful when type=time
    String getTimeFormat()
    {
        return m_owner.getTimeFormat();
    }

    // Returns the timeseparator= attribute
    // The timeseparator is only meaningful when type=time
    String getTimeSeparator()
    {
        return m_owner.getTimeSeparator();
    }

    String getTrim()                                                // @D1A
    {                                                               // @D1A
        return m_owner.getTrim();                                   // @D1A
    }                                                               // @D1A

    boolean isArray()
    {
        return m_owner.isArray();
    }

    // Return number of bytes required to perform toBytes();
    // This is similar to C++ DataObject::FlattenedLengthAt()
    public int byteLength() throws PcmlException 
    {

        int length = getLength();

        switch (getDataType()) 
        {
            case PcmlData.CHAR:
            case PcmlData.INT:
            case PcmlData.FLOAT:
            case PcmlData.BYTE:
                return length;

            case PcmlData.PACKED:
                return length / 2 + 1; // Packed contains 2 digits per byte plus sign

            case PcmlData.ZONED:
                return length;        // Zoned contains 1 digit per byte, sign is in low order digit

            case PcmlData.DATE:
                if (length == 0) {
                  String format = getDateFormat();
                  if (format != null) {
                    length = AS400Date.getByteLength(AS400Date.toFormat(format), PcmlNode.separatorAsChar(getDateSeparator()));
                  }
                }
                return length;

            case PcmlData.TIME:
                if (length == 0) {
                  String format = getTimeFormat();
                  if (format != null) {
                    length = AS400Time.getByteLength(AS400Time.toFormat(format), PcmlNode.separatorAsChar(getTimeSeparator()));
                  }
                }
                return length;

            case PcmlData.TIMESTAMP:
                if (length == 0) {
                  //length = AS400Timestamp.getByteLength(null);
                  length = 26;  // we publicly support only the 26-byte timestamp format
                }
                return length;

            default:
                throw new PcmlException(DAMRI.BAD_DATA_TYPE, new Object[] {Integer.valueOf(getDataType()) , getNameForException()} );

        }
    }

    // Converts an object to the correct type (class) for
    // the defined data tpe.
    // The setValue method allows most data types to be set using 
    // either the Java class associaed with the data type or using
    // a String. This method converts the value to the Java class 
    // defined for the data type. Exceptions such as 
    // NumberFormatException may result.
    private Class getValueClass() throws PcmlException 
    {
        int dataLength = getLength();
        int dataPrecision = getPrecision();
        int dataType = getDataType();

        //try   //@O6D 
        {
            switch (dataType) 
            {
    
                case PcmlData.CHAR:
                    return java.lang.String.class;// @O6C Class.forName("java.lang.String");
                case PcmlData.VARCHAR:
                	return java.lang.String.class;
                case PcmlData.INT:
                    if (dataLength == 2) 
                    {
                        if (dataPrecision == 16) 
                        {
                            return java.lang.Integer.class; //@O6C Class.forName("java.lang.Integer");
                        }
                        else 
                        { // dataPrecision == 15 or defaulted
                            return java.lang.Short.class; //@O6C Class.forName("java.lang.Short");
                        }
                    }
                    else 
                    if (dataLength == 4)                            // @C3A
                    { // dataLength == 4
                        if (dataPrecision == 32) 
                        {
                            return java.lang.Long.class;//@O6C Class.forName("java.lang.Long");
                        }
                        else 
                        { // dataPrecision == 31 or defaulted
                            return java.lang.Integer.class;//@O6C Class.forName("java.lang.Integer");
                        }
                    }
                    else                                            // @C3A
                    if (dataLength == 8)                            // @C3A
                    { // dataLength == 8                            // @C3A
                      if (dataPrecision == 64) 
                      {
                          return java.math.BigInteger.class;//@O6C Class.forName("java.math.BigInteger");
                      }
                      else 
                      { // dataPrecision == 63 or defaulted
                          return java.lang.Long.class;//@O6C Class.forName("java.lang.Long");  // @C3A
                      }  
                    }                                               // @C3A
    
                case PcmlData.PACKED:
                case PcmlData.ZONED:
                    return java.math.BigDecimal.class;//@O6C Class.forName("java.math.BigDecimal");
    
    
                case PcmlData.FLOAT:
                    if (dataLength == 4) 
                    {
                        return java.lang.Float.class;//@O6C Class.forName("java.lang.Float");
                    }
                    else 
                    { // Must be length=8
                        return java.lang.Double.class;//@O6C Class.forName("java.lang.Double");
                    }
    
                case PcmlData.BYTE:
                    return (new byte[0]).getClass();

                case PcmlData.DATE:
                    return java.sql.Date.class;//@O6C Class.forName("java.sql.Date");

                case PcmlData.TIME:
                    return java.sql.Time.class;//@O6C Class.forName("java.sql.Time");

                case PcmlData.TIMESTAMP:
                    return java.sql.Timestamp.class;//@O6C Class.forName("java.sql.Timestamp");


                default:
                    throw new PcmlException(DAMRI.BAD_DATA_TYPE, new Object[] {Integer.valueOf(getDataType()) , getNameForException()} );
    
            } // END: switch (getDataType())
        }
       /* catch (ClassNotFoundException e)  //@O6D 
        {
            throw new PcmlException(DAMRI.CLASS_NOT_FOUND, new Object[] {Integer.valueOf(getDataType()) , getNameForException()} );
        }*/
    }

    // Convert Java object IBM i bytes
    // Returns the number of bytes converted
    public int toBytes(OutputStream bytes, int offset) throws PcmlException 
    {
        Object value = null;
        int dataType = getDataType();
        int dataLength = getLength();
        if (dataLength == 0) { // this can happen with date, time, timestamp
          dataLength = byteLength();
        }
        int bytesConverted = 0;                                     // @B2A

        // Get the Java object and make sure it has been set.
        value = getValue();
        if (value == null) 
        {
            throw new PcmlException(DAMRI.INPUT_VALUE_NOT_SET, new Object[] {getNameForException()} );
        }
        
        // Get a converter from the PcmlDocument node.
        // PcmlDocument will either create a converter or return an existing one.
        AS400DataType converter = m_owner.getDoc().getConverter(dataType, dataLength, getPrecision(), getCcsid(), getDateFormat(), getDateSeparator(), getTimeFormat(), getTimeSeparator());
        byte[] byteArray = new byte[dataLength];
        if (dataType != PcmlData.CHAR)                              // @C6A
        {
            synchronized(converter)                                 // @B1A
            {                                                       // @B1A
                converter.toBytes(value, byteArray);
                bytesConverted = converter.getByteLength();         // @B2A
            }                                                       // @B1A
        }
        else    // We need to use the string type for the conversion   @C6A
        {
            synchronized(converter)                                 // @C6A
            {
                ((AS400Text)converter).toBytes(value, byteArray, 0, m_bidiStringType);  // @C6A
                bytesConverted = converter.getByteLength();         // @C6A
            }
        }
        try { bytes.write(byteArray, 0, bytesConverted); }
        catch (IOException e) { throw new PcmlException(e); }

        // Detail tracing of data conversion
        if (Trace.isTracePCMLOn())                                  // @D3C
        {                                                           // @B2A
            String parseMsg;                                        // @B2A
            if (m_indices.size() > 0)                               // @B2A
            {                                                       // @B2A
                parseMsg = SystemResourceFinder.format(DAMRI.WRITE_DATA_W_INDICES, new Object[] {Integer.toHexString(offset), Integer.toString(bytesConverted), getNameForException(), m_indices.toString(), PcmlMessageLog.toHexString(byteArray, 0, bytesConverted)} ); // @B2A
            }                                                       // @B2A
            else                                                    // @B2A
            {                                                       // @B2A
                parseMsg = SystemResourceFinder.format(DAMRI.WRITE_DATA,           new Object[] {Integer.toHexString(offset), Integer.toString(bytesConverted), getNameForException(), PcmlMessageLog.toHexString(byteArray, 0, bytesConverted)} ); // @B2A
            }                                                       // @B2A
            parseMsg = parseMsg + "\t  " + Thread.currentThread();  // @B2A
            Trace.log(Trace.PCML, parseMsg);                        // @D3C
        }                                                           // @B2A
            
        return bytesConverted;                                      // @B2A
    } // public int toBytes(OutputStream bytes, int offset)

    // Convert IBM i bytes to Java Object
    // Returns the Java Object
    public Object toObject(byte[] bytes) throws PcmlException
    {
        Object newVal = null;
        int dataType = getDataType();
        
        // Get a converter from the PcmlDocument node.
        // PcmlDocument will either create a converter or return an existing one.
        AS400DataType converter = m_owner.getDoc().getConverter(dataType, getLength(), getPrecision(), getCcsid(), getDateFormat(), getDateSeparator(), getTimeFormat(), getTimeSeparator());
        if (dataType != PcmlData.CHAR)                              // @C6A
        {
            synchronized(converter)                                 // @B1A
            {                                                       // @B1A
                newVal = converter.toObject(bytes);
            }                                                       // @B1A
        }
        else    // We need to use the string type for the conversion   @C6A
        {
            synchronized(converter)                                 // @C6A
            {
                newVal = ((AS400Text)converter).toObject(bytes, 0, m_bidiStringType);    // @C6A
            }
        }
        
        // For Strings, trim blanks and nulls appropriately
        if (dataType == PcmlData.CHAR) 
        {
            newVal = trimString((String) newVal, getTrim());                   // @B2A @D1C
        }
        
        // Set the new value
        setValue(newVal);
        
        return newVal;
    } // public int toBytes(byte[] bytes, int offset)

    // Parses array of bytes and stores for later conversion
    // to Java objects. This allows for lazy data translation
    // for performance.
    // Returns the number of bytes consumed from the input byte array
    // Note: This may be larger than the number of bytes saved for this element
    //       because of bytes skipped due to an offset value.
    public int parseBytes(byte[] bytes, int offset, Hashtable offsetStack) throws PcmlException 
    {
        int nbrBytes = 0;
        int skipBytes = 0;


        // Adjust offset if necessary
        // This is driven by the offset= attribute on the  tag
        int myOffset = getOffset();
        if (myOffset != 0) 
        {
            // Handle errors caused by bad offset values
            if (myOffset < 0 || myOffset > bytes.length)
            {
                throw new PcmlException(DAMRI.BAD_OFFSET_VALUE, new Object[] {Integer.valueOf(myOffset), Integer.valueOf(bytes.length), "", getNameForException()} );
            }
            
            // Determine from where the offset is based
            Integer myOffsetbase = null;
            String myOffsetfrom = m_owner.getOffsetfromId();    // Get offsetfrom= element name, if any    @A1C

            // If offsetfrom= was specified with the name of another element, 
            // get the base for the offset from the offset stack. 
            // The offset stack is a stack of beginning offsets for all
            // ancestors of the current element. The offsetfrom= value must be one
            // of these ancestors or an error will be reported.
            if (myOffsetfrom != null) 
            {
                myOffsetbase = (Integer) offsetStack.get(myOffsetfrom);
                if (myOffsetbase == null) 
                {
                    throw new PcmlException(DAMRI.OFFSETFROM_NOT_FOUND, new Object[] {myOffsetfrom, getNameForException()} );
                }
            }
            else
            {
                // If offsetfrom= was specified with an integer literal, use it.
                if (m_owner.getOffsetfrom() >= 0)                   // @A1A
                {                                                   // @A1A
                    myOffsetbase = Integer.valueOf(m_owner.getOffsetfrom()); // @A1A
                }                                                   // @A1A
                // getOffsetfrom() returns -1 to indicate that offset from was not specified.
                // No offsetfrom= was specified, the offset will be relative to the
                // beginning offset of the parent of this elements parent.
                // This is the first (most recent) entry in the offset stack.
                else                                                // @A1A
                {                                                   // @A1A
                    myOffsetfrom = ((PcmlDocNode) m_owner.getParent()).getQualifiedName();
                    myOffsetbase = (Integer) offsetStack.get( myOffsetfrom );
                }                                                   // @A1A
            }
            
            if (myOffsetbase != null)
            {
                myOffset = myOffset + myOffsetbase.intValue();
                // Handle errors caused by bad offset values
                if (myOffset < 0 || myOffset > bytes.length)
                {
                    throw new PcmlException(DAMRI.BAD_TOTAL_OFFSET, new Object[] {Integer.valueOf(myOffset), Integer.valueOf(bytes.length), myOffsetbase, myOffsetfrom, "", getNameForException()} );
                }
            }
            
            // If offset for this element is beyond current offset
            // calculate number of bytes to skip
            if (myOffset > offset) 
            {
                skipBytes = myOffset - offset;
            }

        }

        // Get number of bytes to parse for this value
        nbrBytes = byteLength();

        // Make sure length makes sense
        if (nbrBytes < 0 || nbrBytes > PcmlData.MAX_STRING_LENGTH)
        {
            throw new PcmlException(DAMRI.BAD_DATA_LENGTH, new Object[] {Integer.valueOf(nbrBytes), Integer.valueOf(PcmlData.MAX_STRING_LENGTH), "", getNameForException()} ); // @C4C
        }

        // Make sure we are not trying to access more bytes than available.
        if (offset + skipBytes + nbrBytes > bytes.length)
        {
            throw new PcmlException(DAMRI.INSUFFICIENT_INPUT_DATA, new Object[] {Integer.toString(offset + skipBytes + nbrBytes), Integer.toString(bytes.length), "", getNameForException()} );
        }

        // Finally, make a new byte array, copy the bytes, and save them
        byte [] newBytes = new byte[nbrBytes];
        for (int b = 0; b < nbrBytes; b++) 
        {
            newBytes[b] = bytes[offset+ skipBytes + b];
        }
        setBytes(newBytes);

        // Detail tracing of data parsing
        if (Trace.isTracePCMLOn())                                              // @D3C
        {
            String parseMsg;
            if (m_indices.size() > 0)
            {
                parseMsg = SystemResourceFinder.format(DAMRI.READ_DATA_W_INDICES, new Object[] {Integer.toHexString(offset+ skipBytes), Integer.toString(nbrBytes), getNameForException(), m_indices.toString(), PcmlMessageLog.toHexString(newBytes)} );
            }
            else
            {
                parseMsg = SystemResourceFinder.format(DAMRI.READ_DATA,           new Object[] {Integer.toHexString(offset+ skipBytes), Integer.toString(nbrBytes), getNameForException(), PcmlMessageLog.toHexString(newBytes)} );
            }
            parseMsg = parseMsg + "\t  " + Thread.currentThread();
            Trace.log(Trace.PCML, parseMsg);
        }


        return skipBytes + nbrBytes;
    } // End parseBytes()

    // Resolve an integer value from either a named element or a literal
    private int resolveIntegerValue(int intLiteral, String intId) throws PcmlException 
    {
        PcmlData intNode;
        Object nodeValue;

        if (intId != null)
        {
            intNode = (PcmlData) ((PcmlDocRoot)m_owner.getRootNode()).getElement(intId);
            nodeValue = intNode.getValue(m_indices);
            if (nodeValue instanceof String) 
            {
                return Integer.parseInt((String) nodeValue);
            }
            else if (nodeValue instanceof Number) 
            {
                return ((Number) nodeValue).intValue();
            }
            else if (nodeValue == null) 
            {
                throw new PcmlException(DAMRI.INPUT_VALUE_NOT_SET, new Object[] {intNode.getNameForException()} );
            }
            else
            {
                throw new PcmlException(DAMRI.STRING_OR_NUMBER, new Object[] {nodeValue.getClass().getName(), intNode.getNameForException()} );
            }
        }
        return intLiteral;
    }

    // Converts an object to the correct type (class) for
    // the defined data type.
    // The setValue method allows most data types to be set using 
    // either the Java class associated with the data type or using
    // a String. This method converts the value to the Java class 
    // defined for the data type. Exceptions such as 
    // NumberFormatException may result.
    //
    // This is implemented as a static method so it can be used
    // to verify the init= attribute after parsing the pcml document tags.
    public static Object convertValue(Object newVal, int dataType, int dataLength, int dataPrecision, String nodeNameForException, TimeZone serverTimeZone) throws PcmlException 
    {
        Object convertedVal = null;
        
        if (newVal == null)
            throw new PcmlException(DAMRI.NULL_VALUE, new Object[] {nodeNameForException} );
                            
        switch (dataType) 
        {

            case PcmlData.CHAR:
                if (newVal instanceof String) 
                {
                    convertedVal = newVal;
                }
                else
                {
                    convertedVal = newVal.toString();
                }
                break;

            case PcmlData.INT:
                if (dataLength == 2)
                {
                    if (dataPrecision == 16) 
                    {
                        if (newVal instanceof String) 
                        {
                            convertedVal = Integer.valueOf((String) newVal);
                        }
                        else 
                        {
                            if (newVal instanceof Integer) 
                            {
                                convertedVal = newVal;
                            }
                            else if (newVal instanceof Number) 
                            {
                                convertedVal = Integer.valueOf( ((Number) newVal).intValue() );
                            }
                            else 
                            {
                                throw new PcmlException(DAMRI.STRING_OR_NUMBER, new Object[] {newVal.getClass().getName(), nodeNameForException} );
                            }
                        }
                    }
                    else 
                    { // dataPrecision == 15 or defaulted
                        if (newVal instanceof String) 
                        {
                            convertedVal = Short.valueOf( ((String) newVal) );
                        }
                        else 
                        {
                            if (newVal instanceof Short) 
                            {
                                convertedVal = newVal;
                            }
                            else if (newVal instanceof Number) 
                            {
                                convertedVal = Short.valueOf( ((Number) newVal).shortValue() );
                            }
                            else 
                            {
                                throw new PcmlException(DAMRI.STRING_OR_NUMBER, new Object[] {newVal.getClass().getName(), nodeNameForException} );
                            }
                        }
                    }
                }
                else if (dataLength == 4)                                // @C3A
                {
                    if (dataPrecision == 32) 
                    {
                        if (newVal instanceof String) 
                        {
                            convertedVal = Long.valueOf((String) newVal);
                        }
                        else 
                        {
                            if (newVal instanceof Long) 
                            {
                                convertedVal = newVal;
                            }
                            else if (newVal instanceof Number) 
                            {
                                convertedVal = Long.valueOf( ((Number) newVal).longValue() );
                            }
                            else 
                            {
                                throw new PcmlException(DAMRI.STRING_OR_NUMBER, new Object[] {newVal.getClass().getName(), nodeNameForException} );
                            }
                        }
                    }
                    else 
                    { // dataPrecision == 31 or defaulted
                        if (newVal instanceof String) 
                        {
                            convertedVal = Integer.valueOf((String) newVal);
                        }
                        else 
                        {
                            if (newVal instanceof Integer) 
                            {
                                convertedVal = newVal;
                            }
                            else if (newVal instanceof Number) 
                            {
                                convertedVal = Integer.valueOf( ((Number) newVal).intValue() );
                            }
                            else 
                            {
                                throw new PcmlException(DAMRI.STRING_OR_NUMBER, new Object[] {newVal.getClass().getName(), nodeNameForException} );
                            }
                        }
                    }
                }
                else                                                // @C3A
                { // dataLength == 8                                // @C3A
                    if (dataPrecision == 64) 
                    {
                        if (newVal instanceof String) 
                        {
                            convertedVal = new BigInteger((String) newVal);
                        }
                        else 
                        {
                            if (newVal instanceof BigInteger)
                            {
                                convertedVal = newVal;
                            }
                            else if (newVal instanceof Number) 
                            {
                                long longVal = ((Number) newVal).longValue();
                                convertedVal = new BigInteger(Long.toString(longVal));
                            }
                            else 
                            {
                                throw new PcmlException(DAMRI.STRING_OR_NUMBER, new Object[] {newVal.getClass().getName(), nodeNameForException} );
                            }
                        }
                    }
                    else 
                    { // dataPrecision == 63 or defaulted
                      if (newVal instanceof String)                   // @C3A
                      {                                               // @C3A
                          convertedVal = Long.valueOf((String) newVal);   // @C3A
                      }                                               // @C3A
                      else                                            // @C3A
                      {                                               // @C3A
                          if (newVal instanceof Long)
                          {
                            convertedVal = newVal;
                          }
                          else if (newVal instanceof Number)          // @C3A
                          {                                           // @C3A
                            convertedVal = Long.valueOf( ((Number) newVal).longValue() ); // @C3A
                          }                                           // @C3A
                          else                                        // @C3A
                          {                                           // @C3A
                            throw new PcmlException(DAMRI.STRING_OR_NUMBER, new Object[] {newVal.getClass().getName(), nodeNameForException} ); // @C3A
                          }                                           // @C3A
                      }                                               // @C3A
                    }
                }                                                   // @C3A
                break;

            // Allows input value to be one of: BigDecimal, String or a subclass of Number
            case PcmlData.PACKED:
            case PcmlData.ZONED:
                if (newVal instanceof BigDecimal) 
                {
                    convertedVal = ((BigDecimal)newVal).setScale(dataPrecision, BigDecimal.ROUND_HALF_EVEN);      // @D0C
                }
                else 
                {
                    if (newVal instanceof String) 
                    {
                        convertedVal = (new BigDecimal((String) newVal));
                        if (((BigDecimal)convertedVal).scale() != dataPrecision) {
                          convertedVal = ((BigDecimal)convertedVal).setScale(dataPrecision, BigDecimal.ROUND_HALF_EVEN);     // @D0A
                        }
                    }
                    else
                    {
                        if (newVal instanceof Number) 
                        {
                            convertedVal = (new BigDecimal(((Number) newVal).doubleValue())).setScale(dataPrecision, BigDecimal.ROUND_HALF_EVEN);    // @D0C
                        }
                        else
                        {
                            throw new PcmlException(DAMRI.STRING_OR_NUMBER, new Object[] {newVal.getClass().getName(), nodeNameForException} );
                        }
                    }
                }
                break;


            case PcmlData.FLOAT:
                if (dataLength == 4) 
                {
                    if (newVal instanceof String) 
                    {
                        convertedVal = Float.valueOf((String) newVal);
                    }
                    else 
                    {
                        if (newVal instanceof Float) 
                        {
                            convertedVal = newVal;
                        }
                        else if (newVal instanceof Number) 
                        {
                            convertedVal = Float.valueOf( ((Number) newVal).floatValue() );
                        }
                        else 
                        {
                            throw new PcmlException(DAMRI.STRING_OR_NUMBER, new Object[] {newVal.getClass().getName(), nodeNameForException} );
                        }
                    }
                }
                else 
                { // Must be length=8
                    if (newVal instanceof String) 
                    {
                        convertedVal = Double.valueOf((String) newVal);
                    }
                    else 
                    {
                        if (newVal instanceof Double) 
                        {
                            convertedVal = newVal;
                        }
                        else if (newVal instanceof Number) 
                        {
                            convertedVal = Double.valueOf( ((Number) newVal).doubleValue() );
                        }
                        else 
                        {
                            throw new PcmlException(DAMRI.STRING_OR_NUMBER, new Object[] {newVal.getClass().getName(), nodeNameForException} );
                        }
                    }
                }
                break;

            case PcmlData.BYTE:
                    if (newVal instanceof String) 
                    {
                        StringTokenizer st = new StringTokenizer((String) newVal, " ,");
                        byte byteVal = 0;
                        byte[] byteArray = new byte[dataLength];
                        for (int i=0; i < dataLength; i++)
                        {
                          if (st.hasMoreTokens())
                          {
                            String token = st.nextToken();
                            try {
                              byteVal = Byte.decode(token).byteValue();
                            }
                            catch (NumberFormatException nfe)
                            {
                              // If leftmost bit is on (as in, for example, 0x80), and token isn't prefixed by a minus sign, Byte.decode() throws a NumberFormatException.
                              // For such cases we must use a more accommodating conversion utility.
                              try {
                                byte[] bytes = BinaryConverter.stringToBytes(token);
                                byteVal =  bytes[0];
                              }
                              catch (NumberFormatException e) {
                                // Rethrow original exception; it's more informative.
                                throw nfe;
                              }
                            }
                          }
                          else {
                            // Pad with final token in string
                          }
                          byteArray[i] = byteVal;
                        }
                        convertedVal = byteArray;
                    }
                    else 
                        if (newVal instanceof Number) 
                        {
                            byte byteVal = ((Number) newVal).byteValue();
                            byte[] byteArray = new byte[dataLength];
                            for (int i=0; i < dataLength; i++)
                            {
                                byteArray[i] = byteVal;
                            }
                            convertedVal = byteArray;
                        }
                        else 
                        {
                            throw new PcmlException(DAMRI.STRING_OR_NUMBER, new Object[] {newVal.getClass().getName(), nodeNameForException} );
                        }
                break;

            case PcmlData.DATE:
              if (newVal instanceof String) 
              {
                // Parse XML Schema-style date value: yyyy-mm-dd
                convertedVal = AS400Date.parseXsdString((String)newVal, serverTimeZone);
              }
              else 
              {
                if (newVal instanceof java.sql.Date) 
                {
                  convertedVal = (java.sql.Date)newVal;
                }
                else 
                {
                  throw new PcmlException(DAMRI.STRING_OR_NUMBER, new Object[] {newVal.getClass().getName(), nodeNameForException} );
                }
              }
              break;

            case PcmlData.TIME:
              if (newVal instanceof String) 
              {
                // Parse XML Schema-style time value: hh:mm:ss
                convertedVal = AS400Time.parseXsdString((String)newVal, serverTimeZone);
              }
              else 
              {
                if (newVal instanceof java.sql.Time) 
                {
                  convertedVal = (java.sql.Time)newVal;
                }
                else 
                {
                  throw new PcmlException(DAMRI.STRING_OR_NUMBER, new Object[] {newVal.getClass().getName(), nodeNameForException} );
                }
              }
              break;

            case PcmlData.TIMESTAMP:
              if (newVal instanceof String) 
              {
                // Parse XML Schema-style timestamp value: yyyy-MM-ddTHH:mm:ss.SSSSSSSSS
                convertedVal = AS400Timestamp.parseXsdString((String)newVal, serverTimeZone);
              }
              else 
              {
                if (newVal instanceof java.sql.Timestamp) 
                {
                  convertedVal = (java.sql.Timestamp)newVal;
                }
                else 
                {
                  throw new PcmlException(DAMRI.STRING_OR_NUMBER, new Object[] {newVal.getClass().getName(), nodeNameForException} );
                }
              }
              break;


            default:
                throw new PcmlException(DAMRI.BAD_DATA_TYPE, new Object[] {Integer.valueOf(dataType) , nodeNameForException} );

        } // END: switch (getDataType())

        return convertedVal;
    } 
    
    // Trims blanks and nulls from the ends of a string
    // Returns the trimmed string                               // @D1C
    static String trimString(String str, String trimEnd)        
    {                                                           
        if (trimEnd != null && trimEnd.equals("none"))                              
        {
            return str;                                         
        }
        char[] charBuff = str.toCharArray();                    
        int startOffset = 0;                    // Used when constructing return string 
        int lengthReturned = charBuff.length;   // Used when constructing return string 
        if (trimEnd == null || trimEnd.equals("right") || trimEnd.equals("both") )         
        {                                                       
            // Trim blanks and nulls
            trimRight: while (lengthReturned > 0)               
            {                                                   
                switch (charBuff[lengthReturned-1])             
                {                                               
                    case ' ':                                   
                    case '\0':                                  
                        lengthReturned--;                       
                        break;                                  
                    default:                                    
                        break trimRight;                        
                }                                               
            }                                                   
        }                                                       
        if (trimEnd != null && (trimEnd.equals("left") || trimEnd.equals("both")) )          
        {                                                       
            // Trim blanks and nulls
            trimLeft: while (startOffset < lengthReturned)                            
            {                                                   
                switch (charBuff[startOffset])                          
                {                                               
                    case ' ':                                   
                    case '\0':                                  
                        startOffset++;
                        lengthReturned--;
                        break;                                  
                    default:                                    
                        break trimLeft;                             
                }                                               
            }                                                   
        }
        return new String(charBuff, startOffset, lengthReturned);                  
    }                                                      
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy