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

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

There is a newer version: 11.1
Show newest version
///////////////////////////////////////////////////////////////////////////////
//                                                                             
// JTOpen (IBM Toolbox for Java - OSS version)                              
//                                                                             
// Filename: PSController.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.net.InetAddress;
import java.net.Socket;
import java.util.Vector;

// Represents a connection to a client of the proxy server.
class PSController extends StoppableThread
{
    // Private data.
    private static long nextConnectionId_ = 1000;
    private static Object nextConnectionIdLock_ = new Object();

    private boolean closed_ = false;
    //private PSConfig config_;
    private Socket connectedSocket_;
    private long connectionId_ = -1;
    private PxDSFactory factory_;
    private InputStream input_;
    private boolean ownSocket_ = false;
    private OutputStream output_;
    private boolean running_ = false;
    private PSServerSocketContainerAdapter serverSocket_;
    //private Vector threadGroup_;

    // Constructs a PSController object.
    // @param  threadGroup  The thread group.                               
    // @param  proxyServer  The proxy server.                                  
    // @param  load  The load.
    // @param  loadBalancer  The load balancer.
    // @param  config  The configuration.
    // @param  serverSocket  The server socket container.
    public PSController(Vector threadGroup, ProxyServer proxyServer, PSLoad load, PSLoadBalancer loadBalancer, PSConfig config, PSServerSocketContainerAdapter serverSocket)
    {
        super("PSController-" + serverSocket);

        //threadGroup_ = threadGroup;
        //config_ = config;
        serverSocket_ = serverSocket;

        factory_ = new PxDSFactory();
        factory_.register(new PxBooleanParm());
        factory_.register(new PxIntParm());
        factory_.register(new PxStringParm());
        factory_.register(new PxSerializedObjectParm(null));
        factory_.register(new PxConnectReqSV(threadGroup, this, load, loadBalancer));
        factory_.register(new PxConfigReqSV(config, this));
        factory_.register(new PxEndReqSV(proxyServer, this));
        factory_.register(new PxLoadReqSV(load));

        if (Trace.isTraceOn()) Trace.log(Trace.PROXY, "Px server controller " + this + " opened.");
    }

    public void closeServerSocket()
    {
        if (Trace.isTraceOn()) Trace.log(Trace.PROXY, "Px server controller " + this + " closed.");

        closeSocket();

        try
        {
            serverSocket_.close();
        }
        catch (IOException e)
        {
            Trace.log(Trace.ERROR, e.getMessage(), e);
        }

        closed_ = true;
    }

    public void closeSocket()
    {
        if (ownSocket_)
        {
            try
            {
                if (connectedSocket_ != null) connectedSocket_.close();
            }
            catch (IOException e)
            {
                Trace.log(Trace.ERROR, "Exception closing proxy socket:", e);
            }

            ownSocket_ = false;
        }
    }

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

    // Returns the current requesting client's address.
    // @return  The current requesting client's address.
    public InetAddress getClientAddress()
    {
        return connectedSocket_.getInetAddress();
    }

    // Returns the current requesting client's socket.
    // @return  The current requesting client's socket.
    public Socket getConnectedSocket()
    {
        // Give up ownership of the socket.  A connection owns it now.
        ownSocket_ = false;
        return connectedSocket_;
    }

    // Returns the current requesting client's unique connection id.
    // @return  The current requesting client's unique connection id.
    public long getConnectionId()
    {
        return connectionId_;
    }

    // Returns the input stream used for receiving requests from the current client.
    // @return  The input stream used for receiving requests from the current client.
    public InputStream getInputStream()
    {
        return input_;
    }

    // Returns the output stream used for sending replies to the current client.
    // @return  The output stream used for sending replies to the current client.
    public OutputStream getOutputStream()
    {
        return output_;
    }

    // Runs the controller.
    public void run()
    {
        running_ = true;

        // Loop forever, handling each connection that comes in.
        while (canContinue())
        {
            // If anything goes wrong here, stop the controller.
            try
            {
                connectedSocket_ = serverSocket_.accept();
                // Test note: We see the phrase "Address in use: bind" on occasion when calling this method.  My best guess is that the JDK is printing it!
            }
            catch(Exception e)
            {
                Verbose.println(e);
                Trace.log(Trace.ERROR, "Exception accepting proxy socket:", e);
                break;
            }

            // From here on out, if anything goes wrong, we just loop and try again!
            try
            {
                input_ = new BufferedInputStream(connectedSocket_.getInputStream());
                output_ = new BufferedOutputStream(connectedSocket_.getOutputStream());

                // For now, this class "owns" the socket... at least until a connection gets it for its own use.
                ownSocket_ = true;

                synchronized (nextConnectionIdLock_)
                {
                    connectionId_ = ++nextConnectionId_;
                }

                // Get the next request.
                if (Trace.isTraceProxyOn())
                  Trace.log(Trace.PROXY,this,"calling factory_.getNextDS"); 
                PxReqSV request = (PxReqSV)factory_.getNextDS(input_);
                if (Trace.isTraceProxyOn()) request.dump(Trace.getPrintWriter());

                // Process the request and return the reply, if any.
                PxRepSV reply = (PxRepSV)request.process();
                if (reply != null)
                {
                    reply.setCorrelationId(request.getCorrelationId());
                    synchronized (output_)
                    {
                        if (Trace.isTraceProxyOn()) reply.dump(Trace.getPrintWriter());
                        reply.writeTo(output_);
                        output_.flush();
                    }
                }
            }
            catch (Exception e)
            {
                Verbose.println(e);
                Trace.log(Trace.ERROR, "Exception processing proxy request:", e);
            }
            finally
            {
                closeSocket();
            }
        }

        running_ = false;
    }

    // Stops the thread safely.
    public void stopSafely()
    {
        super.stopSafely();

        // Close the sockets, etc.
        closeServerSocket();

        // Wait for controller loop to finish.  This verifies that the socket was finally closed.  (On some platforms (e.g. Windows) it seems like the close does not take full effect until a few seconds after close() is called.
        try
        {
            while (running_) Thread.sleep(500);
        }
        catch(InterruptedException e)
        {
            // Ignore.
        }

        Verbose.println(ResourceBundleLoader.getText("PROXY_SERVER_ENDED", serverSocket_));
    }

    public String toString()
    {
        return serverSocket_.toString();
    }
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy