src.com.ibm.as400.access.AS400FileImplRemote Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of jt400 Show documentation
Show all versions of jt400 Show documentation
The Open Source version of the IBM Toolbox for Java
///////////////////////////////////////////////////////////////////////////////
//
// JTOpen (IBM Toolbox for Java - OSS version)
//
// Filename: AS400FileImplRemote.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.
//
///////////////////////////////////////////////////////////////////////////////
//
// @A2 - 09/19/2007 - The NULL Field Byte Map array must be set based on the
// maximum number of fields for any format in a given file. Refer to
// more detailed explanation for the corresponding changes in
// DDMObjectDataStream.java (@A1)
//
///////////////////////////////////////////////////////////////////////////////
package com.ibm.as400.access;
import java.beans.PropertyVetoException;
import java.io.IOException;
import java.io.ObjectInputStream;
import java.io.Serializable;
import java.net.UnknownHostException;
import java.util.Vector;
import java.math.BigDecimal; //@D0A 7/15/99
import java.lang.Math; //@A2A
class AS400FileImplRemote extends AS400FileImplBase implements Serializable //@C0C
{
static final long serialVersionUID = 4L;
//-------------------------------------------------------
// VARIABLES
//-------------------------------------------------------
// Flag indicating to ignore replys (used by close and execute).
//@D1D (moved to AS400FileImplBase) private boolean discardReplys_ = false;
// The declared file name. This is an alias for the file which allows
// DDM to process some file requests more quickly. The declared file name is
// determined upon construction and set when the file is opened.
byte[] dclName_ = new byte[8];
// Static variable used to ensure that each AS400File object has a unique
// declared file name.
static long nextDCLName_ = 1;
static final Object nextDCLNameLock_ = new Object();
// S38BUF data for force end of data calls. Contain S38BUF LL, CP and value.
// @F1 - also use this for locking
static private byte[] s38Buffer = {0x00, 0x05, (byte)0xD4, 0x05, 0x00};
// Server
transient AS400Server server_ = null;
// @B1A
private static int lastCorrelationId_ = 0; //@B6C
private static Object correlationIdLock_ = new Object(); //@B6A
// Identify the DDM reply data streams to the AS400Server class.
static
{
AS400Server.addReplyStream(new DDMObjectDataStream(), AS400.RECORDACCESS); //@B5C
AS400Server.addReplyStream(new DDMReplyDataStream(), AS400.RECORDACCESS); //@B5C
}
// Instantiate a default AS400File remote implementation object.
public AS400FileImplRemote()
{
setDCLName();
}
/**
*Closes the file on the server. All file locks held by this connection
*are released. All
*uncommitted transactions against the file are rolled back if commitment
*control has been started.
*@exception AS400Exception If the server returns an error message.
*@exception AS400SecurityException If a security or authority error occurs.
*@exception InterruptedException If this thread is interrupted.
*@exception IOException If an error occurs while communicating with the
*server.
**/
public void close()
throws AS400Exception,
AS400SecurityException,
InterruptedException,
IOException
{
super.close(); //@C0A
// Send the close file data stream request
if (discardReplys_ && (server_ != null))
{
synchronized(server_) //@F1A
{ //@F1A
server_.sendAndDiscardReply(DDMRequestDataStream.getRequestS38CLOSE(dclName_),newCorrelationId());//@M8C
} //@F1A
}
else
{
Vector replys = sendRequestAndReceiveReplies(DDMRequestDataStream.getRequestS38CLOSE(dclName_), /*server_.*/newCorrelationId()); // @B1M
// Reply expected: S38MSGRM, severity code 0
if (!(replys.size() == 1 && verifyS38MSGRM((DDMDataStream)replys.elementAt(0), null, 0)))
{
handleErrorReply(replys, 0);
}
}
}
/**
*Commits all transactions since the last commit boundary. Invoking this
*method will cause all transactions under commitment control for this
*connection to be committed.
*@exception AS400Exception If the server returns an error message.
*@exception AS400SecurityException If a security or authority error occurs.
*@exception ConnectionDroppedException If the connection is dropped
*unexpectedly.
*@exception InterruptedException If this thread is interrupted.
*@exception IOException If an error occurs while communicating with the
*server.
*@exception ServerStartupException If the host server cannot be started.
*@exception UnknownHostException If the server cannot be located.
**/
public void commit()
throws AS400Exception,
AS400SecurityException,
InterruptedException,
IOException
{
// Connect to the server. Note: If we have already connected, that connection
// will be used.
connect();
// Send the commit data stream request
Vector replys = sendRequestAndReceiveReplies(DDMRequestDataStream.getRequestCMMUOW(), newCorrelationId()); //@B6C
// Reply expected: ENDUOWRM, with UOWDSP parameter = 1
if (replys.size() == 1 && ((DDMDataStream)replys.elementAt(0)).getCodePoint() == DDMTerm.ENDUOWRM)
{
// Check the UOWDSP term value
DDMEndUnitOfWorkReply uowReply = new DDMEndUnitOfWorkReply(((DDMDataStream)replys.elementAt(0)).data_);
if (uowReply.getStatus() != 0x01)
{
// Status of logical unit of work committed not returned; should not happen unless
// we are constructing the request wrong.
Trace.log(Trace.ERROR, "Wrong status returned from commit ds", uowReply.data_);
throw new InternalErrorException(InternalErrorException.UNKNOWN, uowReply.getStatus());
}
}
else
{
handleErrorReply(replys, 0);
}
}
/**
*Connects to the server. The name and system must have been
*set at this point via the constructor or the setters.
*@exception AS400SecurityException If a security or authority error occurs.
*@exception ConnectionDroppedException If the connection is dropped
*unexpectedly.
*@exception IOException If an error occurs while communicating with the
*server.
*@exception InterruptedException If this thread is interrupted.
*@exception ServerStartupException If the host server cannot be started.
*@exception UnknownHostException If the server cannot be located.
**/
public void connect()
throws AS400SecurityException, ConnectionDroppedException, IOException,
InterruptedException, ServerStartupException, UnknownHostException
{
if (server_ == null)
{
server_ = system_.getConnection(AS400.RECORDACCESS,
false, /* force new connection */
false /* skip signon server */ ); //@C0C @B5C
}
}
//@D0M 7/15/99 - Moved this code out of ImplBase to here
/**
*Creates the DDS source file to be used to create a physical file based on a user
*supplied RecordFormat.
*The name of the file and the server to which to connect must be set prior
*to invoking this method.
*@see AS400File#AS400File(com.ibm.as400.access.AS400, java.lang.String)
*@see AS400File#setPath
*@see AS400File#setSystem
*@param recordFormat The record format to describe in the DDS source file.
*@param altSeq The value to be specified for the file-level keyword ALTSEQ. If no
*value is to be specified, null may be specified.
*@param ccsid The value to be specified for the file-level keyword CCSID. If no
*value is to be specified, null may be specified.
*@param order The value to be specified to indicate in which order records are to be
*retrieved from the file. Valid values are one of the following file-level keywords:
*
*- FIFO - First in, first out
*
- LIFO - Last in, first out
*
- FCFO - First changed, first out
*
*If no ordering value is to be specified, null may be specified.
*@param ref The value to be specified for the file-level keyword REF. If no
*value is to be specified, null may be specified.
*@param unique Indicates if the file-level keyword UNIQUE is to be specified. True
*indicates that the UNIQUE keyword should be specified; false indicates that it
*should not be specified.
*@param format The value to be specified for the record-level keyword FORMAT. If no
*value is to be specified, null may be specified.
*@param text The value to be specified for the record-level keyword TEXT. If no
*value is to be specified, null may be specified.
*@exception AS400Exception If the server returns an error message.
*@exception AS400SecurityException If a security or authority error occurs.
*@exception ConnectionDroppedException If the connection is dropped unexpectedly.
*@exception InterruptedException If this thread is interrupted.
*@exception IOException If an error occurs while communicating with the server.
*@exception ServerStartupException If the host server cannot be started.
*@exception UnknownHostException If the server cannot be located.
**/
public synchronized void createDDSSourceFile(RecordFormat recordFormat,
String altSeq,
String ccsid,
String order,
String ref,
boolean unique,
String format,
String text)
throws AS400Exception, AS400SecurityException, InterruptedException, IOException
{
// Create the source physical file to hold the DDS source. Note that we create the
// file in library QTEMP. Each server job has its own QTEMP library which is created
// when the job starts and is deleted when the job ends. Using QTEMP allows
// the file to be created regardless of the user's authority and also eliminates
// name collision problems when different jobs are creating files from a record
// format.
AS400Message[] msgs = execute("QSYS/CRTSRCPF FILE(QTEMP/JT400DSSRC) RCDLEN(92) MBR(*FILE) TEXT('JT400 TEMPORARY DDS SOURCE FILE')"); //@B0C @F0C
if (msgs.length > 0)
{
if (msgs[0].getID().equals("CPF5813"))
{
// File already exists from a previous create; clear it.
msgs = execute("QSYS/CLRPFM QTEMP/JT400DSSRC"); //@B0C
if (msgs.length > 0)
{
if (!msgs[0].getID().equals("CPC3101")) // "member cleared"
{
// Clear failed. Throw exception.
Trace.log(Trace.ERROR, "QSYS/CLRPFM QTEMP/JT400DSSRC");
throw new AS400Exception(msgs);
}
}
else
{
Trace.log(Trace.ERROR, "No messages were returned from QSYS/CLRPFM QTEMP/JT400DSSRC");
throw new InternalErrorException("QTEMP/JT400DSSRC", InternalErrorException.UNKNOWN);
}
}
else if (!msgs[0].getID().equals("CPC7301"))
{
// File does not exist and we were unable to create; throw exception
// (CPC7301 = Successful create)
Trace.log(Trace.ERROR, "QSYS/CRTSRCPF FILE(QTEMP/JT400DSSRC) RCDLEN(92) MBR(*FILE) TEXT('JT400 TEMPORARY DDS SOURCE FILE')"); //@F0C
throw new AS400Exception(msgs);
}
}
else
{
// No messages. This shouldn't happen.
Trace.log(Trace.ERROR, "No messages from server.");
throw new InternalErrorException(InternalErrorException.UNKNOWN);
}
//-------------------------------------------------------
// Create the records to be written to the file. These records will contain the
// DDS based on the supplied RecordFormat object.
//-------------------------------------------------------
// Create a RecordFormat object which describes the record format of a source
// physical file.
RecordFormat srcRF = new RecordFormat("JT400DSSRC");
srcRF.addFieldDescription(new ZonedDecimalFieldDescription(new AS400ZonedDecimal(6, 2), "SRCSEQ"));
srcRF.addFieldDescription(new ZonedDecimalFieldDescription(new AS400ZonedDecimal(6, 0), "SRCDAT"));
// - Can't use the system object here because we have no way of filling in the converter
srcRF.addFieldDescription(new CharacterFieldDescription(new AS400Text(80, system_.getCcsid()), "SRCDTA")); //@D0C
Vector lines = new Vector(); // Contains DDS lines to write to source file
String line; // A single line of DDS source
// Create line(s) for any file level keywords - file level keywords must precede
// the line specifying the record format name.
if (altSeq != null)
{
line = STR44 + "ALTSEQ(" + altSeq + ")";
lines.addElement(line);
}
if (ccsid != null)
{
line = STR44 + "CCSID(" + ccsid + ")";
lines.addElement(line);
}
if (order != null)
{
line = STR44 + order;
lines.addElement(line);
}
if (ref != null)
{
line = STR44 + "REF(" + ref + ")";
lines.addElement(line);
}
if (unique)
{
line = STR44 + "UNIQUE";
lines.addElement(line);
}
// Create line for the record format name
line = STR16 + "R ";
// The record format name cannot exceed 10 characters and must be in upper case
if (recordFormat.getName().length() > 10)
{
if (Trace.traceOn_)
{
Trace.log(Trace.WARNING, "Record format name '"+recordFormat.getName()+"' too long. Using '"+recordFormat.getName().substring(0,10)+"' instead.");
}
line += recordFormat.getName().substring(0, 10);
}
else
{
line += recordFormat.getName();
}
lines.addElement(line);
// Create line(s) for any record level keywords. The record level keywords
// must be on the same line or on the lines immediately following the record
// format line.
if (format != null)
{
line = STR44 + "FORMAT(" + format + ")";
lines.addElement(line);
}
if (text != null)
{
if (text.length() > 32)
{ // Text exceeds length left on line - need to continue on next line
line = STR44 + "TEXT('" + text.substring(0, 33) + "-";
lines.addElement(line);
// Add the remainder of the TEXT keyword
line = STR44 + text.substring(34) + "')";
lines.addElement(line);
}
else
{ // Text fits on one line
line = STR44 + "TEXT('" + text + "')";
lines.addElement(line);
}
}
// Create lines for each field description and any keywords for the field
int numberOfFields = recordFormat.getNumberOfFields();
FieldDescription f = null;
String ddsDesc;
String[] dds = null;
int length;
int beginningOffset;
for (int i = 0; i < numberOfFields; ++i)
{
f = recordFormat.getFieldDescription(i);
// Specify the DDS description of the field. The DDS description returned
// from FieldDescription contains the field level keywords as well as the
// description of the field. It is formatted properly for DDS except for
// the preceding blanks. Therefore, we add 18 blanks so that the field
// description starts in column 19 of the line.
dds = f.getDDSDescription();
// Add the fixed portion of the DDS description for the field to the vector
ddsDesc = STR18 + dds[0];
lines.addElement(ddsDesc);
// Add lines containing field level keywords
for (int j = 1; j < dds.length; ++j)
{
ddsDesc = STR44 + dds[j];
length = ddsDesc.length();
beginningOffset = 0;
if (length > 80)
{ // Need to continue the line on the next line
line = ddsDesc.substring(beginningOffset, 79) + "-";
lines.addElement(line);
length -= 79;
beginningOffset = 79;
line = STR44 + ddsDesc.substring(beginningOffset);
lines.addElement(line);
}
else
{ // It all fits on one line
lines.addElement(ddsDesc);
}
}
}
// Create lines for key fields and key field level keywords
numberOfFields = recordFormat.getNumberOfKeyFields();
for (int i = 0; i < numberOfFields; ++i)
{
f = recordFormat.getKeyFieldDescription(i);
// Specify the name of the field
line = STR16 + "K ";
line += f.getDDSName();
lines.addElement(line);
// Specify any key field level keywords
String[] keyFuncs = f.getKeyFieldFunctions();
if (keyFuncs != null)
{
for (short j = 0; j < keyFuncs.length; ++j)
{
line = STR44 + keyFuncs[j];
lines.addElement(line);
}
}
}
// Create an array of records representing each line to be written
// to the file.
Record[] records = new Record[lines.size()];
for (int i = 0; i < records.length; ++i)
{
records[i] = srcRF.getNewRecord();
records[i].setField("SRCSEQ", new BigDecimal(i));
records[i].setField("SRCDAT", new BigDecimal(i));
records[i].setField("SRCDTA", lines.elementAt(i));
}
// Open the DDS source file and write the records. We will write all the records
// at one time, so we specify a blocking factor of records.length on the open().
AS400FileImplRemote src = null; //@B5C @D0C 7/15/99
try
{
//@B5D src = new SequentialFile(system_, "/QSYS.LIB/QTEMP.LIB/JT400DSSRC.FILE");
//@B5 - Can't create a new AS400FileImplBase because it's abstract.
//@B5 - Other alternative is to create a new ImplRemote or
// ImplNative based on the type of this. But can't create
// a new AS400FileImplNative since it doesn't exist at
// compile time.
//@D0D 7/15/99 src = (AS400FileImplBase)this.clone(); //@B5A
//@B5D try
//@B5D {
//@B5D src.setRecordFormat(srcRF);
//@B5D }
//@B5D catch(PropertyVetoException e)
//@B5D { // This is to quiet the compiler
//@B5D }
src = new AS400FileImplRemote(); //@D0A 7/15/99
src.setAll(system_, "/QSYS.LIB/QTEMP.LIB/JT400DSSRC.FILE", srcRF, false, false, false); //@B5A
//@B5D src.open(AS400File.WRITE_ONLY, records.length, AS400File.COMMIT_LOCK_LEVEL_NONE);
src.openFile2(AS400File.WRITE_ONLY, records.length, AS400File.COMMIT_LOCK_LEVEL_NONE, false); //@B5A
src.write(records);
// Close the file
src.close();
}
catch (Exception e)
{ // log error and rethrow exception
Trace.log(Trace.ERROR, "Unable to write records to DDS source file:", e);
try
{
if (src != null)
{
src.close();
}
}
catch (Exception e1)
{ // Ignore it; we will throw the original exception
}
if (e instanceof AS400Exception)
{
throw (AS400Exception) e;
}
else
{
Trace.log(Trace.ERROR, e);
throw new InternalErrorException(InternalErrorException.UNKNOWN, e);
}
}
}
/**
*Deletes the record at the current cursor position.
*@exception AS400Exception If the server returns an error message.
*@exception AS400SecurityException If a security or authority error occurs.
*@exception ConnectionDroppedException If the connection is dropped unexpectedly.
*@exception InterruptedException If this thread is interrupted.
*@exception IOException If an error occurs while communicating with the server.
**/
public void deleteCurrentRecord()
throws AS400Exception,
AS400SecurityException,
InterruptedException,
IOException
{
connect();
// Send the delete record data stream request
Vector replys = sendRequestAndReceiveReplies(DDMRequestDataStream.getRequestS38DEL(dclName_), newCorrelationId()); //@B6C
// Reply expected: S38IOFB
if (!(replys.size() == 1 && ((DDMDataStream)replys.elementAt(0)).getCodePoint() == DDMTerm.S38IOFB))
{
handleErrorReply(replys, 1);
}
}
/**
Executes a command on the server.
@param cmd the command
*@exception AS400SecurityException If a security or authority error occurs.
*@exception ConnectionDroppedException If the connection is dropped unexpectedly.
*@exception InterruptedException If this thread is interrupted.
*@exception IOException If an error occurs while communicating with the server.
**/
public AS400Message[] execute(String cmd)
throws AS400SecurityException, InterruptedException, IOException
{
connect();
// Send the submit command data stream request
Vector replys = null;
if (discardReplys_)
{
synchronized(server_) //@F1A
{ //@F1A
server_.sendAndDiscardReply(DDMRequestDataStream.getRequestS38CMD(cmd, system_),newCorrelationId()); //@C0C @M8C
} //@F1A
}
else
{
replys = sendRequestAndReceiveReplies(DDMRequestDataStream.getRequestS38CMD(cmd, system_), newCorrelationId()); //@C0C @B6C
}
return processReplys(replys);
}
/**
*Throws an appropriate exception based on the type of reply
*data stream specified.
*@param replys The replys to be checked to determine the error.
*@param index The index within replys at which to start.
*@exception AS400Exception If the server returns an error message.
*@exception AS400SecurityException If a security or authority error occurs.
*@exception ConnectionDroppedException If the connection is dropped unexpectedly.
*@exception InterruptedException If this thread is interrupted.
*@exception IOException If an error occurs while communicating with the server.
**/
public void handleErrorReply(Vector replys, int index)
throws AS400Exception,
AS400SecurityException,
InterruptedException,
IOException
{
int size = replys.size();
DDMDataStream reply;
int codePoint;
Vector as400MsgList = new Vector();
for (int i = index; i < size; ++i)
{
reply = (DDMDataStream)replys.elementAt(i);
codePoint = reply.getCodePoint();
switch (codePoint)
{
case DDMTerm.S38MSGRM:
// Because we will normally get more than one AS400Message for an
// error condition, we build a vector of AS400Message arrays and
// throw an AS400Exception with all messages once we have finished
// parsing the replies. Note that the DDMAS400MessageReply class
// extracts all the server messages contained in reply.data_. I.e.
// a single reply may contain more than one message.
DDMAS400MessageReply msgReply = new DDMAS400MessageReply(system_, reply.data_); //@C0C
as400MsgList.addElement(msgReply.getAS400MessageList());
break;
// If any of the following cases occur, we throw the exception and are done.
case DDMTerm.AGNPRMRM:
case DDMTerm.CMDCHKRM:
case DDMTerm.CMDNSPRM:
case DDMTerm.DCLNAMRM:
case DDMTerm.PRCCNVRM:
case DDMTerm.PRMNSPRM:
case DDMTerm.RSCLMTRM:
case DDMTerm.SYNTAXRM:
case DDMTerm.VALNSPRM:
Trace.log(Trace.ERROR, "handleErrorReply()", reply.data_);
throw new InternalErrorException(InternalErrorException.UNKNOWN, codePoint);
default:
// We don't know what the reply is. Throw exception and be done.
Trace.log(Trace.ERROR, "handleErrorReply()", reply.data_);
throw new InternalErrorException(InternalErrorException.DATA_STREAM_UNKNOWN, codePoint);
}
}
// If we get to here, we should have a list of messages to throw
if (as400MsgList.size() > 0)
{
// We need to expand out the Vector of AS400Message[]'s to individual
// AS400Message's in a single array in order to construct our AS400Exception.
int numberOfMessages = 0;
int msgListSize = as400MsgList.size();
int msgNumber = 0;
// Determine number of AS400Message objects we have
for (int i = 0; i < msgListSize; ++i)
{
numberOfMessages += ((AS400Message[])as400MsgList.elementAt(i)).length;
}
// Now populate the single AS400Message[] with the AS400Message objects
AS400Message[] msgs = new AS400Message[numberOfMessages];
AS400Message[] m;
for (int i = 0; i < msgListSize; ++i)
{
m = (AS400Message[])as400MsgList.elementAt(i);
for (int j = 0; j < m.length; ++j)
{
msgs[msgNumber++] = m[j];
}
}
throw new AS400Exception(msgs);
}
Trace.log(Trace.ERROR, "No messages from server.");
throw new InternalErrorException(InternalErrorException.DATA_STREAM_UNKNOWN);
}
/**
*Logs warning messages if tracing is on. This method should be used to log
*warning messages (trace.WARNING) when an operation is successful yet additional
*AS400 messages are sent with the reply.
*@param v The vector of replies containing messages to log.
*@param index The index into v at which to stop pulling out messages.
*@exception AS400Exception If the server returns an error message.
*@exception AS400SecurityException If a security or authority error occurs.
*@exception ConnectionDroppedException If the connection is dropped unexpectedly.
*@exception InterruptedException If this thread is interrupted.
*@exception IOException If an error occurs while communicating with the server.
**/
public void logWarningMessages(Vector v, int index)
throws AS400Exception,
AS400SecurityException,
InterruptedException,
IOException
{
if (Trace.traceOn_)
{
DDMAS400MessageReply msg;
for (int i = 0; i < index; ++i)
{
msg = new DDMAS400MessageReply(system_, ((DDMDataStream)v.elementAt(i)).data_); //@C0C
AS400Message[] msgs = msg.getAS400MessageList();
Trace.log(Trace.WARNING, "AS400FileImplRemote.logWarningMessages():");
for (int j = 0; j < msgs.length; ++j)
{
Trace.log(Trace.WARNING, msgs[j].toString());
}
}
}
}
/**
*Looks for a particular code point in a vector of data streams.
*@param cp The code point for which to look.
*@param v The data streams in which to look.
*@return The index of the datastream in v in which the code point was found.
*If the code point is not found, -1 is returned.
**/
private int lookForCodePoint(int cp, Vector v)
{
int index = -1;
int size = v.size();
DDMDataStream ds;
int offset;
// Search each datastream in v for the code point.
for (int i = 0; i < size && index == -1; ++i)
{
ds = (DDMDataStream)v.elementAt(i);
offset = 8;
// Search through all the code points in an element of v for cp.
while (index == -1 && offset < ds.data_.length)
{
if (ds.get16bit(offset) == cp)
{ // Found the code point
index = i;
}
else
{ // Move on to the next code point
offset += ds.get16bit(offset - 2);
}
}
}
return index;
}
/**
*Opens the file. Helper function to open file for keyed or
*sequential files.
*@param openType
*@param bf blocking factor
*@param access The type of file access for which to open the file.
*@return the open feedback data
*@exception AS400Exception If the server returns an error message.
*@exception AS400SecurityException If a security or authority error occurs.
*@exception ConnectionDroppedException If the connection is dropped unexpectedly.
*@exception InterruptedException If this thread is interrupted.
*@exception IOException If an error occurs while communicating with the server.
*@exception ServerStartupException If the host server cannot be started..
*@exception UnknownHostException If the server cannot be located.
**/
public DDMS38OpenFeedback openFile(int openType, int bf, String access)
throws AS400Exception,
AS400SecurityException,
InterruptedException,
IOException
{
// Ensure that we are connected to the server.
connect();
// Create the user file control block.
byte[] ufcb = createUFCB(openType, bf, access, false);
// Send the open file data stream request
Vector replys = sendRequestAndReceiveReplies(DDMRequestDataStream.getRequestS38OPEN(ufcb, dclName_), newCorrelationId()); //@B6C
// Reply expected: S38OPNFB
int index = lookForCodePoint(DDMTerm.S38OPNFB, replys);
DDMDataStream reply;
if (index != -1)
{
reply = (DDMDataStream)replys.elementAt(index);
openFeedback_ = new DDMS38OpenFeedback(system_, reply.data_); //@C0C
if (Trace.traceOn_)
{
Trace.log(Trace.INFORMATION, "AS400FileImplRemote.openFile()\n" + openFeedback_.toString());
}
if (index != 0)
{ // Informational and/or diagnostic messages were issued. Log them.
logWarningMessages(replys, index);
}
}
else
{
handleErrorReply(replys, 0);
}
return openFeedback_;
}
/**
*Pads data with padByte numBytes starting at start.
*@param data The data to pad.
*@param start The offset in data at which we begin padding.
*@param numBytes The number of bytes to pad.
*@param padChar The character with which to pad.
**/
public void padBytes(byte[] data, int start, int numBytes, byte padChar)
{
for (int i = 0; i < numBytes; ++i)
{
data[start + i] = padChar;
}
}
/**
*Positions the cursor for the file. Which record to position the cursor to is
*determined by the type
*argument.
*@param searchType The type of get to execute. Valid values are:
*
*- TYPE_GET_FIRST
*
- TYPE_GET_NEXT
*
- TYPE_GET_LAST
*
- TYPE_GET_PREV
*
- TYPE_GET_SAME
*@exception AS400Exception If the server returns an error message.
*@exception AS400SecurityException If a security or authority error occurs.
*@exception ConnectionDroppedException If the connection is dropped unexpectedly.
*@exception InterruptedException If this thread is interrupted.
*@exception IOException If an error occurs while communicating with the server.
**/
public Record[] positionCursorAt(int searchType)
throws AS400Exception,
AS400SecurityException,
InterruptedException,
IOException
{
int shr; // Type of locking for the record
// @A1C
if ((openType_ == AS400File.READ_ONLY) || //@C0C
((openType_ == AS400File.READ_WRITE) && readNoUpdate_)) // @A1A //@C0C
{
// Read only
shr = SHR_READ_NORM;
}
else
{ // READ_WRITE, lock the record for update
shr = SHR_UPD_NORM;
}
// Send the request to read. Ignore the data; don't specify
// DATA_NODTA_DTARCD because it is one of the most inefficient
// paths per the DDM server guys. More efficient to get the data and ignore
// it they say.
Vector replys = sendRequestAndReceiveReplies(DDMRequestDataStream.getRequestS38GET(dclName_, searchType, shr, DATA_DTA_DTARCD), newCorrelationId()); //@B6C
int codePoint = ((DDMDataStream)replys.elementAt(0)).getCodePoint();
if (codePoint == DDMTerm.S38IOFB && replys.size() > 1)
{
handleErrorReply(replys, 1);
}
else if (codePoint != DDMTerm.S38BUF)
{
handleErrorReply(replys, 0);
}
Record[] returned = processReadReply(replys, false); // @A1C
return returned;
}
//@RBA
public Record[] positionCursorAtLong(int searchType)
throws AS400Exception,
AS400SecurityException,
InterruptedException,
IOException
{
int shr; // Type of locking for the record
// @A1C
if ((openType_ == AS400File.READ_ONLY) || //@C0C
((openType_ == AS400File.READ_WRITE) && readNoUpdate_)) // @A1A //@C0C
{
// Read only
shr = SHR_READ_NORM;
}
else
{ // READ_WRITE, lock the record for update
shr = SHR_UPD_NORM;
}
// Send the request to read. Ignore the data; don't specify
// DATA_NODTA_DTARCD because it is one of the most inefficient
// paths per the DDM server guys. More efficient to get the data and ignore
// it they say.
Vector replys = sendRequestAndReceiveReplies(DDMRequestDataStream.getRequestS38GET(dclName_, searchType, shr, DATA_DTA_DTARCD), newCorrelationId()); //@B6C
int codePoint = ((DDMDataStream)replys.elementAt(0)).getCodePoint();
if (codePoint == DDMTerm.S38IOFB && replys.size() > 1)
{
handleErrorReply(replys, 1);
}
else if (codePoint != DDMTerm.S38BUF)
{
handleErrorReply(replys, 0);
}
Record[] returned = processReadReplyLong(replys, false); // @A1C
return returned;
}
/**
*Positions the file cursor to after the last record.
*@exception AS400Exception If the server returns an error message.
*@exception AS400SecurityException If a security or authority error occurs.
*@exception ConnectionDroppedException If the connection is dropped unexpectedly.
*@exception InterruptedException If this thread is interrupted.
*@exception IOException If an error occurs while communicating with the server.
**/
public void positionCursorAfterLast()
throws AS400Exception,
AS400SecurityException,
InterruptedException,
IOException
{
// If we are caching records and the cache contains the last record,
// position the cache. Otherwise, position the file and refresh the
// cache if we are caching records.
if (cacheRecords_ && cache_.containsLastRecord())
{
cache_.setPositionAfterLast();
return;
}
if (cacheRecords_)
{
// Invalidate the cache
cache_.setIsEmpty();
}
connect();
// Send the request to force end of data specifying end of the file for
// positioning.
// Need correlation id as we will be chaining an S38BUF object to this request
// and the ids must match.
int id = newCorrelationId(); //@B6C
DDMRequestDataStream req = DDMRequestDataStream.getRequestS38FEOD(dclName_, TYPE_GET_LAST, SHR_READ_NORM_RLS, DATA_NODTA_DTARCD);
req.setIsChained(true); // We will be chaining an S38BUF to this request
req.setHasSameRequestCorrelation(true); // When chaining, must indicate
// that the correlation ids are
// the same.
Vector replys = null;
try
{
synchronized(server_) //@F1A - both datastreams must be written atomically
{ //@F1A
server_.send(req, id); // Send the S38FEOD request
// Although the S38FEOD term description states that
// the S38BUF object is optional, it is not for our purposes.
// Create an empty S38BUF object to send after the S38FEOD
DDMObjectDataStream s38buf = new DDMObjectDataStream(11);
// The class contains static variable s38Buffer which is the S38BUF term with
// a value of 1 byte set to 0x00.
System.arraycopy(s38Buffer, 0, s38buf.data_, 6, 5);
// Send the S38BUF
replys = sendRequestAndReceiveReplies(s38buf, id);
} //@F1A
}
catch (ConnectionDroppedException e)
{
// Connection dropped. Disconnect server and rethrow.
Trace.log(Trace.ERROR, "ConnectionDroppedException:", e);
system_.disconnectServer(server_); //@C0C
//@C1 - Setting the server_ object to null means that
// any operations on this AS400File object after the connection has been
// dropped will result in a NullPointerException. By leaving the server_ object
// around, any subsequent operations should also throw a ConnectionDroppedException.
//@C1D server_ = null;
resetState();
throw e;
}
// Reply expected: S38IOFB, which may be followed by an S38MSGRM
// with id CPF5001 which indicates that the end of file was reached.
// Any other S38MSGRM replies indicate that an error has occurred.
if (((DDMDataStream)replys.elementAt(0)).getCodePoint() == DDMTerm.S38IOFB)
{
if (replys.size() != 1)
{
if (!verifyS38MSGRM((DDMDataStream)replys.elementAt(1), "CPF5001", 0))
{
handleErrorReply(replys, 1);
}
}
}
else
{
handleErrorReply(replys, 0);
}
}
/**
*Positions the file cursor to before the first record.
*@exception AS400Exception If the server returns an error message.
*@exception AS400SecurityException If a security or authority error occurs.
*@exception ConnectionDroppedException If the connection is dropped unexpectedly.
*@exception InterruptedException If this thread is interrupted.
*@exception IOException If an error occurs while communicating with the server.
**/
public void positionCursorBeforeFirst()
throws AS400Exception,
AS400SecurityException,
InterruptedException,
IOException
{
// If we are caching records and the cache contains the first record,
// position the cache. Otherwise, position the file and refresh the
// cache if we are caching records.
if (cacheRecords_ && cache_.containsFirstRecord())
{
cache_.setPositionBeforeFirst();
return;
}
if (cacheRecords_)
{ // Invalidate the cache
cache_.setIsEmpty();
}
connect();
// Send the request to force end of data specifying beginning of the
// file for positioning; this is how we position the cursor to
// before the first record.
// Need correlation id as we will be chaining an S38BUF object to this request
// and the ids must match.
int id = newCorrelationId(); //@B6C
DDMRequestDataStream req = DDMRequestDataStream.getRequestS38FEOD(dclName_, TYPE_GET_FIRST, SHR_READ_NORM_RLS, DATA_NODTA_DTARCD);
req.setIsChained(true); // We will be chaining an S38BUF to this request
req.setHasSameRequestCorrelation(true); // When chaining, must indicate
// that the correlation ids are
// the same.
Vector replys = null;
try
{
synchronized(server_) //@F1A - both datastreams must be written atomically
{ //@F1A
server_.send(req, id); // Send the S38FEOD request
// Although the S38FEOD term description states that
// the S38BUF object is optional, it is not for our purposes.
// Create an empty S38BUF object to send after the S38FEOD
DDMObjectDataStream s38buf = new DDMObjectDataStream(11);
// The class contains static variable s38Buffer which is the S38BUF term with
// a value of 1 byte set to 0x00.
System.arraycopy(s38Buffer, 0, s38buf.data_, 6, 5);
// Send the S38BUF
replys = sendRequestAndReceiveReplies(s38buf, id);
} //@F1A
}
catch (ConnectionDroppedException e)
{
// Connection dropped. Disconnect server and rethrow.
Trace.log(Trace.ERROR, "ConnectionDroppedException:", e);
system_.disconnectServer(server_); //@C0C
//@C1 - Setting the server_ object to null means that
// any operations on this AS400File object after the connection has been
// dropped will result in a NullPointerException. By leaving the server_ object
// around, any subsequent operations should also throw a ConnectionDroppedException.
//@C1D server_ = null;
resetState();
throw e;
}
// Reply expected: S38IOFB, which may be followed by an S38MSGRM
// with id CPF5001 which indicates that the end of file was reached.
// Any other S38MSGRM replies indicate that an error has occurred.
if (((DDMDataStream)replys.elementAt(0)).getCodePoint() == DDMTerm.S38IOFB)
{
if (replys.size() != 1)
{
if (!verifyS38MSGRM((DDMDataStream)replys.elementAt(1), "CPF5001", 0))
{
handleErrorReply(replys, 1);
}
}
}
else
{
handleErrorReply(replys, 0);
}
}
/**
*Positions the file cursor to the specified record number in the file on the server.
*@param recordNumber The record number to which to position the file cursor.
*@exception AS400Exception If the server returns an error message.
*@exception AS400SecurityException If a security or authority error occurs.
*@exception ConnectionDroppedException If the connection is dropped unexpectedly.
*@exception InterruptedException If this thread is interrupted.
*@exception IOException If an error occurs while communicating with the server.
**/
public Record positionCursorToIndex(int recordNumber)
throws AS400Exception,
AS400SecurityException,
InterruptedException,
IOException
{
int shr; // Type of locking for the record
if (cacheRecords_ && !cache_.setPosition(recordNumber)) //@G0A
{
// Invalidate the cache
cache_.setIsEmpty(); //@G0A
}
// @A1C
if ((openType_ == AS400File.READ_ONLY) || //@C0C
((openType_ == AS400File.READ_WRITE) && readNoUpdate_)) // @A1A //@C0C
{
// Read only
shr = SHR_READ_NORM;
}
else
{ // READ_WRITE; get for update
shr = SHR_UPD_NORM;
}
// Send the request to read. Specify 0x08 for type
// to indicate that the record number is determined from the start of the file.
// Note that even though we are only positioning the cursor, we must specify
// DATA_DTA_DTARCD which causes the record to be retrieved. This is because
// the cursor will not be positioned properly if we do not actually get the
// record. This situation occurs when accessing by record number or by key but
// not when accessing sequentially.
// Vector replys = sendRequestAndReceiveReplies(DDMRequestDataStream.getRequestS38GETD(dclName_, recordFormat_, 0x08, shr, DATA_DTA_DTARCD, recordNumber, system_), server_.newCorrelationId()); // @A1D //@C0C
Vector replys = sendRequestAndReceiveReplies(DDMRequestDataStream.getRequestS38GETD(dclName_, recordFormatCTLLName_, 0x08, shr, DATA_DTA_DTARCD, recordNumber, system_), newCorrelationId()); // @A1A //@C0C @B6C
// Reply expected: S38BUF
int codePoint = ((DDMDataStream)replys.elementAt(0)).getCodePoint();
if (codePoint == DDMTerm.S38IOFB && replys.size() > 1)
{
handleErrorReply(replys, 1);
}
else if (codePoint != DDMTerm.S38BUF)
{
handleErrorReply(replys, 0);
}
Record[] returned = processReadReply(replys, false); // @A1C
return returned[0]; // @A1A
// return null; // @A1D
}
//@RBA
public Record positionCursorToIndexLong(long recordNumber)
throws AS400Exception,
AS400SecurityException,
InterruptedException,
IOException
{
int shr; // Type of locking for the record
if (cacheRecords_ && !cache_.setPositionLong(recordNumber)) //@G0A
{
// Invalidate the cache
cache_.setIsEmpty(); //@G0A
}
// @A1C
if ((openType_ == AS400File.READ_ONLY) || //@C0C
((openType_ == AS400File.READ_WRITE) && readNoUpdate_)) // @A1A //@C0C
{
// Read only
shr = SHR_READ_NORM;
}
else
{ // READ_WRITE; get for update
shr = SHR_UPD_NORM;
}
// Send the request to read. Specify 0x08 for type
// to indicate that the record number is determined from the start of the file.
// Note that even though we are only positioning the cursor, we must specify
// DATA_DTA_DTARCD which causes the record to be retrieved. This is because
// the cursor will not be positioned properly if we do not actually get the
// record. This situation occurs when accessing by record number or by key but
// not when accessing sequentially.
// Vector replys = sendRequestAndReceiveReplies(DDMRequestDataStream.getRequestS38GETD(dclName_, recordFormat_, 0x08, shr, DATA_DTA_DTARCD, recordNumber, system_), server_.newCorrelationId()); // @A1D //@C0C
Vector replys = sendRequestAndReceiveReplies(DDMRequestDataStream.getRequestS38GETDLong(dclName_, recordFormatCTLLName_, 0x08, shr, DATA_DTA_DTARCD, recordNumber, system_), newCorrelationId()); // @A1A //@C0C @B6C
// Reply expected: S38BUF
int codePoint = ((DDMDataStream)replys.elementAt(0)).getCodePoint();
if (codePoint == DDMTerm.S38IOFB && replys.size() > 1)
{
handleErrorReply(replys, 1);
}
else if (codePoint != DDMTerm.S38BUF)
{
handleErrorReply(replys, 0);
}
Record[] returned = processReadReplyLong(replys, false); // @A1C
return returned[0]; // @A1A
// return null; // @A1D
}
/**
*Positions the cursor to the first record with the specified key based on the specified
*type of read.
*@param key The values that make up the key with which to find the record.
*@param searchType The type of read. This value is one of the TYPE_GETKEY_* constants.
*@exception AS400Exception If the server returns an error message.
*@exception AS400SecurityException If a security or authority error occurs.
*@exception ConnectionDroppedException If the connection is dropped unexpectedly.
*@exception InterruptedException If this thread is interrupted.
*@exception IOException If an error occurs while communicating with the server.
**/
public Record positionCursorToKey(Object[] key, int searchType)
throws AS400Exception,
AS400SecurityException,
InterruptedException,
IOException
{
int shr; // Type of locking for the record
if (cacheRecords_) //@G0A
{
// Invalidate the cache
cache_.setIsEmpty(); //@G0A
}
// @A1C
if ((openType_ == AS400File.READ_ONLY) || //@C0C
((openType_ == AS400File.READ_WRITE) && readNoUpdate_)) // @A1A //@C0C
{
// Read only
shr = SHR_READ_NORM;
}
else
{ // READ_WRITE
shr = SHR_UPD_NORM;
}
// In order to have the file cursor remain properly positioned, we specify that the record
// is to be returned on the GETK request as opposed to specifying DATA_NODTA_DTARCD. This
// is necessary for the caching support.
// Vector replys = sendRequestAndReceiveReplies(DDMRequestDataStream.getRequestS38GETK(dclName_, recordFormat_, type, shr, DATA_DTA_DTARCD, key, system_), server_.newCorrelationId()); // @A1D
Vector replys = sendRequestAndReceiveReplies(DDMRequestDataStream.getRequestS38GETK(dclName_, recordFormat_, recordFormatCTLLName_, searchType, shr, DATA_DTA_DTARCD, key, system_), newCorrelationId()); // @A1A @B6C
int codePoint = ((DDMDataStream)replys.elementAt(0)).getCodePoint();
if (codePoint == DDMTerm.S38IOFB && replys.size() > 1)
{
handleErrorReply(replys, 1);
}
else if (codePoint != DDMTerm.S38BUF)
{
handleErrorReply(replys, 0);
}
// Record[] returned = processReadReply(replys, true); // @A1C
// return returned[0]; // @A1D
processReadReply(replys, true); // @A1C
return null; // @A1A
}
//@RBA
public Record positionCursorToKeyLong(Object[] key, int searchType)
throws AS400Exception,
AS400SecurityException,
InterruptedException,
IOException
{
int shr; // Type of locking for the record
if (cacheRecords_) //@G0A
{
// Invalidate the cache
cache_.setIsEmpty(); //@G0A
}
// @A1C
if ((openType_ == AS400File.READ_ONLY) || //@C0C
((openType_ == AS400File.READ_WRITE) && readNoUpdate_)) // @A1A //@C0C
{
// Read only
shr = SHR_READ_NORM;
}
else
{ // READ_WRITE
shr = SHR_UPD_NORM;
}
// In order to have the file cursor remain properly positioned, we specify that the record
// is to be returned on the GETK request as opposed to specifying DATA_NODTA_DTARCD. This
// is necessary for the caching support.
// Vector replys = sendRequestAndReceiveReplies(DDMRequestDataStream.getRequestS38GETK(dclName_, recordFormat_, type, shr, DATA_DTA_DTARCD, key, system_), server_.newCorrelationId()); // @A1D
Vector replys = sendRequestAndReceiveReplies(DDMRequestDataStream.getRequestS38GETK(dclName_, recordFormat_, recordFormatCTLLName_, searchType, shr, DATA_DTA_DTARCD, key, system_), newCorrelationId()); // @A1A @B6C
int codePoint = ((DDMDataStream)replys.elementAt(0)).getCodePoint();
if (codePoint == DDMTerm.S38IOFB && replys.size() > 1)
{
handleErrorReply(replys, 1);
}
else if (codePoint != DDMTerm.S38BUF)
{
handleErrorReply(replys, 0);
}
// Record[] returned = processReadReply(replys, true); // @A1C
// return returned[0]; // @A1D
processReadReplyLong(replys, true); // @A1C
return null; // @A1A
}
// @A1A
/**
*Positions the cursor to the first record with the specified key based on the specified
*type of read.
*@param key The byte array that contains the byte values that make up the key with which to find the record.
*@param searchType The type of read. This value is one of the TYPE_GETKEY_* constants.
*@param numberOfKeyFields The number of key fields contained in the byte array key.
*@exception AS400Exception If the server returns an error message.
*@exception AS400SecurityException If a security or authority error occurs.
*@exception ConnectionDroppedException If the connection is dropped unexpectedly.
*@exception InterruptedException If this thread is interrupted.
*@exception IOException If an error occurs while communicating with the server.
**/
public Record positionCursorToKey(byte[] key, int searchType, int numberOfKeyFields)
throws AS400Exception,
AS400SecurityException,
InterruptedException,
IOException
{
int shr; // Type of locking for the record
if (cacheRecords_) //@G0A
{
// Invalidate the cache
cache_.setIsEmpty(); //@G0A
}
if ((openType_ == AS400File.READ_ONLY) ||
((openType_ == AS400File.READ_WRITE) && readNoUpdate_)) // @A1A
{
// Read only
shr = SHR_READ_NORM;
}
else
{ // READ_WRITE
shr = SHR_UPD_NORM;
}
// In order to have the file cursor remain properly positioned, we specify that the record
// is to be returned on the GETK request as opposed to specifying DATA_NODTA_DTARCD. This
// is necessary for the caching support.
// Vector replys = sendRequestAndReceiveReplies(DDMRequestDataStream.getRequestS38GETK(dclName_, recordFormat_, type, shr, DATA_DTA_DTARCD, key, system_, numberOfKeyFields), server_.newCorrelationId()); // @A1D
Vector replys = sendRequestAndReceiveReplies(DDMRequestDataStream.getRequestS38GETK(dclName_, recordFormatCTLLName_, searchType, shr, DATA_DTA_DTARCD, key, system_, numberOfKeyFields), newCorrelationId()); // @A1A @B6C
int codePoint = ((DDMDataStream)replys.elementAt(0)).getCodePoint();
if (codePoint == DDMTerm.S38IOFB && replys.size() > 1)
{
handleErrorReply(replys, 1);
}
else if (codePoint != DDMTerm.S38BUF)
{
handleErrorReply(replys, 0);
}
// Record[] returned = processReadReply(replys, true); // @A1C
// return returned[0]; // @A1D
processReadReply(replys, true); // @A1C
return null; // @A1A
}
//@RBA Add this to support long time record number
public Record positionCursorToKeyLong(byte[] key, int searchType, int numberOfKeyFields)
throws AS400Exception,
AS400SecurityException,
InterruptedException,
IOException
{
int shr; // Type of locking for the record
if (cacheRecords_) //@G0A
{
// Invalidate the cache
cache_.setIsEmpty(); //@G0A
}
if ((openType_ == AS400File.READ_ONLY) ||
((openType_ == AS400File.READ_WRITE) && readNoUpdate_)) // @A1A
{
// Read only
shr = SHR_READ_NORM;
}
else
{ // READ_WRITE
shr = SHR_UPD_NORM;
}
// In order to have the file cursor remain properly positioned, we specify that the record
// is to be returned on the GETK request as opposed to specifying DATA_NODTA_DTARCD. This
// is necessary for the caching support.
// Vector replys = sendRequestAndReceiveReplies(DDMRequestDataStream.getRequestS38GETK(dclName_, recordFormat_, type, shr, DATA_DTA_DTARCD, key, system_, numberOfKeyFields), server_.newCorrelationId()); // @A1D
Vector replys = sendRequestAndReceiveReplies(DDMRequestDataStream.getRequestS38GETK(dclName_, recordFormatCTLLName_, searchType, shr, DATA_DTA_DTARCD, key, system_, numberOfKeyFields), newCorrelationId()); // @A1A @B6C
int codePoint = ((DDMDataStream)replys.elementAt(0)).getCodePoint();
if (codePoint == DDMTerm.S38IOFB && replys.size() > 1)
{
handleErrorReply(replys, 1);
}
else if (codePoint != DDMTerm.S38BUF)
{
handleErrorReply(replys, 0);
}
// Record[] returned = processReadReply(replys, true); // @A1C
// return returned[0]; // @A1D
processReadReplyLong(replys, true); // @A1C
return null; // @A1A
}
/**
*Processes the replys vector for records read. Throws exceptions
*in error cases.
*@param replys The reply datastream(s) containing the record(s) read.
* @param discardRecords should records be discarded
*@return The records read from the system. Returns null if no records were
*read.
*@exception AS400Exception If the server returns an error message.
*@exception AS400SecurityException If a security or authority error occurs.
*@exception InterruptedException If this thread is interrupted.
*@exception IOException If an error occurs while communicating with the server.
**/
// Record[] processReadReply(Vector replys) // @A1D
public Record[] processReadReply(Vector replys, boolean discardRecords) // @A1A
throws AS400Exception,
AS400SecurityException,
InterruptedException,
IOException
{
Record[] returned = null; // Will contain the records read.
int recordIncrement = openFeedback_.getRecordIncrement();
// The format of the reply(s) should be one or more S38BUF objects
// followed by an S38IOFB. However if the end of file was reached or
// if the record to be read was not found in the file, an S38IOFB followed
// be an S38MSGRM indicating CPF5006 or CPF5001 will be returned. If this is
// the case, we return null instead of throwing an exception.
// If an error occurs we may get an S38IOFB followed by S38MSGRM objects indicating
// the server errors that occurred. In that case we throw an exception via
// handleErrorReply. If we only get an S38IOFB back, we also throw an
// exception as an error must have occurred. This case should not happen.
int codePoint = ((DDMDataStream)replys.elementAt(0)).getCodePoint();
if (codePoint == DDMTerm.S38IOFB)
{ // The end of file was reached, the record to be read was not found or an
// error occurred.
if (replys.size() > 1)
{
codePoint = ((DDMDataStream)replys.elementAt(1)).getCodePoint();
if (codePoint == DDMTerm.S38MSGRM)
{ // Check for end of file or record not found messages
DDMAS400MessageReply err = new DDMAS400MessageReply(system_, ((DDMDataStream)replys.elementAt(1)).data_);
String msgId = err.getAS400Message().getID();
if (msgId.equals("CPF5006") || msgId.equals("CPF5001"))
{ // End of file reached or record not found; return null record
return returned;
}
else
{ // Error occurred
handleErrorReply(replys, 1);
}
}
else
{ // Some other error (other than an error) occurred
handleErrorReply(replys, 1);
}
}
else
{ // Only an S38IOFB object was returned. Error situation. Should not occur.
handleErrorReply(replys, 0);
}
}
else if (codePoint == DDMTerm.S38BUF)
{
if (discardRecords)
{ // @A1A
return returned; // @A1A
} // @A1A
// Records were read. Extract format them
// Extract the returned records and the io feedback info
// The S38IOFB will be the last object in the reply(s)
DDMDataStream reply = (DDMDataStream)replys.elementAt(0);
DDMS38IOFB ioFeedback;
boolean largeBuffer;
// If the length of the S38BUF term is greater than 0x7FFF, there
// will be an extra 4 bytes after the S38BUF code point which indicate the length of
// the S38BUF data. We need to special handle these instances, so we use largeBuffer
// to indicate when such a case has occurred.
largeBuffer = (reply.get16bit(6) <= 0x7FFF)? false : true;
if (reply.isChained())
{ // The IO feedback is in the next reply. The S38IOFB data starts at offset
// 10 in the next reply.
ioFeedback = new DDMS38IOFB(((DDMDataStream)replys.elementAt(1)).data_, 10);
}
else
{ // The io feedback info is in this reply.
// if (largeBuffer)
// The length of the record data is in the 4 bytes following the S38BUF
// code point. The S38IOFB data will start at that length + 6 for the header,
// + 8 for the S38BUF LL-CP-extra LL, + 4 for the S38IOFB LL-CP.
// else
// The length of the record data is contained in the LL preceding
// the S38BUF codepoint. The S38IOFB data will start at that length + 6
// for the header, + 4 for the LL-CP of the S38BUF, + 4 for the S38IOFB LL-CP.
int offset = (largeBuffer)? reply.get32bit(10) + 18 : reply.get16bit(6) + 10;
ioFeedback = new DDMS38IOFB(reply.data_, offset);
}
// Extract the record(s) returned; because we are in the if (S38BUF code point found)
// code, we know that there was at least one record returned.
// The S38IOFB contains the number of records read.
int numberOfRecords = ioFeedback.getNumberOfRecordsReturned();
returned = new Record[numberOfRecords];
// if (largeBuffer), the S38BUF CP is followed by 4 bytes of record length info,
// then the record data; otherwise the record data follows the code point
int recordOffset = (largeBuffer)? 14 : 10;
// Determine the offset of the record number within the S38BUF for each record.
// The S38IOFB contains the record length for the record(s). The record length
// consists of the length of the data plus a variable size gap plus the null byte
// field map. The record number offset is 2 bytes more than the record length.
int recordNumberOffset = recordOffset + ioFeedback.getRecordLength() + 2;
// Determine the null byte field map offset. When we opened the file, the
// the S38OPNFB reply contained the offset of the null byte field map in a record.
int nullFieldMapOffset = recordOffset + openFeedback_.getNullFieldByteMapOffset();
// Determine the number of fields in a record from the RecordFormat for this file.
int numFields = recordFormat_.getNumberOfFields();
// If the file has null capable fields, we will need to check the null byte field
// map and set the fields within the Record object as appropriate.
boolean isNullCapable = openFeedback_.isNullCapable();
for (int i = 0; i < numberOfRecords; ++i)
{ // Extract the records from the datastream reply.
returned[i] = recordFormat_.getNewRecord(reply.data_, recordOffset + i * recordIncrement);
// Set any null fields to null
if (isNullCapable)
{ // File has null capable fields
for (int j = 0; j < numFields; ++j)
{ // 0xF1 = field is null, 0xF0 = field is not null
if (reply.data_[nullFieldMapOffset + j + i * recordIncrement] == (byte)0xF1)
{
returned[i].setField(j, null);
}
}
}
// Set the record number. The record number is two bytes after the end of the
// record data and is four bytes long.
try
{
returned[i].setRecordNumber(BinaryConverter.byteArrayToInt(reply.data_, recordNumberOffset + i * recordIncrement));
}
catch (PropertyVetoException e)
{ // We created the Record objects. There is no one to veto anything
} // so this is here to quit the compiler
}
}
else
{ // Error occurred
handleErrorReply(replys, 0);
}
return returned;
}
//@RBA
public Record[] processReadReplyLong(Vector replys, boolean discardRecords) // @A1A
throws AS400Exception,
AS400SecurityException,
InterruptedException,
IOException
{
Record[] returned = null; // Will contain the records read.
int recordIncrement = openFeedback_.getRecordIncrement();
// The format of the reply(s) should be one or more S38BUF objects
// followed by an S38IOFB. However if the end of file was reached or
// if the record to be read was not found in the file, an S38IOFB followed
// be an S38MSGRM indicating CPF5006 or CPF5001 will be returned. If this is
// the case, we return null instead of throwing an exception.
// If an error occurs we may get an S38IOFB followed by S38MSGRM objects indicating
// the server errors that occurred. In that case we throw an exception via
// handleErrorReply. If we only get an S38IOFB back, we also throw an
// exception as an error must have occurred. This case should not happen.
int codePoint = ((DDMDataStream)replys.elementAt(0)).getCodePoint();
if (codePoint == DDMTerm.S38IOFB)
{ // The end of file was reached, the record to be read was not found or an
// error occurred.
if (replys.size() > 1)
{
codePoint = ((DDMDataStream)replys.elementAt(1)).getCodePoint();
if (codePoint == DDMTerm.S38MSGRM)
{ // Check for end of file or record not found messages
DDMAS400MessageReply err = new DDMAS400MessageReply(system_, ((DDMDataStream)replys.elementAt(1)).data_);
String msgId = err.getAS400Message().getID();
if (msgId.equals("CPF5006") || msgId.equals("CPF5001"))
{ // End of file reached or record not found; return null record
return returned;
}
else
{ // Error occurred
handleErrorReply(replys, 1);
}
}
else
{ // Some other error (other than an error) occurred
handleErrorReply(replys, 1);
}
}
else
{ // Only an S38IOFB object was returned. Error situation. Should not occur.
handleErrorReply(replys, 0);
}
}
else if (codePoint == DDMTerm.S38BUF)
{
if (discardRecords)
{ // @A1A
return returned; // @A1A
} // @A1A
// Records were read. Extract format them
// Extract the returned records and the io feedback info
// The S38IOFB will be the last object in the reply(s)
DDMDataStream reply = (DDMDataStream)replys.elementAt(0);
DDMS38IOFB ioFeedback;
boolean largeBuffer;
// If the length of the S38BUF term is greater than 0x7FFF, there
// will be an extra 4 bytes after the S38BUF code point which indicate the length of
// the S38BUF data. We need to special handle these instances, so we use largeBuffer
// to indicate when such a case has occurred.
largeBuffer = (reply.get16bit(6) <= 0x7FFF)? false : true;
if (reply.isChained())
{ // The IO feedback is in the next reply. The S38IOFB data starts at offset
// 10 in the next reply.
ioFeedback = new DDMS38IOFB(((DDMDataStream)replys.elementAt(1)).data_, 10);
}
else
{ // The io feedback info is in this reply.
// if (largeBuffer)
// The length of the record data is in the 4 bytes following the S38BUF
// code point. The S38IOFB data will start at that length + 6 for the header,
// + 8 for the S38BUF LL-CP-extra LL, + 4 for the S38IOFB LL-CP.
// else
// The length of the record data is contained in the LL preceding
// the S38BUF codepoint. The S38IOFB data will start at that length + 6
// for the header, + 4 for the LL-CP of the S38BUF, + 4 for the S38IOFB LL-CP.
int offset = (largeBuffer)? reply.get32bit(10) + 18 : reply.get16bit(6) + 10;
ioFeedback = new DDMS38IOFB(reply.data_, offset);
}
// Extract the record(s) returned; because we are in the if (S38BUF code point found)
// code, we know that there was at least one record returned.
// The S38IOFB contains the number of records read.
int numberOfRecords = ioFeedback.getNumberOfRecordsReturned();
returned = new Record[numberOfRecords];
// if (largeBuffer), the S38BUF CP is followed by 4 bytes of record length info,
// then the record data; otherwise the record data follows the code point
int recordOffset = (largeBuffer)? 14 : 10;
// Determine the offset of the record number within the S38BUF for each record.
// The S38IOFB contains the record length for the record(s). The record length
// consists of the length of the data plus a variable size gap plus the null byte
// field map. The record number offset is 2 bytes more than the record length.
int recordNumberOffset = recordOffset + ioFeedback.getRecordLength() + 2;
// Determine the null byte field map offset. When we opened the file, the
// the S38OPNFB reply contained the offset of the null byte field map in a record.
int nullFieldMapOffset = recordOffset + openFeedback_.getNullFieldByteMapOffset();
// Determine the number of fields in a record from the RecordFormat for this file.
int numFields = recordFormat_.getNumberOfFields();
// If the file has null capable fields, we will need to check the null byte field
// map and set the fields within the Record object as appropriate.
boolean isNullCapable = openFeedback_.isNullCapable();
for (int i = 0; i < numberOfRecords; ++i)
{ // Extract the records from the datastream reply.
returned[i] = recordFormat_.getNewRecord(reply.data_, recordOffset + i * recordIncrement);
// Set any null fields to null
if (isNullCapable)
{ // File has null capable fields
for (int j = 0; j < numFields; ++j)
{ // 0xF1 = field is null, 0xF0 = field is not null
if (reply.data_[nullFieldMapOffset + j + i * recordIncrement] == (byte)0xF1)
{
returned[i].setField(j, null);
}
}
}
// Set the record number. The record number is two bytes after the end of the
// record data and is four bytes long.
try
{
returned[i].setRecordNumberLong(BinaryConverter.byteArrayToUnsignedInt(reply.data_, recordNumberOffset + i * recordIncrement));
}
catch (PropertyVetoException e)
{ // We created the Record objects. There is no one to veto anything
} // so this is here to quit the compiler
}
}
else
{ // Error occurred
handleErrorReply(replys, 0);
}
return returned;
}
/**
Process replys.
@param replys the replys from a request.
* @return array AS400Messages associated with the replies
*@exception AS400SecurityException If a security or authority error occurs.
*@exception InterruptedException If this thread is interrupted.
*@exception IOException If an error occurs while communicating with the server.
**/
public AS400Message[] processReplys(Vector replys)
throws AS400SecurityException, InterruptedException, IOException
{
AS400Message[] msgs = null;
Vector as400MsgList = new Vector();
if (replys != null)
{
for (int i = 0; i < replys.size(); ++i)
{
DDMDataStream reply = (DDMDataStream)replys.elementAt(i);
int codePoint = reply.getCodePoint();
switch (codePoint)
{
case DDMTerm.S38MSGRM:
// Because we will normally get more than one AS400Message for an
// error condition, we build a vector of AS400Message arrays and
// throw an AS400Exception with all messages once we have finished
// parsing the replies. Note that the DDMAS400MessageReply class
// extracts all the server messages contained in reply.data_. I.e.
// a single reply may contain more than one message.
DDMAS400MessageReply msgReply = new DDMAS400MessageReply(system_, reply.data_);
as400MsgList.addElement(msgReply.getAS400MessageList());
break;
// If any of the following cases occur, we throw the exception and are done.
case DDMTerm.AGNPRMRM:
case DDMTerm.CMDCHKRM:
case DDMTerm.CMDNSPRM:
case DDMTerm.DCLNAMRM:
case DDMTerm.PRCCNVRM:
case DDMTerm.PRMNSPRM:
case DDMTerm.RSCLMTRM:
case DDMTerm.SYNTAXRM:
case DDMTerm.VALNSPRM:
Trace.log(Trace.ERROR, "handleErrorReply()", reply.data_);
throw new InternalErrorException(InternalErrorException.UNKNOWN, codePoint);
default:
// We don't know what the reply is. Throw exception and be done.
Trace.log(Trace.ERROR, "handleErrorReply()", reply.data_);
throw new InternalErrorException(InternalErrorException.DATA_STREAM_UNKNOWN, codePoint);
}
}
}
// If we get to here, we should have a list of messages to throw
if (as400MsgList.size() > 0)
{
// We need to expand out the Vector of AS400Message[]'s to individual
// AS400Message's in a single array in order to construct our AS400Exception.
int numberOfMessages = 0;
int msgListSize = as400MsgList.size();
int msgNumber = 0;
// Determine number of AS400Message objects we have
for (int i = 0; i < msgListSize; ++i)
{
numberOfMessages += ((AS400Message[])as400MsgList.elementAt(i)).length;
}
// Now populate the single AS400Message[] with the AS400Message objects
msgs = new AS400Message[numberOfMessages];
AS400Message[] m;
for (int i = 0; i < msgListSize; ++i)
{
m = (AS400Message[])as400MsgList.elementAt(i);
for (int j = 0; j < m.length; ++j)
{
msgs[msgNumber++] = m[j];
}
}
}
return (msgs == null ? new AS400Message[0] : msgs);
}
/**
*Reads the record with the specified record number.
*@param recordNumber The record number of the record to be read. The
*recordNumber must be greater than zero.
*@return The record read. If the record is not found, null is returned.
*@exception AS400Exception If the server returns an error message.
*@exception AS400SecurityException If a security or authority error occurs.
*@exception ConnectionDroppedException If the connection is dropped unexpectedly.
*@exception InterruptedException If this thread is interrupted.
*@exception IOException If an error occurs while communicating with the server.
**/
public Record read(int recordNumber)
throws AS400Exception,
AS400SecurityException,
InterruptedException,
IOException
{
if (cacheRecords_) //@C0A
return super.read(recordNumber); //@C0A
int shr; // Type of locking for the record
// @A1C
if ((openType_ == AS400File.READ_ONLY) ||
((openType_ == AS400File.READ_WRITE) && readNoUpdate_)) // @A1A
{
// Read only
shr = SHR_READ_NORM;
}
else
{ // READ_WRITE
shr = SHR_UPD_NORM;
}
// Vector replys = sendRequestAndReceiveReplies(DDMRequestDataStream.getRequestS38GETD(dclName_, recordFormat_, 0x08, shr, DATA_DTA_DTARCD,recordNumber, // @A1D
Vector replys = sendRequestAndReceiveReplies(DDMRequestDataStream.getRequestS38GETD(dclName_, recordFormatCTLLName_, 0x08, shr, DATA_DTA_DTARCD,recordNumber, system_), newCorrelationId()); // @A1A @B6C
Record[] returned = processReadReply(replys, false); // @A1C
return(returned == null)? null : returned[0];
}
/**
*Reads all the records in the file. Helper function.
*@param fileType The type of file. Valid values are: key or seq
*@return The records read.
*@exception AS400Exception If the server returns an error message.
*@exception AS400SecurityException If a security or authority error occurs.
*@exception ConnectionDroppedException If the connection is dropped unexpectedly.
*@exception InterruptedException If this thread is interrupted.
*@exception IOException If an error occurs while communicating with the server.
*@exception ServerStartupException If the host server cannot be started.
*@exception UnknownHostException If the server cannot be located.
**/
public Record[] readAll(String fileType, int bf) //@D0C
throws AS400Exception,
AS400SecurityException,
InterruptedException,
IOException
{
//@B0A: Changed readAll() to not use the ULDRECF codepoint in the DDM data stream.
// This is because ULDRECF (Unload all records from file) does not handle
// null field values in records. (It throws us a data mapping error on
// the data stream, which would then cause an OutOfMemory exception
// or other strange errors in our code, because the data stream format
// was no longer correct.)
// readAll() now just uses the S38 extensions. It gets the first record
// and then gets the next record in a loop until null is returned.
// Using the S38 extensions gives better performance anyway (supposedly).
// See SequentialFile.readAll() and KeyedFile.readAll() for other changes.
// Other changes were also made in DDMRequestDataStream and DDMTerm.
//@B0A: start block
// readAll is supposed to return at least a Record[] of size 0, never null
Record[] recArray = new Record[0];
synchronized(this) // We synchronize because this file object
{ // isn't supposed to be open (as far as the user knows).
// Use a calculated blocking factor, else use a large blocking factor
//@D0M int bf = 2048/(recordFormat_.getNewRecord().getRecordLength() + 16); //@D0C
//@D0M if (bf <= 0) bf = 1; //@D0C
//@E0 - We don't want to use COMMIT_LOCK_LEVEL_ALL in case commitment control is on because
// inside readAll(), the file isn't supposed to be open, so we should treat it as such.
openFile2(AS400File.READ_ONLY, bf, AS400File.COMMIT_LOCK_LEVEL_NONE, fileType); //@D0C @E0C
// The vector to hold the records as we retrieve them.
Vector allRecords = new Vector();
// The following block was copied from readRecord()
int shr; // Type of locking for the record
if ((openType_ == AS400File.READ_ONLY) ||
((openType_ == AS400File.READ_WRITE) && readNoUpdate_))
{ // Read only
shr = SHR_READ_NORM;
}
else
{ // READ_WRITE; get the record for update
shr = SHR_UPD_NORM;
}
// Get the records
// Initialize returned to be of TYPE_GET_FIRST
// As the loop continues, returned is of TYPE_GET_NEXT
for (Record[] returned = processReadReply(sendRequestAndReceiveReplies(DDMRequestDataStream.getRequestS38GET(dclName_, TYPE_GET_FIRST, shr, DATA_DTA_DTARCD), newCorrelationId()), false); //@B6C
returned != null;
returned = processReadReply(sendRequestAndReceiveReplies(DDMRequestDataStream.getRequestS38GET(dclName_, TYPE_GET_NEXT, shr, DATA_DTA_DTARCD), newCorrelationId()), false)) //@B6C
{
// The reply is an array of records, so add each of them to the vector
for (int i=0; i
0)
{
recArray = new Record[numRecs];
allRecords.copyInto(recArray);
}
close(); // Need to close the file since we opened it earlier.
} // The file is not supposed to be open to the user.
return recArray;
//@B0A: end block
//@B0D: start block
/*
// Connect to the server. Note: If we have already connected, that connection
// will be used.
connect();
// Send the request to read all records. Because the reply object may contain many records
// we need to deal with the possibility of chained replies.
Vector replys = sendRequestAndReceiveReplies(DDMRequestDataStream.getRequestULDRECF(fileType, library_, file_, member_,
system_), server_.newCorrelationId());
Record[] records = null;
int index = lookForCodePoint(DDMTerm.RECORD, replys);
if (index == -1)
{ // Check for any RECAL terms. These contain the active record if we encounter an inactive record. This covers the
// case where records 1, 3, 5, 7, ... have been deleted. In this event we would have only RECAL's returned
// which would contain records 2, 4, 6, 8, ...
index = lookForCodePoint(DDMTerm.RECAL, replys);
}
if (index != -1)
{
// Records returned. Process the reply object(s)
// The last DDMTERM of the last reply received is a RECCNT and contains the number of records read.
int numReplies = replys.size();
DDMDataStream reply = (DDMDataStream)replys.elementAt(numReplies - 1);
int numRead = reply.get32bit(reply.data_.length - 4);
records = new Record[numRead];
if (numRead > 0)
{
int recordNumber = 1;
int recOffset;
int offset;
boolean done;
for (int i = index, j = 0; i < numReplies; ++i)
{
reply = (DDMDataStream)replys.elementAt(i);
done = false;
offset = 10;
while (!done)
{
// Find the next occurrence of a RECORD or RECAL term. We get a RECAL which will contain a
// RECORD when we encounter an inactive (deleted) record position. Otherwise we should just
// get a RECORD term
while (offset < reply.data_.length && ((reply.get16bit(offset - 2) != DDMTerm.RECORD) && (reply.get16bit(offset - 2) != DDMTerm.RECAL)))
{
offset += reply.get16bit(offset - 4);
}
if (offset >= reply.data_.length)
{
break;
}
// Determine offset of the record data and determine the record number
if (reply.get16bit(offset - 2) == DDMTerm.RECAL)
{ // We have encountered a spot for an inactive record. The RECAL term consists
// of the record number of the next active record along with the record data of the
// next active record.
recOffset = offset + 12; // Skip ahead to the record data
recordNumber = reply.get32bit(offset + 4); // Set record number to the record number of the
// active record
}
else
{ // This is an active record; we are already pointing to the record data and the record number
// was set appropriately last time through the loop.
recOffset = offset;
}
if (reply.get16bit(offset - 4) <= 0x7FFF)
{ // The record data immediately follows the code point if the
// record length + 4 is <= 32767
records[j] = recordFormat_.getNewRecord(reply.data_, recOffset);
}
else
{ // The record data starts 4 bytes after the code point if the
// record length + 4 is > 32767. The four bytes immediately
// after the code point contain the actual length of the record
// record data that follows.
records[j] = recordFormat_.getNewRecord(reply.data_, recOffset + 4);
}
// Set the record number of the record
try
{
records[j++].setRecordNumber(recordNumber++);
}
catch(PropertyVetoException e)
{ // We created the Record objects. There is no one to veto anything
} // so this is here to quit the compiler
if (j == numRead)
{
done = true;
}
// Get offset of next record
offset += reply.get16bit(offset - 4);
}
}
}
}
else if (replys.size() == 1 && ((DDMDataStream)replys.elementAt(0)).getCodePoint() == DDMTerm.RECCNT)
{
// If the file contains all deleted records, the only thing returned
// is a RECCNT. If this is the case, return an empty Record[]
return new Record[0];
}
else
{ // Error occurred
handleErrorReply(replys, 0);
}
return (records == null ? new Record[0] : records);
*/
//@B0D: end block
}
//@RBA
public Record[] readAllLong(String fileType, int bf)
throws AS400Exception,
AS400SecurityException,
InterruptedException,
IOException
{
//@B0A: Changed readAll() to not use the ULDRECF codepoint in the DDM data stream.
// This is because ULDRECF (Unload all records from file) does not handle
// null field values in records. (It throws us a data mapping error on
// the data stream, which would then cause an OutOfMemory exception
// or other strange errors in our code, because the data stream format
// was no longer correct.)
// readAll() now just uses the S38 extensions. It gets the first record
// and then gets the next record in a loop until null is returned.
// Using the S38 extensions gives better performance anyway (supposedly).
// See SequentialFile.readAll() and KeyedFile.readAll() for other changes.
// Other changes were also made in DDMRequestDataStream and DDMTerm.
//@B0A: start block
// readAll is supposed to return at least a Record[] of size 0, never null
Record[] recArray = new Record[0];
synchronized(this) // We synchronize because this file object
{ // isn't supposed to be open (as far as the user knows).
// Use a calculated blocking factor, else use a large blocking factor
//@D0M int bf = 2048/(recordFormat_.getNewRecord().getRecordLength() + 16); //@D0C
//@D0M if (bf <= 0) bf = 1; //@D0C
//@E0 - We don't want to use COMMIT_LOCK_LEVEL_ALL in case commitment control is on because
// inside readAll(), the file isn't supposed to be open, so we should treat it as such.
openFile2(AS400File.READ_ONLY, bf, AS400File.COMMIT_LOCK_LEVEL_NONE, fileType); //@D0C @E0C
// The vector to hold the records as we retrieve them.
Vector allRecords = new Vector();
// The following block was copied from readRecord()
int shr; // Type of locking for the record
if ((openType_ == AS400File.READ_ONLY) ||
((openType_ == AS400File.READ_WRITE) && readNoUpdate_))
{ // Read only
shr = SHR_READ_NORM;
}
else
{ // READ_WRITE; get the record for update
shr = SHR_UPD_NORM;
}
// Get the records
// Initialize returned to be of TYPE_GET_FIRST
// As the loop continues, returned is of TYPE_GET_NEXT
for (Record[] returned = processReadReplyLong(sendRequestAndReceiveReplies(DDMRequestDataStream.getRequestS38GET(dclName_, TYPE_GET_FIRST, shr, DATA_DTA_DTARCD), newCorrelationId()), false); //@B6C
returned != null;
returned = processReadReplyLong(sendRequestAndReceiveReplies(DDMRequestDataStream.getRequestS38GET(dclName_, TYPE_GET_NEXT, shr, DATA_DTA_DTARCD), newCorrelationId()), false)) //@B6C
{
// The reply is an array of records, so add each of them to the vector
for (int i=0; i 0)
{
recArray = new Record[numRecs];
allRecords.copyInto(recArray);
}
close(); // Need to close the file since we opened it earlier.
} // The file is not supposed to be open to the user.
return recArray;
}
/**
*Reads the first record with the specified key based on the specified type of read.
*@param key The values that make up the key with which to find the record.
*@param searchType The type of read. This value is one of the TYPE_GETKEY_* constants.
*@return The record read.
*@exception AS400Exception If the server returns an error message.
*@exception AS400SecurityException If a security or authority error occurs.
*@exception ConnectionDroppedException If the connection is dropped unexpectedly.
*@exception InterruptedException If this thread is interrupted.
*@exception IOException If an error occurs while communicating with the server.
**/
public Record read(Object[] key, int searchType)
throws AS400Exception,
AS400SecurityException,
InterruptedException,
IOException
{
int shr; // Type of locking for the record
if ((openType_ == AS400File.READ_ONLY) ||
((openType_ == AS400File.READ_WRITE) && readNoUpdate_)) // @A1A
{
// Read only
shr = SHR_READ_NORM;
}
else
{ // READ_WRITE
shr = SHR_UPD_NORM;
}
// Vector replys = sendRequestAndReceiveReplies(DDMRequestDataStream.getRequestS38GETK(dclName_, recordFormat_, type, shr, DATA_DTA_DTARCD, key, system_), server_.newCorrelationId()); // @A1D
Vector replys = sendRequestAndReceiveReplies(DDMRequestDataStream.getRequestS38GETK(dclName_, recordFormat_, recordFormatCTLLName_, searchType, shr, DATA_DTA_DTARCD, key, system_), newCorrelationId()); // @A1A @B6C
// Call processReadReply to extract the records read (or throw an
// exception if appropriate)
Record[] returned = processReadReply(replys, false); // @A1C
if (cacheRecords_) //@C0A
{
//@C0A
cache_.setIsEmpty(); //@C0A
} //@C0A
return(returned == null)? null : returned[0];
}
//@RBA
public Record readLong(Object[] key, int searchType)
throws AS400Exception,
AS400SecurityException,
InterruptedException,
IOException
{
int shr; // Type of locking for the record
if ((openType_ == AS400File.READ_ONLY) ||
((openType_ == AS400File.READ_WRITE) && readNoUpdate_)) // @A1A
{
// Read only
shr = SHR_READ_NORM;
}
else
{ // READ_WRITE
shr = SHR_UPD_NORM;
}
// Vector replys = sendRequestAndReceiveReplies(DDMRequestDataStream.getRequestS38GETK(dclName_, recordFormat_, type, shr, DATA_DTA_DTARCD, key, system_), server_.newCorrelationId()); // @A1D
Vector replys = sendRequestAndReceiveReplies(DDMRequestDataStream.getRequestS38GETK(dclName_, recordFormat_, recordFormatCTLLName_, searchType, shr, DATA_DTA_DTARCD, key, system_), newCorrelationId()); // @A1A @B6C
// Call processReadReply to extract the records read (or throw an
// exception if appropriate)
Record[] returned = processReadReplyLong(replys, false); // @A1C
if (cacheRecords_) //@C0A
{
//@C0A
cache_.setIsEmpty(); //@C0A
} //@C0A
return(returned == null)? null : returned[0];
}
// @A1A
/**
*Reads the first record with the specified key based on the specified type of read.
*@param key The byte array that contains the byte values that make up the key with which to find the record.
*@param searchType The type of read. This value is one of the TYPE_GETKEY_* constants.
*@param numberOfKeyFields The number of key fields contained in the byte array key.
*@return The record read.
*@exception AS400Exception If the server returns an error message.
*@exception AS400SecurityException If a security or authority error occurs.
*@exception ConnectionDroppedException If the connection is dropped unexpectedly.
*@exception InterruptedException If this thread is interrupted.
*@exception IOException If an error occurs while communicating with the server.
**/
public Record read(byte[] key, int searchType, int numberOfKeyFields)
throws AS400Exception,
AS400SecurityException,
InterruptedException,
IOException
{
int shr; // Type of locking for the record
if ((openType_ == AS400File.READ_ONLY) ||
((openType_ == AS400File.READ_WRITE) && readNoUpdate_)) // @A1A
{
// Read only
shr = SHR_READ_NORM;
}
else
{ // READ_WRITE
shr = SHR_UPD_NORM;
}
// Vector replys = sendRequestAndReceiveReplies(DDMRequestDataStream.getRequestS38GETK(dclName_, recordFormat_, type, shr, DATA_DTA_DTARCD, key, system_, numberOfKeyFields), server_.newCorrelationId()); // @A1D
Vector replys = sendRequestAndReceiveReplies(DDMRequestDataStream.getRequestS38GETK(dclName_, recordFormatCTLLName_, searchType, shr, DATA_DTA_DTARCD, key, system_, numberOfKeyFields), newCorrelationId()); // @A1A @B6C
// Call processReadReply to extract the records read (or throw an
// exception if appropriate)
Record[] returned = processReadReply(replys, false); // @A1C
if (cacheRecords_) //@C0A
{
//@C0A
cache_.setIsEmpty(); //@C0A
} //@C0A
return(returned == null)? null : returned[0];
}
//@RBA
public Record readLong(byte[] key, int searchType, int numberOfKeyFields)
throws AS400Exception,
AS400SecurityException,
InterruptedException,
IOException
{
int shr; // Type of locking for the record
if ((openType_ == AS400File.READ_ONLY) ||
((openType_ == AS400File.READ_WRITE) && readNoUpdate_)) // @A1A
{
// Read only
shr = SHR_READ_NORM;
}
else
{ // READ_WRITE
shr = SHR_UPD_NORM;
}
// Vector replys = sendRequestAndReceiveReplies(DDMRequestDataStream.getRequestS38GETK(dclName_, recordFormat_, type, shr, DATA_DTA_DTARCD, key, system_, numberOfKeyFields), server_.newCorrelationId()); // @A1D
Vector replys = sendRequestAndReceiveReplies(DDMRequestDataStream.getRequestS38GETK(dclName_, recordFormatCTLLName_, searchType, shr, DATA_DTA_DTARCD, key, system_, numberOfKeyFields), newCorrelationId()); // @A1A @B6C
// Call processReadReply to extract the records read (or throw an
// exception if appropriate)
Record[] returned = processReadReplyLong(replys, false); // @A1C
if (cacheRecords_) //@C0A
{
//@C0A
cache_.setIsEmpty(); //@C0A
} //@C0A
return(returned == null)? null : returned[0];
}
/**
*Reads a record from the file. Which record to read is determined by the type
*argument.
*@param searchType The type of get to execute. Valid values are:
*
*- TYPE_GET_FIRST
*
- TYPE_GET_NEXT
*
- TYPE_GET_LAST
*
- TYPE_GET_PREV
*@return the record read
*@exception AS400Exception If the server returns an error message.
*@exception AS400SecurityException If a security or authority error occurs.
*@exception ConnectionDroppedException If the connection is dropped unexpectedly.
*@exception InterruptedException If this thread is interrupted.
*@exception IOException If an error occurs while communicating with the server.
**/
public Record readRecord(int searchType)
throws AS400Exception,
AS400SecurityException,
InterruptedException,
IOException
{
int shr; // Type of locking for the record
// @A1C
if ((openType_ == AS400File.READ_ONLY) ||
((openType_ == AS400File.READ_WRITE) && readNoUpdate_)) // @A1A
{
// Read only
shr = SHR_READ_NORM;
}
else
{ // READ_WRITE; get the record for update
shr = SHR_UPD_NORM;
}
// Send the get S38GET request
Vector replys = sendRequestAndReceiveReplies(DDMRequestDataStream.getRequestS38GET(dclName_, searchType, shr, DATA_DTA_DTARCD), newCorrelationId()); //@B6C
// Call processReadReply to extract the records read (or throw an
// exception if appropriate)
Record[] returned = processReadReply(replys, false); // @A1C
return(returned == null)? null : returned[0];
}
//@RBA
public Record readRecordLong(int searchType)
throws AS400Exception,
AS400SecurityException,
InterruptedException,
IOException
{
int shr; // Type of locking for the record
// @A1C
if ((openType_ == AS400File.READ_ONLY) ||
((openType_ == AS400File.READ_WRITE) && readNoUpdate_)) // @A1A
{
// Read only
shr = SHR_READ_NORM;
}
else
{ // READ_WRITE; get the record for update
shr = SHR_UPD_NORM;
}
// Send the get S38GET request
Vector replys = sendRequestAndReceiveReplies(DDMRequestDataStream.getRequestS38GET(dclName_, searchType, shr, DATA_DTA_DTARCD), newCorrelationId()); //@B6C
// Call processReadReply to extract the records read (or throw an
// exception if appropriate)
Record[] returned = processReadReplyLong(replys, false); // @A1C
return(returned == null)? null : returned[0];
}
/**
*Reads records from the file. The next or previous 'blockingFactor_'
*records are retrieved depending on the direction specified.
*@param direction
*@return the records read
*@exception AS400Exception If the server returns an error message.
*@exception AS400SecurityException If a security or authority error occurs.
*@exception ConnectionDroppedException If the connection is dropped unexpectedly.
*@exception InterruptedException If this thread is interrupted.
*@exception IOException If an error occurs while communicating with the server.
**/
public Record[] readRecords(int direction)
throws AS400Exception,
AS400SecurityException,
InterruptedException,
IOException
{
int searchType = (direction == DDMRecordCache.FORWARD ? TYPE_GET_NEXT :
TYPE_GET_PREV);
// Send the S38GETM request
Vector replys = sendRequestAndReceiveReplies(DDMRequestDataStream.getRequestS38GETM(dclName_, blockingFactor_, searchType, SHR_READ_NORM, DATA_DTA_DTARCD, 0x01), newCorrelationId()); //@B6C
// Call processReadReply to extract the records read (or throw an
// exception if appropriate)
return processReadReply(replys, false); // @A1C
}
//@RBA
public Record[] readRecordsLong(int direction)
throws AS400Exception,
AS400SecurityException,
InterruptedException,
IOException
{
int searchType = (direction == DDMRecordCache.FORWARD ? TYPE_GET_NEXT :
TYPE_GET_PREV);
// Send the S38GETM request
Vector replys = sendRequestAndReceiveReplies(DDMRequestDataStream.getRequestS38GETM(dclName_, blockingFactor_, searchType, SHR_READ_NORM, DATA_DTA_DTARCD, 0x01), newCorrelationId()); //@B6C
// Call processReadReply to extract the records read (or throw an
// exception if appropriate)
return processReadReplyLong(replys, false); // @A1C
}
/**
*Rolls back any transactions since the last commit/rollback boundary. Invoking this
*method will cause all transactions under commitment control for this connection
*to be rolled back. This means that any AS400File object for which a commit
*lock level was specified and that was opened under this connection will have
*outstanding transactions rolled back.
*@exception AS400Exception If the server returns an error message.
*@exception AS400SecurityException If a security or authority error occurs.
*@exception ConnectionDroppedException If the connection is dropped unexpectedly.
*@exception InterruptedException If this thread is interrupted.
*@exception IOException If an error occurs while communicating with the server.
*@exception ServerStartupException If the host server cannot be started.
*@exception UnknownHostException If the server cannot be located.
**/
public void rollback() throws AS400Exception, AS400SecurityException, InterruptedException, IOException
{
// Connect to the server. Note: If we have already connected, that connection
// will be used.
connect();
// Send the rollback data stream request
Vector replys = sendRequestAndReceiveReplies(DDMRequestDataStream.getRequestRLLBCKUOW(), newCorrelationId()); //@B6C
// Reply expected: ENDUOWRM, with UOWDSP parameter = 1
if (replys.size() == 1 && ((DDMDataStream)replys.elementAt(0)).getCodePoint() == DDMTerm.ENDUOWRM)
{
DDMEndUnitOfWorkReply uowReply = new DDMEndUnitOfWorkReply(((DDMDataStream)replys.elementAt(0)).data_);
if (uowReply.getStatus() != 0x02)
{
// Status of logical unit of work committed not returned; should not happen unless
// we are constructing the request wrong.
Trace.log(Trace.ERROR, "AS400FileImplRemote.rollback()",
uowReply.data_);
throw new InternalErrorException(InternalErrorException.UNKNOWN, uowReply.getStatus());
}
}
else
{
handleErrorReply(replys, 0);
}
}
/**
*Sends a request and receives all the replies. This method is used when the potential
*for chained replies exists.
*@param req The request to be sent.
*@param correlationId The correlation id for the request.
*@exception AS400Exception If the server returns an error message.
*@exception ConnectionDroppedException If the connection is dropped unexpectedly.
*@exception InterruptedException If this thread is interrupted.
*@exception IOException If an error occurs while communicating with the server.
**/
private Vector sendRequestAndReceiveReplies(DDMDataStream req, int correlationId)
throws InterruptedException, IOException, AS400SecurityException
{
connect();
DDMDataStream reply = null;
try
{
synchronized(server_) //@F1A - make sure all of our operations are atomic
{ //@F1A - we are synchronized so we don't interrupt any other sends
// that are sensitive.
server_.send(req, correlationId);
reply = (DDMDataStream)server_.receive(correlationId);
} //@F1A
}
catch (ConnectionDroppedException e)
{
// Connection dropped. Disconnect server and rethrow.
Trace.log(Trace.ERROR, "ConnectionDroppedException:", e);
system_.disconnectServer(server_);
//@C1 - Setting the server_ object to null means that
// any operations on this AS400File object after the connection has been
// dropped will result in a NullPointerException. By leaving the server_ object
// around, any subsequent operations should also throw a ConnectionDroppedException.
//@C1D server_ = null;
resetState();
throw e;
}
// Receive all replies from the read into a vector.
Vector replys = new Vector();
while (reply.isChained())
{
replys.addElement(reply);
try
{
reply = (DDMDataStream)server_.receive(correlationId);
}
catch (ConnectionDroppedException e)
{
// Connection dropped. Disconnect server and rethrow.
Trace.log(Trace.ERROR, "ConnectionDroppedException:", e);
system_.disconnectServer(server_);
//@C1 - Setting the server_ object to null means that
// any operations on this AS400File object after the connection has been
// dropped will result in a NullPointerException. By leaving the server_ object
// around, any subsequent operations should also throw a ConnectionDroppedException.
//@C1D server_ = null;
resetState();
throw e;
}
}
// Add the unchained reply to the vector of replys
replys.addElement(reply);
return replys;
}
/**
*Sets the declared file name (DCLNAM). The declared file name for each
*file object must be unique. This method will generate a unique declared file
*name.
**/
public void setDCLName()
{
// Convert nextDCLName_ to a Long and then to a string
long nextDCLName;
synchronized (nextDCLNameLock_) {
nextDCLName = nextDCLName_++; // need to synchronize the incrementation
}
String nextDCLNameAsString = Long.toString(nextDCLName);
// Copy EBCDIC version of nextDCLName to dclNAme_ and blank pad dclName_
// to 8 bytes
for (int i = 0; i < nextDCLNameAsString.length(); i++)
{
char c = nextDCLNameAsString.charAt(i);
dclName_[i] = (byte) (c & 0x000f | 0x00f0);
}
padBytes(dclName_, nextDCLNameAsString.length(),
8 - nextDCLNameAsString.length(), (byte)0x40);
}
/**
*Updates the record at the current cursor position.
*@exception AS400Exception If the server returns an error message.
*@exception AS400SecurityException If a security or authority error occurs.
*@exception ConnectionDroppedException If the connection is dropped unexpectedly.
*@exception InterruptedException If this thread is interrupted.
*@exception IOException If an error occurs while communicating with the server.
**/
public void update(Record record)
throws AS400Exception,
AS400SecurityException,
InterruptedException,
IOException
{
if (record.getRecordLength() != openFeedback_.getRecordLength())
{
Trace.log(Trace.ERROR, "Incorrect record length for file :");
Trace.log(Trace.ERROR, "record.getRecordLength() :" + String.valueOf(record.getRecordLength()));
throw new ExtendedIllegalArgumentException("record", ExtendedIllegalArgumentException. PARAMETER_VALUE_NOT_VALID);
}
connect();
// getObjectS38BUF requires an array of records
Record[] records = new Record[1];
records[0] = record;
// We will be chaining an S38BUF to the S38UPDAT request. This requires that
// the correlation ids be the same.
int correlationId = newCorrelationId(); //@B6C
// Send the S38UPDAT request followed by the S38BUF object containing the
// record with which to update.
DDMRequestDataStream req = DDMRequestDataStream.getRequestS38UPDAT( TYPE_GET_SAME, SHR_UPD_NORM, DATA_DTA_DTARCD, dclName_);
req.setIsChained(true); // Indicate we are chaining an object to the request
req.setHasSameRequestCorrelation(true); // Indicate that they have the same
// correlation ids
// Start of changes for ----------------------------------------- @A2A
// Determine the maxNumberOfFieldsPerFormatInFile for the file
// remembering that multi-format logical files may have a different
// number of getNumberOfFields() per record format.
// DDMObjectDataStream.getObjectS38BUF() needs maxNumberOfFieldsPerFormatInFile
int maxNumberOfFieldsPerFormatInFile = -1;
if (rfCache_ == null)
{
// Use default maxNumberOfFieldsPerFormatInFile == -1
}
else
{
int numberOfRecordFormats = rfCache_.length;
for(int i = 0; i < numberOfRecordFormats; ++i)
{
maxNumberOfFieldsPerFormatInFile =
Math.max(maxNumberOfFieldsPerFormatInFile, rfCache_[i].getNumberOfFields());
}
}
// End of changes for ------------------------------------------- @A2A
// Get the S38BUF object(s) to send after the request
// Because we are updating, there will only be one item in dataToSend
DDMObjectDataStream[] dataToSend =
DDMObjectDataStream.getObjectS38BUF(records, openFeedback_, ssp_, // #SSPDDM1 - add ssp_ parm
maxNumberOfFieldsPerFormatInFile); //@A2C
Vector replys = null; //@F1A
synchronized(server_) //@F1A - both datastreams must be written atomically
{ //@F1A
try
{
server_.send(req, correlationId);
replys = sendRequestAndReceiveReplies(dataToSend[0], correlationId); //@F1M
}
catch (ConnectionDroppedException e)
{
// Connection dropped. Disconnect server and rethrow.
Trace.log(Trace.ERROR, "ConnectionDroppedException:", e);
system_.disconnectServer(server_);
//@C1 - Setting the server_ object to null means that
// any operations on this AS400File object after the connection has been
// dropped will result in a NullPointerException. By leaving the server_ object
// around, any subsequent operations should also throw a ConnectionDroppedException.
//@C1D server_ = null;
resetState();
throw e;
}
} //@F1A
// Reply expected: S38IOFB
if (((DDMDataStream)replys.elementAt(0)).getCodePoint() == DDMTerm.S38IOFB)
{
if (replys.size() != 1)
{
handleErrorReply(replys, 1);
}
}
else
{ // Error occurred
handleErrorReply(replys, 0);
}
}
/**
*Verifies the reply datastream as an S38MSGRM with the specified
*information. If msgID is not null, svrCode is not checked.
*If msgId is null, svrCode is verified.
@param replyParm The DDM reply data stream.
@param msgId The message id that this reply should contain.
@param svrCode The severity code that this reply should contain.
* @return true if datais is a S38MSGRM with specified information
*@exception AS400Exception If the server returns an error message.
*@exception AS400SecurityException If a security or authority error occurs.
*@exception ConnectionDroppedException If the connection is dropped unexpectedly.
*@exception InterruptedException If this thread is interrupted.
*@exception IOException If an error occurs while communicating with the server.
**/
public boolean verifyS38MSGRM(DDMDataStream replyParm, String msgId, int svrCode)
throws AS400Exception,
AS400SecurityException,
InterruptedException,
IOException
{
if (replyParm instanceof DDMReplyDataStream)
{
DDMReplyDataStream reply = (DDMReplyDataStream) replyParm;
DDMAS400MessageReply msgReply = new DDMAS400MessageReply(system_, reply.data_);
AS400Message[] msgs = msgReply.getAS400MessageList();
for (int i = 0; i < msgs.length; ++i)
{
if (msgId != null)
{ // Verify based on msgid only
if (msgs[i].getID().equalsIgnoreCase(msgId))
{
return true;
}
}
else
{ // Verify based on severity code only
if (msgs[i].getSeverity() == svrCode)
{
return true;
}
}
}
}
return false;
}
/**
*Writes an array of records to the file.
*@param records The records to write. The records must have a format
*which matches the record format of this object. To ensure that this
*requirement is met, use the
*RecordFormat.getNewRecord()
*method to obtain default records whose fields can be set appropriately by
*the Java program and then written to the file.
*@exception AS400Exception If the server returns an error message.
*@exception AS400SecurityException If a security or authority error occurs.
*@exception ConnectionDroppedException If the connection is dropped unexpectedly.
*@exception InterruptedException If this thread is interrupted.
*@exception IOException If an error occurs while communicating with the server.
**/
public void write(Record[] records)
throws AS400Exception,
AS400SecurityException,
InterruptedException,
IOException
{
if (records[0].getRecordLength() != openFeedback_.getRecordLength())
{
// Note: The typical cause of this error is that the application neglected to call AS400File.setRecordFormat() prior to calling open() or readAll().
if (Trace.traceOn_)
{
Trace.log(Trace.ERROR, "Incorrect record length for file: " +
"Expected " + String.valueOf(openFeedback_.getRecordLength()) +
", got " + String.valueOf(records[0].getRecordLength()));
}
throw new ExtendedIllegalArgumentException("records", ExtendedIllegalArgumentException. PARAMETER_VALUE_NOT_VALID);
}
connect();
// We will be chaining the S38BUF to the request, so the correlation ids must match
int correlationId = newCorrelationId(); //@B6C
DDMRequestDataStream req = DDMRequestDataStream.getRequestS38PUTM(dclName_);
req.setIsChained(true); // Indicate that the request is chained
req.setHasSameRequestCorrelation(true); // Indicate hat the ids will match
// Start of changes for ----------------------------------------- @A2A
// Determine the maxNumberOfFieldsPerFormatInFile for the file
// remembering that multi-format logical files may have a different
// number of getNumberOfFields() per record format.
// DDMObjectDataStream.getObjectS38BUF() needs maxNumberOfFieldsPerFormatInFile
int maxNumberOfFieldsPerFormatInFile = -1;
if (rfCache_ == null)
{
// Use default maxNumberOfFieldsPerFormatInFile == -1
}
else
{
int numberOfRecordFormats = rfCache_.length;
for(int i = 0; i < numberOfRecordFormats; ++i)
{
maxNumberOfFieldsPerFormatInFile =
Math.max(maxNumberOfFieldsPerFormatInFile, rfCache_[i].getNumberOfFields());
}
}
// End of changes for ------------------------------------------- @A2A
// Get the S38BUF object(s) containing the records to write.
DDMObjectDataStream[] dataToSend =
DDMObjectDataStream.getObjectS38BUF(records, openFeedback_, ssp_, // #SSPDDM1 - add ssp_ parm
maxNumberOfFieldsPerFormatInFile); //@A2A
// It is possible that we will have more than one S38BUF to send. This case
// occurs when the blocking factor is less than the number records to be written.
// In that case we do multiple of S38PUTMs of blocking factor number of records.
Vector replys;
synchronized(server_) //@F1A - both datastreams must be written atomically
{ //@F1A
for (int i = 0; i < dataToSend.length; ++i)
{ // For each S38BUF object, send the S38PUTM followed by the S38BUF
server_.send(req, correlationId);
replys = sendRequestAndReceiveReplies(dataToSend[i], correlationId);
// Reply expected: S38IOFB
if (((DDMDataStream)replys.elementAt(0)).getCodePoint() == DDMTerm.S38IOFB)
{
if (replys.size() != 1)
{
handleErrorReply(replys, 1);
}
}
else
{ // Error occurred
handleErrorReply(replys, 0);
}
}
} //@F1A
}
// @B1A
// @B6C
private int newCorrelationId()
{
synchronized(correlationIdLock_)
{
if (lastCorrelationId_ == 0x7fff) lastCorrelationId_ = 0;
return ++lastCorrelationId_;
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy