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

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

There is a newer version: 20.0.8
Show newest version
///////////////////////////////////////////////////////////////////////////////
//
// JTOpen (IBM Toolbox for Java - OSS version)
//
// Filename:  MemberDescription.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) 2009-2009 International Business Machines Corporation and
// others.  All rights reserved.
//
///////////////////////////////////////////////////////////////////////////////


package com.ibm.as400.access;

import java.io.IOException;
import java.io.UnsupportedEncodingException;
import java.text.ParseException;
import java.text.SimpleDateFormat;
import java.util.Date;
import java.util.HashMap;
import java.util.Map;

/**
 * Represents a database file member and its attributes.
 * MemberDescription objects can be constructed individually, or generated
 * by {@link MemberList#getMemberDescriptions MemberList.getMemberDescriptions()}.
 * A member's attributes can be retrieved by calling {@link #getValue getValue()}
 * and passing one of the attribute identifier constants defined in this class.
 * 

* Note: String data is always stored and returned without trailing blanks. *

* Performance hint: * If you anticipate retrieving multiple attributes for a given member, * first call {@link #refresh refresh()}, which will make a single API call * to retrieve (and cache) all of the member's attribute values. *

* Implementation note: * This class internally calls the "Retrieve Member Description" (QUSRMBRD) API. * Information from formats MBRD0100, MBRD0200, and MBRD0300 is retrieved. * *

* This class is mostly based on a prototype contributed by Mihael Schmidt. * * @see MemberList * @see AS400File * @see ObjectDescription **/ public class MemberDescription { private static final SimpleDateFormat dateTimeFormat_ = new SimpleDateFormat("yyMMddHHmmss"); private static final SimpleDateFormat dateFormat_ = new SimpleDateFormat("yyMMdd"); private static final String QUSRMBRD_FORMAT_100 = "MBRD0100"; private static final String QUSRMBRD_FORMAT_200 = "MBRD0200"; private static final String QUSRMBRD_FORMAT_300 = "MBRD0300"; /** The name of the database file whose member names are placed in the list. Attribute type: java.lang.String.*/ public static final int FILE_NAME = 1; /** The name of the library where the database file is located. Attribute type: java.lang.String. */ public static final int LIBRARY_NAME = 2; /** The name of a member found in the database file. Attribute type: java.lang.String. */ public static final int MEMBER_NAME = 3; /** The type of file found. Attribute type: java.lang.String. */ public static final int FILE_ATTRIBUTE = 4; /** The type of source member if this is a source file. Attribute type: java.lang.String. */ public static final int SOURCE_TYPE = 5; /** The date and time the member was created. Attribute type: java.util.Date.*/ public static final int CREATION_DATE_TIME = 6; /** * For source files, the date and time that this source member was last changed. * For SQL materialized query tables, the date and time that the last SQL Refresh * Table statement refreshed this member. Attribute type: java.util.Date (may be null). */ public static final int LAST_SOURCE_CHANGE_DATE = 7; /** Description of the member found in the database file. Attribute type: java.lang.String. */ public static final int MEMBER_TEXT_DESCRIPTION = 8; /** Whether the database file is a source file or a data file. Attribute type: java.lang.Boolean. */ public static final int SOURCE_FILE = 9; /** * Whether the database file is a remote file (true = remote file, false = local file). * Attribute type: java.lang.Boolean. */ public static final int REMOTE_FILE = 10; /** * Whether the database file is a logical or physical file (true = logical file, false = * physical file). Attribute type: java.lang.Boolean. */ public static final int LOGICAL_FILE = 11; /** * Whether the open data path (ODP) allows sharing with other programs in the same job * (true = ODP sharing is allowed, false = ODP sharing is not allowed). Attribute type: java.lang.Boolean. */ public static final int ODP_SHARING = 12; /** * The number of records that currently exist in this member. * A logical member returns the summarization of index entries. * If the requested physical file member is suspended, the value 0 is returned. * Attribute type: java.lang.Integer. */ public static final int CURRENT_NUMBER_OF_RECORDS = 13; /** * The size of the space that contains the data of the file member, in bytes. * A logical file returns a 0. Attribute type: java.lang.Integer. */ public static final int DATA_SPACE_SIZE = 14; /** * The access path size in bytes for this file member. If the file member is not keyed, * the value 0 is returned. DDM files, which are not from a System/38 or iSeries system, * return value 0. Attribute type: java.lang.Integer. */ public static final int ACCESS_PATH_SIZE = 15; /** * The number of database file members for the logical file member. If the member is a physical * file member, the value is 0. Attribute type: java.lang.Integer. */ public static final int NUMBER_OF_BASED_ON_PHYICAL_FILE_MEMBERS = 16; /** * The date and time this member was changed. The value contains null if the member was * never changed. Attribute type: java.util.Date. */ public static final int CHANGE_DATE_AND_TIME = 17; /** * The date and time that this member was last saved. The value contains null if the member * was never saved. Attribute type: java.util.Date. */ public static final int SAVE_DATE_AND_TIME = 18; /** * The date and time that the member was last restored. The value contains null if the member * was never restored. Attribute type: java.util.Date. */ public static final int RESTORE_DATE_AND_TIME = 19; /** The date that this member expires. Attribute type: java.util.Date. */ public static final int EXPIRATION_DATE = 20; /** * The number of days the member has been used. If the member does not have a last-used date, * the value 0 is returned. Attribute type: java.lang.Integer. */ public static final int NUMBER_OF_DAYS_USED = 21; /** * The century and date this member was last used. The value contains null if the member was * never used. Attribute type: java.util.Date. */ public static final int DATE_LAST_USED = 22; /** * The century and date when the days-used count was last set to 0. If the date is not available * the value contains null. Attribute type: java.util.Date. */ public static final int USE_RESET_DATE = 23; /** * The value to multiply the data space size by to get its true size. Typically this is 1, but * for large files, the value may be greater than 1. If the data space size multiplier is * greater than 1, then the value in the data space size field is not the actual size of the file. * Attribute type: java.lang.Integer. */ public static final int DATA_SPACE_SIZE_MULTIPLIER = 24; /** The value to multiply the access path size by to get its true size. Attribute type: java.lang.Integer. */ public static final int ACCESS_PATH_SIZE_MULTIPLIER = 25; /** The CCSID for the member text description. Attribute type: java.lang.Integer. */ public static final int MEMBER_TEXT_DESCRIPTION_CCSID = 26; /** * The number of deleted records returned in the file member. Keyed logical files return a 0. * DDM files that are not from a System/38 or iSeries system return a 0. If the requested * physical file member is suspended, the value 0 is returned. Attribute type: java.lang.Integer. */ public static final int NUMBER_OF_DELETED_RECORDS = 27; /** * Whether the member's logical file member combines (in one record format) fields from two or * more physical file members (true = join member, false = not a join member). * Attribute type: java.lang.Boolean. */ public static final int JOIN_MEMBER = 28; /** * Specifies, for files with key fields or join logical files, the type of access path * maintenance used for all members of the physical or logical file. Attribute type: java.lang.String. */ public static final int ACCESS_PATH_MAINTENANCE = 29; /** * The kind of SQL file type the file is. If the file isn't an SQL file, blank is returned. * Attribute type: java.lang.String. */ public static final int SQL_FILE_TYPE = 30; /** Whether records in the physical file can be read. Values: Y/N. Attribute type: java.lang.Boolean. */ public static final int ALLOW_READ_OPERATION = 31; /** Whether records can be written to the file. Values: Y/N. Attribute type: java.lang.Boolean. */ public static final int ALLOW_WRITE_OPERATION = 32; /** Whether records in this file can be updated. Values: Y/N. Attribute type: java.lang.Boolean. */ public static final int ALLOW_UPDATE_OPERATION = 33; /** Whether records in this file can be deleted. Values: Y/N. Attribute type: java.lang.Boolean. */ public static final int ALLOW_DELETE_OPERATION = 34; /** * The number of inserted, updated, or deleted records that are processed before the records are * forced into auxiliary storage. A 0 indicates that records are not forced into auxiliary storage. * Attribute type: java.lang.Integer. */ public static final int RECORDS_TO_FORCE_A_WRITE = 35; /** * The maximum allowed percentage of deleted records for each member in the physical file. The * percentage check is made when the member is closed. If the percentage of deleted records is * greater than the value shown, a message is sent to the history log. This field only applies * to physical files and is 0 when either no deleted records are allowed or the file is a * logical file. Attribute type: java.lang.Integer. */ public static final int MAXIMUM_PERCENT_DELETED_RECORDS_ALLOWED = 36; /** * The number of records that can be written to each member of the file before the member size is * automatically extended. This field applies only to physical files and is 0 for logical files. * Attribute type: java.lang.Integer. */ public static final int INITIAL_NUMBER_OF_RECORDS = 37; /** * The maximum number of records that are automatically added to the member when the number of * records in the member is greater than the initial member size. This field applies only to * physical files and is 0 for logical files. Attribute type: java.lang.Integer. */ public static final int INCREMENT_NUMBER_OF_RECORDS = 38; /** * The maximum number of increments automatically added to the member size. This field only * applies to physical files and is 0 for a logical file. Attribute type: java.lang.Integer. */ public static final int MAXIMUM_NUMBER_OF_INCREMENTS = 39; /** * The number of increments that have been added to the member size (data space size). This * field is 0 for logical files because the number of increments only applies to physical files. * Attribute type: java.lang.Integer. */ public static final int CURRENT_NUMBER_OF_INCREMENTS = 40; /** * The actual number of records this member can contain. The value is calculated by multiplying * the increment number of records by the maximum number of increments, and adding the initial * number of records. This field only applies to a physical file and is 0 for a logical file. * Attribute type: java.lang.Integer. */ public static final int RECORD_CAPACITY = 41; /** * The name of a record format selector program that is called when the logical file member * contains more than one logical record format. Attribute type: java.lang.String. */ public static final int RECORD_FORMAT_SELECTOR_PROGRAM_NAME = 42; /** * The library in which the record format selector program resides. This field is blank for * physical files. Attribute type: java.lang.String. */ public static final int RECORD_FORMAT_SELECTOR_LIBRARY_NAME = 43; private AS400 system_; private QSYSObjectPathName pathName_; private HashMap attributes_ = new HashMap(); private final AS400Bin4 intConverter_ = new AS400Bin4(); /** * Constructs an MemberDescription. * @param system The system. * @param path The fully-qualified integrated file system path to the database file member. * Consider using {@link QSYSObjectPathName QSYSObjectPathName} to compose * the fully-qualified path string, * or using {@link #MemberDescription(AS400,QSYSObjectPathName) MemberDescription(AS400,QSYSObjectPathName)} instead. **/ public MemberDescription(AS400 system, String path) { this(system, new QSYSObjectPathName(path)); } /** * Constructs a MemberDescription object. * @param system The system. * @param path The fully-qualified integrated file system path to the database file member. * Consider using {@link QSYSObjectPathName QSYSObjectPathName} to compose the fully-qualified path string. */ public MemberDescription(AS400 system, QSYSObjectPathName path) { if (system == null) throw new NullPointerException("system"); if (path == null) throw new NullPointerException("path"); system_ = system; pathName_ = path; } /** * Constructs a MemberDescription object. * * @param system AS400 system object * @param libraryName Library where the database file is located. * @param objectName The name of the database file. * @param memberName The name of the member within the database file. */ public MemberDescription(AS400 system, String libraryName, String objectName, String memberName) { this(system, new QSYSObjectPathName(libraryName, objectName, memberName, "MBR")); } /** * Constructs a MemberDescription given the specified path to the member with some preloaded * attributes. * * @param system The system. * @param path The fully-qualified integrated file system path to the member. * Consider using {@link QSYSObjectPathName QSYSObjectPathName} to compose the fully-qualified path string. * @param attributes Map with preloaded attributes * * @see MemberList */ MemberDescription(AS400 system, QSYSObjectPathName path, Map attributes) { this(system, path); attributes_.putAll(attributes); } /** * Determine the format to use on the API call depending on the attribute. * * @param attribute Attribute key * @return Format name to be used for the API call. */ private String lookupFormat(int attribute) { String format; switch (attribute) { case FILE_NAME: case LIBRARY_NAME: case MEMBER_NAME: case FILE_ATTRIBUTE: case SOURCE_TYPE: case CREATION_DATE_TIME: case LAST_SOURCE_CHANGE_DATE: case MEMBER_TEXT_DESCRIPTION: case SOURCE_FILE: format = QUSRMBRD_FORMAT_100; break; case REMOTE_FILE: case LOGICAL_FILE: case ODP_SHARING: case CURRENT_NUMBER_OF_RECORDS: case DATA_SPACE_SIZE: case ACCESS_PATH_SIZE: case NUMBER_OF_BASED_ON_PHYICAL_FILE_MEMBERS: case CHANGE_DATE_AND_TIME: case SAVE_DATE_AND_TIME: case RESTORE_DATE_AND_TIME: case EXPIRATION_DATE: case NUMBER_OF_DAYS_USED: case DATE_LAST_USED: case USE_RESET_DATE: case DATA_SPACE_SIZE_MULTIPLIER: case ACCESS_PATH_SIZE_MULTIPLIER: case MEMBER_TEXT_DESCRIPTION_CCSID: case NUMBER_OF_DELETED_RECORDS: format = QUSRMBRD_FORMAT_200; break; case JOIN_MEMBER: case ACCESS_PATH_MAINTENANCE: case SQL_FILE_TYPE: case ALLOW_READ_OPERATION: case ALLOW_WRITE_OPERATION: case ALLOW_UPDATE_OPERATION: case ALLOW_DELETE_OPERATION: case RECORDS_TO_FORCE_A_WRITE: case MAXIMUM_PERCENT_DELETED_RECORDS_ALLOWED: case INITIAL_NUMBER_OF_RECORDS: case INCREMENT_NUMBER_OF_RECORDS: case MAXIMUM_NUMBER_OF_INCREMENTS: case CURRENT_NUMBER_OF_INCREMENTS: case RECORD_CAPACITY: case RECORD_FORMAT_SELECTOR_PROGRAM_NAME: case RECORD_FORMAT_SELECTOR_LIBRARY_NAME: format = QUSRMBRD_FORMAT_300; break; default: if (Trace.traceOn_) Trace.log(Trace.WARNING, "Unrecognized attribute key:", attribute); format = QUSRMBRD_FORMAT_100; } return format; } /** * Returns the requested member attribute information object. If the value is not in the cache * it will be retrieved from the system. * * @param attribute Attribute to be retrieved (either from cache or from system) * * @return Requested member attribute * * @throws ObjectDoesNotExistException If a system object necessary for the call does not exist on the system. * @throws InterruptedException If this thread is interrupted. * @throws IOException If an error occurs while communicating with the system. * @throws ErrorCompletingRequestException If an error occurs before the request is completed. * @throws AS400SecurityException If a security or authority error occurs. * @throws AS400Exception If the program on the server sends an escape message. */ public Object getValue(int attribute) throws AS400Exception, AS400SecurityException, ErrorCompletingRequestException, IOException, InterruptedException, ObjectDoesNotExistException { final Integer key = Integer.valueOf(attribute); Object value = null; value = attributes_.get(key); // If the value has not yet been retrieved from the server, the value object is still null. // Retrieve it from the server and set the value object at last. if (value == null) { value = retrieve(attribute); // add value to the cache attributes_.put(key, value); } return value; } /** * Retrieves (and caches) all attributes of this member from the system. * * @throws ObjectDoesNotExistException If a system object necessary for the call does not exist on the system. * @throws InterruptedException If this thread is interrupted. * @throws IOException If an error occurs while communicating with the system. * @throws ErrorCompletingRequestException If an error occurs before the request is completed. * @throws AS400SecurityException If a security or authority error occurs. * @throws AS400Exception If the program on the server sends an escape message. **/ public void refresh() throws AS400Exception, AS400SecurityException, ErrorCompletingRequestException, InterruptedException, IOException, ObjectDoesNotExistException { retrieve(RECORD_FORMAT_SELECTOR_LIBRARY_NAME); } /** * Sets the attribute of the member. Any previous information of the same attribute will be * replaced. * * @param attribute Attribute key (Integer) * @param value member information object to be set */ void setAttribute(int attribute, Object value) { attributes_.put(Integer.valueOf(attribute), value); } /** * This method makes the actual call to the server and gets the information about the member. * All other member information which belong to the same format are also retrieved. * * @param attribute Attribute to be retrieved and returned * * @return The value of the attribute * * @throws ObjectDoesNotExistException If a system object necessary for the call does not exist on the system. * @throws InterruptedException If this thread is interrupted. * @throws IOException If an error occurs while communicating with the system. * @throws ErrorCompletingRequestException If an error occurs before the request is completed. * @throws AS400SecurityException If a security or authority error occurs. * @throws AS400Exception If the program on the server sends an escape message. */ private Object retrieve(int attribute) throws AS400Exception, AS400SecurityException, ErrorCompletingRequestException, IOException, InterruptedException, ObjectDoesNotExistException { final String format = lookupFormat(attribute); ProgramParameter[] parameters = buildProgramParameters(format); ProgramCall program = new ProgramCall(system_, "/QSYS.LIB/QUSRMBRD.PGM", parameters); // this API is not threadsafe if (program.run()) { readMemberInfo(parameters[0].getOutputData(), format); return attributes_.get(Integer.valueOf(attribute)); } else { throw new AS400Exception(program.getMessageList()); } } /** * Builds the program parameter list for a program call to QUSRMBRD (Retrieve Member Description). * * @return Program parameter list for QUSRMBRD */ private ProgramParameter[] buildProgramParameters(String format) throws UnsupportedEncodingException { Converter charConverter = new Converter(system_.getCcsid(), system_); ProgramParameter[] parameterList = new ProgramParameter[7]; // Receiver variable: parameterList[0] = new ProgramParameter(332); // Length of receiver variable: parameterList[1] = new ProgramParameter(intConverter_.toBytes(332)); // Format name: parameterList[2] = new ProgramParameter(charConverter.stringToByteArray(format)); // Qualified database file name: parameterList[3] = new ProgramParameter(charConverter.stringToByteArray(pathName_.toQualifiedObjectName())); // Database member name: String memberName = pathName_.getMemberName(); if (memberName.length() == 0) { memberName = "*FIRST"; if (Trace.traceOn_) Trace.log(Trace.DIAGNOSTIC, "Defaulting to member *FIRST", pathName_.getPath()); } parameterList[4] = new ProgramParameter(new AS400Text(10,system_.getCcsid()).toBytes(memberName)); //@AF9C // Override processing: (1 == "overrides are processed") parameterList[5] = new ProgramParameter(charConverter.stringToByteArray("1")); // Error code: parameterList[6] = new ErrorCodeParameter(); return parameterList; } // Returns the offset of the specified field, in the MBRD* structures. private static final int offsetOf(int attribute) { switch (attribute) { // Fields in format MBRD0100: case FILE_NAME: return 8; case LIBRARY_NAME: return 18; case MEMBER_NAME: return 28; case FILE_ATTRIBUTE: return 38; case SOURCE_TYPE: return 48; case CREATION_DATE_TIME: return 58; case LAST_SOURCE_CHANGE_DATE: return 71; case MEMBER_TEXT_DESCRIPTION: return 84; case SOURCE_FILE: return 134; // Additional fields in format MBRD0200: case REMOTE_FILE: return 135; case LOGICAL_FILE: return 136; case ODP_SHARING: return 137; case CURRENT_NUMBER_OF_RECORDS: return 140; // offset 144: number of deleted records case DATA_SPACE_SIZE: return 148; case ACCESS_PATH_SIZE: return 152; case NUMBER_OF_BASED_ON_PHYICAL_FILE_MEMBERS: return 156; case CHANGE_DATE_AND_TIME: return 160; case SAVE_DATE_AND_TIME: return 173; case RESTORE_DATE_AND_TIME: return 186; case EXPIRATION_DATE: return 199; // offset 206: reserved case NUMBER_OF_DAYS_USED: return 212; case DATE_LAST_USED: return 216; case USE_RESET_DATE : return 223; // offset 230: reserved case DATA_SPACE_SIZE_MULTIPLIER: return 232; case ACCESS_PATH_SIZE_MULTIPLIER: return 236; case MEMBER_TEXT_DESCRIPTION_CCSID: return 240; case NUMBER_OF_DELETED_RECORDS: return 256; // Additional fields in format MBRD0300: case JOIN_MEMBER: return 266; case ACCESS_PATH_MAINTENANCE: return 267; case SQL_FILE_TYPE: return 268; // offset 278: reserved case ALLOW_READ_OPERATION: return 279; case ALLOW_WRITE_OPERATION: return 280; case ALLOW_UPDATE_OPERATION: return 281; case ALLOW_DELETE_OPERATION: return 282; // offset 283: reserved case RECORDS_TO_FORCE_A_WRITE: return 284; case MAXIMUM_PERCENT_DELETED_RECORDS_ALLOWED: return 288; case INITIAL_NUMBER_OF_RECORDS: return 292; case INCREMENT_NUMBER_OF_RECORDS: return 296; case MAXIMUM_NUMBER_OF_INCREMENTS: return 300; case CURRENT_NUMBER_OF_INCREMENTS: return 304; case RECORD_CAPACITY: return 308; case RECORD_FORMAT_SELECTOR_PROGRAM_NAME: return 312; case RECORD_FORMAT_SELECTOR_LIBRARY_NAME: return 322; default: Trace.log(Trace.ERROR, "Unrecognized attribute key:", attribute); throw new InternalErrorException(InternalErrorException.UNKNOWN, attribute); } } /** * Reads all available information from the output (receiver) bytes regardless of the attribute to * be retrieved. The attribute to be retrieved determines in which content and format the data is sent. * * @param entryBytes Raw bytes of the entry * @param format Content and format name * * @throws UnsupportedEncodingException If the ccsid is not supported. */ private void readMemberInfo(byte[] entryBytes, String format) throws UnsupportedEncodingException { final CharConverter charConverter = new CharConverter(system_.getCcsid(), system_); // Fields returned in all formats: setAttribute(FILE_NAME, charConverter.byteArrayToString(entryBytes, offsetOf(FILE_NAME), 10).trim()); setAttribute(LIBRARY_NAME, charConverter.byteArrayToString(entryBytes, offsetOf(LIBRARY_NAME), 10).trim()); setAttribute(FILE_ATTRIBUTE, charConverter.byteArrayToString(entryBytes, offsetOf(FILE_ATTRIBUTE), 10).trim()); setAttribute(MEMBER_NAME, charConverter.byteArrayToString(entryBytes, offsetOf(MEMBER_NAME), 10).trim()); setAttribute(SOURCE_TYPE, charConverter.byteArrayToString(entryBytes, offsetOf(SOURCE_TYPE), 10).trim()); setAttribute(MEMBER_TEXT_DESCRIPTION, charConverter.byteArrayToString(entryBytes, offsetOf(MEMBER_TEXT_DESCRIPTION), 50).trim()); setAttribute(LAST_SOURCE_CHANGE_DATE, // Note: This field can contain hex zeros. transformDate(charConverter.byteArrayToString(entryBytes, offsetOf(LAST_SOURCE_CHANGE_DATE), 13))); setAttribute(CREATION_DATE_TIME, transformDate(charConverter.byteArrayToString(entryBytes, offsetOf(CREATION_DATE_TIME), 13))); setAttribute(SOURCE_FILE, transformBoolean(charConverter.byteArrayToString(entryBytes, offsetOf(SOURCE_FILE), 1))); // Fields returned in both formats 200 and 300: if (format != QUSRMBRD_FORMAT_100) { setAttribute(REMOTE_FILE, transformBoolean(charConverter.byteArrayToString(entryBytes, offsetOf(REMOTE_FILE), 1))); setAttribute(LOGICAL_FILE, transformBoolean(charConverter.byteArrayToString(entryBytes, offsetOf(LOGICAL_FILE), 1))); setAttribute(CURRENT_NUMBER_OF_RECORDS, Integer.valueOf(intConverter_.toInt(entryBytes, offsetOf(CURRENT_NUMBER_OF_RECORDS)))); setAttribute(ODP_SHARING, transformBoolean(charConverter.byteArrayToString(entryBytes, offsetOf(ODP_SHARING), 1))); setAttribute(DATA_SPACE_SIZE, Integer.valueOf(intConverter_.toInt(entryBytes, offsetOf(DATA_SPACE_SIZE)))); setAttribute(ACCESS_PATH_SIZE, Integer.valueOf(intConverter_.toInt(entryBytes, offsetOf(ACCESS_PATH_SIZE)))); setAttribute(NUMBER_OF_BASED_ON_PHYICAL_FILE_MEMBERS, Integer.valueOf(intConverter_.toInt(entryBytes, offsetOf(NUMBER_OF_BASED_ON_PHYICAL_FILE_MEMBERS)))); setAttribute(CHANGE_DATE_AND_TIME, transformDate(charConverter.byteArrayToString(entryBytes, offsetOf(CHANGE_DATE_AND_TIME), 13))); setAttribute(RESTORE_DATE_AND_TIME, transformDate(charConverter.byteArrayToString(entryBytes, offsetOf(RESTORE_DATE_AND_TIME), 13))); setAttribute(SAVE_DATE_AND_TIME, transformDate(charConverter.byteArrayToString(entryBytes, offsetOf(SAVE_DATE_AND_TIME), 13))); setAttribute(EXPIRATION_DATE, transformDate(charConverter.byteArrayToString(entryBytes, offsetOf(EXPIRATION_DATE), 13))); setAttribute(NUMBER_OF_DAYS_USED, Integer.valueOf(intConverter_.toInt(entryBytes, offsetOf(NUMBER_OF_DAYS_USED)))); setAttribute(DATE_LAST_USED, transformDate(charConverter.byteArrayToString(entryBytes, offsetOf(DATE_LAST_USED), 13))); setAttribute(USE_RESET_DATE, transformDate(charConverter.byteArrayToString(entryBytes, offsetOf(USE_RESET_DATE), 13))); setAttribute(DATA_SPACE_SIZE_MULTIPLIER, Integer.valueOf(intConverter_.toInt(entryBytes, offsetOf(DATA_SPACE_SIZE_MULTIPLIER)))); setAttribute(ACCESS_PATH_SIZE_MULTIPLIER, Integer.valueOf(intConverter_.toInt(entryBytes, offsetOf(ACCESS_PATH_SIZE_MULTIPLIER)))); setAttribute(MEMBER_TEXT_DESCRIPTION_CCSID, Integer.valueOf(intConverter_.toInt(entryBytes, offsetOf(MEMBER_TEXT_DESCRIPTION_CCSID)))); setAttribute(NUMBER_OF_DELETED_RECORDS, Integer.valueOf(intConverter_.toInt(entryBytes, offsetOf(NUMBER_OF_DELETED_RECORDS)))); } // Fields returned only in format 300: if (QUSRMBRD_FORMAT_300.equals(format)) { setAttribute(JOIN_MEMBER, transformBoolean(charConverter.byteArrayToString(entryBytes, offsetOf(JOIN_MEMBER), 1))); setAttribute(ACCESS_PATH_MAINTENANCE, charConverter.byteArrayToString(entryBytes, offsetOf(ACCESS_PATH_MAINTENANCE), 1)); setAttribute(SQL_FILE_TYPE, charConverter.byteArrayToString(entryBytes, offsetOf(SQL_FILE_TYPE), 10).trim()); setAttribute(ALLOW_READ_OPERATION, transformBoolean(charConverter.byteArrayToString(entryBytes, offsetOf(ALLOW_READ_OPERATION), 1))); setAttribute(ALLOW_WRITE_OPERATION, transformBoolean(charConverter.byteArrayToString(entryBytes, offsetOf(ALLOW_WRITE_OPERATION), 1))); setAttribute(ALLOW_UPDATE_OPERATION, transformBoolean(charConverter.byteArrayToString(entryBytes, offsetOf(ALLOW_UPDATE_OPERATION), 1))); setAttribute(ALLOW_DELETE_OPERATION, transformBoolean(charConverter.byteArrayToString(entryBytes, offsetOf(ALLOW_DELETE_OPERATION), 1))); setAttribute(RECORDS_TO_FORCE_A_WRITE, Integer.valueOf(intConverter_.toInt(entryBytes, offsetOf(RECORDS_TO_FORCE_A_WRITE)))); setAttribute(MAXIMUM_PERCENT_DELETED_RECORDS_ALLOWED, Integer.valueOf(intConverter_.toInt(entryBytes, offsetOf(MAXIMUM_PERCENT_DELETED_RECORDS_ALLOWED)))); setAttribute(INITIAL_NUMBER_OF_RECORDS, Integer.valueOf(intConverter_.toInt(entryBytes, offsetOf(INITIAL_NUMBER_OF_RECORDS)))); setAttribute(INCREMENT_NUMBER_OF_RECORDS, Integer.valueOf(intConverter_.toInt(entryBytes, offsetOf(INCREMENT_NUMBER_OF_RECORDS)))); setAttribute(MAXIMUM_NUMBER_OF_INCREMENTS, Integer.valueOf(intConverter_.toInt(entryBytes, offsetOf(MAXIMUM_NUMBER_OF_INCREMENTS)))); setAttribute(CURRENT_NUMBER_OF_INCREMENTS, Integer.valueOf(intConverter_.toInt(entryBytes, offsetOf(CURRENT_NUMBER_OF_INCREMENTS)))); setAttribute(RECORD_CAPACITY, Integer.valueOf(intConverter_.toInt(entryBytes, offsetOf(RECORD_CAPACITY)))); setAttribute(RECORD_FORMAT_SELECTOR_PROGRAM_NAME, charConverter.byteArrayToString(entryBytes, offsetOf(RECORD_FORMAT_SELECTOR_PROGRAM_NAME), 10).trim()); setAttribute(RECORD_FORMAT_SELECTOR_LIBRARY_NAME, charConverter.byteArrayToString(entryBytes, offsetOf(RECORD_FORMAT_SELECTOR_LIBRARY_NAME), 10).trim()); } } /** * Parses the date string which should be in the format CYYMMDDHHMMSS or CYYMMDD; * * @param dateString Date string * * @return Date object representing the passed date string or null if the * date string is null, empty or not parseable. */ private Date transformDate(String dateString) { Date retVal = null; if (dateString == null) { // nothing => returns null } else { dateString = dateString.trim(); if (dateString.length() == 0) { // nothing => returns null } else { try { if (dateString.length() == 13) { synchronized(dateTimeFormat_) { retVal = dateTimeFormat_.parse(dateString.substring(1)); // skip the century digit } } else if (dateString.length() == 7) { synchronized(dateFormat_) { retVal = dateFormat_.parse(dateString.substring(1)); // skip the century digit } } else { if (Trace.traceOn_) Trace.log(Trace.WARNING, "Date string has unrecognized format:", dateString); } } catch(ParseException pe) { // return null if (Trace.traceOn_) Trace.log(Trace.ERROR, "Ignored error while parsing date string: " + dateString, pe); } } } return retVal; } /** * Transforms the value of 1 to true and everything else to false. * * @param value * @return Boolean object representing 1 as true and everything else as false */ private Boolean transformBoolean(String value) { // Assume that the caller has verified that the argument is non-null. if (value.equals("1")) { return Boolean.TRUE; } else { return Boolean.FALSE; } } /** * Returns the IFS path of this member. * * @return IFS path to this member. */ public String getPath() { return pathName_.getPath(); } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy