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

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

The newest version!
///////////////////////////////////////////////////////////////////////////////
//                                                                             
// JTOpen (IBM Toolbox for Java - OSS version)                              
//                                                                             
// Filename: PcmlDocument.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-2003 International Business Machines Corporation and     
// others. All rights reserved.                                                
//                                                                             
///////////////////////////////////////////////////////////////////////////////

package com.ibm.as400.data;

import com.ibm.as400.access.AS400;
import com.ibm.as400.access.AS400DataType;
import com.ibm.as400.access.AS400Text;
import com.ibm.as400.access.AS400Bin2;
import com.ibm.as400.access.AS400UnsignedBin2;
import com.ibm.as400.access.AS400Bin4;
import com.ibm.as400.access.AS400UnsignedBin4;
import com.ibm.as400.access.AS400Bin8;                              // @C4A
import com.ibm.as400.access.AS400UnsignedBin8;
import com.ibm.as400.access.AS400PackedDecimal;
import com.ibm.as400.access.AS400ZonedDecimal;
import com.ibm.as400.access.AS400Float4;
import com.ibm.as400.access.AS400Float8;
import com.ibm.as400.access.AS400ByteArray;
import com.ibm.as400.access.AS400Date;
import com.ibm.as400.access.AS400Time;
import com.ibm.as400.access.AS400Timestamp;
import com.ibm.as400.access.AS400Message;
import com.ibm.as400.access.AS400SecurityException;
import com.ibm.as400.access.ObjectDoesNotExistException;
import com.ibm.as400.access.ErrorCompletingRequestException;
import com.ibm.as400.access.ProgramCall;


import java.io.IOException;
import java.io.ObjectInputStream;                                   // @C1A

import java.io.PrintWriter;                                         //@E1A
import java.io.OutputStream;                                        //@E1A
import com.ibm.as400.access.Trace;                                  //@E1A

import java.net.UnknownHostException;


import java.util.Enumeration;
import java.util.LinkedList;
import java.util.TimeZone;

import com.ibm.as400.access.BinaryConverter;

/**
*/
class PcmlDocument extends PcmlDocRoot
{
    // Used for XPCML testing
    boolean compareSucceeded=true;


    static final long serialVersionUID = -8169008879805188674L;

    // New attributes should be added to the end of this array
    private static final String PCMLATTRIBUTES[] = {
        "version"
    };

    private static final int VERSION_1_ATTRIBUTE_COUNT = 1;

    private static final String DEFAULT_DATE_SEPARATOR = "hyphen"; // default separator for 'date'
    private static final String DEFAULT_TIME_SEPARATOR = "period"; // default separator for 'time'

    private String m_docName;
    private String m_XsdName;     //@E1A


    // The following attributes added for PCML v2.0
    private String m_Version;     // version=, string literal          @B1A

    private static AS400Bin2          m_Bin2   = new AS400Bin2();
    private static AS400UnsignedBin2  m_UBin2  = new AS400UnsignedBin2();
    private static AS400Bin4          m_Bin4   = new AS400Bin4();
    private static AS400UnsignedBin4  m_UBin4  = new AS400UnsignedBin4();
    private static AS400Bin8          m_Bin8   = new AS400Bin8();   // @C4A
    private static AS400UnsignedBin8  m_UBin8  = new AS400UnsignedBin8();
    private static AS400Float4        m_Float4 = new AS400Float4();
    private static AS400Float8        m_Float8 = new AS400Float8();
    private static AS400PackedDecimal m_Packed_15_5 = new AS400PackedDecimal(15, 5);
    private static AS400ZonedDecimal  m_Zoned_15_5 = new AS400ZonedDecimal(15, 5);
    // We can't have static versions of these, since they all depend on the timezone of the system. 
    //private static AS400Date          m_Date       = new AS400Date();
    //private static AS400Time          m_Time       = new AS400Time();
    // private static AS400Timestamp     m_Timestamp  = new AS400Timestamp();

    private long   correlationID_ = 0;                       // @C8A

    // Transient data not stored durial serialization
    // These are a cache of the most common converters.
    // NOTE: For AS400Text converters, the cached objects are 'thrown away'
    // when the IBM i system object changes via setAs400(). This will cause
    // the new system (and CCSID) to be used to construct new converters.
    protected transient AS400Text          m_Text_1  = null; // Create at run time
    protected transient AS400Text          m_Text_10 = null; // Create at run time
    protected transient AS400ByteArray     m_Byte_1  = null; // Create at run time
    protected transient AS400ByteArray     m_Byte_2  = null; // Create at run time
    protected transient AS400ByteArray     m_Byte_3  = null; // Create at run time
    protected transient AS400ByteArray     m_Byte_4  = null; // Create at run time
                                     // @D0C: Made the above converters protected.

    private AS400              m_as400;                             // @C1C
    private int                m_as400Vrm = -1;                     // @C1C

    private transient PcmlSpecificationException m_PcmlSpecificationException;
    private transient boolean m_bSerializingWithData = false;       // @C1A
    private transient long m_DeserializationTs = 0;                 // @C1A
    private transient PcmlProgram m_pcmlProgram;
    private transient Object correlationIDLock_ = new Object();


    // @E1A -- String constant for use in XPCML
    private static final String XMLNS_STRING =  " xmlns:xsi=" + "\"" + "http://www.w3.org/2001/XMLSchema-instance" + "\"" +
                          "\n" + "       xsi:noNamespaceSchemaLocation=";

    /**
    */
    PcmlDocument(PcmlAttributeList attrs, String docName)           // @C3A
    {
        super();                                                    // @C5A
        m_PcmlSpecificationException = null;

        setNodeType(PcmlNodeType.DOCUMENT);                         // @C3A
        m_XmlAttrs = attrs;                                         // @C3A
        m_docName = docName;
        m_XsdName="";                                               //@E1A

        // Set the pcml version number
        m_Version = getAttributeValue("version");
    }

    // All pcml document nodes support clone.
    // This is a deep clone in that the result is to
    // clone an entire subtree. This method recursively
    // clones its children.
    public Object clone()                                           // @C5A
    {                                                               // @C5A
        PcmlDocument doc = (PcmlDocument) super.clone();            // @C5A

        // Add the entire cloned tree of objects to the document hash table
        doc.addToHashtable(doc);                                    // @C5A

        return doc;                                                 // @C5A
    }                                                               // @C5A

	// Custom deserialization
	private void readObject(ObjectInputStream in)
		throws IOException, ClassNotFoundException                  // @C1A
	{                                                               // @C1A
		// Set timestamp for deserialization (load) of the document
		// This timestamp is used to set all data and dimension
		// timestamps during deserialization because the
		// system clock is not necessarily synchronized with the
		// clock when the document was serialized (due to transfering
		// the serialized document to another system).
		// Classes in the document that maintain timestamps will use
		// this value as the current timestamp during readObject().
		// Currently, PcmlDataValues and PcmlDataVector use timestamps.
		// Note that timestamps are only an issue when serializing "with data".
		// The timestamp sensitive classes are not serialized when serializing
		// without data.
          correlationIDLock_ = new Object();
          m_DeserializationTs = getCorrelationID();                  // @C1A @C8C

		// Default deserialization
		in.defaultReadObject();                                     // @C1A
		
		if (m_as400 == null)                                        // @C1A
		    m_as400Vrm = -1;                                        // @C1A
		
		// Perform deserialization post-processing
		readObjectPostprocessing();
	}                                                               // @C1A

    /**
    */
    AS400 getAs400()
    {
        return m_as400;
    }

    /**
    */
    long getDeserializationTimestamp()                              // @C1A
    {                                                               // @C1A
        return m_DeserializationTs;                                 // @C1A
    }                                                               // @C1A

   /**
    * Return the list of valid attributes for the data element.
    **/
    String[] getAttributeList()                                 // @C6A
    {
        return PCMLATTRIBUTES;                                  // @C6A
    }

    /**
    */
    boolean isSerializingWithData()                                 // @C1A
    {                                                               // @C1A
        return m_bSerializingWithData;                              // @C1A
    }                                                               // @C1A

    /**
    */
    void setSerializingWithData(boolean b)                          // @C1A
    {                                                               // @C1A
        m_bSerializingWithData = b;                                 // @C1A
    }                                                               // @C1A

    /**
    */
    int getAs400VRM() throws PcmlException
    {
		if (m_as400Vrm == -1)
		{
			// Get the VRM of the host
			// Maybe should optimize this by caching initially by callProgram and
			// making available to all children
			try
			{
				m_as400Vrm = getAs400().getVRM();
			}
	        catch (AS400SecurityException e)
	        {
	            throw new PcmlException(e);
	        }
	        catch (UnknownHostException e)
	        {
	            throw new PcmlException(e);
	        }
	        catch (IOException e)
	        {
	            throw new PcmlException(e);
	        }
		}

		return m_as400Vrm;
    }

    /**
    */
    String getDocName()
    {
        return m_docName;
    }

    /**
    */
    String getVersion()                                             // @B1A
    {                                                               // @B1A
        return m_Version;                                           // @B1A
    }                                                               // @B1A

    /*
    */
    synchronized void setAs400(AS400 sys)
    {
        m_as400 = sys;
        m_as400Vrm = -1;

        // Release any CCSID sensitive data converters
        // The CCSID of this system may be different from previous system
        m_Text_1 = null;                                            // @C1A
        m_Text_10 = null;                                           // @C1A
    }


    /*
     Called by PcmlDataValues, PcmlDocument, and RfmlDocument.
    */
    protected AS400DataType getConverter(int dataType , int dataLength, int dataPrecision, int ccsid, String dateFormat, String dateSeparator, String timeFormat, String timeSeparator)
                                         throws PcmlException
    {
        switch (dataType)
        {

            case PcmlData.CHAR:
                // If the requested CCSID is not the same as
                // the system's default host CCSID, always create
                // a new converter. I.E. We only cach converters
                // with CCSIDs that match the system object.
                if (ccsid != m_as400.getCcsid())                    // @C2C
				{
                    return new AS400Text(dataLength, ccsid, m_as400);
				}

                switch (dataLength)
                {
                    case 1:
                        if (m_Text_1 == null)
                        {
                            m_Text_1 = new AS400Text(dataLength, ccsid, m_as400);
                        }
                        return m_Text_1;
                    case 10:
                        if (m_Text_10 == null)
                        {
                            m_Text_10 = new AS400Text(dataLength, ccsid, m_as400);
                        }
                        return m_Text_10;
                    default:
                        return new AS400Text(dataLength, ccsid, m_as400);

                }

            case PcmlData.INT:
                if (dataLength == 2)
                {
                    if (dataPrecision == 16)
                        return m_UBin2;
                    else // must be dataPrecision == 15
                        return m_Bin2;
                }
                else if (dataLength == 4)
                {
                    if (dataPrecision == 32)
                        return m_UBin4;
                    else // must be dataPrecision == 31
                        return m_Bin4;
                }
                else                                                // @C4A
                { // must be datalength == 8                        // @C4A
                    if (dataPrecision == 64)
                        return m_UBin8;
                    else // must be dataPrecision == 63
                        return m_Bin8;
                }                                                   // @C4A

            case PcmlData.PACKED:
                if (dataLength == 15 && dataPrecision == 5)
                    return m_Packed_15_5;
                else
                    return new AS400PackedDecimal(dataLength, dataPrecision);

            case PcmlData.ZONED:
                if (dataLength == 15 && dataPrecision == 5)
                    return m_Zoned_15_5;
                else
                    return new AS400ZonedDecimal(dataLength, dataPrecision);

            case PcmlData.FLOAT:
                if (dataLength == 4)
                {
                    return m_Float4;
                }
                else
                { // must be datalength == 8
                    return m_Float8;
                }

            case PcmlData.BYTE:
                switch (dataLength)
                {
                    case 1:
                        if (m_Byte_1 == null)
                        {
                            m_Byte_1 = new AS400ByteArray(dataLength);
                        }
                        return m_Byte_1;
                    case 2:
                        if (m_Byte_2 == null)
                        {
                            m_Byte_2 = new AS400ByteArray(dataLength);
                        }
                        return m_Byte_2;
                    case 3:
                        if (m_Byte_3 == null)
                        {
                            m_Byte_3 = new AS400ByteArray(dataLength);
                        }
                        return m_Byte_3;
                    case 4:
                        if (m_Byte_4 == null)
                        {
                            m_Byte_4 = new AS400ByteArray(dataLength);
                        }
                        return m_Byte_4;
                    default:
                        return new AS400ByteArray(dataLength);

                }

            case PcmlData.DATE:
              //
              // We can't use a static one because that assumes the timezone is known. 
              //if (dateFormat == null ||
              //    (dateFormat.equals("ISO") &&  // default date format
              //     (dateSeparator == null || dateSeparator.equals(DEFAULT_DATE_SEPARATOR))))
              //{
              //  return m_Date;
              //}
              //else  // create a date converter
              {
                int format = AS400Date.toFormat(dateFormat);
                // Note: The format value is validated in PcmlData.checkAttributes().
                if (dateSeparator == null) {
                  return new AS400Date(getTimeZone(), format); // assume the default separator
                }
                else
                {  // Convert the separator name ('comma', 'hyphen', etc) to a character.
                  return new AS400Date(getTimeZone(), format, separatorAsChar(dateSeparator));
                }
              }

            case PcmlData.TIME:
              // We can't use a static one because that assumes the timezone is known. 
              //if (timeFormat == null ||
              //    (timeFormat.equals("ISO") &&  // default time format
              //     (timeSeparator == null || timeSeparator.equals(DEFAULT_TIME_SEPARATOR))))
              //{
              //  return m_Time;
              //}
              //else  // create a time converter
              {
                int format = AS400Time.toFormat(timeFormat);
                // Note: The format value is validated in PcmlData.checkAttributes().
                if (timeSeparator == null) {
                  return new AS400Time(getTimeZone(), format); // assume the default separator
                }
                else
                {  // Convert the separator name ('comma', 'hyphen', etc) to a character.
                  return new AS400Time(getTimeZone(), format,separatorAsChar(timeSeparator));
                }
              }

            case PcmlData.TIMESTAMP:
              
              return new AS400Timestamp(getTimeZone()); 

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

        } // END: switch (getDataType())
    }


    /**
    */
    synchronized boolean callProgram(String name)
						           throws AS400SecurityException,
						                  ObjectDoesNotExistException,
						                  InterruptedException,
						                  ErrorCompletingRequestException,
						                  IOException,
						                  PcmlException
    {
        m_pcmlProgram = getProgramNode(name);
        return m_pcmlProgram.callProgram(m_as400);
    }

    /**
    Returns an "errno" value for the named service program element.
    

The named program element must be defined as service program entrypoint. The value returned is the "errno" value resulting from the most recent call to the program. If the program has not been called, zero is returned. @return The integer "errno" value for the named service program element. @param name The name of the <program> element in the PCML document. @exception PcmlException If an error occurs. */ synchronized int getErrno(String name) throws PcmlException // @B1A { // @B1A return getProgramNode(name).getErrno(); // @B1A } // @B1A /** Returns an int return value for the named service program element.

The named program element must be defined as service program entrypoint. The value returned is the integer return value from the most recent call to the program. If the program has not been called, zero is returned. @return The integer return value for the named service program element. @param name The name of the <program> element in the PCML document. @exception PcmlException If an error occurs. */ synchronized int getIntReturnValue(String name) throws PcmlException // @B1A { // @B1A return getProgramNode(name).getIntReturnValue(); // @B1A } // @B1A /** */ synchronized int getIntValue(String name) throws PcmlException { return getIntValue(name, new PcmlDimensions()); } /** */ synchronized int getIntValue(String name, PcmlDimensions indices) throws PcmlException { Object value; value = getValue(name, indices); if (value instanceof String) { return Integer.parseInt((String) value); } else if (value instanceof Number) { return ((Number) value).intValue(); } else { throw new PcmlException(DAMRI.STRING_OR_NUMBER, new Object[] {value.getClass().getName(), name} ); } } /** */ synchronized String getStringValue(String name, int type) throws PcmlException // @C7A { return getStringValue(name, new PcmlDimensions(), type); // @C7A } /** Returns a long value that replaces a timestamp. This value is used by PcmlDataValues to control data conversion, PcmlDataVector to keep track of when the data dimension changes, and PcmlDocument for serialization. */ long getCorrelationID() // @C8A { synchronized (correlationIDLock_) { return ++correlationID_; // @C8A } } /** */ synchronized String getStringValue(String name, PcmlDimensions indices, int type) throws PcmlException // @C7A { return getDataNode(name).getStringValue(indices, type); // @C7A } /** */ synchronized int getOutputsize(String name) throws PcmlException { return getOutputsize(name, new PcmlDimensions()); } /** */ synchronized int getOutputsize(String name, PcmlDimensions indices) throws PcmlException { PcmlNode node = (PcmlDocNode) getElement(name); if (node != null && node instanceof PcmlData) { return ((PcmlData) node).getOutputsize(indices); } else if (node != null && node instanceof PcmlStruct) { return ((PcmlStruct) node).getOutputsize(indices); } else { return 0; } } /** Returns the list of IBM i system messages returned from running the program. An empty list is returned if the program has not been run yet. @return The array of messages returned by the server for the program. */ synchronized AS400Message[] getMessageList(String name) throws PcmlException { return getProgramNode(name).getMessageList(); } /** Returns the option for how many messages will be retrieved for the specified program. @return A constant indicating how many messages will be retrieved. **/ synchronized int getMessageOption(String name) throws PcmlException { return getProgramNode(name).getMessageOption(); } /** Returns the ProgramCall object that was used in the most recent invocation of {@link #callProgram() callProgram()}. @return The ProgramCall object; null if callProgram has not been called. **/ synchronized ProgramCall getProgramCall() { return ( m_pcmlProgram == null ? null : m_pcmlProgram.getProgramCall() ); } /** */ synchronized Object getValue(String name) throws PcmlException { return getValue(name, new PcmlDimensions()); } /** */ synchronized Object getValue(String name, PcmlDimensions indices) throws PcmlException { return getDataNode(name).getValue(indices); } /** */ synchronized boolean isArray(String name) throws PcmlException { return getDataNode(name).isArray(); } /** */ synchronized boolean isInArray(String name) throws PcmlException { return getDataNode(name).isInArray(); } // Set a named data value /** */ synchronized void setValue(String name, Object value) throws PcmlException { setValue(name, value, new PcmlDimensions()); } // Set a named data value using dimension indices /** */ synchronized void setValue(String name, Object value, PcmlDimensions indices) throws PcmlException { getDataNode(name).setValue(value, indices); } /** */ synchronized void setStringValue(String name, String value, int type) throws PcmlException // @C7A { setStringValue(name, value, new PcmlDimensions(), type); // @C7A } /** */ synchronized void setStringValue(String name, String value, PcmlDimensions indices, int type) throws PcmlException // @C7A { getDataNode(name).setStringValue(value, indices, type); // @C7A } //@AI6A synchronized void setCharArrayValue(String name, char[] value, int type) throws PcmlException { setCharArrayValue(name, value, new PcmlDimensions(), type); } //@AI6A synchronized void setCharArrayValue(String name, char[] value, PcmlDimensions indices, int type) throws PcmlException { getDataNode(name).setCharArrayValue(value, indices, type); } // Overrides the threadsafe= attribute void setThreadsafeOverride(String program, boolean threadsafe) // @C6A throws PcmlException { // @C6A getProgramNode(program).setThreadsafeOverride(threadsafe); // @C6A } // @C6A // Set the path of the program to be called void setPath(String program, String path) // @D1A throws PcmlException { // @D1A getProgramNode(program).setPath(path); // @D1A } // @D1A // gets the override of the threadsafe= attribute boolean getThreadsafeOverride(String program) // @C6A throws PcmlException { // @C6A return getProgramNode(program).getThreadsafeOverride(); // @C6A } // @C6A /** Specifies the option for how many messages should be retrieved for the specified program. By default, to preserve compatability, only the messages sent to the program caller and only up to ten messages are retrieved. This property will only take effect on systems that support the new option. @param messageOption A constant indicating how many messages to retrieve. **/ synchronized void setMessageOption(String program, int messageOption) throws PcmlException { getProgramNode(program).setMessageOption(messageOption); } // Add a subtree to the document's hashtable. // This is called to complete the document cloneing process. void addToHashtable(PcmlDocNode newChild) // @C5A { // @C5A String qName; // Qualified name of child // @C5A Enumeration children; // @C5A PcmlDocNode child; // @C5A children = newChild.getChildren(); // @C5A if (children == null) // @C5A return; // @C5A LinkedList queue = new LinkedList(); //@F4 queue.add(newChild);//@F4 while (!queue.isEmpty()) {//@F4 child = (PcmlDocNode) queue.getFirst();//@F4 queue.removeFirst();//@F4 children = child.getChildren();//@F4 while (children.hasMoreElements()) {//@F4 child = (PcmlDocNode) children.nextElement();//@F4 qName = child.getQualifiedName();//@F4 if (!qName.equals("")) {//@F4 if (this.addElement(child) != null) {//@F4 this.addPcmlSpecificationError(DAMRI.MULTIPLE_DEFINE, new Object[] { qName });//@F4 }//@F4 }//@F4 queue.add(child);//@F4 }//@F4 } } // @C5A /** */ private PcmlProgram getProgramNode(String name) throws PcmlException { PcmlNode node; node = getElement(name); if (node instanceof PcmlProgram) { return (PcmlProgram) node; } else { if (node == null) throw new PcmlException(DAMRI.ELEMENT_NOT_FOUND, new Object[] {name, ""} ); else throw new PcmlException(DAMRI.WRONG_ELEMENT_TYPE, new Object[] {name, ""} ); } } /** */ private PcmlData getDataNode(String name) throws PcmlException { PcmlNode node; node = getElement(name); if (node instanceof PcmlData) { return (PcmlData) node; } else { if (node == null) throw new PcmlException(DAMRI.ELEMENT_NOT_FOUND, new Object[] {name, ""} ); else throw new PcmlException(DAMRI.WRONG_ELEMENT_TYPE, new Object[] {name, ""} ); } } // Returns the Pcml specification error (if any). PcmlSpecificationException getPcmlSpecificationException() { return m_PcmlSpecificationException; } // Add a Pcml specification error to the list of errors. void addPcmlSpecificationError(String key, Object[] args) { // If an exception object doesn't exist, create one if (m_PcmlSpecificationException == null) { m_PcmlSpecificationException = new PcmlSpecificationException(SystemResourceFinder.format(DAMRI.FAILED_TO_VALIDATE, new Object[] { m_docName })); } // Now add the message m_PcmlSpecificationException.addMessage(SystemResourceFinder.format(key, args)); } /** @E1A -- NEW XPCML METHODS -- **/ /** Generates XPCML representing the data contained in this program object. Throws a runtime exception if this object contains no data. @param outStream The output stream to which to write the text. @exception IOException If an error occurs while writing the data. @exception XmlException If an error occurs while processing XPCML. **/ void generateXPCML(String pgmName,OutputStream outStream) throws IOException, XmlException { PrintWriter xmlFile = new PrintWriter(outStream); //@E1A -- New variables for XPCML. These are used to control dimensioning // of a node. int num_dim =0; //@E1A int cur_dim=0; //@E1A int[] dimArray = {0,0,0,0,0,0,0,0,0,0}; PcmlDimensions dims = new PcmlDimensions(dimArray); //@E1A try { // Start XPCML stream by writing out xpcml tag // xmlFile.println(""); // @A1c xmlFile.println(""); xmlFile.print(""); } else xmlFile.println("'xpcml.xsd' >"); // xmlFile.println(" xmlns:xsi=" + "\"" + "http://www.w3.org/2001/XMLSchema-instance" + "\"" + // "\n" + " xsi:noNamespaceSchemaLocation='xpcml.xsd'>"); xmlFile.println(); // Check if pgmName is null. If so, generate XPCML for entire PCML node tree. If not, just // generate XPCML for node tree associated with the given program name. if (pgmName == null) generateXPCML(this, this, xmlFile, "",num_dim, cur_dim, dims); else { // Get node for the program requested generateXPCML(this, getProgramNode(pgmName), xmlFile, "", num_dim, cur_dim, dims); } xmlFile.println(""); if (xmlFile.checkError()) // Note: This flushes the stream. { Trace.log(Trace.ERROR, "Error when writing PCML to OutputStream."); } } finally { xmlFile.close(); // Note: close() flushes the stream first. } } /** Generates XPCML representing the data contained in the specified node. Throws a runtime exception if this object contains no data. @param pcmlDocNode The root node of the tree. @param node The node to generate PCML for. @param writer The writer to which to write the text. @param indent The indentation with which to prepend the generated XML. @exception IOException If an error occurs while writing the data. @exception XmlException If an error occurs while processing XPCML. **/ private void generateXPCML(PcmlDocument pcmlDocNode, PcmlDocNode node, PrintWriter writer, String indent, int num_dimensions, int current_dimension, PcmlDimensions dimensions) throws IOException, XmlException { int tempInt; String lastTag=""; boolean tagDone=false; // Define variable to be used to get count values of nodes PcmlDimensions dim = new PcmlDimensions(); dim.add(0); // Reset all dimensions if this is a or node if (node.getNodeType() == PcmlNodeType.PROGRAM || node.getNodeType() == PcmlNodeType.DOCUMENT) { for (int i = 0; i < dimensions.size(); ++i) { dimensions.set(i,0); } current_dimension = 0; num_dimensions = 0; } // Check if this is the first time this node has been processed. The initial value for CountReps // is -10. Get the count value for the node to determine how many times the node needs to be processed. if (node.getCountReps() == -10) { int nodeType = node.getNodeType(); if (nodeType == PcmlNodeType.DATA && ((PcmlData) node).getXPCMLCount(dim) != 0) { // If the node is a data node and it has a count set tempInt to the count tempInt = ( (PcmlData) node).getXPCMLCount(dim); } else if (nodeType== PcmlNodeType.STRUCT && ((PcmlStruct) node).getXPCMLCount(dim) != 0) { // If the node is a struct node and it has a count set tempInt to the count tempInt = ( (PcmlStruct) node).getXPCMLCount(dim); } else tempInt = 1; node.setCountReps(tempInt); // Set the number of reps for the node. // set current dimension dimensions.set(current_dimension, 0); } // Start the tag. Transform a 0 ) { if (dimensions.at(current_dimension)== 0) { // Check if this is a user defined element if (node.getCondensedName() != "") { writer.print(indent); writer.print("<"+ node.getCondensedName()); lastTag="arrayOfStringParm"; } else { writer.print(indent); writer.print(" 0 ) { if (dimensions.at(current_dimension)== 0) { // Check if this is a user defined element if (node.getCondensedName() != "") { writer.print(indent); writer.print("<"+ node.getCondensedName()); lastTag="arrayOfStructParm"; } else { writer.print(indent); writer.print(" 0 ) { if (dimensions.at(current_dimension)== 0) { // Check if this is a user defined element if (node.getCondensedName() != "") { writer.print(indent); writer.print("<"+ node.getCondensedName()); lastTag="arrayOfHexBinaryParm"; } else { writer.print(indent); writer.print(" 0 ) // if (node.getAttributeValue("count") != null ) { if (dimensions.at(current_dimension)== 0) { // Check if this is a user defined element if (node.getCondensedName() != "") { writer.print(indent); writer.print("<"+ node.getCondensedName()); lastTag="arrayOfUnsignedIntParm"; } else { writer.print(indent); writer.print(" 0 ) { if (dimensions.at(current_dimension)== 0) { // Check if this is a user defined element if (node.getCondensedName() != "") { writer.print(indent); writer.print("<"+ node.getCondensedName()); lastTag="arrayOfIntParm"; } else { writer.print(indent); writer.print(" 0 ) { if (dimensions.at(current_dimension)== 0) { // Check if this is a user defined element if (node.getCondensedName() != "") { writer.print(indent); writer.print("<"+ node.getCondensedName()); lastTag="arrayOfUnsignedShortParm"; } else { writer.print(indent); writer.print(" 0 ) { if (dimensions.at(current_dimension)== 0) { // Check if this is a user defined element if (node.getCondensedName() != "") { writer.print(indent); writer.print("<"+ node.getCondensedName()); lastTag="arrayOfShortParm"; } else { writer.print(indent); writer.print(" 0 ) // if (node.getAttributeValue("count") != null ) { if (dimensions.at(current_dimension)== 0) { // Check if this is a user defined element if (node.getCondensedName() != "") { writer.print(indent); writer.print("<"+ node.getCondensedName()); lastTag="arrayOfUnsignedLongParm"; } else { writer.print(indent); writer.print(" 0 ) { if (dimensions.at(current_dimension)== 0) { // Check if this is a user defined element if (node.getCondensedName() != "") { writer.print(indent); writer.print("<"+ node.getCondensedName()); lastTag="arrayOfLongParm"; } else { writer.print(indent); writer.print(" 0 ) { if (dimensions.at(current_dimension)== 0) { // Check if this is a user defined element if (node.getCondensedName() != "") { writer.print(indent); writer.print("<"+ node.getCondensedName()); lastTag="arrayOfFloatParm"; } else { writer.print(indent); writer.print(" 0 ) { if (dimensions.at(current_dimension)== 0) { // Check if this is a user defined element if (node.getCondensedName() != "") { writer.print(indent); writer.print("<"+ node.getCondensedName()); lastTag="arrayOfDoubleParm"; } else { writer.print(indent); writer.print(" 0 ) { if (dimensions.at(current_dimension)== 0) { // Check if this is a user defined element if (node.getCondensedName() != "") { writer.print(indent); writer.print("<"+ node.getCondensedName()); lastTag="arrayOfZonedDecimalParm"; } else { writer.print(indent); writer.print(" 0 ) { if (dimensions.at(current_dimension)== 0) { // Check if this is a user defined element if (node.getCondensedName() != "") { writer.print(indent); writer.print("<"+ node.getCondensedName()); lastTag="arrayOfPackedDecimalParm"; } else { writer.print(indent); writer.print(" 0 ) { if (dimensions.at(current_dimension)== 0) { // Check if this is a user defined element if (node.getCondensedName() != "") { writer.print(indent); writer.print("<"+ node.getCondensedName()); lastTag="arrayOfDateParm"; } else { writer.print(indent); writer.print(" 0 ) { if (dimensions.at(current_dimension)== 0) { // Check if this is a user defined element if (node.getCondensedName() != "") { writer.print(indent); writer.print("<"+ node.getCondensedName()); lastTag="arrayOfTimeParm"; } else { writer.print(indent); writer.print(" 0 ) { if (dimensions.at(current_dimension)== 0) { // Check if this is a user defined element if (node.getCondensedName() != "") { writer.print(indent); writer.print("<"+ node.getCondensedName()); lastTag="arrayOfTimestampParm"; } else { writer.print(indent); writer.print(" 0 ) { if (dimensions.at(current_dimension)== 0) { // Check if this is a user defined element if (node.getCondensedName() != "") { writer.print(indent); writer.print("<"+ node.getCondensedName()); lastTag="arrayOfStruct"; } else { writer.print(indent); writer.print(" 0) { // Don't output attributes working on an array element entry } else if (node.getNodeType() == PcmlNodeType.STRUCT && dimensions.at(current_dimension) > 0) { } else { for (int i=0; i"); } // Add struct_i tag for struct arrays if (node.getNodeType() == PcmlNodeType.DATA && ((PcmlData)node).getDataType()==PcmlData.STRUCT && ((PcmlData) node).getXPCMLCount(dim) > 0 ) { if (dimensions.at(current_dimension) > 0) { writer.println(indent+ " "); } else { writer.println(indent+ " " ); } lastTag="struct_i"; } if (node.getNodeType() == PcmlNodeType.STRUCT && ((PcmlStruct) node).getXPCMLCount(dim) > 0 ) { if (dimensions.at(current_dimension) > 0) { writer.println(indent+ " "); } else { writer.println(indent+ " " ); } lastTag="struct_i"; } // if node = "program" write out "paramterList" tag if ( node.getNodeType() == PcmlNodeType.PROGRAM) { writer.println(indent + ""); } Enumeration children = node.getChildren(); // Process the children of the current node while (children.hasMoreElements()) { PcmlDocNode child = (PcmlDocNode) children.nextElement(); // Reset all dimensions if this is tag if (child.getNodeType() == PcmlNodeType.PROGRAM) { for (int i = 0; i < dimensions.size(); ++i) { dimensions.set(i, 0); } current_dimension = 0; num_dimensions = 0; } // Reset all dimensions if this is a tag without a parent if (child.getNodeType() == PcmlNodeType.STRUCT && child.getParent().getName().trim().length()==0 ) { for (int i = 0; i < dimensions.size(); ++i) { dimensions.set(i, 0); } current_dimension = 0; num_dimensions = 0; } // Increase dimensions by 1 if this is the first child, i.e., we've gone down the tree if (child.getChildNbr()==0 && ( (node.getNodeType() == PcmlNodeType.DATA && ((PcmlData) node).getXPCMLCount(dim) > 0 ) || (node.getNodeType() == PcmlNodeType.STRUCT && ((PcmlStruct) node).getXPCMLCount(dim) > 0 ) ) ) { num_dimensions++; current_dimension++; if (num_dimensions > 9) dimensions.add(0); } // Call generateXPCML to generate XPCML for the child generateXPCML(pcmlDocNode,child, writer, indent+" ", num_dimensions, current_dimension, dimensions); } // end while more children // Write the end tag. // If node = "program" write out "paramterList" tag if ( node.getNodeType() == PcmlNodeType.PROGRAM) writer.println(indent + ""); if ( node.getNodeType() == PcmlNodeType.DATA) { int dataType = ((PcmlData) node).getDataType(); writer.print(indent); switch( dataType ) { case PcmlData.CHAR: if ( ((PcmlData) node).getXPCMLCount(dim) <= 0 ) { if (node.getCondensedName() != "") writer.print(""); else writer.print(""); } else { // Processing multiples of the node. Should end with "); if (node.getCountReps() == 0) { if (node.getCondensedName() != "") writer.print(""); else { // All done with elements. Add end array tag writer.println( ""); } } } break; case PcmlData.STRUCT: if ( ((PcmlData) node).getXPCMLCount(dim) <= 0 ) { if (node.getCondensedName() != "") writer.print(""); else writer.println(""); } else { // Processing multiples of the node. Should end with "); if (node.getCountReps() == 0) { if (node.getCondensedName() != "") writer.print(indent+""); else { // All done with elements. Add end array tag writer.println(indent+""); } } } break; case PcmlData.BYTE: if ( ((PcmlData) node).getXPCMLCount(dim) <= 0 ) { if (node.getCondensedName() != "") writer.print(""); else writer.print(""); } else { // Processing multiples of the node. Should end with "); if (node.getCountReps() == 0) { if (node.getCondensedName() != "") writer.print(""); else { // All done with elements. Add end array tag writer.println( ""); } } } break; case PcmlData.INT: if (node.getAttributeValue("length").equals("4") && node.getAttributeValue("precision") != null && node.getAttributeValue("precision").equals("32") ) { if ( ((PcmlData) node).getXPCMLCount(dim) <= 0 ) { if (node.getCondensedName() != "") writer.print(""); else writer.print(""); } else { // Processing multiples of the node. Should end with "); if (node.getCountReps() == 0) { if (node.getCondensedName() != "") writer.print(""); else { // All done with elements. Add end array tag writer.println( ""); } } } } else if (node.getAttributeValue("length").equals("4")) { if ( ((PcmlData) node).getXPCMLCount(dim) <= 0 ) { if (node.getCondensedName() != "") writer.print(""); else writer.print(""); } else { // Processing multiples of the node. Should end with "); if (node.getCountReps() == 0) { if (node.getCondensedName() != "") writer.print(""); else { // All done with elements. Add end array tag writer.println( ""); } } } } else if (node.getAttributeValue("length").equals("2") && node.getAttributeValue("precision") != null && node.getAttributeValue("precision").equals("16")) { if ( ((PcmlData) node).getXPCMLCount(dim) <= 0 ) { if (node.getCondensedName() != "") writer.print(""); else writer.print(""); } else { // Processing multiples of the node. Should end with "); if (node.getCountReps() == 0) { if (node.getCondensedName() != "") writer.print(""); else { // All done with elements. Add end array tag writer.println( ""); } } } } else if (node.getAttributeValue("length").equals("2")) { if ( ((PcmlData) node).getXPCMLCount(dim) <= 0 ) { if (node.getCondensedName() != "") writer.print(""); else writer.print(""); } else { // Processing multiples of the node. Should end with "); if (node.getCountReps() == 0) { if (node.getCondensedName() != "") writer.print(""); else { // All done with elements. Add end array tag writer.println( ""); } } } } else if (node.getAttributeValue("length").equals("8")) { if ( ((PcmlData) node).getXPCMLCount(dim) <= 0 ) { if (node.getCondensedName() != "") writer.print(""); else writer.print(""); } else { // Processing multiples of the node. Should end with "); if (node.getCountReps() == 0) { if (node.getCondensedName() != "") writer.print(""); else { // All done with elements. Add end array tag writer.println( ""); } } } } break; case PcmlData.FLOAT: if (node.getAttributeValue("length").equals("4")) { if ( ((PcmlData) node).getXPCMLCount(dim) <= 0 ) { if (node.getCondensedName() != "") writer.print(""); else writer.print(""); } else { // Processing multiples of the node. Should end with "); if (node.getCountReps() == 0) { if (node.getCondensedName() != "") writer.print(""); else { // All done with elements. Add end array tag writer.println( ""); } } } } else if (node.getAttributeValue("length").equals("8")) { if ( ((PcmlData) node).getXPCMLCount(dim) <= 0 ) { if (node.getCondensedName() != "") writer.print(""); else writer.print(""); } else { // Processing multiples of the node. Should end with "); if (node.getCountReps() == 0) { if (node.getCondensedName() != "") writer.print(""); else { // All done with elements. Add end array tag writer.println( ""); } } } } break; case PcmlData.ZONED: if ( ((PcmlData) node).getXPCMLCount(dim) <= 0 ) { if (node.getCondensedName() != "") writer.print(""); else writer.print(""); } else { // Processing multiples of the node. Should end with "); if (node.getCountReps() == 0) { if (node.getCondensedName() != "") writer.print(""); else { // All done with elements. Add end array tag writer.println( ""); } } } break; case PcmlData.PACKED: if ( ((PcmlData) node).getXPCMLCount(dim) <= 0 ) { if (node.getCondensedName() != "") writer.print(""); else writer.print(""); } else { // Processing multiples of the node. Should end with "); if (node.getCountReps() == 0) { if (node.getCondensedName() != "") writer.print(""); else { // All done with elements. Add end array tag writer.println( ""); } } } break; case PcmlData.DATE: if ( ((PcmlData) node).getXPCMLCount(dim) <= 0 ) { if (node.getCondensedName() != "") writer.print(""); else writer.print(""); } else { // Processing multiples of the node. Should end with "); if (node.getCountReps() == 0) { if (node.getCondensedName() != "") writer.print(""); else { // All done with elements. Add end array tag writer.println( ""); } } } break; case PcmlData.TIME: if ( ((PcmlData) node).getXPCMLCount(dim) <= 0 ) { if (node.getCondensedName() != "") writer.print(""); else writer.print(""); } else { // Processing multiples of the node. Should end with "); if (node.getCountReps() == 0) { if (node.getCondensedName() != "") writer.print(""); else { // All done with elements. Add end array tag writer.println( ""); } } } break; case PcmlData.TIMESTAMP: if ( ((PcmlData) node).getXPCMLCount(dim) <= 0 ) { if (node.getCondensedName() != "") writer.print(""); else writer.print(""); } else { // Processing multiples of the node. Should end with "); if (node.getCountReps() == 0) { if (node.getCondensedName() != "") writer.print(""); else { // All done with elements. Add end array tag writer.println( ""); } } } break; default: throw new PcmlException(DAMRI.BAD_DATA_TYPE, new Object[] {Integer.valueOf(dataType) , "*"} ); } // switch(dataType) } // PcmlNodeType.DATA else if (node.getNodeType() == PcmlNodeType.STRUCT) { writer.print(indent); if ( ((PcmlStruct) node).getXPCMLCount(dim) <= 0 ) { if (node.getCondensedName() != "") writer.print(""); else writer.println(""); } else { // Processing multiples of the node. Should end with "); if (node.getCountReps() == 0) { if (node.getCondensedName() != "") writer.print(indent+""); else { // All done with elements. Add end array tag writer.println(indent+ ""); } } } } else if (node.getNodeType() == PcmlNodeType.DOCUMENT) { } else writer.println(indent + ""); if ( node.getNodeType() == PcmlNodeType.PROGRAM) writer.println(); // Reduce current and number of dimensions by 1 since their are no // more children on this node (stepping back up the tree a level) if ((node.getNodeType() == PcmlNodeType.DATA && ((PcmlData) node).getXPCMLCount(dim) > 0 ) || (node.getNodeType() == PcmlNodeType.STRUCT && ((PcmlStruct) node).getXPCMLCount(dim) > 0 )) { current_dimension--; num_dimensions--; } if (node.getCountReps() > 0) { // This node has a count greater than 0 so we need to process it again... // Calculate index Integer countVal= Integer.valueOf(0); if (node.getNodeType() == PcmlNodeType.DATA && ((PcmlData) node).getXPCMLCount(dim) != 0) countVal = Integer.valueOf( ((PcmlData )node).getXPCMLCount(dim)); if (node.getNodeType() == PcmlNodeType.STRUCT && ((PcmlStruct) node).getXPCMLCount(dim) != 0) countVal = Integer.valueOf( ((PcmlStruct )node).getXPCMLCount(dim)); if (node.getNodeType() != PcmlNodeType.DATA && node.getNodeType() != PcmlNodeType.STRUCT) countVal = Integer.valueOf(0); dimensions.set(current_dimension, countVal.intValue() - node.getCountReps()); // Reprocess the node because count is > 0 so we need to print it out at least one more time generateXPCML(pcmlDocNode, node, writer, indent, num_dimensions, current_dimension, dimensions); } else { // We're done with this node so reset its count to -10 node.setCountReps(-10); } } else // node has no children { // Write out value of node as string value if (node.getNodeType() == PcmlNodeType.DATA && !node.getAttributeValue("type").equals("struct")) { // Get the value of the data node and print out as a string String strVal=""; // String tempVal=""; Object objVal; int[] indices = new int[1]; indices[0] = node.getCountReps(); if (node.getQualifiedName().trim().equals("")) { // Can't set values of unnamed nodes strVal=""; } else { try { if (node.getAttributeValue("type").equals("byte")) { int length = ((PcmlData)node).getLength(dimensions); byte[] byteVal = new byte[length]; byteVal = (byte[]) pcmlDocNode.getValue(node.getQualifiedName(), dimensions); if (byteVal != null) { // Now hexBinary encode the bytes strVal = BinaryConverter.bytesToString(byteVal); } } else { strVal =""; objVal = pcmlDocNode.getValue(node.getQualifiedName(), dimensions); if (objVal != null) { if (objVal instanceof java.sql.Date) { strVal = AS400Date.toXsdString(objVal, getTimeZone()); } else if (objVal instanceof java.sql.Time) { strVal = AS400Time.toXsdString(objVal, getTimeZone()); } else if (objVal instanceof java.sql.Timestamp) { strVal = AS400Timestamp.toXsdString(objVal, getTimeZone()); } else { strVal = objVal.toString(); strVal = escapeSpecialCharaceters(strVal); //@M1 } } } } catch (PcmlException e) { Trace.log(Trace.PCML, "No value for node: " + node.getQualifiedName() ); } catch (java.lang.IllegalArgumentException e) { Trace.log(Trace.PCML, "Bad count value for node: " + node.getQualifiedName() ); } } // Check if CountReps > 0. If so, then dealing with an array if ( ((PcmlData) node).getXPCMLCount(dim) > 0 ) { // Outputting an array element. Check which type first if (lastTag.equals("arrayOfStringParm") || lastTag.equals("arrayOfIntParm") || lastTag.equals("arrayOfLongParm") || lastTag.equals("arrayOfUnsignedLongParm") || lastTag.equals("arrayOfUnsignedIntParm") || lastTag.equals("arrayOfShortParm") || lastTag.equals("arrayOfUnsignedShortParm") || lastTag.equals("arrayOfFloatParm") || lastTag.equals("arrayOfDoubleParm") || lastTag.equals("arrayOfZonedDecimalParm") || lastTag.equals("arrayOfPackedDecimalParm") || lastTag.equals("arrayOfHexBinaryParm") || lastTag.equals("arrayOfDateParm") || lastTag.equals("arrayOfTimeParm") || lastTag.equals("arrayOfTimestampParm") ) writer.println(">"); if (dimensions.at(current_dimension) > 0) { writer.print(indent + " " + strVal); } else { writer.print(indent + " " + strVal); } } else writer.print(">" + strVal); // Check if CountReps > 0. If so, then dealing with an array if ( ((PcmlData) node).getXPCMLCount(dim) > 0 ) { // Outputting an array element. writer.println(""); tagDone = true; } else { writer.println(""); tagDone = true; } } // Finish the start tag. if (!tagDone) writer.println("/>"); if (node.getCountReps() > 0) // Only called when data has a count > 0 { // Calculate index int countVal; if (node.getNodeType() == PcmlNodeType.DATA && ((PcmlData) node).getXPCMLCount(dim) != 0) countVal = ((PcmlData )node).getXPCMLCount(dim); else if (node.getNodeType() == PcmlNodeType.STRUCT && ((PcmlStruct) node).getXPCMLCount(dim) != 0) countVal = ((PcmlStruct)node).getXPCMLCount(dim); else countVal = 0; int countV=countVal-node.getCountReps(); dimensions.set(current_dimension, countVal - node.getCountReps()); generateXPCML(pcmlDocNode,node, writer, indent, num_dimensions, current_dimension, dimensions); } else { // Check if CountReps > 0. If so, then dealing with an array if ( ((PcmlData) node).getXPCMLCount(dim) > 0 ) { // Outputting an array element. Check which type first int dataType = ((PcmlData) node).getDataType(); writer.print(indent); switch( dataType ) { case PcmlData.CHAR: if (node.getCondensedName() != "") writer.println(""); else writer.println(""); break; case PcmlData.INT: if (node.getAttributeValue("length").equals("4") && node.getAttributeValue("precision") != null && node.getAttributeValue("precision").equals("32") ) { if (node.getCondensedName() != "") writer.println(""); else writer.println(""); } else if (node.getAttributeValue("length").equals("4") ) { if (node.getCondensedName() != "") writer.println(""); else writer.println(""); } else if (node.getAttributeValue("length").equals("2") && node.getAttributeValue("precision") != null && node.getAttributeValue("precision").equals("16") ) { if (node.getCondensedName() != "") writer.println(""); else writer.println(""); } else if (node.getAttributeValue("length").equals("2") ) { if (node.getCondensedName() != "") writer.println(""); else writer.println(""); } else if (node.getAttributeValue("length").equals("8") ) { if (node.getCondensedName() != "") writer.println(""); else writer.println(""); } break; case PcmlData.FLOAT: if (node.getAttributeValue("length").equals("8") ) { if (node.getCondensedName() != "") writer.println(""); else writer.println(""); } else if (node.getAttributeValue("length").equals("4") ) { if (node.getCondensedName() != "") writer.println(""); else writer.println(""); } break; case PcmlData.ZONED: if (node.getCondensedName() != "") writer.println(""); else writer.println(""); break; case PcmlData.PACKED: if (node.getCondensedName() != "") writer.println(""); else writer.println(""); break; case PcmlData.BYTE: if (node.getCondensedName() != "") writer.println(""); else writer.println(""); break; case PcmlData.DATE: if (node.getCondensedName() != "") writer.println(""); else writer.println(""); break; case PcmlData.TIME: if (node.getCondensedName() != "") writer.println(""); else writer.println(""); break; case PcmlData.TIMESTAMP: if (node.getCondensedName() != "") writer.println(""); else writer.println(""); break; default: break; } } node.setCountReps(-10); } } // end else if string value writer.flush(); } // end generateXPCML // ****************************** // @E0 -- New methods for XPCML * // ****************************** void setXsdName(String xsdName) { m_XsdName = xsdName; } String getXsdName() { return m_XsdName; } // ****************************** // @E0 -- New method for XPCML * // ****************************** void copyValues(PcmlDocNode root, PcmlDocNode node) throws IOException, XmlException { //@E1A -- New variables for XPCML. These are used to control dimensioning // of a node. int num_dim =0; //@E1A int cur_dim=0; //@E1A int[] dims = {0,0,0,0,0,0,0,0,0,0}; PcmlDimensions dimensions = new PcmlDimensions(dims); String[] struct = new String[30]; String[] structR = new String[30]; int count=0; copyValues(root, node, num_dim, cur_dim, dimensions ,struct, structR, count); } void copyValues(PcmlDocNode root, PcmlDocNode node, int num_dimensions, int current_dimension, PcmlDimensions dimensions,String[] struct, String[] structR, int count) throws IOException, XmlException { int tempInt; // Define variable to be used to get count values of nodes PcmlDimensions dim = new PcmlDimensions(); dim.add(0); if (count < 0 ) count=0; // Reset all dimensions if this is a or node if (node.getNodeType() == PcmlNodeType.PROGRAM || node.getNodeType() == PcmlNodeType.DOCUMENT) { for (int i = 0; i < dimensions.size(); ++i) { dimensions.set(i,0); } current_dimension = 0; num_dimensions = 0; for (int i=0; i< 10 ; ++i) { struct[i] = ""; structR[i] = ""; } count=0; } // Check if this is the first time this node has been processed. The initial value for CountReps // is -10. Get the count value for the node to determine how many times the node needs to be processed. if (node.getNodeType() == PcmlNodeType.DATA && node.getAttributeValue("type").equals("struct")) { PcmlDocNode parent = (PcmlDocNode) node.getParent(); if ( parent.getNodeType()==PcmlNodeType.DATA && parent.getAttributeValue("type").equals("struct")) count++; else count=0; // Check if parent is a struct ref also structR[count] = node.getName(); struct[count] = node.getAttributeValue("struct"); } if (node.getCountReps() == -10) { int nodeType = node.getNodeType(); if (nodeType == PcmlNodeType.DATA && ((PcmlData) node).getXPCMLCount(dim) != 0 ) { // If the node is a data node and it has a count set tempInt to the count tempInt = ( (PcmlData) node).getXPCMLCount(dim); } else if (nodeType== PcmlNodeType.STRUCT && ((PcmlStruct) node).getXPCMLCount(dim) != 0) { // If the node is a struct node and it has a count set tempInt to the count tempInt = ( (PcmlStruct) node).getXPCMLCount(dim); } else tempInt = 1; node.setCountReps(tempInt); // Set the number of reps for the node. // set current dimension dimensions.set(current_dimension,0); } // Reduce the count for this node by 1. if (node.getCountReps() != 0) { node.setCountReps(node.getCountReps() - 1); } if (node.hasChildren()) { Enumeration children = node.getChildren(); // Process the children of the current node while (children.hasMoreElements()) { PcmlDocNode child = (PcmlDocNode) children.nextElement(); // Reset all dimensions if this is tag if (child.getNodeType() == PcmlNodeType.PROGRAM) { for (int i = 0; i < dimensions.size(); ++i) { dimensions.set(i,0); } current_dimension = 0; num_dimensions = 0; for (int i=0; i< 30 ; ++i) { struct[i] = ""; structR[i] = ""; } count=0; } // Reset all dimensions if this is a tag without a parent if (child.getNodeType() == PcmlNodeType.STRUCT && child.getParent().getName().trim().length()==0 ) { for (int i = 0; i < dimensions.size(); ++i) { dimensions.set(i,0); } current_dimension = 0; num_dimensions = 0; for (int i=0; i< 30 ; ++i) { struct[i] = ""; structR[i] = ""; } count=0; } // Increase dimensions by 1 if this is the first child, i.e., we've gone down the tree if (child.getChildNbr()==0 && ( (node.getNodeType() == PcmlNodeType.DATA && ((PcmlData) node).getXPCMLCount(dim) > 0 ) || (node.getNodeType() == PcmlNodeType.STRUCT && ((PcmlStruct) node).getXPCMLCount(dim) > 0 ) ) ) { num_dimensions++; current_dimension++; if (num_dimensions > 9) dimensions.add(0); } copyValues(root, child, num_dimensions, current_dimension, dimensions, struct, structR, count); } // end while more children // Reduce current and number of dimensions by 1 since their are no // more children on this node (stepping back up the tree a level) if ((node.getNodeType() == PcmlNodeType.DATA && ((PcmlData) node).getXPCMLCount(dim) > 0 ) || (node.getNodeType() == PcmlNodeType.STRUCT && ((PcmlStruct) node).getXPCMLCount(dim) > 0 )) { current_dimension--; num_dimensions--; count--; } if (node.getCountReps() > 0) { // This node has a count greater than 0 so we need to process it again... // Calculate index Integer countVal= Integer.valueOf(0); if (node.getNodeType() == PcmlNodeType.DATA && ((PcmlData) node).getXPCMLCount(dim) != 0) countVal = Integer.valueOf( ((PcmlData )node).getXPCMLCount(dim)); if (node.getNodeType() == PcmlNodeType.STRUCT && ((PcmlStruct) node).getXPCMLCount(dim) != 0) countVal = Integer.valueOf( ((PcmlStruct )node).getXPCMLCount(dim)); if (node.getNodeType() != PcmlNodeType.DATA && node.getNodeType() != PcmlNodeType.STRUCT) countVal = Integer.valueOf(0); dimensions.set(current_dimension, (countVal.intValue() - node.getCountReps())); // Reprocess the node because count is > 0 so we need to print it out at least one more time copyValues(root, node, num_dimensions, current_dimension, dimensions, struct, structR, count); } else { // We're done with this node so reset its count to -10 node.setCountReps(-10); } } else // node has no children { // Copy the node value to the new node if (node.getNodeType() == PcmlNodeType.DATA && !node.getAttributeValue("type").equals("struct")) { // Get the value of the data node and print out as a string String strVal=""; String tempVal=""; Object objVal; int[] indices = new int[1]; indices[0] = node.getCountReps(); if (node.getQualifiedName().trim().equals("")) { // Can't set values of unnamed nodes objVal=null; } else { try { objVal=null; if ( node.getUsage() != PcmlDocNode.OUTPUT ) { objVal = ((PcmlData)node).getValue(dimensions); if (objVal == null) { // Not set yet so see if the referenced node has values and set it to if ( ( (PcmlDocNode)node.getParent()).getNodeType()==PcmlNodeType.DATA && ( (PcmlDocNode)node.getParent()).getAttributeValue("type").equals("struct") || ( (PcmlDocNode)node.getParent()).getNodeType()==PcmlNodeType.STRUCT) { // Parent is a struct parm. Check if struct parm referenced value // is set and, if so, copy that value here... boolean augmented=false; PcmlDocNode parent = (PcmlDocNode) node.getParent(); while ( parent != null && parent.getNodeType() != PcmlNodeType.PROGRAM && augmented==false ) { if (parent.getNodeType() == PcmlNodeType.DATA && parent.getAttributeValue("type").equals("struct") ) augmented=true; parent=(PcmlDocNode) parent.getParent(); } if (augmented) { String[] nodeName = new String[count+1]; for (int i=0; i<= count ; i++) { nodeName[i] = ""; } boolean found=false; for (int i=0; i<=count && !found; ++i) { nodeName[i] = struct[i]; for (int j=i+1; j<=count; j++) { nodeName[i]= nodeName[i] + "." + structR[j]; } String fullName = node.getQualifiedName(); String restOfName = fullName.substring(fullName.indexOf(structR[count])+structR[count].length()+1); nodeName[i] = nodeName[i] + "." + restOfName; String nodeToGet = nodeName[i]; PcmlDocRoot rootOfTree = root.getRootNode(); PcmlData nodeToCopy=null; try { nodeToCopy = (PcmlData) rootOfTree.getElement(nodeToGet); } catch (NullPointerException e) { nodeToCopy = null; } if (nodeToCopy != null) { int copyDim = nodeToCopy.getNbrOfDimensions(); int[] copy_dim = new int[copyDim]; for (int j=0; j< copyDim; ++j) { copy_dim[j] = dimensions.at( ((PcmlData) node).getNbrOfDimensions() - copyDim + j); } PcmlDimensions copy_dimensions = new PcmlDimensions(copy_dim); objVal = nodeToCopy.getValue(copy_dimensions); if (objVal != null) { ((PcmlData) node).setValue(objVal, dimensions); found=true; } } } } } } } } catch (PcmlException e) { Trace.log(Trace.PCML, "No value for node: " + node.getQualifiedName() ); } catch (java.lang.IllegalArgumentException e) { Trace.log(Trace.PCML, "Bad count value for node: " + node.getQualifiedName() ); } } // Check if CountReps > 0. If so, then dealing with an array if (node.getCountReps() > 0) // Only called when data has a count > 0 { // Calculate index int countVal; if (node.getNodeType() == PcmlNodeType.DATA && ((PcmlData) node).getXPCMLCount(dim) != 0) countVal = ((PcmlData )node).getXPCMLCount(dim); else if (node.getNodeType() == PcmlNodeType.STRUCT && ((PcmlStruct) node).getXPCMLCount(dim) != 0) countVal = ((PcmlStruct)node).getXPCMLCount(dim); else countVal = 0; dimensions.set(current_dimension, (countVal - node.getCountReps())); copyValues(root, node, num_dimensions, current_dimension, dimensions, struct, structR, count); } else { node.setCountReps(-10); } } // end else if string value } } public TimeZone getTimeZone() { return AS400.getDefaultTimeZone(getAs400()); } //@M1A START private static final char[] SPECIAL_LT = {'&','l','t',';'}; private static final char[] SPECIAL_GT = {'&','g','t',';'}; private static final char[] SPECIAL_APOS = {'&','a','p','o','s',';'}; private static final char[] SPECIAL_QUOT = {'&','q','u','o','t',';'}; private static final char[] SPECIAL_AMP = {'&','a','m','p',';'}; /** * Compares portions of two character arrays for equality. Returns true if * the specified portions match. * @param array1 First array to compare. * @param offset1 Offset in first array to begin comparison. * @param array2 Second array to compare. * @param offset2 Offset in second array to begin comparison. * @param length Number of elements to compare. * @return True if both portions are identical, else false. */ private static boolean compareCharArrays(char[] array1, int offset1, char[] array2, int offset2, int length) { if(array1.length-offset1': output.append(">"); break; case '\'': output.append("'"); break; case '"': output.append("""); break; case '&': //Need to determine if it's followed by special sequence if(compareCharArrays(data, i, SPECIAL_AMP, 0, SPECIAL_AMP.length)) { output.append("&"); i+= (SPECIAL_AMP.length-1); //-1 because the loop will add 1 } else if(compareCharArrays(data, i, SPECIAL_LT, 0, SPECIAL_LT.length)) { output.append("<"); i+= (SPECIAL_GT.length-1); } else if(compareCharArrays(data, i, SPECIAL_GT, 0, SPECIAL_GT.length)) { output.append(">"); i+= (SPECIAL_GT.length-1); } else if(compareCharArrays(data, i, SPECIAL_APOS, 0, SPECIAL_APOS.length)) { output.append("'"); i+= (SPECIAL_APOS.length-1); } else if(compareCharArrays(data, i, SPECIAL_QUOT, 0, SPECIAL_QUOT.length)) { output.append("""); i+= (SPECIAL_GT.length-1); } else output.append("&"); break; default: output.append(data[i]); } } return output.toString(); } //@M1A END }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy