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

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

There is a newer version: 20.0.8
Show newest version
///////////////////////////////////////////////////////////////////////////////
//
// JTOpen (IBM  Toolbox for Java - OSS version)
//
// Filename:  UserSpaceImplRemote.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.access;

import java.beans.PropertyVetoException;
import java.io.FileNotFoundException;
import java.io.IOException;

// The UserSpaceImplRemote class is the remote implementation of the user space class.
class UserSpaceImplRemote implements UserSpaceImpl
{
    // The server where the user space is located.
    protected AS400ImplRemote system_ = null;
    // The full path name of the user space.
    protected String path_ = null;
    // The library that contains the user space.
    protected String library_ = null;
    // The name of the user space.
    protected String name_ = null;
    // Use ProgramCall instead of IFS.
    protected boolean mustUseProgramCall_ = false;
    // Use sockets instead of native methods when running natively.
    protected boolean mustUseSockets_ = false;
    // The string to byte data converter.
    protected ConverterImplRemote converter_;
    // Qualified user space name parameter, set on first touch.
    private ProgramParameter nameParameter_ = null;
    // Error code parameter for API's, set on first touch.
    private ProgramParameter errorCodeParameter_ = null;

    // Impl object for remote command server delete, getAttributes, setAttributes.
    protected RemoteCommandImpl remoteCommand_;
    // Whether to call remote commands on-thread.
    private Boolean runOnThread_ = RemoteCommandImpl.OFF_THREAD;

    // The integrated file system object used for read and write.
    private IFSRandomAccessFileImplRemote file_;

    // Throw or return an exception based on the message list from a remote program call.
    private AS400Exception buildException() throws AS400SecurityException, ObjectDoesNotExistException
    {
        // Get the message list.
        AS400Message[] messageList = remoteCommand_.getMessageList();
        // Get the message id of the first message.
        String id = messageList[0].getID();

        // Throw appropriate exceptions for not existing or not authorized.
        if (id.equals("CPF9801") || id.equals("CPF2105"))
        {
            Trace.log(Trace.ERROR, "Object does not exist: " + path_);
            throw new ObjectDoesNotExistException(path_, ObjectDoesNotExistException.OBJECT_DOES_NOT_EXIST);
        }
        if (id.equals("CPF9802") || id.equals("CPF2189"))
        {
            Trace.log(Trace.ERROR, "User is not authorized to object: " + path_);
            throw new AS400SecurityException(path_, AS400SecurityException.OBJECT_AUTHORITY_INSUFFICIENT);
        }
        if (id.equals("CPF9810") || id.equals("CPF2209") || id.equals("CPF2110"))
        {
            Trace.log(Trace.ERROR, "Library does not exist: " + path_);
            throw new ObjectDoesNotExistException(path_, ObjectDoesNotExistException.LIBRARY_DOES_NOT_EXIST);
        }
        if (id.equals("CPF9820") || id.equals("CPF2182"))
        {
            Trace.log(Trace.ERROR, "User is not authorized to library: " + path_);
            throw new AS400SecurityException(path_, AS400SecurityException.LIBRARY_AUTHORITY_INSUFFICIENT);
        }
        if (id.equals("CPF2283"))
        {
            String authorizationListName = "/QSYS.LIB/" + converter_.byteArrayToString(messageList[0].getSubstitutionData()).trim() + ".AUTL";
            Trace.log(Trace.ERROR, "Object does not exist: " + authorizationListName);
            throw new ObjectDoesNotExistException(authorizationListName, ObjectDoesNotExistException.OBJECT_DOES_NOT_EXIST);
        }
        // Else return exception for messages.
        return new AS400Exception(messageList);
    }

    // Closes our file stream to the user space (if a stream has been created),
    // and releases any system resources associated with the stream.
    // This will not close the connection to the Host Server job held by the associated AS400 object.
    public void close() throws IOException
    {
        if (file_ != null)
        {
            // Close the random access file stream.
            file_.close();
            // Clear the file connection info.
            file_ = null;
        }
    }

    // Creates a user space.
    public void create(byte[] domainBytes, int length, boolean replace, String extendedAttribute, byte initialValue, String textDescription, String authority) throws AS400SecurityException, ErrorCompletingRequestException, InterruptedException, IOException, ObjectDoesNotExistException
    {
        // Close the file stream to the user space (if one already exists).
        close();

        // Setup qualified user space name parameter.
        setupNameParameter();
        // Setup error code parameter.
        setupErrorCodeParameter();
        // Setup program call parameters.
        ProgramParameter[] parameters = new ProgramParameter[]
        {
            // Qualified user space name, input, char(20).
            nameParameter_,
            // Extended attributes, input, char(10).
            new ProgramParameter(padByteArray(converter_.stringToByteArray(extendedAttribute), 10)),
            // Initial size, input, binary(4).
            new ProgramParameter(BinaryConverter.intToByteArray(length)),
            // Initial value, input, char(1).
            new ProgramParameter(new byte[] { initialValue }),
            // Public authority, input, char(10).
            new ProgramParameter(padByteArray(converter_.stringToByteArray(authority), 10)),
            // Text description, input, char(50).
            new ProgramParameter(padByteArray(converter_.stringToByteArray(textDescription), 50)),
            // Replace, input, char(10), EBCDIC "*YES" or "*NO".
            new ProgramParameter((replace) ? new byte[] { 0x5C, (byte)0xE8, (byte)0xC5, (byte)0xE2, 0x40, 0x40, 0x40, 0x40, 0x40, 0x40 } : new byte[] { 0x5C, (byte)0xD5, (byte)0xD6, 0x40, 0x40, 0x40, 0x40, 0x40, 0x40, 0x40 } ),
            // Error code, input/output, char(*).
            errorCodeParameter_,
            // Domain, input, char(10).
            new ProgramParameter(domainBytes),
            // Transfer size request, input, binary(4).
            new ProgramParameter(new byte[] { 0x00, 0x00, 0x00, 0x00 } ),
            // Optimum space alignment, input, char(1).
            new ProgramParameter(new byte[] { length > 16773120 ? (byte)0xF0 : (byte)0xF1 } )
        };

        // Setup for remote program call.
        if (remoteCommand_ == null) {
          setupRemoteCommand();
        }
        // Run create user space (QUSCRTUS) API.  This is a threadsafe API.
        if (!remoteCommand_.runProgram("QSYS", "QUSCRTUS", parameters, runOnThread_))
        {
            // Throw the returned messages.
            throw buildException();
        }
    }

    // Return a byte array padded to the correct length.
    private static byte[] padByteArray(byte[] bytes, int length)
    {
        byte[] result = new byte[length];
        System.arraycopy(bytes, 0, result, 0, bytes.length);
        for (int i = bytes.length; i < length; ++i) result[i] = 0x40;
        return result;
    }

    // Deletes a user space.
    public void delete() throws AS400SecurityException, ErrorCompletingRequestException, InterruptedException, IOException, ObjectDoesNotExistException
    {
        // Close the file stream to the user space (if one exists), to avoid locking problems.
        close();

        // Setup qualified user space name parameter.
        setupNameParameter();
        // Setup error code parameter.
        setupErrorCodeParameter();
        // Setup program call parameters.
        ProgramParameter[] parameters = new ProgramParameter[]
        {
            // Qualified user space name, input, char(20).
            nameParameter_,
            // Error code, input/output, char(*).
            errorCodeParameter_
        };

        // Setup for remote program call.
        if (remoteCommand_ == null) {
          setupRemoteCommand();
        }
        // Run delete user space (QUSDLTUS) API.  This is a threadsafe API.
        if (!remoteCommand_.runProgram("QSYS", "QUSDLTUS", parameters, runOnThread_))
        {
            // Throw the returned messages.
            throw buildException();
        }
    }

    // Returns the initial value used for filling in the user space during creation and extension.
    public byte getInitialValue() throws AS400SecurityException, ErrorCompletingRequestException, InterruptedException, IOException, ObjectDoesNotExistException
    {
        return retrieveAttributes()[13];
    }

    // Returns the size in bytes of the user space.
    public int getLength() throws AS400SecurityException, ErrorCompletingRequestException, InterruptedException, IOException, ObjectDoesNotExistException
    {
        return BinaryConverter.byteArrayToInt(retrieveAttributes(), 8);
    }

    // Indicates if the user space is auto extendible.
    public boolean isAutoExtendible() throws AS400SecurityException, ErrorCompletingRequestException, InterruptedException, IOException, ObjectDoesNotExistException
    {
        return retrieveAttributes()[12] == (byte)0xF1;
    }

    // Retrieve the user space attributes.
    protected byte[] retrieveAttributes() throws AS400SecurityException, ErrorCompletingRequestException, InterruptedException, IOException, ObjectDoesNotExistException
    {
        // Close the file stream to the user space (if one exists), to avoid locking problems.
        close();

        // Setup qualified user space name parameter.
        setupNameParameter();
        // Setup error code parameter.
        setupErrorCodeParameter();
        // Setup program call parameters.
        ProgramParameter[] parameters = new ProgramParameter[]
        {
            // Receiver variable, output, char(*), ask for 24 bytes.
            new ProgramParameter(24),
            // Length of receiver variable, input, binary(4), 24 bytes.
            new ProgramParameter(new byte[] { 0x00, 0x00, 0x00, 0x18 } ),
            // Format name, input, char(8), EBCDIC "SPCA0100".
            new ProgramParameter(new byte[] { (byte)0xE2, (byte)0xD7, (byte)0xC3, (byte)0xC1, (byte)0xF0, (byte)0xF1, (byte)0xF0, (byte)0xF0 } ),
            // Qualified user space name, input, char(20).
            nameParameter_,
            // Error code, input/output, char(*).
            errorCodeParameter_
        };

        // Setup for remote program call.
        if (remoteCommand_ == null) {
          setupRemoteCommand();
        }
        // Run retrieve user space attributes (QUSRUSAT) API.  This is a threadsafe API.
        if (!remoteCommand_.runProgram("QSYS", "QUSRUSAT", parameters, runOnThread_))
        {
            // Throw the returned messages.
            throw buildException();
        }
        // Return the data returned from the program.
        return parameters[0].getOutputData();
    }

    // Read from user space.
    public int read(byte[] dataBuffer, int userSpaceOffset, int dataOffset, int length) throws AS400SecurityException, ErrorCompletingRequestException, InterruptedException, IOException, ObjectDoesNotExistException
    {
        // Use remote program call implementation or file implementation.
        if (mustUseProgramCall_)
        {
            // Setup qualified user space name parameter.
            setupNameParameter();
            // Setup program call parameters.
            ProgramParameter[] parameters = new ProgramParameter[]
            {
                // Qualified user space name, input, char(20).
                nameParameter_,
                // Starting position, input, binary(4), add 1 for 1 based offset.
                new ProgramParameter(BinaryConverter.intToByteArray(userSpaceOffset + 1)),
                // Length of data, input, binary(4).
                new ProgramParameter(BinaryConverter.intToByteArray(length)),
                // Receiver variable, output, char(*).
                new ProgramParameter(length)
                // Omit error code optional parameter.
            };

            // Setup for remote program call.
            if (remoteCommand_ == null) {
              setupRemoteCommand();
            }
            // Run retrieve user space (QUSRTVUS) API.  This is a threadsafe API.
            if (!remoteCommand_.runProgram("QSYS", "QUSRTVUS", parameters, runOnThread_))
            {
                String id = remoteCommand_.getMessageList()[0].getID();
                if (!id.equals("CPF3C14") && !id.equals("CPD3C14"))
                {
                    // Throw the returned messages.
                    throw buildException();
                }
                int userSpaceLength = getLength();
                if (userSpaceLength < userSpaceOffset) return -1;
                length = userSpaceLength - userSpaceOffset;
                try
                {
                    parameters[2].setInputData(BinaryConverter.intToByteArray(length));
                    parameters[3].setOutputDataLength(length);
                }
                catch (PropertyVetoException e)
                {
                    Trace.log(Trace.ERROR, "Unexpected PropertyVetoException:", e);
                    throw new InternalErrorException(InternalErrorException.UNEXPECTED_EXCEPTION);
                }
                if (!remoteCommand_.runProgram("QSYS", "QUSRTVUS", parameters, runOnThread_))
                {
                    // Throw the returned messages.
                    throw buildException();
                }
            }
            // Copy output data into user's array.
            System.arraycopy(parameters[3].getOutputData(), 0, dataBuffer, dataOffset, length);
            return length;
        }
        else
        {
            // Setup for file.
            setupFile();

            // Seek to correct offset.
            file_.seek(userSpaceOffset);
            // Read from the user space, return number of bytes read.
            return file_.read(dataBuffer, dataOffset, length, false);
        }
    }

    // Sets the auto extend attribute.
    public void setAutoExtendible(boolean autoExtendibility) throws AS400SecurityException, ErrorCompletingRequestException, InterruptedException, IOException, ObjectDoesNotExistException
    {
        // Number of attributes is 1, key is 3, length of attribute is 1 byte, value is EBCDIC "1" or "0".
        changeAttributes(new byte[] { 0x00, 0x00, 0x00, 0x01, 0x00, 0x00, 0x00, 0x03, 0x00, 0x00, 0x00, 0x01, autoExtendibility ? (byte)0xF1 : (byte)0xF0 });
    }

    // Sets the initial value to be used during user space creation or extension.
    public void setInitialValue(byte initialValue) throws AS400SecurityException, ErrorCompletingRequestException, InterruptedException, IOException, ObjectDoesNotExistException
    {
        // Number of attributes is 1, key is 2, length of attribute is 1 byte, value is as specified.
        changeAttributes(new byte[] { 0x00, 0x00, 0x00, 0x01, 0x00, 0x00, 0x00, 0x02, 0x00, 0x00, 0x00, 0x01, initialValue });
    }

    // Sets the size of the user space.  Valid values are 1 through 1,6776,704.
    public void setLength(int length) throws AS400SecurityException, ErrorCompletingRequestException, InterruptedException, IOException, ObjectDoesNotExistException
    {
        // Number of attributes is 1, key is 1, length of attribute is 4 bytes, use 4 bytes of  0 to hold space for value.
        byte[] attributeBytes = new byte[] { 0x00, 0x00, 0x00, 0x01, 0x00, 0x00, 0x00, 0x01, 0x00, 0x00, 0x00, 0x04, 0x00, 0x00, 0x00, 0x00 };
        // Set the length into the 12th position of the byte array.
        BinaryConverter.intToByteArray(length, attributeBytes, 12);
        changeAttributes(attributeBytes);
    }

    // Change the user space attributes.
    protected void changeAttributes(byte[] attributeBytes) throws AS400SecurityException, ErrorCompletingRequestException, InterruptedException, IOException, ObjectDoesNotExistException
    {
        // Close the file stream to the user space (if one exists), to avoid locking problems.
        close();

        // Setup qualified user space name parameter.
        setupNameParameter();
        // Setup error code parameter.
        setupErrorCodeParameter();
        // Setup program call parameters.
        ProgramParameter[] parameters = new ProgramParameter[]
        {
            // Returned library name, output, char(10).
            new ProgramParameter(10),
            // Qualified user space name, input, char(20).
            nameParameter_,
            // Attribtes to change, input, char(*).
            new ProgramParameter(attributeBytes),
            // ErrorCode, input/output, char(*).
            errorCodeParameter_
        };

        // Setup for remote program call.
        if (remoteCommand_ == null) {
          setupRemoteCommand();
        }
        // Run change user space attributes (QUSCUSAT) API.  This is a threadsafe API.
        if (!remoteCommand_.runProgram("QSYS", "QUSCUSAT", parameters, runOnThread_))
        {
            // Throw the returned messages.
            throw buildException();
        }
    }

    // Set needed implementation properties.
    public void setProperties(AS400Impl system, String path, String name, String library, boolean mustUseProgramCall, boolean mustUseSockets)
    {
        system_ = (AS400ImplRemote)system;
        path_ = path;
        library_ = library;
        name_ = name;
        mustUseProgramCall_ = mustUseProgramCall;
        mustUseSockets_ = mustUseSockets;
    }

    // Setup error code program parameter object on first touch.  Synchronized to protect instance variables.  This method can safely be called multiple times because it checks for a previous call before changing the instance variables.
    private void setupErrorCodeParameter()
    {
        // If not already setup.
        if (errorCodeParameter_ == null)
        {
            // Set error code parameter to eight bytes of zero's.  This causes the messages to be issued to the API caller.
            errorCodeParameter_ = new ProgramParameter(new byte[8]);
        }
    }

    // Setup file object on first touch.  Synchronized to protect instance variables.  This method can safely be called multiple times because it checks for a previous call before changing the instance variables.
    private synchronized void setupFile() throws AS400SecurityException, IOException, ObjectDoesNotExistException
    {
        // If not already setup.
        if (file_ == null)
        {
            try
            {
                // Create the file descriptor implementation object.
                IFSFileDescriptorImplRemote fd_ = new IFSFileDescriptorImplRemote();
                // Set the necessary properties into it.
                fd_.initialize(0, this, path_, IFSRandomAccessFile.SHARE_ALL, system_);

                // Create the file implementation object.
                file_ = new IFSRandomAccessFileImplRemote();
                // Set the necessary properties into it.
                file_.setFD(fd_);
                file_.setMode("rw");
                file_.setExistenceOption(IFSRandomAccessFile.OPEN_OR_FAIL);
                // Open the file.
                file_.open();
            }
            catch (FileNotFoundException e)
            {
                Trace.log(Trace.ERROR, "Object does not exist: " + path_, e);
                throw new ObjectDoesNotExistException(path_, ObjectDoesNotExistException.OBJECT_DOES_NOT_EXIST);
            }
            catch (ExtendedIOException e)
            {
                if (e.getReturnCode() == ExtendedIOException.ACCESS_DENIED)
                {
                    Trace.log(Trace.ERROR, "User is not authorized to object: " + path_, e);
                    throw new AS400SecurityException(path_, AS400SecurityException.OBJECT_AUTHORITY_INSUFFICIENT);
                }
                if (e.getReturnCode() == ExtendedIOException.REQUEST_NOT_SUPPORTED && library_.equals("QTEMP"))
                {
                    // File server cannot access QTEMP library.
                    Trace.log(Trace.WARNING, "File server cannot access QTEMP, use mustUseProgramCall option.");
                }
                Trace.log(Trace.ERROR, "Error opening file: " + path_, e);
                throw e;
            }
        }
    }

    // Setup qualified user space name program parameter object on first touch.  Synchronized to protect instance variables.  This method can safely be called multiple times because it checks for a previous call before changing the instance variables.
    private synchronized void setupNameParameter() throws IOException
    {
        // If not already setup.
        if (nameParameter_ == null)
        {
            // Get a converter for the system objects CCSID.
            converter_ = ConverterImplRemote.getConverter((system_).getCcsid(), system_);
            // The name and library of the user space used in program call.  Start with 20 EBCDIC spaces (" ").
            byte[] qualifiedUserSpaceName = new byte[] { 0x40, 0x40, 0x40, 0x40, 0x40, 0x40, 0x40, 0x40, 0x40, 0x40, 0x40, 0x40, 0x40, 0x40, 0x40, 0x40, 0x40, 0x40, 0x40, 0x40 };
            // Put the converted object name at the beginning of the array.
            converter_.stringToByteArray(name_, qualifiedUserSpaceName, 0);
            // Put the converted library name at position ten.
            converter_.stringToByteArray(library_, qualifiedUserSpaceName, 10);
            // Create the program parameter object.
            nameParameter_ = new ProgramParameter(qualifiedUserSpaceName);
        }
    }

    // Setup remote command object on first touch.  Synchronized to protect instance variables.  This method can safely be called multiple times because it checks for a previous call before changing the instance variables.
    protected synchronized void setupRemoteCommand() throws IOException
    {
      // If not already setup.
      if (remoteCommand_ == null)
      {
        boolean runningNatively = false;
        if (system_.canUseNativeOptimizations())
        {
          try
          {
            remoteCommand_ = (RemoteCommandImpl)Class.forName("com.ibm.as400.access.RemoteCommandImplNative").newInstance();
            // Avoid direct reference - it can cause NoClassDefFoundError at class loading time on Sun JVM's.
            runningNatively = true;
          }
          catch (Throwable e) {
            // A ClassNotFoundException would be unexpected, since canUseNativeOptions() returned true.
            Trace.log(Trace.WARNING, "Unable to instantiate class RemoteCommandImplNative.", e);
          }
        }
        if (remoteCommand_ == null)
        {
          remoteCommand_ = new RemoteCommandImplRemote();
        }
        remoteCommand_.setSystem(system_);

        // Note: All the API's that are called from this class, are threadsafe API's.
        // However, we need to stay consistent with the Toolbox's default threadsafety behavior.
        // So we'll indicate that the remote commands can safely be run on-thread (but only if the threadSafe property isn't set to 'false').  This will enable applications to use UserSpace when running on IBM i and using a profile that is disabled or has password *NONE.

        if (runningNatively && !mustUseSockets_)
        {
          // Abide by the setting of the thread-safety property (if it's set).
          String propVal = ProgramCall.getThreadSafetyProperty();
          if (propVal == null || !propVal.equals("false"))
          {
            runOnThread_ = RemoteCommandImpl.ON_THREAD;
          }
        }
      }
    }

    // Remote implementation of write.
    public void write(byte[] dataBuffer, int userSpaceOffset, int dataOffset, int length, int force) throws AS400SecurityException, ErrorCompletingRequestException, InterruptedException, IOException, ObjectDoesNotExistException
    {
        // Use remote program call implementation or file implementation.
        if (mustUseProgramCall_)
        {
            // Setup qualified user space name parameter.
            setupNameParameter();
            // Allocate parameter array bytes.
            byte[] inputData = new byte[length];
            // Copy data into parameter array.
            System.arraycopy(dataBuffer, dataOffset, inputData, 0, length);
            // Setup program call parameters.
            ProgramParameter[] parameters = new ProgramParameter[]
            {
                // Qualified user space name, input, char(20).
                nameParameter_,
                // Starting position, input, binary(4), add 1 for 1 based offset.
                new ProgramParameter(BinaryConverter.intToByteArray(userSpaceOffset + 1)),
                // Length of data, input, binary(4).
                new ProgramParameter(BinaryConverter.intToByteArray(length)),
                // Input data, input, char(*).
                new ProgramParameter(inputData),
                // Force changes to auxiliary storage, input, char(1).
                new ProgramParameter(new byte[] { (byte)(0xF0 | force) } )
                // Omit error code optional parameter.
            };

            // Setup for remote program call.
            if (remoteCommand_ == null) {
              setupRemoteCommand();
            }
            // Run change user space (QUSCHGUS) API.  This is a threadsafe API.
            if (!remoteCommand_.runProgram("QSYS", "QUSCHGUS", parameters, runOnThread_))
            {
                // Throw the returned messages.
                throw buildException();
            }
        }
        else
        {
            // Setup for file.
            setupFile();
            // Seek to correct offset.
            file_.seek(userSpaceOffset);
            // Set force option.
            switch (force)
            {
                case UserSpace.FORCE_SYNCHRONOUS:
                    file_.setForceToStorage(true);
                    break;
                case UserSpace.FORCE_ASYNCHRONOUS:
                    file_.setForceToStorage(false);
                    break;
            }
            // Write data.
            file_.writeBytes(dataBuffer, dataOffset, length);
        }
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy