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

com.ibm.as400.micro.JdbcMeService Maven / Gradle / Ivy

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

package com.ibm.as400.micro;

import com.ibm.as400.access.Trace;
import java.io.*;
import java.sql.*;
import java.util.*;

class JdbcMeService implements Service
{
  private static final String copyright = "Copyright (C) 1997-2001 International Business Machines Corporation and others.";

    private Vector connections_;
    private Vector statements_;
    private Hashtable statementresults_; // DOMINO
    private Hashtable map_;
    private Properties properties_;

    private MicroDataInputStream in_;
    private MicroDataOutputStream out_;
    private int dataFlowType_;

    // Note: for a multi-threaded client, we would have to synchronize
    //       access to this.  Of course, we would have to do a lot more
    //       than that. :)
    private int NextObjectId_;


    private ConnectionHandler connectionHandler_;
    private StatementHandler statementHandler_;
    private ResultSetHandler resultSetHandler_;


    /**
    Constructor so we do not have to pass the streams around in all
    the helper functions.
    **/
    public JdbcMeService()
    {
        // Create the data structures to hold our JDBC objects.
        connections_ = new Vector();
        statements_  = new Vector();
        map_         = new Hashtable();
        statementresults_ = new Hashtable(); // DOMINO

        NextObjectId_ = 1;
        dataFlowType_ = MEConstants.DATA_FLOW_LIMITED;

        registerDrivers();
    }


    /**
    **/
    public void setDataStreams(MicroDataInputStream in, MicroDataOutputStream out)
    {
        in_ = in;
        out_ = out;

        // Create the JDBC internal handlers.
        connectionHandler_ = new ConnectionHandler(this, in_, out_);
        statementHandler_ = new StatementHandler(this, in_, out_);
        resultSetHandler_ = new ResultSetHandler(this, in_, out_);
    }


    /**
    Tells the caller whether or not this service request
    handler can handle the specified request being input.
    **/
    public boolean acceptsRequest(int functionId)
    {

        if ((functionId > 999) && (functionId < 2000))
            return true;

        return false;
    }


    /**
    Routes the specified function request to the correct
    internal handler.
    **/
    public void handleRequest(int functionId) throws IOException 
    {
        // Output the function id for debugging.
        if (Trace.isTraceOn())
            Trace.log(Trace.PROXY, "Function id is " + Integer.toHexString(functionId));

        // First check for and handle any service requests.
        // Just like the various JDBC object handlers have a process
        // method for dealing with requests, so do service handlers.
        if (isServiceRequest(functionId))
        {
            process(functionId);
            return;
        }


        // If a new connection is needed, do it inline here.  As there
        // is not Driver object, there is no JDBC object to follow in
        // the data stream.
        if (functionId == MEConstants.CONN_NEW)
        {
            if (Trace.isTraceOn())
                Trace.log(Trace.PROXY, "JDBC Service: The request is here. " + Integer.toHexString(functionId));

            boolean trace = Trace.isTraceOn();

            if (trace)
                Trace.setTraceOn(false);

            // Input Parm - the url to use to get the connection.
           String url = in_.readUTF();

           if (trace)
               Trace.setTraceOn(true);
            
            // Create the connection
            try
            {
                Connection c = DriverManager.getConnection(url);

                // Add it to our vector.
                connections_.addElement(c);

                // Put the object into our map.
                int objectId = getNextObjectId();
                map_.put(Integer.valueOf(objectId), Integer.valueOf(c.hashCode()));
                out_.writeInt(objectId);
                out_.flush();
            }
            catch (SQLException e)
            {
                handleException(e);
            }
            catch (Exception e)
            {
                if (Trace.isTraceOn())
                    Trace.log(Trace.ERROR, e);
            }

            return;
        }

        // The first value on all JDBC requests that are not for a new
        // connections or service requests is an object id.
        int handle = in_.readInt();

        // Get the object that is represented by the JDBC handle.
        Object object = processJdbcObject(handle);

        // Do the work - I don't really like this layout... perhaps
        // I will change it as Fred suggested to do things more OO.
        if (object instanceof Connection)
        {
            connectionHandler_.process((Connection) object, functionId);
        }
        else if (object instanceof Statement)
        {
            statementHandler_.process((Statement) object, functionId);
        }
        else if (object instanceof ResultSet)
        {
            resultSetHandler_.process((ResultSet) object, functionId);
        }
        // DOMINO START
        else
        {
            if (Trace.isTraceOn())
                Trace.log(Trace.PROXY, "Error, unknown Jdbc object < " + object + ">");
            try
            {
                throw new SQLException("Error, unknown Jdbc object < " + object + ">");
            }
            catch (SQLException e)
            {
                handleException(e);
            }
        }
        // DOMINO END
    }


    /**
    Figure out what the handle type is so that we can figure
    out what the function indentifier means.
 
    TODO:  We will probably change this so that that handle
    is really the object hash code instead of the CLI handle.
    Otherwise we can't be JDBC driver neutral.
     * @param handle 
     * @return Object
     * @throws IOException 
    **/
    public Object  processJdbcObject(int handle) throws IOException 
    {
        // Get the object hashcode based off of the user input handle.
        Integer oObjectHash = (Integer) map_.get(Integer.valueOf(handle));
        int iObjectHash = oObjectHash.intValue();

        Integer mapValue = null;

        try
        {
// DOMINO ADD
            // Loop through all our result sets
            // The Hostserver requires JDK 1.2, so using this
            // method should be fine.
            Collection col = statementresults_.values();
            Iterator   it  = col.iterator();
            while (it.hasNext())
            {
                ResultSet rs = (ResultSet) it.next();
                int   rsHandle = rs.hashCode();
                if (iObjectHash == rsHandle)
                    return rs;
            }
// DOMINO END ADD

// DOMINO REORDER
            // Loop through all our statements
            for (int i = 0; i < statements_.size(); i++)
            {
                Statement s = (Statement) statements_.elementAt(i);
                int stmtHandle = s.hashCode();
                if (iObjectHash == stmtHandle)
                    return s;
            }

            // Loop through all our connections...
            for (int i = 0; i < connections_.size(); i++)
            {
                // Get the next connection object.
                Connection c = (Connection) connections_.elementAt(i);
                // Get the connection's hashcode
                int connHandle = c.hashCode();
                // Compare it.
                if (iObjectHash == connHandle)
                    return c;
            }
        }
        catch (Exception e)
        {
            if (Trace.isTraceOn())
                Trace.log(Trace.ERROR, e);
        }

        if (Trace.isTraceOn())
            Trace.log(Trace.ERROR, "ERROR! returning null from processJdbcObject!");
        
        return null;
    }


    /**
    This function can handle adding statements, preparedStatements,
    and callableStatements to the internal data structures.
     * @param s 
    **/
    public void addStatement(Statement s)
    {
        statements_.addElement(s);
        // don't add a statementresults entry until the result
        // set is created.
    }

    /**
    This function adds connections to the internal data structures.
     * @param c 
    **/
    public void addConnection(Connection c)
    {
        connections_.addElement(c);
    }

    /**
     * This function adds result sets to the internal data structure
     * @param s 
     * @param rs 
     * @throws SQLException 
     */
    public void addResultSet(Statement s, ResultSet rs) throws SQLException 
    {     // DOMINO
        // The DOMINO Jdbc driver doesn't implement
        // the getStatement() method. We can't use it
        // here. 'lotus.jdbc.domino.DominoDriver'
        //// Statement s = rs.getStatement();

        // statementresults key is statement, value is the resultset
        statementresults_.remove(s);   // 'close' existing result set
        statementresults_.put(s, rs);  // Add the new one.
    }

    /**
    This function can handle removing statements, preparedStatements,
    and callableStatements to the internal data structures.
     * @param s 
    **/
    public void removeStatement(Statement s)
    {
        statements_.remove(s);
        // Also remove the result set associated with the statement
        // if there is one.
        statementresults_.remove(s);
    }

    public void removeResultSet(ResultSet rs) throws SQLException 
    {   // DOMINO
        // The DOMINO Jdbc driver doesn't implement
        // the getStatement() method. We can't use it
        // here. 'lotus.jdbc.domino.DominoDriver'
        //// Statement s = rs.getStatement();
        //// statementresults.remove(s);

        // Remove the statement->resultset mapping by going
        // backwards its a bit slower.
        Collection col = statementresults_.values();
        // The Collection is a live view who's backing
        // is the Hashtable, removing this element, removes
        // it from the hash table too.
        col.remove(rs);
    }

    /**
    This function removes connections to the internal data structures.
     * @param c 
    **/
    public void removeConnection(Connection c)
    {
        // Before removing the connection from our data storage, we
        // want to remove the statements from storage for that connection.
        Connection toTry = null;
        for (int i = 0; i < statements_.size(); i++)
        {
            Statement s = (Statement) statements_.elementAt(i);
            try
            {
                toTry = s.getConnection();
            }
            catch (SQLException e)
            {
                if (Trace.isTraceOn())
                    Trace.log(Trace.ERROR, "Exception thrown trying to get the connection for a statement.", e);
            }

            if (c == toTry)
            {
                if (Trace.isTraceOn())
                    Trace.log(Trace.PROXY, "Implicitly closing a statement because of connection close");

                removeStatement(s);
            }
        }
        connections_.remove(c);
    }


    /**
    This is the generic exception handler for all SQLExceptions
    that can happen.
     * @param e 
     * @throws IOException 
    **/
    public void handleException(SQLException e) throws IOException 
    {
        if (Trace.isTraceOn())
            Trace.log(Trace.ERROR, e);

        out_.writeInt(-1);
        
        String s = e.getSQLState();
        
        if (s == null)
            s = "null";

        out_.writeUTF(s);
        
        s = e.getMessage();
        
        if (s == null)
            s = "null";

        out_.writeUTF(s);
        out_.flush();
    }


    /**
    This function returns the next usable object id..
     * @return next usable object id
    **/
    public int getNextObjectId()
    {
        // TODO:  at some point, it should be changed to reuse opened
        // object ids from holes that get created in its map.
        int returnValue = NextObjectId_;
        NextObjectId_++;
        
        return returnValue;
    }


    /**
    This method encapsulates the map out of the service handlers.
     * @param object 
     * @return int
    **/
    public int mapObject(Object object)
    {
        int objectId = getNextObjectId();
        map_.put(Integer.valueOf(objectId), Integer.valueOf(object.hashCode()));
        
        return objectId;
    }


    /**
    Get a list of JDBC drivers that the server should attempt to make available.
    **/
    public void registerDrivers()
    {
        String driver = "com.ibm.as400.access.AS400JDBCDriver";
        
        if (Trace.isTraceOn())
            Trace.log(Trace.PROXY, "Loading driver: " + driver);
        try
        {
            Class.forName(driver);
        }
        catch (ClassNotFoundException e)
        {
            if (Trace.isTraceOn())
                Trace.log(Trace.PROXY, "Failed to load driver " + driver);
        }
    }



    /**
     * @param funcId 
     * @return true if funcId is a service request
    **/
    public boolean isServiceRequest(int funcId)
    {
        // Check for function id in the JDBCME service request range.
        if ((funcId > 1900) && (funcId < 1999))
            return true;

        return false;
    }



    /**
    Process service requests
     * @param funcId 
     * @throws IOException 
    **/
    public void process(int funcId) throws IOException
    {
        switch (funcId)
        {
        case MEConstants.JDBCME_DATA_TYPE_FLOW:
            setDataFlowType();
            break;
        default:
            // TODO:  This is an exception condition...
            System.out.println("Error - JDBC-ME Service request unrecognized - function code: " + funcId);
            break;
        }
    }


    /**
    Allows JDBCME service handlers to determine the type of
    data type flows the user application is interested in.
     * @return data file type
    **/
    public int getDataFlowType()
    {
        return dataFlowType_;
    }


    /**
     * @throws IOException 
    **/
    public void setDataFlowType() throws IOException
    {
        // TODO:  Verification should be done that this is
        //        a value value passed in and stuff like
        //        that.  Also, the function should be broken
        //        out so that appropriate JavaDoc can be
        //        created for it.  
        int type = in_.readInt();
        if ((type == MEConstants.DATA_FLOW_ALL) ||
            (type == MEConstants.DATA_FLOW_LIMITED) ||
            (type == MEConstants.DATA_FLOW_STRINGS_ONLY))
        {
            dataFlowType_ = type;
            out_.writeInt(1);
            out_.flush();
        }
        else
        {
            handleServiceException("An invalid setting was passed for setting the data flow type: " + type);
        }
    }


    /**
    Handles exceptions that happen at a service level.
     * @param message 
     * @throws IOException 
    **/
    public void handleServiceException(String message) throws IOException 
    {
        out_.writeInt(-1);
        out_.writeUTF("JDBC");
        out_.writeUTF(message);
        out_.flush();
    }

}







© 2015 - 2025 Weber Informatics LLC | Privacy Policy