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

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

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

package com.ibm.as400.access;

import java.util.Vector;
import java.io.IOException;


/**
 * The PrintObjectList class is an abstract base class for the various types of network print object lists.
**/
abstract class PrintObjectListImplRemote implements PrintObjectListImpl, Runnable
{
  private static final String copyright = "Copyright (C) 1997-2002 International Business Machines Corporation and others.";

  // These instance variable are persistent.
  private NPCPAttributeIDList attrsToRetrieve_;
  private NPCPID idFilter_;          // for certain lists an idcodepoint may be used to filter
  private NPCPSelection selection_;  // selection filter
  private AS400ImplRemote system_;   // @A4 - changed from AS400 to AS400ImplRemote
  private int typeOfObject_ = 0;     // indicates object type

  // These instance variables are not persistent.
  private transient boolean completed_ = false;
  private transient boolean closed_ = false;
  private final Object closeLock_ = new Object();
  private transient boolean waiting_ = false;
  private final Object waitLock_ = new Object();
  private transient int waitingForItem_ = -1;
  private transient Vector cpidList_ = null;                 // @A5A
  private transient Vector cpattrList_ = null;               // @A5A
  private transient int numItems_ = 0;

  private transient Exception anyException_ = null;
  private transient Vector printObjectListListeners_ = null;

  private boolean useCache_ = true;

  /**
    *Adds the specified PrintObjectList listener to receive
    *PrintObjectList events from this print object list.
    *
    * @see #removePrintObjectListListener
    * @param listener The PrintObjectList listener.
    **/
  public void addPrintObjectListListener(PrintObjectListListener listener)
  {
    if (printObjectListListeners_ == null) printObjectListListeners_ = new Vector();
    synchronized(printObjectListListeners_)
    {
      printObjectListListeners_.addElement(listener);
    }
  }


  /**
   * This method does all of the datastream work and object creations.
  **/
  private void buildList()
  throws AS400Exception,
  AS400SecurityException,
  ErrorCompletingRequestException,
  InterruptedException,
  IOException,
  RequestNotSupportedException
  {

    NPSystem npSystem = NPSystem.getSystem(getSystem());
    if (npSystem != null)
    {
      NPConversation conversation = npSystem.getConversation();
      try
      {
        if (conversation != null)
        {
          boolean     fTossData = false;
          boolean     fMoreData = true;
          int         count     = 0;
          // @B1D int         ccsid = conversation.getHostCCSID();
          ConverterImpl converter = conversation.getConverter();        // @B1A
          AS400Server server = conversation.getServer();
          int correlation = server.newCorrelationId();
          NPDataStream request = new NPDataStream(typeOfObject_);
          NPDataStream reply;
          request.setCorrelation(correlation);
          request.setAction(NPDataStream.LIST);
          // @B1D request.setHostCCSID(ccsid);
          request.setConverter(converter);                              // @B1A
          if (selection_ != null)
          {
            request.addCodePoint(selection_);
          }
          if (idFilter_ != null)
          {
            request.addCodePoint(idFilter_);
          }
          if (attrsToRetrieve_ != null)
          {
            request.addCodePoint(attrsToRetrieve_);
          }
          else
          {
            request.addCodePoint(getDefaultAttrsToList());
          }
          server.send(request, correlation);
          do
          {
            reply = (NPDataStream)server.receive(correlation);
            if (reply == null)
            {
              // throw execption - internal error.
              fMoreData = false;
              Trace.log(Trace.ERROR, "buildList: Null reply from AS400Server.receive()!");
              throw new InternalErrorException(InternalErrorException.PROTOCOL_ERROR);
            }
            else
            {
              count++;
              fMoreData = !(reply.isLastReply());
              if (!fTossData)
              {
                // we are going to keep this reply so set the
                // ccsid so it can tranlsate its text correctly
                // @B1D reply.setHostCCSID(ccsid);
                reply.setConverter(converter);                            // @B1A
                int rc = reply.getReturnCode();
                if (rc == NPDataStream.RET_OK)
                {
                  // @A5D PrintObject npObject = newNPObject(getSystem(), reply);
                  NPCPID cpid = newNPCPID(reply);                        // @A5A
                  NPCPAttribute cpattr = newNPCPAttribute(reply);        // @A5A
                  if (cpid != null)                                      // @A5C
                  {
                    synchronized(closeLock_) // In case someone tries to close us before we're finished.
                    {
                      if (closed_ || completed_) // Don't add the item if we've been closed already.
                      {
                        // the list has been closed before
                        // it completed, start tossing any
                        // incoming data...
                        fTossData = true;
                      }
                      else
                      {
                        if (useCache_) cpidList_.addElement(cpid);
                        if (useCache_) cpattrList_.addElement(cpattr);
                        ++numItems_;
                        // tell any listeners an object was added to the list.                  // @A2A
                        firePrintObjectList(PrintObjectListEvent.OBJECT_ADDED, cpid, cpattr, null); // @A5A

                        if (waiting_ && waitingForItem_ <= numItems_ && waitingForItem_ > -1)
                        {
                          // Only notify if someone is waiting to be notified.
                          synchronized(waitLock_)
                          {
                            waitLock_.notifyAll();
                          }
                        }
                      }
                    }
                  }
                }
                else
                {
                  if (rc == NPDataStream.RET_CPF_MESSAGE)
                  {
                    NPCPAttribute cpCPFMessage = (NPCPAttribute)reply.getCodePoint(NPCodePoint.ATTRIBUTE_VALUE);
                    if (cpCPFMessage != null)
                    {
                      String strCPFMessageID = cpCPFMessage.getStringValue(PrintObject.ATTR_MSGID);
                      String strCPFMessageText = cpCPFMessage.getStringValue(PrintObject.ATTR_MSGTEXT);
                      String strCPFMessageHelp = cpCPFMessage.getStringValue(PrintObject.ATTR_MSGHELP);
                      if (Trace.traceOn_) Trace.log(Trace.ERROR, "buildList: CPF Message("+strCPFMessageID+") = " + strCPFMessageText + ", HelpText= " +strCPFMessageHelp); //@CRS
                      // Create an AS400Message object
                      AS400Message msg = new AS400Message(strCPFMessageID, strCPFMessageText);
                      msg.setHelp(strCPFMessageHelp);
                      AS400Exception e = new AS400Exception(msg);
                      // throw an exception containing our CPF message.
                      // our catcher will actually fire the error event.
                      throw e;
                    }
                  }
                  else
                  {
                    if (rc == NPDataStream.RET_EMPTY_LIST)
                    {
                      // the list is empty, that isn't an error.
                    }
                    else
                    {
                      // look at RC and throw appropriate exception
                      Trace.log(Trace.ERROR, "buildList: Host Return Code" + rc);
                      // we get back a 4 (INV_REQ_ACT) if we try to list
                      // AFP Resources on pre V3R7 systems..
                      if (rc == NPDataStream.RET_INV_REQ_ACT)
                      {
                        throw new RequestNotSupportedException(conversation.getAttribute(PrintObject.ATTR_NPSLEVEL),
                                                               RequestNotSupportedException.SYSTEM_LEVEL_NOT_CORRECT);
                      }
                      else
                      {
                        throw new ErrorCompletingRequestException(ErrorCompletingRequestException.AS400_ERROR,
                                                                  "QNPSERVS RC = " + rc);
                      }
                    }
                  }
                }
              }
            }
            // every 15 datastreams that we get we'll call to the
            // garbage collector to keep the VM from running out of
            // of memory
//@C0 - This is a serious performance bottleneck and should no longer be
//      necessary under today's "modern" JVMs.
//@C0D                      if ((count % 15) == 0)
//@C0D                      {
//@C0D                         System.gc();
//@C0D                      }
          } while (fMoreData);
        }
      }
      finally
      {
        npSystem.returnConversation(conversation);
      }
    }
  }



  // The sub classes have default constructors implemented
  // for JavaBean support in visual builders. We need to
  // check the run time state of the object. The sub classes
  // may add additional checks by having their own
  // checkRunTimeState(), but will call super.checkRunTimeState()
  // to get this check.
  void checkRunTimeState()
  {
    if (getSystem() == null)
    {
      Trace.log(Trace.ERROR, "checkRunTimeState: Parameter 'system' has not been set.");
      throw new ExtendedIllegalStateException(
                                             "system", ExtendedIllegalStateException.PROPERTY_NOT_SET);
    }
  }



  /**
   * Closes the list so that objects in the list can be garbage collected.
  **/
  public void close()
  {
    synchronized(closeLock_)
    {
      closed_ = true;
      cpidList_ = null;               // @A5A
      cpattrList_ = null;             // @A5A
      numItems_ = 0; //@CRS
      anyException_ = null;
    }

    // tell any listeners the list was closed.
    firePrintObjectList(PrintObjectListEvent.CLOSED, null, null, null); // @A5C

    // At this point, all other threads that are using us should not be
    // firing events.
  }



  // The JavaBeans 1.0 Specification strongly recommends to avoid
  // using a synchronized method to fire an event. We use a
  // synchronized block to locate the target listeners and then
  // call the event listeners from unsynchronized code.
  private void firePrintObjectList(int id,
                                   NPCPID cpid,                       // @A5A
                                   NPCPAttribute cpattr,              // @A5A
                                   Exception exception )
  {
    // Return if no listeners are registered.
    if (printObjectListListeners_ == null) //@CRS
    {
      return;
    }

    PrintObjectListEvent event;

    // Now that we know we have listeners, we construct
    // the event object. We could have passed an event
    // oject to firePrintObjectList() but that would be
    // extra overhead if there were no listeners.
    if (exception !=null)
    {
      event = new PrintObjectListEvent(this, exception);
    }
    else if (cpid != null)                                      // @A5D
    {
      event = new PrintObjectListEvent(this, cpid, cpattr);    // @A5D
    }
    else
    {
      event = new PrintObjectListEvent(this, id);
    }

    synchronized(printObjectListListeners_) // We don't want listeners being added/removed while we're firing an event.
    {
      int size = printObjectListListeners_.size();

      for (int i=0; i < size; i++)
      {
        switch (id)
        {
          // OBJECT_ADDED is the most frequent case.
          case PrintObjectListEvent.OBJECT_ADDED:
            ((PrintObjectListListener)printObjectListListeners_.elementAt(i)).listObjectAdded(event);
            break;

          case PrintObjectListEvent.CLOSED:
            ((PrintObjectListListener)printObjectListListeners_.elementAt(i)).listClosed(event);
            break;

          case PrintObjectListEvent.COMPLETED:
            ((PrintObjectListListener)printObjectListListeners_.elementAt(i)).listCompleted(event);
            break;

          case PrintObjectListEvent.ERROR_OCCURRED:
            ((PrintObjectListListener)printObjectListListeners_.elementAt(i)).listErrorOccurred(event);
            break;

          case PrintObjectListEvent.OPENED:
            ((PrintObjectListListener)printObjectListListeners_.elementAt(i)).listOpened(event);
            break;
        }
      }
    }
  }



  /**
    * Non-externalized abstract method that the child-classes implement to
    * set the default attributes to retrieve on the list
    **/
  abstract NPCPAttributeIDList getDefaultAttrsToList();


  // @A5A
  public synchronized NPCPID getNPCPID(int index)
  {
    return(NPCPID)cpidList_.elementAt(index);
  }

  // @A5A
  public synchronized NPCPAttribute getNPCPAttribute(int index)
  {
    return(NPCPAttribute)cpattrList_.elementAt(index);
  }


  // This method is called by subclassed ImplRemotes
  NPCPSelection getSelectionCP()
  {
    return selection_;
  }


  /**
   * Returns the system name. This method is primarily provided for visual
   * application builders that support JavaBeans.
   *
   * @return The system on which the objects in the list exist.
   **/
  final public AS400ImplRemote getSystem()
  {
    return system_;
  }



  /**
   * Checks if a list that was opened asynchronously has completed.
   * If any exception occurred while the list was being retrieved, it will
   * be thrown here.
   *
   * @exception AS400Exception If the server returns an error message.
   * @exception AS400SecurityException If a security or authority error occurs.
   * @exception ConnectionDroppedException If the connection is dropped unexpectedly.
   * @exception ErrorCompletingRequestException If an error occurs before the request is completed.
   * @exception InterruptedException If this thread is interrupted.
   * @exception IOException If an error occurs while communicating with the server.
   * @exception RequestNotSupportedException If the requested function is not supported because
   *                                         the server operating system is not at the correct level.
   * @return true if the list is completely built; false otherwise.
   **/
  public boolean isCompleted()
  throws  AS400Exception,
  AS400SecurityException,
  ConnectionDroppedException,
  ErrorCompletingRequestException,
  InterruptedException,
  IOException,
  RequestNotSupportedException
  {
    if (anyException_ != null)
    {
      rethrowException();
    }

    return completed_;
  }



  /**
   * Non-externalized abstract method that child-classes implement to
   * create the correct type of object from the reply datastream
   **/
  abstract NPCPID newNPCPID(NPDataStream reply);                      // @A5A



  // @A5A
  NPCPAttribute newNPCPAttribute(NPDataStream reply)
  {
    return(NPCPAttribute)reply.getCodePoint(NPCodePoint.ATTRIBUTE_VALUE);   // may return null
  }


  /**
   * Builds the list asynchronously.  This method starts a thread
   * to build the list and then returns. The caller may register
   * listeners to obtain status about the list, or call isCompleted(),
   * waitForItem(), or waitForListToComplete().
   **/
  public void openAsynchronously()
  {
    checkRunTimeState();

    // Start thread to build the list. We are runnable, and
    // so our run() method just calls openSynchronously().
    Thread t = new Thread(this);
    t.start();
  }


  /**
   * Builds the list synchronously. This method will not
   * return until the list has been built completely.
   * The caller may then call the getObjects() method
   * to get an enumeration of the list.
   *
   * @exception AS400Exception If the server returns an error message.
   * @exception AS400SecurityException If a security or authority error occurs.
   * @exception ConnectionDroppedException If the connection is dropped unexpectedly.
   * @exception ErrorCompletingRequestException If an error occurs before the request is completed.
   * @exception InterruptedException If this thread is interrupted.
   * @exception IOException If an error occurs while communicating with the server.
   * @exception RequestNotSupportedException If the requested function is not supported because the
   *              server operating system is not at the correct level.
   **/
  public void openSynchronously()
  throws  AS400Exception,
  AS400SecurityException,
  ConnectionDroppedException,
  ErrorCompletingRequestException,
  InterruptedException,
  IOException,
  RequestNotSupportedException
  {
    // check the run time state of the object. To support JavaBeans
    // the sub-classes provide null constructors.
    checkRunTimeState();

    AS400ImplRemote theSystem = getSystem();
    if (theSystem != null)
    {                                           // @A1A
      NPSystem npSystem = NPSystem.getSystem(theSystem);              // @A1A
      if (npSystem != null)                                           // @A1A
      {
        // @A1A
        NPConversation conversation = npSystem.getConversation();    // @A1A
        npSystem.returnConversation(conversation);                   // @A1A
      }                                                               // @A1A
    }                                                                  // @A1A

    cpidList_ = new Vector();                                       // @A5A
    cpattrList_ = new Vector();                                     // @A5A
    numItems_ = 0;
    completed_ = false;                                             // @A2A
    closed_ = false;                                                // @A6A
    anyException_ = null;                                           // @A2A

    // tell any listeners the list was opened.                         // @A2A
    firePrintObjectList(PrintObjectListEvent.OPENED, null, null, null); // @A2A @A5C

    try                                                                // @A2A
    {
      buildList();
    }                                                                  // @A2A
    catch (Exception e)                                               // @A2A
    {
      anyException_ = e;                                           // @A2A

      // tell any listeners an error occurred.                        // @A2A
      firePrintObjectList(PrintObjectListEvent.ERROR_OCCURRED, null, null, e); // @A2A @A5C
    }                                                                  // @A2A
    finally                                                            // @A2A
    {
      // tell any listeners the list completed.                       // @A2A
      firePrintObjectList(PrintObjectListEvent.COMPLETED, null, null, null);  // @A2A @A5C
      synchronized(waitLock_) // Notify anyone who is waiting for an item or the list to complete.
      {
        completed_ = true;
        waitLock_.notifyAll();
      }
    }                                                                  // @A2A

    // we caught the exception above so that we could notify any       // @A2A
    // listeners. Now, rethrow the exception for applications          // @A2A
    // that may not be using listeners.                                // @A2A
    if (anyException_ != null)                                        // @A2A
    {
      // @A2A
      rethrowException();                                             // @A2A
    }                                                                  // @A2A
  }



  /**
    *Removes the specified PrintObjectList listener
    *so that it no longer receives PrintObjectList events
    *from this print object list.
    *
    * @see #addPrintObjectListListener
    * @param listener The PrintObjectList listener.
    **/
  public void removePrintObjectListListener( PrintObjectListListener listener )
  {
    if (printObjectListListeners_ != null)
    {
      synchronized(printObjectListListeners_)
      {
        printObjectListListeners_.removeElement(listener);
      }
    }
  }



  /**
   * Resets the list of object attributes to retrieve.
   **/
  public void resetAttributesToRetrieve()
  {
    attrsToRetrieve_ = null;
  }



  /**
   * Resets the list filter back to default values.
   **/
  public void resetFilter()
  {
    selection_.reset();
    idFilter_ = null;      // effectively resets the id Codepoint filter;
  }



  // Rethrow any exception.
  private void rethrowException()
  throws  AS400Exception,
  AS400SecurityException,
  ConnectionDroppedException,
  ErrorCompletingRequestException,
  InterruptedException,
  IOException,
  RequestNotSupportedException
  {
    if (anyException_ instanceof AS400Exception)
      throw (AS400Exception)anyException_;
    if (anyException_ instanceof AS400SecurityException)
      throw (AS400SecurityException)anyException_;
    if (anyException_ instanceof ConnectionDroppedException)
      throw (ConnectionDroppedException)anyException_;
    if (anyException_ instanceof ErrorCompletingRequestException)
      throw (ErrorCompletingRequestException)anyException_;
    if (anyException_ instanceof InterruptedException)
      throw (InterruptedException)anyException_;
    if (anyException_ instanceof IOException)
      throw (IOException)anyException_;
    if (anyException_ instanceof RequestNotSupportedException)
      throw (RequestNotSupportedException)anyException_;

    // runtime exceptions

    if (anyException_ instanceof ExtendedIllegalStateException)
      throw (ExtendedIllegalStateException)anyException_;
    if (anyException_ instanceof NullPointerException)
      throw (NullPointerException)anyException_;

    // If we get here we are getting an exception we overlooked,
    // trace it for debugging.

    Trace.log(Trace.ERROR, "rethrowException: Exception was not rethrown.");
  }



  /**
   * This is called automatically when Thread.start() is called by openAsynchronously().
   **/
  public void run()
  {
    try
    {
      openSynchronously();
    }
    catch (Exception e)
    {
      anyException_ = e;
    }
  }



  /**
   * Sets the attributes of the object that should be returned in the list.
   * This method can be used to speed up the listing if
   * only interested in a few attributes for each item in the list.
   *
   * @param attributes An array of attribute IDs that define which
   *   object attributes will be retrieved for each item in the list
   *   when the list is opened.
   *
   * @see PrintObject
   **/
  public void setAttributesToRetrieve(int[] attributes)
  {
    // check params
    if (attributes == null)
    {
      Trace.log(Trace.ERROR, "setAttributesToRetrieve: Parameter 'attributes' is null.");
      throw new NullPointerException("attributes");
    }

    if (attrsToRetrieve_ != null)
    {
      attrsToRetrieve_.reset();
    }
    else
    {
      attrsToRetrieve_ = new NPCPAttributeIDList();
    }

    for (int i = 0; i




© 2015 - 2025 Weber Informatics LLC | Privacy Policy