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

src.com.ibm.as400.access.NPCPAttributeValue Maven / Gradle / Ivy

///////////////////////////////////////////////////////////////////////////////
//                                                                             
// JTOpen (IBM Toolbox for Java - OSS version)                              
//                                                                             
// Filename: NPCPAttributeValue.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.access;

import java.io.UnsupportedEncodingException;
import java.util.Hashtable;
import java.util.Enumeration;

/**
  * NPCPAttributeValue class - class for an attribute value list code point used with
  * the network print server's data stream.
  * This class is derived from NPCodePoint and will be used to build a code
  * point that has as its data a list of attribute.  Each attribute consist of
  * its ID, its length, its type and an offset to its data.
  *
  * The layout of an Attribute value codepoint in memory is:
  *
  *       ------------------------------
  *       | LLLL | CPID |     data     |
  *       ------------------------------
  *       LLLL - four byte code point length
  *       CPID - code point ID (2 bytes)
  *       data - code point data as follows:
  *
  *       ----------------------------------------------------------------------------------
  *       |nn | LEN | ID1 | tt | llll | ofof | ...... | IDnn| tt | llll | ofof |   values  |
  *       ----------------------------------------------------------------------------------
  *
  *         nn   - two byte total # of attributes in code point
  *         LEN  - two byte length of each attribute entry, right
  *                now this will be 12 (0x0C).
  *         IDx  - two byte attribute ID
  *         tt   - two byte type of attribute
  *         llll - four byte length of attribute value
  *         ofof - four byte offset from beginning of code point to
  *                attribute value.
  *         values - list of values for attributes.
  *
  *  There are many codepoints that fall into this structure for their data.  This
  *   class - NPCPAttributeValue - is an Abstract base class for these other classes.
  *   This class contains all the logic to build the raw data from a list of NPAttributes
  *   and to build a list of NPAttribute from the raw data.
  *
*/

abstract class NPCPAttributeValue extends NPCodePoint
{
  private static final String copyright = "Copyright (C) 1997-2000 International Business Machines Corporation and others.";



    static final long serialVersionUID = 4L;


    // private data members
    private Hashtable attrTable_;
    private boolean   fDataOutOfDate_;  // is base codepoint raw data out of date?
    private boolean   fListOutOfDate_;  // is our list out of date?

    // package scope data members
    final static String emptyString ="";

    // public methods and members
    private final static int LEN_ATTR_HEADER = 4;   // size of header on this data
    private final static int LEN_ATTR_ENTRY  = 12;  // size of each entry

    NPCPAttributeValue(NPCPAttributeValue cp)
    {
       super(cp);                             // call parent's copy ctor
       fDataOutOfDate_ = cp.fDataOutOfDate_;
       fListOutOfDate_ = cp.fListOutOfDate_;
       attrTable_ = new Hashtable(41);
       if (fDataOutOfDate_)
       {
          int i;
          NPAttribute attr;
          // if the data is out of date, then the list must be up to date
          // so we must copy the list over attribute by attribute
          for (Enumeration list = cp.attrTable_.elements(); list.hasMoreElements();)
          {
             attr = (NPAttribute)list.nextElement();
             if (attr != null)
             {
                try
                {
                   attrTable_.put(new Integer(attr.getID()), attr.clone());
                }
                catch(java.lang.CloneNotSupportedException e )
                {
                   // should never happen unless a certain NPAttribute subclass
                   // didn't override the clone() method!  We will throw a runtime error here
                   Trace.log(Trace.ERROR, " NPCPAttribributeValue: Error cloning new attribute" + e);
                   throw new InternalErrorException(e.toString(), InternalErrorException.UNKNOWN);
                }
             }
          }
       } else {
          // the raw data is upto date we'll rebuild the list of attrs when we
          // need to
          fListOutOfDate_ = true;
       }
    }
   /**
    * basic constructor that takes the ID and no data - child class passes in correct ID
    */
    NPCPAttributeValue(int ID)
    {
       super(ID);                            // construct codepoint with this ID
       fDataOutOfDate_ = false;
       fListOutOfDate_ = false;
       attrTable_ = new Hashtable(41);
    }

   /**
    * constructor that takes the ID and data - child class passes in correct ID
    * data should have the form described at the top of the class (nn len ID1...)
    */
    NPCPAttributeValue( int ID, byte[] data )
    {
       super(ID, data);                     // construct codepoint with this ID
       fDataOutOfDate_ = false;
       fListOutOfDate_ = true;
       attrTable_ = new Hashtable(41);
    }

    // override getLength from NPCodePoint class
    // returns total length of code point (data and header)
    int getLength()
    {
       if (fDataOutOfDate_)
       {
          updateData();
       }
       return super.getLength();
    }
    
    // @B2A - Added method below to be package scope only! (for use by PrintObject.java)
    boolean getfListOutOfDate()         // @B2A
    {                                   // @B2A
        return fListOutOfDate_;         // @B2A
    }                                   // @B2A
    

    void setDataBuffer( byte[] dataBuffer, int datalen, int offset)
    {
        fListOutOfDate_ = true;
        fDataOutOfDate_ = false;
        super.setDataBuffer(dataBuffer, datalen, offset);
    }

    // get current data buffer
    byte[] getDataBuffer()
    {
        if (fDataOutOfDate_)
        {
           updateData();
        }
        return super.getDataBuffer();
    }

    // get current data buffer and make it big enough to handle this many bytes
    byte[] getDataBuffer(int dataLength)
    {
        if (fDataOutOfDate_)
        {
           updateData();
        }
        fListOutOfDate_ = true;
        return super.getDataBuffer(dataLength);
    }

    // override reset() method to wipe out our data
    void reset()
    {
        zeroAttrTable();
        fListOutOfDate_ = false;
        fDataOutOfDate_ = false;
        super.reset();
    }

    /**
      * setAttrValue - add or change the attribute associated with this ID to this
      *                string value
      **/
    void setAttrValue(int attrID, String value)
    {
       if (!NPAttribute.idIsValid(attrID))
       {
          throw(new ExtendedIllegalArgumentException(PrintObject.getAttributeName(attrID),
             ExtendedIllegalArgumentException.PARAMETER_VALUE_NOT_VALID));
       }

       if (fListOutOfDate_)
       {
          updateList();
       }

       // if the attribute ID is less than 0 this is an IFS type attribute
       // and must be broken down into its object name and library.
       if (attrID < 0)
       {
           NPAttributeIFS ifsAttr =
              NPAttributeIFS.ifsAttrs[java.lang.Math.abs(attrID) - 1];

           // if the string starts with a "*" that means it is a special
           // value and we should just set the name field to this
           if (value.startsWith("*"))
           {
               attrTable_.put(new Integer(ifsAttr.nameID_),
                              new NPAttrString(ifsAttr.nameID_, value));            // @B1C
           } else {

              QSYSObjectPathName ifsPath = null;
              // if the type static, check it in QSYSObjectPathName ctor
              if (ifsAttr.typeID_ != 0)
              {
                 ifsPath = new QSYSObjectPathName(value);
              } else {
                 ifsPath = new QSYSObjectPathName(value, ifsAttr.typeString_);
              }

              attrTable_.put(new Integer(ifsAttr.nameID_),
                             new NPAttrString(ifsAttr.nameID_,
                                              ifsPath.getObjectName())); // @B1C

              attrTable_.put(new Integer(ifsAttr.libraryID_),
                             new NPAttrString(ifsAttr.libraryID_,
                                              ifsPath.getLibraryName()));// @B1C

               // if the type is dynamic, set this also
               if (ifsAttr.typeID_ != 0)
               {
                   // convert the type from "OVL" to "*OVL" first
                   String strType = "*";
                   strType += ifsPath.getObjectType();
                   attrTable_.put(new Integer(ifsAttr.typeID_),
                                  new NPAttrString(ifsAttr.typeID_,
                                                   strType));            // @B1C
               }
           }
       } else {
          attrTable_.put(new Integer(attrID), new NPAttrString(attrID, value));  // @B1C
       }
       fDataOutOfDate_ = true;

    }

    /**
      * getStringValue - get the attribute value associated with this ID.
      *            Throws an exception if the attribute ID is not a
      *              valid attribute ID.
      *            Throws an execption if the attribute is not a String
      *              type attribute.
      *
      * @param  attrID attribute ID from 1 to NPAttribute.MAX_ATTR_ID  or one of
      *                the special negative attribute IDs for IFS Path attributes.
      * @return String reference if successful.  If this reference is null
      *           the attribute wasn't found in this code point.
      *             represents an attribute that is not a String type attribute.
      * @see NPObject class
      **/
    String getStringValue(int attrID)
    {
       boolean fRC = false;
       String  rcString = null;
       Object attr;

       if (!NPAttribute.idIsValid(attrID))
       {
          throw(new ExtendedIllegalArgumentException(PrintObject.getAttributeName(attrID),
             ExtendedIllegalArgumentException.PARAMETER_VALUE_NOT_VALID));
       }

       if (fListOutOfDate_)
       {
          updateList();
       }

       // if the attribute ID is less than 0 then it is an IFS type
       // attribute ID and we must build the string up from its components
       if (attrID < 0)
       {
           String name = null, lib = null, type = null;
           // attribute is an IFS attribute, we must build the string here
           NPAttributeIFS ifsAttr =
                NPAttributeIFS.ifsAttrs[java.lang.Math.abs(attrID) - 1];


           // get the string for the object name
           // if this is null we'll end up returning null for the IFS path
           attr = attrTable_.get(new Integer(ifsAttr.nameID_));
           if (attr != null)
           {
              name = ((NPAttrString)attr).get();
           }

           // get the string for the object lib
           if (name != null)
           {/* check for default case when resource are *INLINE   @B5A@B6C */
            //  if (0 == name.compareTo("F1DFLT"))                 /* @B5A@B6C */
          //    {
            //    lib = "          "; /* Library will be blank in this case @B5A@B6D */
              //} else {                                           /* @B5A@B6D */
                 attr = attrTable_.get(new Integer(ifsAttr.libraryID_));
                 if (attr != null)
                {
                   lib  = ((NPAttrString)attr).get();
                   if ( 0 == lib.compareTo("")){
                     lib = "          ";
                   }
                }
             // }    /*  end Else            @B5A@B6D */
           }

           // get the string for the object type
           if (ifsAttr.typeID_ != 0)
           {
               attr = attrTable_.get(new Integer(ifsAttr.typeID_));
               if (attr != null)
               {
                  // check that attr is a String type
                  // WHAT to do when it is not?  Resource Type is an int!
                  type = ((NPAttrString)attr).get();
                  if ( (type != null) && type.startsWith("*"))
                  {
                    type = type.substring(1, type.length());// @B2A correct subs
                 // type = type.substring(1, type.length()-1); @B3D
                  }
               }
           } else {
               type = ifsAttr.typeString_;
           }

           // if the name is a special value (like *NONE or *FRONTOVL)
           // then don't return an IFS path name because the library
           // would be blank and the type might not make sense.
           if (name != null)
           {
               if (name.startsWith("*"))
               {
                   rcString = name;
               } else {
                  rcString = QSYSObjectPathName.toPath(lib, name, type);
               }
           }

       } else {

          attr = attrTable_.get(new Integer(attrID));
          if (attr != null)
          {
              // make sure this an NPAttrString class object
              if (attr instanceof NPAttrString)
              {
                 NPAttrString attrString = (NPAttrString)attr;
                 // rcString = new String(attrString.get());
                 rcString = attrString.get();
              } else {
                 throw(new ExtendedIllegalArgumentException(PrintObject.getAttributeName(attrID),
                    ExtendedIllegalArgumentException.PARAMETER_VALUE_DATATYPE_NOT_VALID));
              }
          }
       }
       return rcString;         // may be null
    }

    /**
      * setAttrValue - add or change the attribute associated with this ID to this
      *                int value
      **/
    void setAttrValue(int attrID, int value)
    {
       if (!NPAttribute.idIsValid(attrID))
       {
          throw(new ExtendedIllegalArgumentException(PrintObject.getAttributeName(attrID),
             ExtendedIllegalArgumentException.PARAMETER_VALUE_NOT_VALID));
       }

       if (fListOutOfDate_)
       {
          updateList();
       }
       attrTable_.put(new Integer(attrID), new NPAttrBin4(attrID, value));
       fDataOutOfDate_ = true;
    }

    /**
      * getIntValue - get the attribute value associated with this ID.
      *            Throws an exception if the attribute ID is not a
      *             valid attribute ID.
      *            Throws an execption if the attribute is not a int
      *             type attribute.
      *
      * @param  attrID attribute ID from 1 to NPAttribute.MAX_ATTR_ID
      * @return Int reference if successful.  If this reference is null
      *           the attribute wasn't found in this code point.
      *             represents an attribute that is not a Integer type attribute.
      * @see NPObject class
      **/
    Integer getIntValue(int attrID)
    {
       boolean fRC = false;
       Integer   rcInt = null;
       Object attr;

       if (!NPAttribute.idIsValid(attrID))
       {
          throw(new ExtendedIllegalArgumentException(PrintObject.getAttributeName(attrID),
             ExtendedIllegalArgumentException.PARAMETER_VALUE_NOT_VALID));
       }

       if (fListOutOfDate_)
       {
          updateList();
       }
       attr = attrTable_.get(new Integer(attrID));
       if (attr != null)
       {
          // make sure this an NPAttrBin4 class object
          if (attr instanceof NPAttrBin4)
          {
             NPAttrBin4 attrBin4= (NPAttrBin4)attr;
             rcInt = new Integer(attrBin4.get());
          } else {
             throw(new ExtendedIllegalArgumentException(PrintObject.getAttributeName(attrID),
                ExtendedIllegalArgumentException.PARAMETER_VALUE_DATATYPE_NOT_VALID));
          }
       }

       return rcInt;
    }

    /**
      * setAttrValue - add or change the attribute associated with this ID to this
      *                float value
     **/
    void setAttrValue(int attrID, float value)
    {
       if (!NPAttribute.idIsValid(attrID))
       {
          throw(new ExtendedIllegalArgumentException(PrintObject.getAttributeName(attrID),
             ExtendedIllegalArgumentException.PARAMETER_VALUE_NOT_VALID));
       }

       if (fListOutOfDate_)
       {
          updateList();
       }
       attrTable_.put(new Integer(attrID), new NPAttrFloat(attrID, value));
       fDataOutOfDate_ = true;
    }


    /**
      * getFloatValue - get the attribute value associated with this ID.
      *            Throws an exception if the attribute ID is not a
      *             valid attribute ID.
      *            Throws an execption if the attribute is not a float
      *             type attribute.
      *
      * @param  attrID attribute ID from 1 to NPAttribute.MAX_ATTR_ID
      * @return Float reference if successful.  If this reference is null
      *           the attribute wasn't found in this code point.
      *             represents an attribute that is not a Float type attribute.
      * @see NPObject class
      **/
    Float getFloatValue(int attrID)
    {
       boolean fRC = false;
       Float   rcFloat = null;
       Object attr;

       if (!NPAttribute.idIsValid(attrID))
       {
          throw(new ExtendedIllegalArgumentException(PrintObject.getAttributeName(attrID),
             ExtendedIllegalArgumentException.PARAMETER_VALUE_NOT_VALID));
       }

       if (fListOutOfDate_)
       {
          updateList();
       }
       attr = attrTable_.get(new Integer(attrID));
       if (attr != null)
       {
          // make sure this an NPAttrFloat class object
          if (attr instanceof NPAttrFloat)
          {
             NPAttrFloat attrFloat= (NPAttrFloat)attr;
             rcFloat = new Float(attrFloat.get());
          } else {
             throw(new ExtendedIllegalArgumentException(PrintObject.getAttributeName(attrID),
                ExtendedIllegalArgumentException.PARAMETER_VALUE_DATATYPE_NOT_VALID));
          }
       }
       return rcFloat;
    }

    /**
      * addUpdateAttributes - merges the passed in AttributeValue codepoint with this one
      *                   by adding any attributes that aren't already in this codepoint
      *                   and replacing any that are with the ones from the parameter.
     **/
     void addUpdateAttributes(NPCPAttributeValue cp)
     {
        if (fListOutOfDate_)
        {
           updateList();
        }
        if (cp.fListOutOfDate_)
        {
            cp.updateList();
        }
        NPAttribute attr;
        for (Enumeration e = cp.attrTable_.elements(); e.hasMoreElements(); )
        {
            attr = (NPAttribute)e.nextElement();
            if (attr != null)
            {
                attrTable_.put(new Integer(attr.getID()), attr);
            }
        }

        fDataOutOfDate_ = true;   /* @A1A */

     }

     /**
       * removeAttribute - removes the attribute with the specified ID from the code point.
       * Nothing happens if the attribute currently is not in the code point.
       * runtime exception is thrown if the attribute ID is not valid
       **/
     void removeAttribute(int attrID)
     {
         if (!NPAttribute.idIsValid(attrID))
         {
             throw(new ExtendedIllegalArgumentException(PrintObject.getAttributeName(attrID),
                 ExtendedIllegalArgumentException.PARAMETER_VALUE_NOT_VALID));
         }
         if (fListOutOfDate_)
         {
             updateList();
         }
         // if the attribute ID is less than 0 then it is an IFS type
         // attribute ID and we must remove all parts of the IFS name (name, lib and maybe type)
         if (attrID < 0)
         {

             // attribute is an IFS attribute, we must build the string here
             NPAttributeIFS ifsAttr =
                  NPAttributeIFS.ifsAttrs[java.lang.Math.abs(attrID) - 1];


             // remove the string for the object name
             attrTable_.remove(new Integer(ifsAttr.nameID_));
             // remove the string for the object library
             attrTable_.remove(new Integer(ifsAttr.libraryID_));

             // get the string for the object type
             if (ifsAttr.typeID_ != 0)
             {
                 attrTable_.remove(new Integer(ifsAttr.typeID_));
             }
       } else {
           // just a simple attribute, remove it from the table
           attrTable_.remove(new Integer(attrID));
       }
     }


    // private methods
    private void updateData()
    {
       int dataLength, elements, dataOffset, valueOffset;
       byte[] data;
       NPAttribute attr;

       elements = 0;
       dataLength =  LEN_ATTR_HEADER;
       for (Enumeration e = attrTable_.elements(); e.hasMoreElements(); )
       {
          attr = (NPAttribute)e.nextElement();

          if (attr != null)
          {
             dataLength += LEN_ATTR_ENTRY;   // add the 12 bytes needed to describe this attr
             dataLength += attr.getHostLength(converter_);  // add bytes for host length     @B1C
             elements++;                          // bump number of found elements
          }
       }


       data = super.getDataBuffer(dataLength);   // get buffer for the raw data

       // set the number of attributes in the codepoint
       dataOffset = super.getOffset();
       BinaryConverter.unsignedShortToByteArray(elements, data, dataOffset);
       dataOffset += 2;

       // set the length (12) of each attribute header
       BinaryConverter.unsignedShortToByteArray(LEN_ATTR_ENTRY, data, dataOffset);
       dataOffset += 2;

       // initialize where the first value will be placed in the buffer (after all
       // of the attribute header entries)
       valueOffset = dataOffset + LEN_ATTR_ENTRY * elements;

       // add each attribute to the data
       for (Enumeration e = attrTable_.elements(); e.hasMoreElements() && (elements != 0);)
       {
          attr = (NPAttribute)e.nextElement();
          if (attr != null)
          {
             // set 2 byte attr ID
             BinaryConverter.unsignedShortToByteArray(attr.getID(), data, dataOffset);
             dataOffset += 2;

             // set 2 byte type
             BinaryConverter.unsignedShortToByteArray(attr.getType(), data, dataOffset);
             dataOffset += 2;

             // set 4 byte length
             BinaryConverter.intToByteArray(attr.getHostLength(converter_), data, dataOffset);       // @B1C
             dataOffset += 4;

             // set 4 byte value offset
             BinaryConverter.intToByteArray(valueOffset+NPCodePoint.LEN_HEADER, data, dataOffset);
             dataOffset += 4;

             // set actual value
             System.arraycopy(attr.getHostData(converter_), 0,    // source                          // @B1C
                              data, valueOffset,        // dest
                              attr.getHostLength(converter_));    // len                             // @B1C
             valueOffset += attr.getHostLength(converter_);                                          // @B1C

             elements--;
          }
       }
       fDataOutOfDate_ = false;
    }  // updateData()

    private void updateList()
    {
       byte[] data;
       int i;
       byte nullbyte = (byte)'\0';
      
       NPAttribute attr = null;
       
       // zero out table and the rebuild based on data
       zeroAttrTable();
       data = super.getDataBuffer();
       if ( (data != null) && (data.length >= LEN_ATTR_HEADER) )
       {
          long dataLength;
          int  elements, offset;
          // @B1D Converter converterObj;
          // @B1D try
          // @B1D {
          // @B1D   // @A2C changed below from g.getConverter(hostCCSID_);
          // @B1D    converterObj = new Converter(hostCCSID_);
          // @B1D }
          // @B1D catch (java.io.UnsupportedEncodingException e)
          // @B1D {
          // @B1D     Trace.log(Trace.ERROR,
          // @B1D             " UnsupportedEncodingException for ccsid = " + hostCCSID_ +
          // @B1D             ".  Exception text = " + e);
          // @B1D     // @A2C changed below from Converter.getConverter()
          // @B1D     converterObj = new Converter();
          // @B1D }
          dataLength = (long)super.getDataLength();
          offset = super.getOffset();
          if (dataLength > LEN_ATTR_HEADER)
          {
             elements = BinaryConverter.byteArrayToShort(data, offset);
             offset += 2;
             if (elements != 0)
             {
                int size;
                size = BinaryConverter.byteArrayToShort(data, offset);
                offset += 2;
                if ( (size >= LEN_ATTR_ENTRY) && (dataLength >= (offset+elements*size)) )
                {
                   int ID, type, length, valueOffset;
                   while (elements != 0)
                   {
                      ID = BinaryConverter.byteArrayToShort(data, offset);
                      offset += 2;

                      type = BinaryConverter.byteArrayToShort(data, offset);
                      offset += 2;

                      length = BinaryConverter.byteArrayToInt(data, offset);
                      offset += 4;

                      // valueOffset is from the beginning of the code point
                      // since we only have the codepoint data here, we have
                      // to adjust it
                      valueOffset = BinaryConverter.byteArrayToInt(data, offset);
                      offset += 4;
                      valueOffset -= NPCodePoint.LEN_HEADER;

                      switch (type)
                      {
                         case NPAttribute.FOUR_BYTE:
                         case NPAttribute.FOUR_BYTE_ENU:
                            attr = new NPAttrBin4(ID, data, valueOffset,
                                                   length);
                            break;
                         case NPAttribute.STRING:
                         case NPAttribute.STRING_ENU:
                            attr = new NPAttrString(ID, data, valueOffset,
                                                     length, converter_);                    // @B1C
                            break;
                         case NPAttribute.FLOAT:
                            attr = new NPAttrFloat(ID, data, valueOffset,
                                                    length);
                            break;
                         case NPAttribute.LISTSTRING:                                        // @B4A
                            // lists of strings will contain single null per field
                            // and double null to signify list end                              @B4A
                            
                            for ( i = 0; i < length; i++){                                   // @B4A
                                if ((data[i + valueOffset] == nullbyte)&& (i < length -1)){
                                     if (data[i + valueOffset + 1] != nullbyte){
                                        data[i + valueOffset] = (byte)'\u007A';
                            // uses ':' as a delimiter between fields            @B4A                                                                               
                                     }
                                 }
                                 }
                            attr = new NPAttrString(ID, data, valueOffset,
                                                    length, converter_);
                            break;
                         default:
                            // unknown attribute type - could be new?
                            // System.out.println(" unknown type = " + type);
                      }
                      attrTable_.put(new Integer(ID), attr);

                      elements--;
                      offset += size - LEN_ATTR_ENTRY;  // if the attribute entry was bigger than we know it
                   }
                }
             }
          }
       }
       fListOutOfDate_ = false;
    }  // updateList()

    private void zeroAttrTable()
    {

       attrTable_.clear();
    }

    
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy