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

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

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

package com.ibm.as400.access;

import java.beans.PropertyVetoException;
import java.io.IOException;
import java.util.Calendar;
import java.util.Date;


/**
 * Represents a licensed product on the system.
 * The {@link #isInstalled isInstalled()} method should be called to verify that
 * the product is installed on the system. If the product is not installed, other information
 * returned by getters in this class might not be valid.
**/
public class Product
{
  private static final String USERSPACE_NAME = "JT4PTF    QTEMP     ";
  private static final String USERSPACE_PATH = "/QSYS.LIB/QTEMP.LIB/JT4PTF.USRSPC";

  private boolean loaded_ = false; // Have we loaded the common (PRDR0100 format) values
  private boolean loadedLoadID_ = false; // Have we loaded the load ID yet
  private boolean partiallyLoaded_ = false; // Were we constructed from a ProductList
  private boolean loadedOptions_ = false; // Have we loaded our option information
  private boolean loadedDescriptionText_ = false; // Have we loaded the description yet
  private boolean loaded500_ = false; // Have we loaded the PRDR0500 format values
  private boolean loaded800_ = false; // Have we loaded the PRDR0500 format values
  private boolean error100_ = false; // Did we get an error retrieving the PRDR0100 format values
  private boolean error500_ = false; // Did we get an error retrieving the PRDR0500 format values
  private boolean error800_ = false; // Did we get an error retrieving the PRDR0800 format values

  private AS400 system_;            // never null
  private String productID_;        // never null
  private String productOption_;    // never null
  private String releaseLevel_;     // never null
  private String descriptionID_;
  private String descriptionText_;
  private String messageFile_;
  private boolean installed_ = true; // Default to true; if the API throws an exception, switch to false.
  private String loadID_;

  // PRDR0100 format   (fields that are meaningfully returned in all formats)
  // Note: The loaded_ flag indicates that the following fields have been loaded.
  //       The loadID_ field gets its own separate flag: loadedLoadID_
  private String loadType_;
  private String symbolicLoadState_;
  private boolean loadErrorIndicator_;
  private String loadState_;
  private boolean supported_;
  private String registrationType_;
  private String registrationValue_;
  private String primaryLanguageLoadID_;
  private String minimumTargetRelease_;
  private String minimumBaseVRM_;
  private int requirementsMet_;
  private String level_;

  // PRDR0500 format
  private boolean allowsMultipleReleases_;
  private Date releaseDate_;
  private String firstCopyright_;
  private String currentCopyright_;
  // messageFile_;
  // product options and their properties
  private boolean allowsDynamicNaming_;
  private String minimumVRM_;

  private boolean allowsMixedReleases_;

  // PRDR0800 format
  private ProductDirectoryInformation[] directories_;
  private Product[] options_;
  private int chunkSize_ = 8192;



  /**
   * Constant indicating that the product load is defined but
   * the product load object for this load does not exist. When
   * a product definition is created, a code load is defined for
   * each product option, and language loads can be defined.
  **/
  public static final String LOAD_STATE_DEFINED_NO_OBJECT = "10";

  /**
   * Constant indicating that the product load object exists, but
   * before it can be saved using the Save Licensed Program (SAVLICPGM)
   * command, it must be packaged with either the Package Product
   * Option (PKGPRDOPT) command or the Package Product Option
   * (QSZPKGPO) API.
  **/
  public static final String LOAD_STATE_DEFINED_OBJECT_EXISTS = "20";

  /**
   * Constant indicating that a Restore Licensed Program (RSTLICPGM)
   * command did not complete successfully. A preoperation exit program
   * failed. The product being replaced had been packaged, but not installed.
  **/
  public static final String LOAD_STATE_PACKAGED_RESTORE_FAILED_EXIT_PROGRAM_FAILED = "3E";

  /**
   * Constant indicating that a Restore Licensed Program (RSTLICPGM)
   * command failed. A preoperation exit program did not fail. The product
   * being replaced had been packaged, but not installed.
  **/
  public static final String LOAD_STATE_PACKAGED_RESTORE_FAILED = "3F";

  /**
   * Constant indicating that the product load object for this load has
   * been packaged with the Package Product Option (PKGPRDOPT) command
   * or the Package Product Option (QSZPKGPO) API.
  **/
  public static final String LOAD_STATE_PACKAGED = "30";

  /**
   * Constant indicating that the product load object for this load has
   * been packaged with the Package Product Option (PKGPRDOPT) command
   * or the Package Product Option (QSZPKGPO) API, but either a
   * development library or folder was renamed when the product does
   * not allow dynamic naming, or the product definition or product load
   * for a packaged load was renamed or moved to another library.
  **/
  public static final String LOAD_STATE_PACKAGED_RENAMED = "32";

  /**
   * Constant indicating that the product load object for this load has
   * been packaged with the Package Product Option (PKGPRDOPT) command
   * or the Package Product Option (QSZPKGPO) API, but an object was
   * found to be damaged the last time that the Check Product Option
   * (CHKPRDOPT) command or Save Licensed Program (SAVLICPGM) command
   * was used for this load.
  **/
  public static final String LOAD_STATE_PACKAGED_DAMAGED = "33";

  /**
   * Constant indicating that the product load object for this load has
   * been packaged with the Package Product Option (PKGPRDOPT) command
   * or the Package Product Option (QSZPKGPO) API, but either an attempt
   * was made to delete the product load using the Delete Licensed Program
   * (DLTLICPGM) command and the attempt failed, or a
   * packaged object was missing the last time the Check Product Option
   * (CHKPRDOPT) command or Save Licensed Program (SAVLICPGM) command was
   * used for this load.
  **/
  public static final String LOAD_STATE_PACKAGED_DELETED = "34";

  /**
   * Constant indicating that a Restore Licensed Program (RSTLICPGM) command
   * is in progress. The product being replaced had been packaged, but
   * not installed.
  **/
  public static final String LOAD_STATE_PACKAGED_RESTORE_IN_PROGRESS = "35";

  /**
   * Constant indicating that a Delete Licensed Program (DLTLICPGM) command
   * is in progress. The product being deleted had been packaged, but
   * not installed.
  **/
  public static final String LOAD_STATE_PACKAGED_DELETE_IN_PROGRESS = "38";

  /**
   * Constant indicating that a Restore Licensed Program (RSTLICPGM) command
   * is in progress. The product being replaced had been installed.
  **/
  public static final String LOAD_STATE_INSTALLED_RESTORE_IN_PROGRESS = "50";

  /**
   * Constant indicating that a Delete Licensed Program (DLTLICPGM) command
   * is in progress. The product being deleted had been installed.
  **/
  public static final String LOAD_STATE_INSTALLED_DELETE_IN_PROGRESS = "53";

  /**
   * Constant indicating that this product is an IBM-supplied product and
   * it is not compatible with the currently installed release level of the System i operating system.
   * An error occurred when the product was restored or when the operating system was installed.
   * The IBM-supplied product is at a release level earlier than V2R2M0, which
   * is not supported by the Save Licensed Program (SAVLICPGM) command.
  **/
  public static final String LOAD_STATE_IBM_SUPPLIED_NOT_COMPATIBLE = "59";

  /**
   * Constant indicating that a Restore Licensed Program (RSTLICPGM) command
   * did not complete successfully. A preoperation exit program failed. The
   * product being replaced had been installed.
  **/
  public static final String LOAD_STATE_INSTALLED_RESTORE_FAILED_EXIT_PROGRAM_FAILED = "6E";

  /**
   * Constant indicating that a Restore Licensed Program (RSTLICPGM) command
   * failed. The failure was not a preoperation exit program or postoperation
   * exit program. The product being replaced had been installed.
  **/
  public static final String LOAD_STATE_INSTALLED_RESTORE_FAILED = "6F";

  /**
   * Constant indicating that the product load object for this load was loaded
   * on to the system by the Restore Licensed Program (RSTLICPGM) command.
  **/
  public static final String LOAD_STATE_RESTORED = "60";

  /**
   * Constant indicating that the product load object for this load was loaded
   * on to the system by the Restore Licensed Program (RSTLICPGM) command but
   * a postoperation exit program failed.
  **/
  public static final String LOAD_STATE_RESTORED_EXIT_PROGRAM_FAILED = "61";

  /**
   * Constant indicating that an installed library or folder was renamed, but
   * the product does not allow dynamic naming.
  **/
  public static final String LOAD_STATE_RESTORED_RENAMED = "62";

  /**
   * Constant indicating that the product load object for this load was installed
   * by the Restore Licensed Program (RSTLICPGM) command, but an object is
   * damaged.
  **/
  public static final String LOAD_STATE_RESTORED_DAMAGED = "63";

  /**
   * Constant indicating that the product load object for this load was installed
   * by the Restore Licensed Program (RSTLICPGM) command, but either an object
   * was found to be missing when the Check Product Option (CHKPRDOPT) command
   * or the Save Licensed Program (SAVLICPGM) command was used, or an error
   * occurred while the Delete Licensed Program (DLTLICPGM) command was being
   * used.
  **/
  public static final String LOAD_STATE_RESTORED_DELETED = "64";

  /**
   * Constant indicating that the Check Product Option (CHKPRDOPT) command was
   * used for this product load, but the postoperation exit program failed or
   * indicated that an error was found.
  **/
  public static final String LOAD_STATE_CHECK_ERROR = "67";

  /**
   * Constant indicating that the product load was installed successfully. If an
   * object was missing or was damaged, but the problem was corrected, using
   * the Check Product Option (CHKPRDOPT) command sets the state back to this.
  **/
  public static final String LOAD_STATE_INSTALLED = "90";



  /**
   * Constant indicating a product load type of *CODE.
  **/
  public static final String LOAD_TYPE_CODE = "*CODE";

  /**
   * Constant indicating a product load type of *LNG.
  **/
  public static final String LOAD_TYPE_LANGUAGE = "*LNG";



  /**
   * Constant indicating a product option of *BASE.
  **/
  public static final String PRODUCT_OPTION_BASE = "*BASE";

  /**
   * Constant indicating a feature ID of *CODE.
  **/
  public static final String PRODUCT_FEATURE_CODE = "*CODE";

  /**
   * Constant indicating a product ID of *OPSYS.
  **/
  public static final String PRODUCT_ID_OPERATING_SYSTEM = "*OPSYS";

  /**
   * Constant indicating a release level of *CUR.
  **/
  public static final String PRODUCT_RELEASE_CURRENT = "*CUR";

  /**
   * Constant indicating a release level of *ONLY.
  **/
  public static final String PRODUCT_RELEASE_ONLY = "*ONLY";

  /**
   * Constant indicating a release level of *PRV.
  **/
  public static final String PRODUCT_RELEASE_PREVIOUS = "*PRV";

  /**
   * Constant indicating that the release level of the product
   * should be determined at runtime by the system.
  **/
  public static final String PRODUCT_RELEASE_ANY = "ANY";



  /**
   * Constant indicating that the registration type *PHONE was
   * specified when the product load or product definition was
   * created.
  **/
  public static final String REGISTRATION_TYPE_PHONE = "02";

  /**
   * Constant indicating that the registration type is the
   * same as the registration type for the operating system.
  **/
  public static final String REGISTRATION_TYPE_SYSTEM = "04";

  /**
   * Constant indicating that the registration type *CUSTOMER was
   * specified when the product load or product definition was
   * created.
  **/
  public static final String REGISTRATION_TYPE_CUSTOMER = "08";



  /**
   * Constant indicating that there is not enough information
   * available to determine if the release requirements have
   * been met. This will be the value if the load type is
   * LOAD_TYPE_LANGUAGE.
  **/
  public static final int REQUIREMENTS_UNKNOWN = 0;

  /**
   * Constant indicating that the releases of the *BASE and option
   * meet all requirements.
  **/
  public static final int REQUIREMENTS_MET = 1;

  /**
   * Constant indicating that the release of the option is too
   * old compared to the *BASE.
  **/
  public static final int REQUIREMENTS_OPTION_TOO_OLD = 2;

  /**
   * Constant indicating that the release of the *BASE is too
   * old compared to the option.
  **/
  public static final int REQUIREMENTS_BASE_TOO_OLD = 3;



  /**
   * Constant indicating that the load is defined but the product load
   * object for this load does not exist.
  **/
  public static final String SYMBOLIC_LOAD_STATE_DEFINED = "*DEFINED";

  /**
   * Constant indicating that the product load object for this load
   * exists. It must be packaged with the Package Product Option (PKGPRDOPT)
   * command before it can be saved using the Save Licensed Program (SAVLICPGM) command.
  **/
  public static final String SYMBOLIC_LOAD_STATE_CREATED = "*CREATED";

  /**
   * Constant indicating that the product load object for this load has
   * been packaged with the Package Product Option (PKGPRDOPT) command.
  **/
  public static final String SYMBOLIC_LOAD_STATE_PACKAGED = "*PACKAGED";

  /**
   * Constant indicating that either the product load object has been damaged
   * (if this option is something other than the base option or the load type
   * is a language load), or the product definition for this product ID and
   * release level has been damaged or the product load object has been damaged
   * (if this option is for the base option and code load type).
  **/
  public static final String SYMBOLIC_LOAD_STATE_DAMAGED = "*DAMAGED";

  /**
   * Constant indicating that either a Restore Licensed Program (RSTLICPGM)
   * function is in progress, a Delete Licensed Program (DLTLICPGM) function
   * is in progress, or the product was created previous to V2R2M0 and there
   * was an error during the process of converting product information.
  **/
  public static final String SYMBOLIC_LOAD_STATE_LOADED = "*LOADED";

  /**
   * Constant indicating that the product load object for this load was loaded
   * on to the system by the Restore Licensed Program (RSTLICPGM) command.
  **/
  public static final String SYMBOLIC_LOAD_STATE_INSTALLED = "*INSTALLED";



  /**
   * Constructs a Product object. The following default values are used:
   * 
    *
  • productOption - PRODUCT_OPTION_BASE *
  • releaseLevel - PRODUCT_RELEASE_ANY *
  • featureID - PRODUCT_FEATURE_CODE *
* @param system The system. * @param productID The product identifier. **/ public Product(AS400 system, String productID) { this(system, productID, PRODUCT_OPTION_BASE, PRODUCT_RELEASE_ANY, PRODUCT_FEATURE_CODE); } /** * Constructs a Product object. The following default values are used: *
    *
  • releaseLevel - PRODUCT_RELEASE_ANY *
  • featureID - PRODUCT_FEATURE_CODE *
* @param system The system. * @param productID The product identifier. * @param productOption The product option. **/ public Product(AS400 system, String productID, String productOption) { this(system, productID, productOption, PRODUCT_RELEASE_ANY, PRODUCT_FEATURE_CODE); } /** * Constructs a Product object. * @param system The system. * @param productID The product identifier. * @param productOption The product option. * @param releaseLevel The release level of the product. * @param featureID The product feature identifier. **/ public Product(AS400 system, String productID, String productOption, String releaseLevel, String featureID) { if (system == null) throw new NullPointerException("system"); if (productID == null) throw new NullPointerException("productID"); if (productOption == null) throw new NullPointerException("productOption"); if (releaseLevel == null) throw new NullPointerException("releaseLevel"); if (featureID == null) throw new NullPointerException("featureID"); String id = productID.toUpperCase().trim(); if (id.length() != 7 && !id.equals(PRODUCT_ID_OPERATING_SYSTEM)) { throw new ExtendedIllegalArgumentException("productID("+id+")", ExtendedIllegalArgumentException.LENGTH_NOT_VALID); } String option = productOption.toUpperCase().trim(); if (option.equals(PRODUCT_OPTION_BASE)) { option = "0000"; } if (option.length() > 4) { throw new ExtendedIllegalArgumentException(id+".productOption("+option+")", ExtendedIllegalArgumentException.LENGTH_NOT_VALID); } while (option.length() < 4) { option = "0"+option; } String level = releaseLevel.toUpperCase().trim(); if (level.length() != 6 && level.length() != 0 && !level.equals(PRODUCT_RELEASE_CURRENT) && !level.equals(PRODUCT_RELEASE_ONLY) && !level.equals(PRODUCT_RELEASE_PREVIOUS) && !level.equals(PRODUCT_RELEASE_ANY)) { throw new ExtendedIllegalArgumentException("releaseLevel", ExtendedIllegalArgumentException.LENGTH_NOT_VALID); } String load = featureID.toUpperCase().trim(); if (load.length() != 4 && !load.equals(PRODUCT_FEATURE_CODE)) { throw new ExtendedIllegalArgumentException("featureID", ExtendedIllegalArgumentException.LENGTH_NOT_VALID); } system_ = system; productID_ = id; productOption_ = option; releaseLevel_ = level; loadID_ = load; } /** * Constructs a Product object. Sets featureID to PRODUCT_FEATURE_CODE. * @param system The system. * @param productID The product identifier. * @param productOption The product option. * @param releaseLevel The release level of the product. * @param loadType The type of product load. * @param languageID The language feature ID for the product. **/ public Product(AS400 system, String productID, String productOption, String releaseLevel, String loadType, String languageID) { this(system, productID, productOption, releaseLevel, PRODUCT_FEATURE_CODE); if (loadType == null) throw new NullPointerException("loadType"); if (languageID == null) throw new NullPointerException("languageID"); loadType_ = loadType; primaryLanguageLoadID_ = languageID; } /** * This constructor is used by the 0500 format. **/ Product(AS400 system, String id, String option, String level, String feature, boolean allow, String msgID, String minVRM) { this(system, id, option, level, feature); allowsDynamicNaming_ = allow; descriptionID_ = msgID; minimumVRM_ = minVRM; loadedOptions_ = true; } /** * This constructor is used by ProductList. **/ Product(AS400 system, String id, String option, String level, String descriptionID, String descriptionText, String messageFile, boolean installed, boolean supported, String regType, String regValue) { system_ = system; productID_ = id; if (option.equals(PRODUCT_OPTION_BASE)) { option = "0000"; } else if (option.length() > 4) { option = option.substring(1, 5); // Chop off the first 0. } productOption_ = option; releaseLevel_ = level; descriptionID_ = descriptionID; descriptionText_ = descriptionText; messageFile_ = messageFile; installed_ = installed; supported_ = supported; registrationType_ = regType; registrationValue_ = regValue; partiallyLoaded_ = true; } /** * Indicates if the names of product libraries and root folders * for this product option can be dynamically changed without * causing a product error. * @return true if the product can by dynamically named, false if it cannot. **/ public boolean allowsDynamicNaming() throws AS400Exception, AS400SecurityException, ErrorCompletingRequestException, InterruptedException, IOException, ObjectDoesNotExistException { if (!loadedOptions_) fillInOptionInformation(); return allowsDynamicNaming_; } /** * Indicates if this product allows mixed releases between its *BASE * and options. * @return true if the *BASE option and other options of this product * can be at different release levels, false if they must all be at * the same release level. **/ public boolean allowsMixedReleases() throws AS400Exception, AS400SecurityException, ErrorCompletingRequestException, InterruptedException, IOException, ObjectDoesNotExistException { if (!loaded500_) refresh(500); return allowsMixedReleases_; } /** * Indicates if this product can be installed at a release level different * from the current release level without installing over a current release. * @return true if the product can be installed at a different release level, * false if it cannot be installed at a different release level without * installing over the current release. **/ public boolean allowsMultipleReleases() throws AS400Exception, AS400SecurityException, ErrorCompletingRequestException, InterruptedException, IOException, ObjectDoesNotExistException { if (!loaded500_) refresh(500); return allowsMultipleReleases_; } /** * Helper method. **/ private void fillInOptionInformation() throws AS400Exception, AS400SecurityException, ErrorCompletingRequestException, InterruptedException, IOException, ObjectDoesNotExistException { Product[] products = getProductOptions(); for (int i=0; i *
  • {@link #LOAD_STATE_DEFINED_NO_OBJECT LOAD_STATE_DEFINED_NO_OBJECT} *
  • {@link #LOAD_STATE_DEFINED_OBJECT_EXISTS LOAD_STATE_DEFINED_OBJECT_EXISTS} *
  • {@link #LOAD_STATE_PACKAGED_RESTORE_FAILED_EXIT_PROGRAM_FAILED LOAD_STATE_PACKAGED_RESTORE_FAILED_EXIT_PROGRAM_FAILED} *
  • {@link #LOAD_STATE_PACKAGED_RESTORE_FAILED LOAD_STATE_PACKAGED_RESTORE_FAILED} *
  • {@link #LOAD_STATE_PACKAGED LOAD_STATE_PACKAGED} *
  • {@link #LOAD_STATE_PACKAGED_RENAMED LOAD_STATE_PACKAGED_RENAMED} *
  • {@link #LOAD_STATE_PACKAGED_DAMAGED LOAD_STATE_PACKAGED_DAMAGED} *
  • {@link #LOAD_STATE_PACKAGED_DELETED LOAD_STATE_PACKAGED_DELETED} *
  • {@link #LOAD_STATE_PACKAGED_RESTORE_IN_PROGRESS LOAD_STATE_PACKAGED_RESTORE_IN_PROGRESS} *
  • {@link #LOAD_STATE_PACKAGED_DELETE_IN_PROGRESS LOAD_STATE_PACKAGED_DELETE_IN_PROGRESS} *
  • {@link #LOAD_STATE_INSTALLED_RESTORE_IN_PROGRESS LOAD_STATE_INSTALLED_RESTORE_IN_PROGRESS} *
  • {@link #LOAD_STATE_INSTALLED_DELETE_IN_PROGRESS LOAD_STATE_INSTALLED_DELETE_IN_PROGRESS} *
  • {@link #LOAD_STATE_IBM_SUPPLIED_NOT_COMPATIBLE LOAD_STATE_IBM_SUPPLIED_NOT_COMPATIBLE} *
  • {@link #LOAD_STATE_INSTALLED_RESTORE_FAILED_EXIT_PROGRAM_FAILED LOAD_STATE_INSTALLED_RESTORE_FAILED_EXIT_PROGRAM_FAILED} *
  • {@link #LOAD_STATE_INSTALLED_RESTORE_FAILED LOAD_STATE_INSTALLED_RESTORE_FAILED} *
  • {@link #LOAD_STATE_RESTORED LOAD_STATE_RESTORED} *
  • {@link #LOAD_STATE_RESTORED_EXIT_PROGRAM_FAILED LOAD_STATE_RESTORED_EXIT_PROGRAM_FAILED} *
  • {@link #LOAD_STATE_RESTORED_RENAMED LOAD_STATE_RESTORED_RENAMED} *
  • {@link #LOAD_STATE_RESTORED_DAMAGED LOAD_STATE_RESTORED_DAMAGED} *
  • {@link #LOAD_STATE_RESTORED_DELETED LOAD_STATE_RESTORED_DELETED} *
  • {@link #LOAD_STATE_CHECK_ERROR LOAD_STATE_CHECK_ERROR} *
  • {@link #LOAD_STATE_INSTALLED LOAD_STATE_INSTALLED} * * @return The load state. * @see #getSymbolicLoadState **/ public String getLoadState() throws AS400Exception, AS400SecurityException, ErrorCompletingRequestException, InterruptedException, IOException, ObjectDoesNotExistException { if (!loaded_) refresh(100); return loadState_; } /** * Returns the type of product load for which information was retrieved. * Possible values are: *
      *
    • {@link #LOAD_TYPE_CODE LOAD_TYPE_CODE} *
    • {@link #LOAD_TYPE_LANGUAGE LOAD_TYPE_LANGUAGE} *
    * @return The load type. **/ public String getLoadType() throws AS400Exception, AS400SecurityException, ErrorCompletingRequestException, InterruptedException, IOException, ObjectDoesNotExistException { if (!loaded_) refresh(100); return loadType_; } /** * Returns the minimum release level that is allowed for the option * that will run with the current level of the *BASE option for the product. * This method is only applicable if mixed releases are allowed. * Possible values are a release level (e.g. "V5R1M0") or "*MATCH" * which indicates the release of the option matches that of *BASE. * @return The minimum required release level of this product option. * @see #allowsMixedReleases * @see #getMinimumRequiredReleaseForBase **/ public String getMinimumRequiredRelease() throws AS400Exception, AS400SecurityException, ErrorCompletingRequestException, InterruptedException, IOException, ObjectDoesNotExistException { if (!loadedOptions_) fillInOptionInformation(); return minimumVRM_; } /** * Returns the minimum release level that is allowed for the *BASE option * that will run with the current level of the option for the product. This * method is only applicable if mixed releases are allowed and if the * load type is *CODE. Possible values are a release level (e.g. "V5R1M0") * or "*MATCH" which indicates the release of the option matches that * of *BASE. * @return The minimum required release level of the base product option. * @see #allowsMixedReleases * @see #getMinimumRequiredRelease **/ public String getMinimumRequiredReleaseForBase() throws AS400Exception, AS400SecurityException, ErrorCompletingRequestException, InterruptedException, IOException, ObjectDoesNotExistException { if (!loaded_) refresh(100); return minimumBaseVRM_; } /** * Returns the minimum operating system release to which the Save Licensed Program * (SAVLICPGM) command will allow the product to be saved. * @return The minimum target release (e.g. "V5R1M0"). **/ public String getMinimumTargetRelease() throws AS400Exception, AS400SecurityException, ErrorCompletingRequestException, InterruptedException, IOException, ObjectDoesNotExistException { if (!loaded_) refresh(100); return minimumTargetRelease_; } /** * Returns the primary language feature ID for this product. *

    * For code loads, this is the primary language of the product option; * that is, it is the National Language Version (NLV) of the language * that is installed in the libraries. It is "" if no language is * installed in the libraries for the code load. *

    * For language loads (e.g. "2938"), it is "". * @return The primary language feature ID, or "" if no language is * installed or this product is a language load. * @see #getLoadType **/ public String getPrimaryLanguageFeatureID() throws AS400Exception, AS400SecurityException, ErrorCompletingRequestException, InterruptedException, IOException, ObjectDoesNotExistException { if (!loaded_) refresh(100); return primaryLanguageLoadID_; } /** * Returns the product ID for this product. * @return The product ID (e.g. "5722JC1"). **/ public String getProductID() { return productID_; } /** * Returns the product option for this product. * @return The product option (e.g. "*BASE" or "0012"). **/ public String getProductOption() { return productOption_; } /** * Returns the list of product options for this product ID. * @return The array of products. **/ public Product[] getProductOptions() throws AS400Exception, AS400SecurityException, ErrorCompletingRequestException, InterruptedException, IOException, ObjectDoesNotExistException { if (!loaded500_) refresh(500); return options_; } /** * Returns the list of Program Temporary Fixes (PTFs) on * the system for this product. * @param includeSupersededPTFs Specify true to include any * superseded PTFs in the list; false otherwise. * @return The array of PTFs. If there are no PTFs for the * product, this method returns an array of size 0. **/ public PTF[] getPTFs(boolean includeSupersededPTFs) throws AS400Exception, AS400SecurityException, ErrorCompletingRequestException, InterruptedException, IOException, ObjectDoesNotExistException { if (!isInstalled()) return new PTF[0]; int ccsid = system_.getCcsid(); ConvTable conv = ConvTable.getTable(ccsid, null); ProgramParameter[] parms = new ProgramParameter[4]; try { parms[0] = new ProgramParameter(conv.stringToByteArray(USERSPACE_NAME)); parms[0].setParameterType(ProgramParameter.PASS_BY_REFERENCE); byte[] prodInfo = new byte[50]; AS400Text text4 = new AS400Text(4, ccsid, system_); AS400Text text6 = new AS400Text(6, ccsid, system_); AS400Text text7 = new AS400Text(7, ccsid, system_); AS400Text text10 = new AS400Text(10, ccsid, system_); text7.toBytes(productID_, prodInfo, 0); text6.toBytes(getReleaseLevel(), prodInfo, 7); text4.toBytes(productOption_, prodInfo, 13); if (loadID_ == null) refresh(100); text10.toBytes((loadID_.equals(PRODUCT_FEATURE_CODE) ? "*ALL" : loadID_), prodInfo, 17); //text10.toBytes("*ALL", prodInfo, 17); prodInfo[27] = includeSupersededPTFs ? (byte)0xF1 : (byte)0xF0; // '1' or '0' parms[1] = new ProgramParameter(prodInfo); parms[1].setParameterType(ProgramParameter.PASS_BY_REFERENCE); parms[2] = new ProgramParameter(conv.stringToByteArray("PTFL0100")); parms[2].setParameterType(ProgramParameter.PASS_BY_REFERENCE); parms[3] = new ProgramParameter(new byte[4]); // error code parms[3].setParameterType(ProgramParameter.PASS_BY_REFERENCE); } catch (PropertyVetoException pve) { // will never happen if (Trace.traceOn_) Trace.log(Trace.ERROR, pve); } ServiceProgramCall pc = new ServiceProgramCall(system_, "/QSYS.LIB/QPZLSTFX.SRVPGM", "QpzListPTF", ServiceProgramCall.NO_RETURN_VALUE, parms); // Note: The called API is not thread-safe. // Determine the needed scope of synchronization. Object lockObject; boolean willRunProgramsOnThread = pc.isStayOnThread(); if (willRunProgramsOnThread) { // The calls will run in the job of the JVM, so lock for entire JVM. lockObject = USERSPACE_PATH; } else { // The calls will run in the job of the Remote Command Host Server, so lock on the connection. lockObject = system_; } byte[] buf = null; synchronized(lockObject) { UserSpace us = new UserSpace(system_, "/QSYS.LIB/QTEMP.LIB/JT4PTF.USRSPC"); us.setMustUseProgramCall(true); if (!willRunProgramsOnThread) { us.setMustUseSockets(true); // Force the use of sockets when running natively but not on-thread. // We have to do it this way since UserSpace will otherwise make a // native ProgramCall, and will use a different QTEMP library than that used by the host server. } try { us.create(256*1024, true, "", (byte)0, "User space for PTF list", "*EXCLUDE"); if (!pc.run()) { AS400Message[] messages = pc.getMessageList(); if (messages.length == 1 && (messages[0].getID().equalsIgnoreCase("CPF6601") || // No PTF activity exists messages[0].getID().equalsIgnoreCase("CPF35BE"))) // Product not supported or installed { return new PTF[0]; } throw new AS400Exception(pc.getMessageList()); } int size = us.getLength(); buf = new byte[size]; us.read(buf, 0); } finally { // Delete the temporary user space, to allow other threads to re-create and use it. try { us.delete(); } catch (Exception e) { Trace.log(Trace.ERROR, "Exception while deleting temporary user space", e); } } } int startingOffset = BinaryConverter.byteArrayToInt(buf, 124); int numEntries = BinaryConverter.byteArrayToInt(buf, 132); int entrySize = BinaryConverter.byteArrayToInt(buf, 136); int entryCCSID = BinaryConverter.byteArrayToInt(buf, 140); conv = ConvTable.getTable(entryCCSID, null); int offset = 0; PTF[] ptfs = new PTF[numEntries]; for (int i=0; i *

  • {@link #REGISTRATION_TYPE_PHONE REGISTRATION_TYPE_PHONE} *
  • {@link #REGISTRATION_TYPE_SYSTEM REGISTRATION_TYPE_SYSTEM} *
  • {@link #REGISTRATION_TYPE_CUSTOMER REGISTRATION_TYPE_CUSTOMER} * * @return The registration type. * @see #getRegistrationValue **/ public String getRegistrationType() throws AS400Exception, AS400SecurityException, ErrorCompletingRequestException, InterruptedException, IOException, ObjectDoesNotExistException { if (!loaded_ && !partiallyLoaded_) refresh(100); return registrationType_; } /** * Returns the registration value associated with this product. The * registration type and registration value together make up the * registration ID for the product. * @return The registration value. * @see #getRegistrationType **/ public String getRegistrationValue() throws AS400Exception, AS400SecurityException, ErrorCompletingRequestException, InterruptedException, IOException, ObjectDoesNotExistException { if (!loaded_ && !partiallyLoaded_) refresh(100); return registrationValue_; } /** * Returns the value specified for the release date when the product * definition for this product load was created. If no release date * was specified for the product, then null is returned. * @return The release date, or null if there is no release date. **/ public Date getReleaseDate() throws AS400Exception, AS400SecurityException, ErrorCompletingRequestException, InterruptedException, IOException, ObjectDoesNotExistException { if (!loaded500_) refresh(500); return releaseDate_; } /** * Returns the release level for this product. For example: "V5R1M0". * If any of the special values were specified when this object was constructed, * the real release level will be retrieved from the system. * @return The release level. **/ public String getReleaseLevel() throws AS400Exception, AS400SecurityException, ErrorCompletingRequestException, InterruptedException, IOException, ObjectDoesNotExistException { if (releaseLevel_.equals(PRODUCT_RELEASE_CURRENT) || releaseLevel_.equals(PRODUCT_RELEASE_PREVIOUS) || releaseLevel_.equals(PRODUCT_RELEASE_ONLY) || releaseLevel_.equals(PRODUCT_RELEASE_ANY)) { refresh(100); } return releaseLevel_; } /** * Returns the reason why the release requirements between the base and * option may or may not be in error. (When a product allows mixed releases * between its base and option, certain requirements must be met). If the * load type is LOAD_TYPE_LANGUAGE, then this method returns REQUIREMENTS_UNKNOWN. * Possible values are: *
      *
    • {@link #REQUIREMENTS_UNKNOWN REQUIREMENTS_UNKNOWN} *
    • {@link #REQUIREMENTS_MET REQUIREMENTS_MET} *
    • {@link #REQUIREMENTS_OPTION_TOO_OLD REQUIREMENTS_OPTION_TOO_OLD} *
    • {@link #REQUIREMENTS_BASE_TOO_OLD REQUIREMENTS_BASE_TOO_OLD} *
    * @return The reason why requirements are met or not. * @see #getLoadType **/ public int getRequirementsMet() throws AS400Exception, AS400SecurityException, ErrorCompletingRequestException, InterruptedException, IOException, ObjectDoesNotExistException { if (!loaded_) refresh(100); return requirementsMet_; } /** * Returns the symbolic load state for which information was retrieved. * This value, in conjunction with the load error, can be used to determine * if the load is installed correctly. Possible values are: *
      *
    • {@link #SYMBOLIC_LOAD_STATE_DEFINED SYMBOLIC_LOAD_STATE_DEFINED} *
    • {@link #SYMBOLIC_LOAD_STATE_CREATED SYMBOLIC_LOAD_STATE_CREATED} *
    • {@link #SYMBOLIC_LOAD_STATE_PACKAGED SYMBOLIC_LOAD_STATE_PACKAGED} *
    • {@link #SYMBOLIC_LOAD_STATE_DAMAGED SYMBOLIC_LOAD_STATE_DAMAGED} *
    • {@link #SYMBOLIC_LOAD_STATE_LOADED SYMBOLIC_LOAD_STATE_LOADED} *
    • {@link #SYMBOLIC_LOAD_STATE_INSTALLED SYMBOLIC_LOAD_STATE_INSTALLED} *
    * @return The symbolic load state. * @see #getLoadState * @see #isLoadInError **/ public String getSymbolicLoadState() throws AS400Exception, AS400SecurityException, ErrorCompletingRequestException, InterruptedException, IOException, ObjectDoesNotExistException { return getSymbolicLoadState(true); // normal error processing } /** Utility method to enable retrieval of product load state without cluttering the job log with "Product information not found" messages. @param logErrorsInJobLog Whether errors are to be logged and reported normally. If false, errors are to be returned in the 'error code' parameter (and not logged in the job log). **/ private String getSymbolicLoadState(boolean logErrorsInJobLog) throws AS400Exception, AS400SecurityException, ErrorCompletingRequestException, InterruptedException, IOException, ObjectDoesNotExistException { if (!loaded_) refresh(100, logErrorsInJobLog); return symbolicLoadState_; } /** * Returns the system. * @return The system. **/ public AS400 getSystem() { return system_; } /** * Indicates whether or not this product is installed on the system. * @return true if the product is installed, false if it is not. **/ public boolean isInstalled() throws AS400Exception, AS400SecurityException, ErrorCompletingRequestException, InterruptedException, IOException, ObjectDoesNotExistException { if (partiallyLoaded_) return installed_; try { // First arg says: Don't clutter the job log with CPF0C1F "Product information not found" message (if the product isn't installed). return getSymbolicLoadState(false).equals(SYMBOLIC_LOAD_STATE_INSTALLED); } catch(AS400Exception e) { AS400Message[] messages = e.getAS400MessageList(); if (messages.length == 1 && messages[0].getID().equalsIgnoreCase("CPF0C1F")) // Product info not found { return false; } throw e; } } /** * Indicates if there is a known error for this product load. This does * not mean that product is necessarily installed. Check the symbolic load * state to determine if the product load is installed or not. * @return true if an error was found the last time that the state of * this load was checked or updated, false if no error was found. * @see #getSymbolicLoadState **/ public boolean isLoadInError() throws AS400Exception, AS400SecurityException, ErrorCompletingRequestException, InterruptedException, IOException, ObjectDoesNotExistException { if (!loaded_) refresh(100); return loadErrorIndicator_; } /** * Indicates whether this product feature is currently supported. A feature * can be supported by using the Work with Supported Products (WRKSPTPRD) * command in the System Manager. * @return true if the feature is supported, false if it is not. **/ public boolean isSupported() throws AS400Exception, AS400SecurityException, ErrorCompletingRequestException, InterruptedException, IOException, ObjectDoesNotExistException { if (!loaded_ && !partiallyLoaded_) refresh(100); return supported_; } /** * Refreshes the current values and settings for this Product by retrieving all of them from the system. * The getter methods implicitly refresh the necessary value if the value being sought has not been retrieved yet. * @return true if some or all of the values were successfully refreshed; false if the system found no information * for this product. **/ public boolean refresh() throws AS400Exception, AS400SecurityException, ErrorCompletingRequestException, InterruptedException, IOException, ObjectDoesNotExistException { loadedLoadID_ = false; // disregard any previously-loaded value // If no previous error with formats 0500 or 0100, get the format 0500 values. if (!error500_ && !error100_) refresh(500); // If no previous error with formats 0800 or 0100, get the format 0800 values. if (!error800_ && !error100_) refresh(800); // If there were errors with formats 0500 and 0800, and no errors with format 0100, get the format 0100 values. if (error500_ && error800_ && !error100_) refresh(100); // If no previous error with format 0500, get description text. if (!error500_) { loadedDescriptionText_ = false; getDescriptionText(); } fillInOptionInformation(); return !error100_; } /** * Does the real work based on the format requested. * Formats currently supported are 100, 500, and 800. Note that * all formats are supersets of the 100 format. **/ private void refresh(int whichFormat) throws AS400Exception, AS400SecurityException, ErrorCompletingRequestException, InterruptedException, IOException, ObjectDoesNotExistException { refresh(whichFormat, true); } /** * Does the real work based on the format requested. * Formats currently supported are 100, 500, and 800. Note that * all formats are supersets of the 100 format. * @param logErrorsInJobLog Whether errors are to be logged and reported normally. * If false, errors are to be returned in the 'error code' parameter (and not logged in the job log). **/ private void refresh(int whichFormat, boolean logErrorsInJobLog) throws AS400Exception, AS400SecurityException, ErrorCompletingRequestException, InterruptedException, IOException, ObjectDoesNotExistException { if (releaseLevel_.equals(PRODUCT_RELEASE_ANY)) { // First try ONLY, then PREVIOUS, then CURRENT, since there will // almost always be a product definition for CURRENT. error100_ = false; releaseLevel_ = PRODUCT_RELEASE_ONLY; try { refresh(whichFormat); return; } catch(AS400Exception x) {} if (error100_) { error100_ = false; releaseLevel_ = PRODUCT_RELEASE_PREVIOUS; try { refresh(whichFormat); return; } catch(AS400Exception x) {} if (error100_) { error100_ = false; releaseLevel_ = PRODUCT_RELEASE_CURRENT; refresh(whichFormat); return; } } } String format = null; int len = 0; switch(whichFormat) { case 100: if (error100_) return; // No point in trying again. format = "PRDR0100"; len = 108; break; case 500: if (error500_) return; // No point in trying again. format = "PRDR0500"; len = 108+49+chunkSize_; // Don't need all the info. break; case 800: if (error800_) return; // No point in trying again. format = "PRDR0800"; len = 108+17+chunkSize_; // In real life, len = 108+17 + (50+primaryPath+installPath+(10*numberOfObjectAuthorities)*numberOfEntries break; default: format = "PRDR0100"; whichFormat = 100; len = 108; break; } int ccsid = system_.getCcsid(); ConvTable conv = ConvTable.getTable(ccsid, null); ProgramParameter[] parms = new ProgramParameter[6]; parms[0] = new ProgramParameter(len); // receiver variable parms[1] = new ProgramParameter(BinaryConverter.intToByteArray(len)); // length of receiver variable parms[2] = new ProgramParameter(conv.stringToByteArray(format)); // format name byte[] productInfo = new byte[36]; AS400Text text4 = new AS400Text(4, ccsid, system_); AS400Text text6 = new AS400Text(6, ccsid, system_); AS400Text text7 = new AS400Text(7, ccsid, system_); AS400Text text10 = new AS400Text(10, ccsid, system_); text7.toBytes(productID_, productInfo, 0); text6.toBytes(releaseLevel_, productInfo, 7); String option = productOption_; if (option.equals(PRODUCT_OPTION_BASE) || whichFormat == 500) // PRDR0500 needs 0000 { option = "0000"; } text4.toBytes(option, productInfo, 13); text10.toBytes((loadID_ == null || whichFormat == 500) ? PRODUCT_FEATURE_CODE : loadID_, productInfo, 17); // PRDR0500 needs *CODE BinaryConverter.intToByteArray(36, productInfo, 28); BinaryConverter.intToByteArray(ccsid, productInfo, 32); parms[3] = new ProgramParameter(productInfo); // product information // 'error code' parameter if (logErrorsInJobLog) { parms[4] = new ErrorCodeParameter(); // normal error processing } else { parms[4] = new ErrorCodeParameter(true,false); // return any errors via the 'error code' parameter; don't clutter the job log } parms[5] = new ProgramParameter(conv.stringToByteArray("PRDI0200")); // product information format name ProgramCall pc = new ProgramCall(system_, "/QSYS.LIB/QSZRTVPR.PGM", parms); // Note: The called API is not thread-safe. boolean pcSucceeded = pc.run(); AS400Message[] messages = null; if (logErrorsInJobLog) // normal error processing { if (!pcSucceeded) { messages = pc.getMessageList(); // get errors from message list } } else { // If any errors occurred, they were returned in the 'error code' parm, rather than in the message list. ErrorCodeParameter errParm = (ErrorCodeParameter)parms[4]; String msgID = errParm.getMessageID(); if (msgID != null) { pcSucceeded = false; // Compose a message list, as if we had called the program with a normal 'error code' parm. try { String substData = errParm.getSubstitutionData(); MessageFile mf = new MessageFile(system_, "/QSYS.LIB/QCPFMSG.MSGF"); AS400Message msg = mf.getMessage(msgID, substData); messages = new AS400Message[1]; messages[0] = msg; } catch (Throwable t) { Trace.log(Trace.ERROR, "Error when retrieving error messages.", t); } } } if (!pcSucceeded) { if (messages.length == 1 && messages[0].getID().equalsIgnoreCase("CPF0C1F")) { // It's OK, there just wasn't a product definition. if (Trace.traceOn_) { Trace.log(Trace.DIAGNOSTIC, "No product definition found for "+toString()+" using format "+whichFormat+"."); } installed_ = false; if (whichFormat != 100) { refresh(100); if (whichFormat == 500) { releaseDate_ = null; firstCopyright_ = null; currentCopyright_ = null; options_ = new Product[0]; loaded500_ = true; error500_ = true; } else if (whichFormat == 800) { directories_ = new ProductDirectoryInformation[0]; loaded500_ = true; error800_ = true; } return; } else { // The 0500 format may work since it uses a different product option than what we are set to. // But the 0100 or 0800 formats may still fail since we switch back to use the option we were given. error100_ = true; } } throw new AS400Exception(messages); } byte[] outputData = parms[0].getOutputData(); int bytesReturned = BinaryConverter.byteArrayToInt(outputData, 0); int bytesAvailable = BinaryConverter.byteArrayToInt(outputData, 4); if (bytesReturned < bytesAvailable) { chunkSize_ = bytesAvailable; refresh(whichFormat); return; } releaseLevel_ = conv.byteArrayToString(outputData, 19, 6); if (whichFormat != 500) { // Since 0500 uses *CODE, we don't want to reset it. loadID_ = conv.byteArrayToString(outputData, 29, 4); loadedLoadID_ = true; } loadType_ = conv.byteArrayToString(outputData, 33, 10).trim(); symbolicLoadState_ = conv.byteArrayToString(outputData, 43, 10); loadErrorIndicator_ = !conv.byteArrayToString(outputData, 53, 10).trim().equals("*NONE"); // *ERROR for error, *NONE for not. loadState_ = conv.byteArrayToString(outputData, 63, 2); supported_ = conv.byteArrayToString(outputData, 65, 1).equals("1"); registrationType_ = conv.byteArrayToString(outputData, 66, 2); registrationValue_ = conv.byteArrayToString(outputData, 68, 14); primaryLanguageLoadID_ = conv.byteArrayToString(outputData, 88, 4); minimumTargetRelease_ = conv.byteArrayToString(outputData, 92, 6); minimumBaseVRM_ = conv.byteArrayToString(outputData, 98, 6); requirementsMet_ = (int)(outputData[104] & 0x000F); // 0xF0 = 0, 0xF1 = 1, etc... level_ = conv.byteArrayToString(outputData, 105, 3); loaded_ = true; // indicate that the above 0100 fields have been loaded if (whichFormat == 500) { int offset = BinaryConverter.byteArrayToInt(outputData, 84); allowsMultipleReleases_ = (outputData[offset++] == (byte)0xF1); // '1' is yes, '0' is no. boolean y2k = (outputData[offset++] == (byte)0xF1); // '1' is 20xx; '0' is 19xx String d = conv.byteArrayToString(outputData, offset, 6); Date releaseDate = null; if (d.trim().length() == 6) { Calendar cal = AS400Calendar.getGregorianInstance(); cal.clear(); cal.set(Integer.parseInt(d.substring(0,2)) + (y2k ? 2000 : 1900), // year Integer.parseInt(d.substring(2,4))-1, // month is zero based Integer.parseInt(d.substring(4,6))); // day releaseDate = cal.getTime(); } releaseDate_ = releaseDate; offset += 6; firstCopyright_ = conv.byteArrayToString(outputData, offset, 4); offset += 4; currentCopyright_ = conv.byteArrayToString(outputData, offset, 4).trim(); offset += 4; String fileName = conv.byteArrayToString(outputData, offset, 10).trim(); offset += 10; String fileLibrary = conv.byteArrayToString(outputData, offset, 10).trim(); offset += 10; if (fileName.length() > 0) { messageFile_ = QSYSObjectPathName.toPath(fileLibrary, fileName, "MSGF"); } else { messageFile_ = ""; } int numOptions = BinaryConverter.byteArrayToInt(outputData, offset); offset += 4; int entryLength = BinaryConverter.byteArrayToInt(outputData, offset); offset += 4; int offsetToOptions = BinaryConverter.byteArrayToInt(outputData, offset); offset += 4; allowsMixedReleases_ = (outputData[offset] == (byte)0xF1); // '1' allows, '0' doesn't. options_ = new Product[numOptions]; for (int i=0; i




  • © 2015 - 2025 Weber Informatics LLC | Privacy Policy