src.com.ibm.as400.access.QSYSObjectPathName Maven / Gradle / Ivy
Show all versions of jt400 Show documentation
///////////////////////////////////////////////////////////////////////////////
//
// JTOpen (IBM Toolbox for Java - OSS version)
//
// Filename: QSYSObjectPathName.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.
//
///////////////////////////////////////////////////////////////////////////////
//
// Usage notes:
//
// For IBM Toolbox for Java developers, this class is generally used in one
// of the following ways:
//
// 1) QSYSObjectPathName ifs = new QSYSObjectPathName(myIFSString);
// if (!ifs.getObjectType().equals("MYTYPE1") &&
// !ifs.getObjectType().equals("MYTYPE2"))
// throw new IllegalPathNameException(myIFSString,
// IllegalPathNameException.OBJECT_TYPE_NOT_VALID);
//
// 2) QSYSObjectPathName ifs
// = new QSYSObjectPathName(myIFSString, "MYTYPE");
//
// For public usage, see the class prolog.
//
///////////////////////////////////////////////////////////////////////////////
// @A1 - 12/04/2007 - Added code to better handle IASPs in QSYS paths.
// parse() - Code to handle /QSYS.LIB and /iasp/qsys.lib without any
// further object types.
// buildPathName() -
// QSYSObjectPathName - new constructor which takes IASP parameter
// toPath() - additional method which takes IASP parameter
//
///////////////////////////////////////////////////////////////////////////////
package com.ibm.as400.access;
import java.beans.PropertyChangeListener;
import java.beans.PropertyChangeSupport;
import java.beans.PropertyVetoException;
import java.beans.VetoableChangeListener;
import java.beans.VetoableChangeSupport;
import java.io.Serializable;
import java.util.EventListener;
/**
Provides an integrated file system path name that represents an object in the QSYS library file system.
QSYSObjectPathName objects generate the following events:
- PropertyChangeEvent
- VetoableChangeEvent
This object can be used in two ways:
- To parse the integrated file system path name to ensure correct syntax and to extract the library, object, member, and object type.
- To build an integrated file system path name from a library, object, and member or object type.
IllegalPathNameExceptions are thrown if errors are found.
Objects in the QSYS file system have integrated file system names with the following format for objects or members in a library other than QSYS:
/QSYS.LIB/library.LIB/object.type
/QSYS.LIB/library.LIB/object.FILE/member.MBR
For objects or members that reside in QSYS, this format is used:
/QSYS.LIB/object.type
/QSYS.LIB/object.FILE/member.MBR
For example:
/QSYS.LIB/QGPL.LIB/CRTLIB.CMD
/QSYS.LIB/QGPL.LIB/ACCOUNTS.FILE/PAYABLE.MBR
/QSYS.LIB/CRTLIB.CMD
/QSYS.LIB/ACCOUNTS.FILE/PAYABLE.MBR
In an integrated file system path name, special values, such as *ALL, that begin with an asterisk are
not depicted with an asterisk but with leading and trailing percent signs (%ALL%).
In the integrated file system, an asterisk is a wildcard character.
The following special values are recognized by this class:
Library name: %ALL%(*ALL), %ALLUSR%(*ALLUSR), %CURLIB%(*CURLIB), %LIBL%(*LIBL), %USRLIBL%(*USRLIBL)
Object name: %ALL%(*ALL)
Member name: %ALL%(*ALL), %FILE%(*FILE), %FIRST%(*FIRST), %LAST%(*LAST) %NONE%(*NONE)
The path name will be in uppercase. If case needs to be preserved for a library, object, or member name, quotation marks should be used around the names. For example:
QSYSObjectPathName path = new QSYSObjectPathName("/QSYS.LIB/\"MixedCase\".LIB/\"lowercase\".FILE");
Examples:
- This code will extract the pieces of an integrated file system name that represents a file:
QSYSObjectPathName path = new QSYSObjectPathName("/QSYS.LIB/QGPL.LIB/ACCOUNTS.FILE");
System.out.println(path.getLibraryName()); // Will print "QGPL"
System.out.println(path.getObjectName()); // Will print "ACCOUNTS"
System.out.println(path.getObjectType()); // Will print "FILE"
- This code will extract the pieces of an integrated file system name that represents a member:
QSYSObjectPathName path = new QSYSObjectPathName("/QSYS.LIB/QGPL.LIB/ACCOUNTS.FILE/PAYABLE.MBR");
System.out.println(path.getLibraryName()); // Will print "QGPL"
System.out.println(path.getObjectName()); // Will print "ACCOUNTS"
System.out.println(path.getMemberName()); // Will print "PAYABLE"
System.out.println(path.getObjectType()); // Will print "MBR"
- This code will build an integrated file system name for a file:
QSYSObjectPathName path = new QSYSObjectPathName("QGPL", "ACCOUNTS", "FILE");
// Will print "/QSYS.LIB/QGPL.LIB/ACCOUNTS.FILE"
System.out.println(path.getPath());
- This code will build an integrated file system name for a member:
QSYSObjectPathName path = new QSYSObjectPathName("QGPL", "ACCOUNTS", "PAYABLE", "MBR");
// Will print "/QSYS.LIB/QGPL.LIB/ACCOUNTS.FILE/PAYABLE.MBR"
System.out.println(path.getPath());
**/
public class QSYSObjectPathName implements Serializable
{
static final long serialVersionUID = 4L;
// Full IFS path name.
private String path_ = "";
// Library name.
private String libraryName_ = "";
// Object name.
private String objectName_ = "";
// Member name.
private String memberName_ = "";
// Object type.
private String objectType_ = "";
// Object ASP name
private String aspName_ = ""; //@A1A
private static QSYSObjectPathName verified_ = null; // to quiet the static analyzer
// List of property change event bean listeners.
private transient PropertyChangeSupport propertyChangeListeners_ = null; // Set on first add.
// List of vetoable change event bean listeners.
private transient VetoableChangeSupport vetoableChangeListeners_ = null; // Set on first add.
/**
Constructs a QSYSObjectPathName object. It creates an integrated file system path name for an object in the QSYS file system.
**/
public QSYSObjectPathName()
{
super();
if (Trace.traceOn_) Trace.log(Trace.DIAGNOSTIC, "Constructing QSYSObjectPathName object.");
}
/**
Constructs a QSYSObjectPathName object. It creates an integrated file system path name for an object in the QSYS file system.
@param path The fully qualified integrated file system name of an object in the QSYS file system.
**/
public QSYSObjectPathName(String path)
{
super();
if (Trace.traceOn_) Trace.log(Trace.DIAGNOSTIC, "Constructing QSYSObjectPathName object, path: " + path);
if (path == null)
{
Trace.log(Trace.ERROR, "Parameter 'path' is null.");
throw new NullPointerException("path");
}
parse(path);
}
// Constructs a QSYSObjectPathName object. It creates an integrated file system path name for an object in the QSYS file system.
// @param path The fully qualified integrated file system name of an object in the QSYS file system.
// @param objectType The type of object path should represent.
QSYSObjectPathName(String path, String objectType)
{
this(path);
// Ensure that the type is correct.
if (!objectType_.equals(objectType.toUpperCase()))
{
Trace.log(Trace.ERROR, "Object type is not valid, path: '" + path + "'");
throw new IllegalPathNameException(path, IllegalPathNameException.OBJECT_TYPE_NOT_VALID);
}
}
/**
Constructs a QSYSObjectPathName object. It builds an integrated file system path name to represent the object.
QSYSObjectPathName ifsName = new QSYSObjectPathName("library", "name", "type");
// This line will print "/QSYS.LIB/LIBRARY.LIB/NAME.TYPE".
System.out.println(ifsName.getPath());
@param libraryName The library in which the object exists. It must be 1-10 characters.
@param objectName The name of the object. It must be 1-10 characters.
@param objectType The type of the object. It must be 1-6 characters. This is the IBM i system abbreviation for the type of object, for example, LIB for library, or CMD for command. Types can be found by prompting for the OBJTYPE parameter on commands such as WRKOBJ.
**/
public QSYSObjectPathName(String libraryName, String objectName, String objectType)
{
super();
if (Trace.traceOn_) Trace.log(Trace.DIAGNOSTIC, "Constructing QSYSObjectPathName object, libraryName: " + libraryName + " objectName: " + objectName + " objectType: " + objectType);
// Verify arguments will make a valid QSYS IFS path name.
checkLibraryName(libraryName);
checkObjectName(objectName);
checkObjectType(objectType);
// If quoted, store as mixed case, else store as uppercase.
libraryName_ = toQSYSName(libraryName);
// If quoted, store as mixed case, else store as uppercase.
objectName_ = toQSYSName(objectName);
objectType_ = objectType.toUpperCase();
path_ = buildPathName(libraryName_, objectName_, "", objectType_);
}
/**
Constructs a QSYSObjectPathName object. It builds an integrated file system path name to represent the member.
QSYSObjectPathName ifsName = new QSYSObjectPathName("library", "name", "member", "MBR");
// This line will print "/QSYS.LIB/LIBRARY.LIB/NAME.FILE/MEMBER.MBR".
System.out.println(ifsName.getPath());
@param libraryName The library in which the object exists. It must be 1-10 characters.
@param objectName The name of the object. It must be 1-10 characters.
@param memberName The name of the member. It must be 1-10 characters.
@param objectType The type of the object. This must be "MBR".
**/
public QSYSObjectPathName(String libraryName, String objectName, String memberName, String objectType)
{
// Note the code in this method is similar to that in the toPath method, and code changes here should most likely be reflected there as well.
super();
if (Trace.traceOn_) Trace.log(Trace.DIAGNOSTIC, "Constructing QSYSObjectPathName object, libraryName: " + libraryName + " objectName: " + objectName + " memberName: " + memberName + " objectType: " + objectType);
// Verify arguments will make a valid QSYS IFS path name.
checkLibraryName(libraryName);
checkObjectName(objectName);
checkMemberName(memberName);
checkObjectTypeIsMember(objectType);
// If quoted, store as mixed case, else store as uppercase.
libraryName_ = toQSYSName(libraryName);
// If quoted, store as mixed case, else store as uppercase.
objectName_ = toQSYSName(objectName);
memberName_ = toQSYSName(memberName);
objectType_ = "MBR";
path_ = buildPathName(libraryName_, objectName_, memberName_, objectType_);
}
/**
Constructs a QSYSObjectPathName object. It builds an integrated file system path name to represent the member.
QSYSObjectPathName ifsName = new QSYSObjectPathName("iasp1", "library", "name", "member", "MBR");
// This line will print "/IASP1/QSYS.LIB/LIBRARY.LIB/NAME.FILE/MEMBER.MBR".
System.out.println(ifsName.getPath());
@param aspName The IASP on which the object exists. It must be 1-10 characters.
@param libraryName The library in which the object exists. It must be 1-10 characters.
@param objectName The name of the object. It must be 1-10 characters.
@param memberName The name of the member. It must be 1-10 characters.
@param objectType The type of the object. This must be "MBR".
**/
public QSYSObjectPathName(String aspName, String libraryName, String objectName, String memberName, String objectType) //@A1A
{
// Note the code in this method is similar to that in the toPath method, and code changes here should most likely be reflected there as well.
super();
if (Trace.traceOn_) Trace.log(Trace.DIAGNOSTIC, "Constructing QSYSObjectPathName object, libraryName: " + libraryName + " objectName: " + objectName + " memberName: " + memberName + " objectType: " + objectType);
// Verify arguments will make a valid QSYS IFS path name.
checkAspName(aspName);
checkLibraryName(libraryName);
checkObjectName(objectName);
checkMemberName(memberName);
checkObjectTypeIsMember(objectType);
// If quoted, store as mixed case, else store as uppercase.
libraryName_ = toQSYSName(libraryName);
// If quoted, store as mixed case, else store as uppercase.
objectName_ = toQSYSName(objectName);
memberName_ = toQSYSName(memberName);
objectType_ = "MBR";
aspName_ = aspName;
path_ = buildPathName(aspName_, libraryName_, objectName_, memberName_, objectType_);
}
/**
Adds a listener to be notified when the value of any bound property is changed. The propertyChange method will be called.
@param listener The listener object.
**/
public void addPropertyChangeListener(PropertyChangeListener listener)
{
if (Trace.traceOn_) Trace.log(Trace.DIAGNOSTIC, "Adding property change listener.");
checkListener(listener);
synchronized (this)
{
// If first add.
if (propertyChangeListeners_ == null)
{
propertyChangeListeners_ = new PropertyChangeSupport(this);
}
propertyChangeListeners_.addPropertyChangeListener(listener);
}
}
/**
Adds a listener to be notified when the value of any constrained property is changed. The vetoableChange method will be called.
@param listener The listener object.
**/
public void addVetoableChangeListener(VetoableChangeListener listener)
{
if (Trace.traceOn_) Trace.log(Trace.DIAGNOSTIC, "Adding vetoable change listener.");
checkListener(listener);
synchronized (this)
{
// If first add.
if (vetoableChangeListeners_ == null)
{
vetoableChangeListeners_ = new VetoableChangeSupport(this);
}
vetoableChangeListeners_.addVetoableChangeListener(listener);
}
}
// Builds the full path name from the pieces (ASP name, library name, object name, object type, member name), if enough data is available. Returns an empty string if not successful.
// @return The integrated file system path name. An empty String will be returned if the path cannot be created.
private static String buildPathName(String aspName, String libraryName, String objectName, String memberName, String objectType)//@A1A
{
if (Trace.traceOn_) Trace.log(Trace.DIAGNOSTIC, "QSYSObjectPathName buildPathName(asp): aspName: " + aspName + " libraryName: " + libraryName + " objectName: " + objectName + " memberName: " + memberName + " objectType: " + objectType);
// First build the path (without the ASP name), then prepend the asp name
String result1 = buildPathName(libraryName, objectName, memberName, objectType);
StringBuffer result2 = new StringBuffer(75);
if (!result1.equals(""))
{
if (!aspName.equals(""))
{
result2.append("/");
result2.append(aspName); // Prepend the ASP name
}
result2.append(result1);
if (Trace.traceOn_) Trace.log(Trace.DIAGNOSTIC, "QSYSObjectPathName buildPathName(asp): result: " + result2.toString());
return result2.toString();
}
else
{
if (Trace.traceOn_) Trace.log(Trace.DIAGNOSTIC, "QSYSObjectPathName buildPathName(asp): result: EMPTY STRING");
return "";
}
}
// Builds the full path name from the pieces (library name, object name, object type, member name), if enough data is available. Returns an empty string if not successful.
// @return The integrated file system path name. An empty String will be returned if the path cannot be created.
private static String buildPathName(String libraryName, String objectName, String memberName, String objectType)
{
if (Trace.traceOn_) Trace.log(Trace.DIAGNOSTIC, "QSYSObjectPathName buildPathName(): libraryName: " + libraryName + " objectName: " + objectName + " memberName: " + memberName + " objectType: " + objectType);
if (!libraryName.equals("") && !objectType.equals("") && (!memberName.equals("") || !objectType.equals("MBR"))) //@A1C
{
StringBuffer result = new StringBuffer(64);
result.append("/QSYS.LIB");
if (!libraryName.equals("QSYS"))
{
// Add "/xyz.LIB"
result.append("/");
result.append(convertLibraryName(libraryName));
result.append(".LIB");
}
// For the special case of "/QSYS.LIB", the objectName is empty string
if (!objectName.equals(""))
{
result.append("/");
result.append(convertObjectName(objectName));
if (!memberName.equals(""))
{
result.append(".FILE/");
result.append(convertMemberName(memberName));
result.append(".MBR");
}
else
{
result.append(".");
result.append(objectType);
}
}
if (Trace.traceOn_) Trace.log(Trace.DIAGNOSTIC, "QSYSObjectPathName buildPathName(): result: " + result.toString());
return result.toString();
}
if (Trace.traceOn_) Trace.log(Trace.DIAGNOSTIC, "QSYSObjectPathName buildPathName(): result: EMPTY STRING");
return "";
}
static void checkObjectTypeIsMember(String objectType)
{
if (objectType == null)
{
Trace.log(Trace.ERROR, "Parameter 'objectType' is null.");
throw new NullPointerException("objectType");
}
if (!objectType.toUpperCase().equals("MBR"))
{
Trace.log(Trace.ERROR, "Value of parameter 'objectType' is not valid: " + objectType);
throw new ExtendedIllegalArgumentException("objectType (" + objectType + ")", ExtendedIllegalArgumentException.PARAMETER_VALUE_NOT_VALID);
}
}
static void checkMemberName(String memberName)
{
if (memberName == null)
{
Trace.log(Trace.ERROR, "Parameter 'memberName' is null.");
throw new NullPointerException("memberName");
}
if (memberName.length() < 1 || memberName.length() > 10)
{
Trace.log(Trace.ERROR, "Length of parameter 'memberName' is not valid: '" + memberName + "'");
throw new ExtendedIllegalArgumentException("memberName (" + memberName + ")", ExtendedIllegalArgumentException.LENGTH_NOT_VALID);
}
}
static void checkListener(EventListener listener)
{
if (listener == null)
{
Trace.log(Trace.ERROR, "Parameter 'listener' is null.");
throw new NullPointerException("listener");
}
}
// New Method to verify ASP name is valid @A1A
static void checkAspName(String aspName) //@A1A
{
if (aspName == null)
{
Trace.log(Trace.ERROR, "Parameter 'aspName' is null.");
throw new NullPointerException("aspName");
}
if (aspName.length() < 1 || aspName.length() > 10)
{
Trace.log(Trace.ERROR, "Length of parameter 'aspName' is not valid: '" + aspName + "'");
throw new ExtendedIllegalArgumentException("aspName (" + aspName + ")", ExtendedIllegalArgumentException.LENGTH_NOT_VALID);
}
}
static void checkLibraryName(String libraryName)
{
if (libraryName == null)
{
Trace.log(Trace.ERROR, "Parameter 'libraryName' is null.");
throw new NullPointerException("libraryName");
}
if (libraryName.length() < 1 || libraryName.length() > 10)
{
Trace.log(Trace.ERROR, "Length of parameter 'libraryName' is not valid: '" + libraryName + "'");
throw new ExtendedIllegalArgumentException("libraryName (" + libraryName + ")", ExtendedIllegalArgumentException.LENGTH_NOT_VALID);
}
}
static void checkObjectName(String objectName)
{
if (objectName == null)
{
Trace.log(Trace.ERROR, "Parameter 'objectName' is null.");
throw new NullPointerException("objectName");
}
if (objectName.length() < 1 || objectName.length() > 10)
{
Trace.log(Trace.ERROR, "Length of parameter 'objectName' is not valid: '" + objectName + "'");
throw new ExtendedIllegalArgumentException("objectName (" + objectName + ")", ExtendedIllegalArgumentException.LENGTH_NOT_VALID);
}
}
static void checkObjectType(String objectType)
{
if (objectType == null)
{
Trace.log(Trace.ERROR, "Parameter 'objectType' is null.");
throw new NullPointerException("objectType");
}
if (objectType.length() < 1 || objectType.length() > 6)
{
Trace.log(Trace.ERROR, "Length of parameter 'objectType' is not valid: '" + objectType + "'");
throw new ExtendedIllegalArgumentException("objectType (" + objectType + ")", ExtendedIllegalArgumentException.LENGTH_NOT_VALID);
}
}
// Converts the library name to the integrated file system pathname value by handling special values.
// @param libraryName The library in which the object exists.
// @return The library name in integrated file system path name form.
private static String convertLibraryName(String libraryName)
{
// Check to see if we have a special value.
if (libraryName.equals("*LIBL")) return "%LIBL%";
if (libraryName.equals("*CURLIB")) return "%CURLIB%";
if (libraryName.equals("*USRLIBL")) return "%USRLIBL%";
if (libraryName.equals("*ALL")) return "%ALL%";
if (libraryName.equals("*ALLUSR")) return "%ALLUSR%";
return libraryName;
}
// Converts the member name to the integrated file system pathname value by handling special values.
// @param memberName The name of the member.
// @return The member name in integrated file system path name form.
private static String convertMemberName(String memberName)
{
// Check to see if we have a special value.
if (memberName.equals("*FIRST")) return "%FIRST%";
if (memberName.equals("*LAST")) return "%LAST%";
if (memberName.equals("*FILE")) return "%FILE%";
if (memberName.equals("*ALL")) return "%ALL%";
if (memberName.equals("*NONE")) return "%NONE%";
return memberName;
}
// Converts the object name to the integrated file system pathname value by handling special values.
// @param objectName The name of the object.
// @return The object name in the integrated file system path name form.
static private String convertObjectName(String objectName)
{
// Check for special value.
if (objectName.equals("*ALL")) return "%ALL%";
return objectName;
}
/**
Returns the ASP on which the object resides.
@return The name of the ASP. If the ASP name has not been set, an empty string is returned.
**/
public String getAspName() //@A1A
{
if (Trace.traceOn_) Trace.log(Trace.DIAGNOSTIC, "Getting ASP name: " + aspName_);
return aspName_;
}
/**
Returns the library in which the object resides.
@return The name of the library. If the library name has not been set, an empty string is returned.
**/
public String getLibraryName()
{
if (Trace.traceOn_) Trace.log(Trace.DIAGNOSTIC, "Getting library name: " + libraryName_);
return libraryName_;
}
/**
Returns the name of the member. If this object does not represent a member, an empty string is returned.
@return The name of the member. If the member name has not been set, an empty string is returned.
**/
public String getMemberName()
{
if (Trace.traceOn_) Trace.log(Trace.DIAGNOSTIC, "Getting member name: " + memberName_);
return memberName_;
}
/**
Returns the name of the object this path name represents. If this object represents a member, the object name is the name of the file in which the member exists.
@return The name of the object. If the object name has not been set, an empty string is returned.
**/
public String getObjectName()
{
if (Trace.traceOn_) Trace.log(Trace.DIAGNOSTIC, "Getting object name: " + objectName_);
return objectName_;
}
/**
Returns type of object this path name represents. Type is the IBM i system abbreviation for the type of object, for example, LIB for library, or CMD for command. Types can be found by prompting for the OBJTYPE parameter on commands such as WRKOBJ.
@return The type of the object. If the object type has not been set, an empty string is returned.
**/
public String getObjectType()
{
if (Trace.traceOn_) Trace.log(Trace.DIAGNOSTIC, "Getting object type: " + objectType_);
return objectType_;
}
/**
Returns the fully qualified integrated file system path name.
@return The fully qualified integrated file system path name.
If the path has not been set, an empty string is returned.
**/
public String getPath()
{
if (Trace.traceOn_) Trace.log(Trace.DIAGNOSTIC, "Getting path: " + path_);
return path_;
}
// Uppercase to simplify parsing.
// Extracts the library name, object name, object type, and member name from the full integrated file system path string. Verifies the full integrated file system path is in fact a valid integrated file system path name for an object in the QSYS file system. Fills in instance data.
// @param path The integrated file system path name to parse.
private void parse(String path)
{
String upperCasePath = toQSYSName(path);
//------------------------------------------------------
// Process the prefix.
//------------------------------------------------------
// Take into account possible IASP prefix to QSYS path. @A1A
// So determine index of "/QSYS.LIB" @A1A
int indexOfQsysLib = upperCasePath.indexOf("/QSYS.LIB"); //@A1A
// Required prefix.
if (indexOfQsysLib == -1) //@A1C
{
Trace.log(Trace.ERROR, "Specified path does not include '/QSYS.LIB':", path);
throw new IllegalPathNameException(path, IllegalPathNameException.QSYS_PREFIX_MISSING);
}
// Extract the name of the IASP @A1A
if (indexOfQsysLib > 0) //@A1A
{
// Assume absolute path starts with delimiter
aspName_ = upperCasePath.substring(1, indexOfQsysLib); //@A1A
}
// Special case the "/QSYS.LIB" scenario (no other nested objects specified) //@A1A
if ((upperCasePath.substring(indexOfQsysLib).equals("/QSYS.LIB")) || (upperCasePath.substring(indexOfQsysLib).equals("/QSYS.LIB/")))
{
libraryName_ = "QSYS"; //@A1A
objectType_ = "LIB"; //@A1A
path_ = buildPathName(aspName_, libraryName_, objectName_, memberName_, objectType_); //@A1A
return; //@A1A
}
//------------------------------------------------------
// Process library.
//------------------------------------------------------
// Move to start of first nested object within "/QSYS.LIB/"
int currentOffset = 10; // Length of "/QSYS.LIB/"
if (!aspName_.equals("")) //@A1A
{
// Move past "/iaspname"
currentOffset = currentOffset + 1 + aspName_.length(); //@A1A
}
// Find suffix after library name.
int nextOffset = upperCasePath.indexOf(".LIB/", currentOffset);
// If a qualifying library name was specified.
if (nextOffset > currentOffset)
{
// If quoted, store as mixed case, else store as uppercase.
libraryName_ = toQSYSName(path.substring(currentOffset, nextOffset));
// Disallow /QSYS.LIB/QSYS.LIB.
if (libraryName_.equals("QSYS"))
{
Trace.log(Trace.ERROR, "Object in library QSYS specified incorrectly, path: '" + path + "'");
throw new IllegalPathNameException(path, IllegalPathNameException.QSYS_SYNTAX_NOT_VALID);
}
if (libraryName_.charAt(0) == '%')
{
// Possibly a "special" library name.
if (libraryName_.equals("%LIBL%")) libraryName_ = "*LIBL";
else if (libraryName_.equals("%CURLIB%")) libraryName_ = "*CURLIB";
else if (libraryName_.equals("%USRLIBL%")) libraryName_ = "*USRLIBL";
else if (libraryName_.equals("%ALL%")) libraryName_ = "*ALL";
else if (libraryName_.equals("%ALLUSR%")) libraryName_ = "*ALLUSR";
}
// Move past ".LIB/" to the first character of object name.
currentOffset = nextOffset + 5;
}
else if (nextOffset == -1)
{
// No qualifying library name was specified, set library name to QSYS.
libraryName_ = "QSYS";
}
else
{
// A ".LIB" immediately followed "/QSYS.LIB/".
Trace.log(Trace.ERROR, "Length of the library name is not valid, path: '" + path + "'");
throw new IllegalPathNameException(path, IllegalPathNameException.LIBRARY_LENGTH_NOT_VALID);
}
// If name is > 10 chars.
if (libraryName_.length() > 10)
{
Trace.log(Trace.ERROR, "Length of the library name is not valid, path: '" + path + "'");
throw new IllegalPathNameException(path, IllegalPathNameException.LIBRARY_LENGTH_NOT_VALID);
}
//------------------------------------------------------
// Process object type.
//------------------------------------------------------
// Find last period in path name (object.type delimiter).
nextOffset = upperCasePath.lastIndexOf('.');
// If no type specified or type > 6 chars.
if (nextOffset < currentOffset || upperCasePath.length() - nextOffset - 1 > 6)
{
Trace.log(Trace.ERROR, "Length of the object type is not valid, path: '" + path + "'");
throw new IllegalPathNameException(path, IllegalPathNameException.TYPE_LENGTH_NOT_VALID);
}
objectType_ = upperCasePath.substring(nextOffset + 1);
//------------------------------------------------------
// Process member name.
//------------------------------------------------------
// Only needs to be done if the type is MBR.
if (objectType_.equals("MBR"))
{
// Find end of the object (file) name.
int memberOffset = upperCasePath.lastIndexOf(".FILE/", nextOffset);
// If not found or not in object name portion of path.
if (memberOffset == -1 || memberOffset < currentOffset)
{
Trace.log(Trace.ERROR, "Member is not contained in a file, path: '" + path + "'");
throw new IllegalPathNameException(path, IllegalPathNameException.MEMBER_WITHOUT_FILE);
}
// Move to first char of member name.
memberOffset += 6;
// Check that member name is 1-10 chars
if (nextOffset < memberOffset || nextOffset - memberOffset > 10)
{
Trace.log(Trace.ERROR, "Length of the member name is not valid, path: '" + path + "'");
throw new IllegalPathNameException(path, IllegalPathNameException.MEMBER_LENGTH_NOT_VALID);
}
// The member name is syntactically correct.
// If quoted, store as mixed case, else store as uppercase.
memberName_ = toQSYSName(path.substring(memberOffset, nextOffset));
if (memberName_.charAt(0) == '%')
{
// Check for special member values.
if (memberName_.equals("%FIRST%")) memberName_ = "*FIRST";
else if (memberName_.equals("%LAST%")) memberName_ = "*LAST";
else if (memberName_.equals("%FILE%")) memberName_ = "*FILE";
else if (memberName_.equals("%ALL%")) memberName_ = "*ALL";
else if (memberName_.equals("%NONE%")) memberName_ = "*NONE";
}
// Move to period after object name.
nextOffset = memberOffset - 6;
}
//------------------------------------------------------
// Process object name.
//------------------------------------------------------
// Try to catch the case where the library was specified incorrectly, by testing if the 'object name' has a slash in it but is not quoted, and the library was thought to be QSYS.
if (libraryName_.equals("QSYS") && upperCasePath.substring(currentOffset, nextOffset).indexOf('/') != -1 && upperCasePath.charAt(currentOffset) != '\"')
{
Trace.log(Trace.ERROR, "Library not specified correctly, path: '" + path + "'");
throw new IllegalPathNameException(path, IllegalPathNameException.LIBRARY_SPECIFICATION_NOT_VALID);
}
// Check that object name is 1-10 chars.
if (nextOffset < currentOffset || nextOffset - currentOffset > 10)
{
Trace.log(Trace.ERROR, "Length of the object name is not valid, path: '" + path + "'");
throw new IllegalPathNameException(path, IllegalPathNameException.OBJECT_LENGTH_NOT_VALID);
}
// The object name is syntactically correct.
// If quoted, store as mixed case, else store as uppercase.
objectName_ = toQSYSName(path.substring(currentOffset, nextOffset));
// Check for special object values.
if (objectName_.equals("%ALL%")) objectName_ = "*ALL";
// Fill in path_ instance var (with correct case).
path_ = buildPathName(aspName_, libraryName_, objectName_, memberName_, objectType_);
}
/**
Removes a listener from the change list. If the listener is not on the list, do nothing.
@param listener The listener object.
**/
public void removePropertyChangeListener(PropertyChangeListener listener)
{
if (Trace.traceOn_) Trace.log(Trace.DIAGNOSTIC, "Removing property change listener.");
checkListener(listener);
// If we have listeners.
if (propertyChangeListeners_ != null)
{
propertyChangeListeners_.removePropertyChangeListener(listener);
}
}
/**
Removes a listener from the veto list. If the listener is not on the list, do nothing.
@param listener The listener object.
**/
public void removeVetoableChangeListener(VetoableChangeListener listener)
{
if (Trace.traceOn_) Trace.log(Trace.DIAGNOSTIC, "Removing vetoable change listener.");
checkListener(listener);
// If we have listeners.
if (vetoableChangeListeners_ != null)
{
vetoableChangeListeners_.removeVetoableChangeListener(listener);
}
}
/**
Sets the IASP in which the object resides. This is a bound and constrained property. Note that changes to this property also affect the pathName property.
@param aspName The IASP in which the object exists. It must be 1-10 characters.
@exception PropertyVetoException If the change was vetoed.
**/
public void setAspName(String aspName) throws PropertyVetoException //@A1A
{
// Check for valid parameter.
checkAspName(aspName);
// Process the asp name, if quoted, store as mixed case, else store as uppercase.
String newAspName = toQSYSName(aspName);
// Build new path.
String newPath = buildPathName(newAspName, libraryName_, objectName_, memberName_, objectType_);
if (propertyChangeListeners_ == null && vetoableChangeListeners_ == null)
{
// Make the changes.
aspName_ = newAspName;
path_ = newPath;
}
else
{
String oldAspName = aspName_;
String oldPath = path_;
if (vetoableChangeListeners_ != null)
{
vetoableChangeListeners_.fireVetoableChange("aspName", oldAspName, newAspName);
vetoableChangeListeners_.fireVetoableChange("path", oldPath, newPath);
}
// Make the changes.
aspName_ = newAspName;
path_ = newPath;
if (propertyChangeListeners_ != null)
{
propertyChangeListeners_.firePropertyChange("aspName", oldAspName, newAspName);
propertyChangeListeners_.firePropertyChange("path", oldPath, newPath);
}
}
}
/**
Sets the library in which the object resides. This is a bound and constrained property. Note that changes to this property also affect the pathName property.
@param libraryName The library in which the object exists. It must be 1-10 characters.
@exception PropertyVetoException If the change was vetoed.
**/
public void setLibraryName(String libraryName) throws PropertyVetoException
{
// Check for valid parameter.
checkLibraryName(libraryName);
// Process the library name, if quoted, store as mixed case, else store as uppercase.
String newLibraryName = toQSYSName(libraryName);
// Build new path.
String newPath = buildPathName(aspName_, newLibraryName, objectName_, memberName_, objectType_);
if (propertyChangeListeners_ == null && vetoableChangeListeners_ == null)
{
// Make the changes.
libraryName_ = newLibraryName;
path_ = newPath;
}
else
{
String oldLibraryName = libraryName_;
String oldPath = path_;
if (vetoableChangeListeners_ != null)
{
vetoableChangeListeners_.fireVetoableChange("libraryName", oldLibraryName, newLibraryName);
vetoableChangeListeners_.fireVetoableChange("path", oldPath, newPath);
}
// Make the changes.
libraryName_ = newLibraryName;
path_ = newPath;
if (propertyChangeListeners_ != null)
{
propertyChangeListeners_.firePropertyChange("libraryName", oldLibraryName, newLibraryName);
propertyChangeListeners_.firePropertyChange("path", oldPath, newPath);
}
}
}
/**
Sets the name of the member. If a value other than an empty String ("") is specified, the object type is set to MBR. This is a bound and constrained property. Note that changes to this property also affect the objectType and pathName properties.
@param memberName The name of the member. It must be 10 characters or less. An empty String ("") can be passed to indicate this object does not represent a member.
@exception PropertyVetoException If the change was vetoed.
**/
public void setMemberName(String memberName) throws PropertyVetoException
{
// Check for valid parameter.
if (memberName == null)
{
Trace.log(Trace.ERROR, "Parameter 'memberName' is null.");
throw new NullPointerException("memberName");
}
if (memberName.length() > 10)
{
Trace.log(Trace.ERROR, "Length of parameter 'memberName' is not valid: '" + memberName + "'");
throw new ExtendedIllegalArgumentException("memberName (" + memberName + ")", ExtendedIllegalArgumentException.LENGTH_NOT_VALID);
}
// If quoted, store as mixed case, else store as uppercase.
String newMemberName = toQSYSName(memberName);
// The type of an object with a member is always MBR.
String newObjectType = memberName.length() != 0 ? "MBR" : objectType_;
// Build new path name.
String newPath = buildPathName(aspName_, libraryName_, objectName_, newMemberName, newObjectType);
if (propertyChangeListeners_ == null && vetoableChangeListeners_ == null)
{
// Make the changes.
memberName_ = newMemberName;
objectType_ = newObjectType;
path_ = newPath;
}
else
{
String oldMemberName = memberName_;
String oldObjectType = objectType_;
String oldPath = path_;
if (vetoableChangeListeners_ != null)
{
vetoableChangeListeners_.fireVetoableChange("memberName", oldMemberName, newMemberName);
vetoableChangeListeners_.fireVetoableChange("objectType", oldObjectType, newObjectType);
vetoableChangeListeners_.fireVetoableChange("path", oldPath, newPath);
}
// Make the changes.
memberName_ = newMemberName;
objectType_ = newObjectType;
path_ = newPath;
if (propertyChangeListeners_ != null)
{
propertyChangeListeners_.firePropertyChange("memberName", oldMemberName, newMemberName);
propertyChangeListeners_.firePropertyChange("objectType", oldObjectType, newObjectType);
propertyChangeListeners_.firePropertyChange("path", oldPath, newPath);
}
}
}
/**
Sets the name of the object this path name represents. If this object represents a member, the object name is the name of the file that the member is in. This is a bound and constrained property. Note that changes to this property also affect the pathName property.
@param objectName The name of the object. It must be 1-10 characters.
@exception PropertyVetoException If the change was vetoed.
**/
public void setObjectName(String objectName) throws PropertyVetoException
{
// Check for valid parameter.
checkObjectName(objectName);
// Process object name. if quoted, store as mixed case else store as uppercase.
String newObjectName = toQSYSName(objectName);
// Build new path name.
String newPath = buildPathName(aspName_, libraryName_, newObjectName, memberName_, objectType_);
if (propertyChangeListeners_ == null && vetoableChangeListeners_ == null)
{
// Make the changes.
objectName_ = newObjectName;
path_ = newPath;
}
else
{
String oldObjectName = objectName_;
String oldPath = path_;
if (vetoableChangeListeners_ != null)
{
vetoableChangeListeners_.fireVetoableChange("objectName", oldObjectName, newObjectName);
vetoableChangeListeners_.fireVetoableChange("path", oldPath, newPath);
}
// Make the changes.
objectName_ = newObjectName;
path_ = newPath;
if (propertyChangeListeners_ != null)
{
propertyChangeListeners_.firePropertyChange("objectName", oldObjectName, newObjectName);
propertyChangeListeners_.firePropertyChange("path", oldPath, newPath);
}
}
}
/**
Sets type of object this path name represents. If the type is not MBR, the member name property will be set to an empty string. The value will be uppercased. This is a bound and constrained property. Note that changes to this property also affect the memberName and pathName properties.
@param objectType The type of the object. It must be 1-6 characters. This is the IBM i system abbreviation for the type of object, for example, LIB for library, or CMD for command. Types can be found by prompting for the OBJTYPE parameter on commands such as WRKOBJ.
@exception PropertyVetoException If the change was vetoed.
**/
public void setObjectType(String objectType) throws PropertyVetoException
{
// Check for valid parameter.
checkObjectType(objectType);
// Process object type.
String newObjectType = objectType.toUpperCase();
String newMemberName = newObjectType.equals("MBR") ? memberName_ : "";
// Build new path name.
String newPath = buildPathName(aspName_, libraryName_, objectName_, newMemberName, newObjectType);
if (propertyChangeListeners_ == null && vetoableChangeListeners_ == null)
{
// Make the changes.
memberName_ = newMemberName;
objectType_ = newObjectType;
path_ = newPath;
}
else
{
String oldMemberName = memberName_;
String oldObjectType = objectType_;
String oldPath = path_;
if (vetoableChangeListeners_ != null)
{
vetoableChangeListeners_.fireVetoableChange("memberName", oldMemberName, newMemberName);
vetoableChangeListeners_.fireVetoableChange("objectType", oldObjectType, newObjectType);
vetoableChangeListeners_.fireVetoableChange("path", oldPath, newPath);
}
// Make the changes.
objectType_ = newObjectType;
memberName_ = newMemberName;
path_ = newPath;
if (propertyChangeListeners_ != null)
{
propertyChangeListeners_.firePropertyChange("memberName", oldMemberName, newMemberName);
propertyChangeListeners_.firePropertyChange("objectType", oldObjectType, newObjectType);
propertyChangeListeners_.firePropertyChange("path", oldPath, newPath);
}
}
}
/**
Sets the integrated file system path name for this object. This is a bound and constrained property. Note that changes to this property also affect the libraryName, memberName, objectName, and objectType properties.
@param path The fully qualified integrated file system name of an object in the QSYS file system.
@exception PropertyVetoException If the change was vetoed.
**/
public void setPath(String path) throws PropertyVetoException
{
// Verify path is valid IFS path name.
QSYSObjectPathName ifs = new QSYSObjectPathName(path);
if (propertyChangeListeners_ == null && vetoableChangeListeners_ == null)
{
// Make the changes.
path_ = ifs.path_;
libraryName_ = ifs.libraryName_;
objectName_ = ifs.objectName_;
memberName_ = ifs.memberName_;
objectType_ = ifs.objectType_;
}
else
{
String newPath = ifs.path_;
String newLibraryName = ifs.libraryName_;
String newObjectName = ifs.objectName_;
String newMemberName = ifs.memberName_;
String newObjectType = ifs.objectType_;
String oldPath = path_;
String oldLibraryName = libraryName_;
String oldObjectName = objectName_;
String oldMemberName = memberName_;
String oldObjectType = objectType_;
if (vetoableChangeListeners_ != null)
{
vetoableChangeListeners_.fireVetoableChange("path", oldPath, newPath);
vetoableChangeListeners_.fireVetoableChange("libraryName", oldLibraryName, newLibraryName);
vetoableChangeListeners_.fireVetoableChange("objectName", oldObjectName, newObjectName);
vetoableChangeListeners_.fireVetoableChange("memberName", oldMemberName, newMemberName);
vetoableChangeListeners_.fireVetoableChange("objectType", oldObjectType, newObjectType);
}
// Make the changes.
path_ = newPath;
libraryName_ = newLibraryName;
objectName_ = newObjectName;
memberName_ = newMemberName;
objectType_ = newObjectType;
if (propertyChangeListeners_ != null)
{
propertyChangeListeners_.firePropertyChange("path", oldPath, newPath);
propertyChangeListeners_.firePropertyChange("libraryName", oldLibraryName, newLibraryName);
propertyChangeListeners_.firePropertyChange("objectName", oldObjectName, newObjectName);
propertyChangeListeners_.firePropertyChange("memberName", oldMemberName, newMemberName);
propertyChangeListeners_.firePropertyChange("objectType", oldObjectType, newObjectType);
}
}
}
/**
Builds an integrated file system path name to represent the object.
@param libraryName The library the object is in. It must be 1-10 characters.
@param objectName The name of the object. It must be 1-10 characters.
@param objectType The type of the object. It must be 1-6 characters. This is the IBM i system abbreviation for the type of object, for example, LIB for library, or CMD for command. Types can be found by prompting for the OBJTYPE parameter on commands such as WRKOBJ.
@return The integrated file system name for the object.
**/
public static String toPath(String libraryName, String objectName, String objectType)
{
// Note the code in this method is similar to that in the construct method, and code changes here should most likely be reflected there as well.
// Verify arguments will make a valid QSYS IFS path name.
checkLibraryName(libraryName);
checkObjectName(objectName);
checkObjectType(objectType);
libraryName = toQSYSName(libraryName);
objectName = toQSYSName(objectName);
objectType = objectType.toUpperCase();
return buildPathName(libraryName, objectName, "", objectType);
}
/**
Builds an integrated file system path name to represent the member.
@param libraryName The library the object is in. It must be 1-10 characters.
@param objectName The name of the object. It must be 1-10 characters.
@param memberName The name of the member. It must be 1-10 characters.
@param objectType The type of the object. This must be "MBR".
@return The integrated file system name for the object.
**/
public static String toPath(String libraryName, String objectName, String memberName, String objectType)
{
// Note the code in this method is similar to that in the 4-parm ctor, and code changes here should most likely be reflected there as well.
// Verify arguments will make a valid QSYS IFS path name.
checkLibraryName(libraryName);
checkObjectName(objectName);
checkMemberName(memberName);
checkObjectTypeIsMember(objectType);
libraryName = toQSYSName(libraryName);
objectName = toQSYSName(objectName);
memberName = toQSYSName(memberName);
objectType = objectType.toUpperCase();
return buildPathName(libraryName, objectName, memberName, objectType);
}
/**
Builds an integrated file system path name to represent the member.
@param aspName The IASP on which the object exists. It must be 1-10 characters.
@param libraryName The library the object is in. It must be 1-10 characters.
@param objectName The name of the object. It must be 1-10 characters.
@param memberName The name of the member. It must be 1-10 characters.
@param objectType The type of the object. This must be "MBR".
@return The integrated file system name for the object.
**/
// Similar to previous toPath() method but include aspName parameter
public static String toPath(String aspName, String libraryName, String objectName, String memberName, String objectType)//@A1A
{
// Note the code in this method is similar to that in the 4-parm ctor, and code changes here should most likely be reflected there as well.
// Verify arguments will make a valid QSYS IFS path name.
checkAspName(aspName);
checkLibraryName(libraryName);
checkObjectName(objectName);
checkMemberName(memberName);
checkObjectTypeIsMember(objectType);
libraryName = toQSYSName(libraryName);
objectName = toQSYSName(objectName);
memberName = toQSYSName(memberName);
objectType = objectType.toUpperCase();
aspName = aspName.toUpperCase();
return buildPathName(aspName, libraryName, objectName, memberName, objectType);
}
/**
Utility method for selectively uppercasing the characters in a string, for use as an IBM i *NAME value. This method is used, for example, when uppercasing object names, library names, and member names. Characters within double-quotes are left as-is.
@param name The name to be uppercased.
@return The *NAME value, selectively uppercased.
**/
public static final String toQSYSName(String name)
{
// Uppercase all unquoted characters _except_ the "Latin small letter 'a' with grave" (\u00E0), which for CCSID 297 (French) gets converted to EBCDIC x7C, an invariant character that is allowed in *NAME strings.
if (name.indexOf('\u00E0') == -1 && // no special characters
name.indexOf('\"') == -1) { // and no quotes
return name.toUpperCase();
}
else { // selectively uppercase unquoted characters
StringBuffer result = new StringBuffer();
int length = name.length();
boolean inQuotes = false;
for (int i=0; i 10) {
if (Trace.traceOn_) Trace.log(Trace.WARNING, "Truncating object name to 10 characters:" , objectName_);
buffer.replace(0, 10, objectName_.substring(0,10));
}
else
buffer.replace(0, objectName_.length(), objectName_);
// Set the library name into the second 10 characters.
if (libraryName_.length() > 10) {
if (Trace.traceOn_) Trace.log(Trace.WARNING, "Truncating library name to 10 characters:" , libraryName_);
buffer.replace(10, 20, libraryName_.substring(0,10));
}
else
buffer.replace(10, libraryName_.length()+10, libraryName_);
return buffer.toString();
}
// Validates a path.
// @param path The fully qualified integrated file system name of an object in the QSYS file system.
// @param objectType The type of object path should represent.
static final void validatePath(String path, String objectType)
{
// The constructor will throw an exception if the path is not valid.
verified_ = new QSYSObjectPathName(path, objectType);
}
}