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

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

///////////////////////////////////////////////////////////////////////////////
//                                                                             
// JTOpen (IBM Toolbox for Java - OSS version)                              
//                                                                             
// Filename: PxClientConnectionAdapter.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.BufferedInputStream;
import java.io.BufferedOutputStream;
import java.io.InputStream;
import java.io.IOException;
import java.io.OutputStream;
import java.lang.reflect.Constructor;
import java.lang.reflect.InvocationTargetException;
import java.net.*;

//
// Tunneling is another mechanism to get to the server.  This class now
// has two ways to get to the server:
//    Traditional -- a socket connection is made to the server and a
//    deamon thread is started to handly data coming back on the connection.
//    Tunnel -- A URL connection is made for each data flow.  The connection
//    is made to our tunnel servelet running in the http server.
//


/**
The PxClientConnectionAdapter class represents the connection
to a proxy server.
**/
public abstract class PxClientConnectionAdapter
{
  private static final String copyright = "Copyrixght (C) 1997-2000 International Business Machines Corporation and others.";




    // Private data.
    private boolean                     closed_             = false;
    private InputStream                 input_              = null;
    private OutputStream                output_             = null;
    private PxClientReadDaemon          readDaemon_         = null;
    private PxSocketContainerAdapter    socket_             = null;
    private SSLOptions                  sslOptions_         = null;

                                                                    // tunnel_ is used by the
                                                                    // subclass ProxyClientConnection
                                                                    // so it cannot be private
            boolean                     tunnel_             = false;           // @D1a
    private long                        clientId_           = -1;              // @D1a @D2C
    private URL                         tunnelURL_          = null;            // @D1a
    private String localName = null; 

    protected PxClientConnectionAdapter (String proxyServer, SSLOptions secure)
    {
        // @D2D clientId_ = new byte[8];
        sslOptions_ = secure;
        open (proxyServer);
    }


/**
Closes the connection to the proxy server.
      // @D1a question -- should we do something in the tunneling case
      //      to clean up the server?
**/
    public void close ()
    {
        if (!tunnel_)                                                          // @D1a
        {
           if (Trace.isTraceProxyOn ())
               Trace.log (Trace.PROXY, "Closing a connection to proxy server.");

           readDaemon_.stopSafely();

           // I am using separate try-catch blocks to make sure
           // that everything gets closed.
           try {
               input_.close ();
           }
           catch (IOException e) {
               if (Trace.isTraceErrorOn ())
                   Trace.log (Trace.ERROR, e.getMessage (), e);
               throw new ProxyException (ProxyException.CONNECTION_DROPPED);
           }

           try {
             output_.close ();
           }
           catch (IOException e) {
               if (Trace.isTraceErrorOn ())
                   Trace.log (Trace.ERROR, e.getMessage (), e);
               throw new ProxyException (ProxyException.CONNECTION_DROPPED);
           }

           // $$ Question for Jim, his new class skipped the above
           //    two funtions -- input_.close() and output_.close().
           //    If a good idea then remove from here as well.
           try {
               socket_.close ();
           }
           catch (IOException e) {
               if (Trace.isTraceErrorOn ())
                   Trace.log (Trace.ERROR, e.getMessage (), e);
               throw new ProxyException (ProxyException.CONNECTION_DROPPED);
           }
        }

        closed_ = true;
    }



    protected void finalize ()
        throws Throwable
    {
        if (closed_ == false)
            close ();
        super.finalize ();
    }



    public PxDSFactory getFactory()
    {
        return readDaemon_.getFactory();
    }


    public void open (String proxyServer)
    {
        boolean secure = (sslOptions_ != null && sslOptions_.proxyEncryptionMode_ != SecureAS400.PROXY_SERVER_TO_SERVER);

        if (Trace.isTraceOn()) Trace.log(Trace.PROXY, "Opening a connection to proxy server "
                                                     + proxyServer
                                                     + " (secure=" + secure + ").");


        // Parse the proxy server name, port number (and protocol if tunneling)
        localName    = proxyServer;
        String protocolName = null;

        int port = -1;

        // determine if we are going with traditional or tunnel proxy.
        // Assume any string with a :// wants to use the tunnel.  This would
        // be http://, https://, etc.
        if (proxyServer.indexOf("://") > 0)                                    // @D1a
        {
           tunnel_ = true;
           // the name of the server is everything beyond the ://
           localName    = proxyServer.substring(proxyServer.indexOf(":") + 3);
           protocolName = proxyServer.substring(0, proxyServer.indexOf(":"));
        }

        // now strip the port of the end of the server name (if one exists)
        int colon = localName.indexOf(':');
        if (colon >= 0)
        {
            if (colon < localName.length() - 1)
                port = Integer.parseInt(localName.substring (colon + 1));
            localName = localName.substring(0, colon);
        }


        if (! tunnel_)                                                         // @D1a
        {
            if (port < 0)
               port =  secure ? ProxyConstants.SECURE_PORT_NUMBER : ProxyConstants.PORT_NUMBER;       //$B1C
            openTraditional(localName, port, secure);
        }
        else
        {
           // when openTunnel comes back move creating tunnelURL_ to the try/catch.
           openTunnel(protocolName, localName, port);                                                         // @D1a
        }
    }

    // this method used to be part of open.  It was split out when
    // tunneling was added.
    void openTraditional(String name, int port, boolean secure)                                // @D1a
    {
        // Open the socket and streams.
        try {
            if (secure) {
            	// Call view reflection to remove dependency on sslight.zip
            	Class classPxSecureSocketContainer = Class.forName("com.ibm.as400.access.PxSecureSocketContainer");
            	Class[] parameterTypes = new Class[3]; 
            	parameterTypes[0] = "".getClass(); 
            	parameterTypes[1] = Integer.TYPE; 
            	parameterTypes[2] = Class.forName("com.ibm.as400.access.SSLOptions"); 
            	Constructor constructor = classPxSecureSocketContainer.getConstructor(parameterTypes);
            	Object[] initargs = new Object[3]; 
            	initargs[0] = name; 
            	initargs[1] = new Integer(port); 
            	initargs[2] = sslOptions_; 
                socket_ = (PxSocketContainerAdapter) constructor.newInstance(initargs);
            } else
                socket_ = new PxSocketContainer (name, port);
            output_     = new BufferedOutputStream (socket_.getOutputStream());
            input_      = new BufferedInputStream(new RetryInputStream(socket_.getInputStream())); // @A2C

            readDaemon_ = new PxClientReadDaemon(input_);
            readDaemon_.start();

            if (Trace.isTraceProxyOn ())
                Trace.log (Trace.PROXY, "Connection established.");
        }
        catch (ClassNotFoundException e) { 
            if (Trace.isTraceErrorOn ())
                Trace.log (Trace.ERROR, "Error when opening connection to proxy server (ClassNotFound", e);
            throw new ProxyException (ProxyException.CONNECTION_NOT_ESTABLISHED);
        }
        catch (NoSuchMethodException e) { 
            if (Trace.isTraceErrorOn ())
                Trace.log (Trace.ERROR, "Error when opening connection to proxy server (NoSuchMethodException", e);
            throw new ProxyException (ProxyException.CONNECTION_NOT_ESTABLISHED);
        }
        catch (IllegalAccessException e) {
            if (Trace.isTraceErrorOn ())
                Trace.log (Trace.ERROR, "Error when opening connection to proxy server (IllegalAccessException", e);
            throw new ProxyException (ProxyException.CONNECTION_NOT_ESTABLISHED);
        }
        catch (IOException e) {
            if (Trace.isTraceErrorOn ())
                Trace.log (Trace.ERROR, "Error when opening connection to proxy server (openio", e);
            throw new ProxyException (ProxyException.CONNECTION_NOT_ESTABLISHED);
        } catch (IllegalArgumentException e) {
            if (Trace.isTraceErrorOn ())
                Trace.log (Trace.ERROR, "Error when opening connection to proxy server (IllegalArgumentException", e);
            throw new ProxyException (ProxyException.CONNECTION_NOT_ESTABLISHED);
		} catch (InstantiationException e) {
            if (Trace.isTraceErrorOn ())
                Trace.log (Trace.ERROR, "Error when opening connection to proxy server (InstantiationException", e);
            throw new ProxyException (ProxyException.CONNECTION_NOT_ESTABLISHED);
		} catch (InvocationTargetException e) {
            if (Trace.isTraceErrorOn ())
                Trace.log (Trace.ERROR, "Error when opening connection to proxy server (InvocationTargetException", e);
            throw new ProxyException (ProxyException.CONNECTION_NOT_ESTABLISHED);
		}
    }


    // @D1a New method
    void openTunnel(String protocol, String name, int port)
    {
       try
       {
          readDaemon_ = new PxClientReadDaemon();
          readDaemon_.register(new PxAcceptRepCV());

          if (port < 0)
             tunnelURL_ = new URL(protocol, name, "/servlet/com.ibm.as400.access.TunnelProxyServer");
          else
             tunnelURL_ = new URL(protocol, name, port, "/servlet/com.ibm.as400.access.TunnelProxyServer");
       }
       catch (IOException e)
       {
          if (Trace.isTraceErrorOn ())
              Trace.log (Trace.ERROR, "Error when opening connection to proxy server", e);
          throw new ProxyException (ProxyException.CONNECTION_NOT_ESTABLISHED);
       }
    }


    // @D1a New method
    private URLConnection tunnelSend(PxReqCV request)
    {
       try
       {
          URLConnection connection_;

          // @D2D if (clientId_ == null)
          // @D2D    request.setClientId(new byte[8]);
          // @D2D else
             request.setClientId(clientId_);

          // connection_ = (HttpURLConnection) tunnelURL_.openConnection();
          connection_ = tunnelURL_.openConnection();

          connection_.setUseCaches(false);
          connection_.setDoOutput(true);
          connection_.setDoInput(true);

          connection_.setRequestProperty("Content-type", "application/octet-stream");
          connection_.setRequestProperty("Connection",   "Keep-Alive");
          // connection_.setRequestMethod("POST");
          // connection_.setFollowRedirects(false);
          // connection.setRequestProperty("Content-length", " " + bytes.length)

          // connection_.connect();
          OutputStream connectionOut = connection_.getOutputStream();

          if (Trace.isTraceProxyOn())
              request.dump (Trace.getPrintWriter ());

          request.writeTo(connectionOut);
          connectionOut.flush();
          // connectionOut.close();

          return connection_;
       }
       catch (Exception e)
       {
          if (Trace.isTraceErrorOn ())
                Trace.log (Trace.ERROR, "Error when opening connection to proxy server", e);
             throw new ProxyException (ProxyException.CONNECTION_NOT_ESTABLISHED);
       }
    }



   Object tunnelReceive(long correlationId, URLConnection connection_)
        throws InvocationTargetException
   {
      try
      {
          InputStream connectionIn = connection_.getInputStream();

          PxRepCV reply;
          Object returnValue;
          try
          {
             reply = (PxRepCV) readDaemon_.getReply(correlationId, connectionIn);
             returnValue = reply.process ();
             clientId_ = reply.getClientId();
             return returnValue;
          }
          catch (IOException e) {
             if (Trace.isTraceErrorOn ())
                 Trace.log (Trace.ERROR, "Error when receiving reply from proxy server", e);
             throw new ProxyException (ProxyException.CONNECTION_DROPPED);
          }
      }
      catch (InvocationTargetException ite)
      {
         throw ite;
      }
      catch (Exception e)
      {
         if (Trace.isTraceErrorOn ())
                Trace.log (Trace.ERROR, "Error when opening connection to proxy server", e);
            throw new ProxyException (ProxyException.CONNECTION_NOT_ESTABLISHED);
      }
   }



    // Note: This method should NOT be synchronized.  If a thread is waiting to         // @A1A
    // receive a reply, it should not block other threads from sending requests.        // @A1A
    // (This is the point of the read daemon thread.)                                   // @A1A
    private Object receive (long correlationId)                                         // @A1C
        throws InvocationTargetException
    {
        Object returnValue = null;

        PxRepCV reply;
        try {
            reply = (PxRepCV) readDaemon_.getReply(correlationId);
            returnValue = reply.process ();
        }
        catch (IOException e) {
            if (Trace.isTraceErrorOn ())
                Trace.log (Trace.ERROR, "Error when receiving reply from proxy server", e);
            throw new ProxyException (ProxyException.CONNECTION_DROPPED);
        }

        return returnValue;
    }



/**
Sends a request to the proxy server.  No reply is expected.

@param request  The request.
**/
    protected synchronized void send (PxReqCV request)
    {

        if (! tunnel_)                                                     // @D1a
        {
           if (Trace.isTraceProxyOn())
              request.dump (Trace.getPrintWriter ());

           try {
              request.writeTo (output_);
              output_.flush ();
           }
           catch (IOException e) {
               if (Trace.isTraceErrorOn ())
                   Trace.log (Trace.ERROR, e.getMessage (), e);
               throw new ProxyException (ProxyException.CONNECTION_DROPPED);
           }
        }
        else
        {
           URLConnection connection_ = tunnelSend(request);                // @D1a
           // $$1 do something with the connection
        }
    }



/**
Sends a request to the proxy server and receives and processes a reply.

@param request  The request.
@return         The returned object, or null if none.
**/
    // Note: This method should NOT be synchronized.  If a thread is waiting to         // @A1A
    // receive a reply, it should not block other threads from sending requests.        // @A1A
    // (This is the point of the read daemon thread.)                                   // @A1A
    protected Object sendAndReceive (PxReqCV request)                                   // @A1C
        throws InvocationTargetException
    {
        if (! tunnel_)                                                     // @D1a
        {
           send (request);
           return receive (request.getCorrelationId());
        }
        else                                                               // @D1a
        {
           // cannot use HttpURLConnection in IE or Netscape so use URLConnection instead

           URLConnection connection_ = tunnelSend(request);                // @D1a
           Object o = tunnelReceive(request.getCorrelationId(), connection_);  // @D1a
           // return tunnelReceive(request.getCorrelationId(), connection_);  // @D1a
           return o;
        }                                                                  // @D1a
    }


    public String getSystemName() { 
      return localName; 
    }

}





© 2015 - 2025 Weber Informatics LLC | Privacy Policy