src.com.ibm.as400.access.NPDataStream 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: NPDataStream.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-2000 International Business Machines Corporation and
// others. All rights reserved.
//
///////////////////////////////////////////////////////////////////////////////
package com.ibm.as400.access;
import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStream;
/**
*Base class of Network Print data streams. This same class is used
*for both request and reply data streams.
**/
class NPDataStream extends ClientAccessDataStream
{
private static final String copyright = "Copyright (C) 1997-2000 International Business Machines Corporation and others.";
static final int NP_SERVER_ID = 0xE003;
// Network Print Server Template.
static final int TEMPLATE_LENGTH = 12;
static final int ACTION_ID_OFFSET = HEADER_LENGTH + 0;
static final int FLAGS_OFFSET = HEADER_LENGTH + 2;
static final int RC_OFFSET = HEADER_LENGTH + 6;
static final int EO_OFFSET = HEADER_LENGTH + 8;
static final int LAST_REPLY_MASK = (int)0x80000000;
static final int CODE_POINT_OFFSET = HEADER_LENGTH + TEMPLATE_LENGTH;
// @B1D // Network Print Server "objects".
// @B1D static final int SPOOLED_FILE = 0x0001;
// @B1D static final int WRITER_JOB = 0x0002;
// @B1D static final int PRINTER_DEVICE = 0x0003;
// @B1D static final int OUTPUT_QUEUE = 0x0004;
// @B1D static final int PRINTER_FILE = 0x0005;
// @B1D static final int LIBRARY = 0x0006;
// @B1D static final int RESOURCE = 0x0007;
// @B1D static final int NP_SERVER = 0x0008;
// Network Print Server "actions".
static final int CREATE = 0x0001;
static final int OPEN = 0x0002;
static final int READ = 0x0003;
static final int WRITE = 0x0004;
static final int CLOSE = 0x0005;
static final int HOLD = 0x0006;
static final int RELEASE = 0x0007;
static final int START = 0x0008;
static final int END = 0x0009;
static final int DELETE = 0x000A;
static final int MOVE = 0x000B;
static final int SEND = 0x000C;
static final int CALL_EXIT_PROGRAM = 0x000D;
static final int CHANGE_ATTRIBUTES = 0x000E;
static final int RETRIEVE_ATTRIBUTES = 0x000F;
static final int RETRIEVE_ATTRIBUTE_INFO = 0x0010;
static final int RETRIEVE_MESSAGE = 0x0011;
static final int ANSWER_MESSAGE = 0x0012;
static final int WATCH = 0x0013;
static final int CHECK_AUTHORITY = 0x0014;
static final int PURGE = 0x0015;
static final int LIST = 0x0016;
static final int SEEK = 0x0017;
static final int TELL = 0x0018;
static final int SELECT_PAGE = 0x0019; /* @A1A */
static final int OPEN_MODIFIED_SPLF = 0x001A; /* @A2A */
static final int COPY = 0x001B; /* @C1A */
// Network Print Server return codes
static final int RET_OK = 0x0000;
static final int RET_INV_REQ_STRUCT = 0x0001;
static final int RET_INV_REQ_ID = 0x0002;
static final int RET_INV_ACT_ID = 0x0003;
static final int RET_INV_REQ_ACT = 0x0004;
static final int RET_INV_CODEPOINT = 0x0005;
static final int RET_INV_ATTR = 0x0006;
static final int RET_INV_ATTRVALUE = 0x0007;
static final int RET_NOT_AUTHORIZED = 0x0008;
static final int RET_CPF_MESSAGE = 0x0009;
static final int RET_INV_SPLF_HANDLE = 0x000A;
static final int RET_SPLF_CREATE_ERR = 0x000B;
static final int RET_CL_ERROR = 0x000C;
static final int RET_SPLF_NOT_OPEN = 0x000D;
static final int RET_SPLF_NO_MESSAGE = 0x000E;
static final int RET_SPLF_OPEN_FAILED= 0x000F;
static final int RET_SEEK_OFF_BAD = 0x0010;
static final int RET_SEEK_FAILED = 0x0011;
static final int RET_READ_INCOMPLETE = 0x0012;
static final int RET_READ_EOF = 0x0013;
static final int RET_EMPTY_LIST = 0x0014;
static final int RET_FUNCTION_NOT_SUP= 0x0015;
static final int RET_CANNOT_ACC_PRTF = 0x0016;
static final int RET_CANNOT_ACC_PRTF_ATTR = 0x0017;
static final int RET_WRITER_NOT_FOUND = 0x0018;
static final int RET_RETURN_CP_MISSING = 0x0019;
static final int RET_NO_EXIT_PGM = 0x001A;
static final int RET_EXIT_PGM_DENIED = 0x001B;
static final int RET_NLV_NOT_AVAILABLE = 0x001C;
static final int RET_INV_BYTE_COUNT = 0x001D; /* @A1A */
static final int RET_SPLF_NOT_FOUND = 0x001E; /* @A1A */
static final int RET_INV_DATA_TYPE = 0x001F; /* @A1A */
static final int RET_READ_OUT_OF_RANGE = 0x0020; /* @A1A */
static final int RET_PAGE_OUT_OF_RANGE = 0x0021; /* @A1A */
private int reqRepID_;
private int correlation_;
private int actionID_;
// @B1D private int hostCCSID_;
private ConverterImpl converter_; // @B1C
//
// we could use a hashTable (dictionary) of codepoints using their ID as
// hashing keys but that is probably excessive since there are only 16 codepoints
// currently and all IDs are sequential starting at ID # 1, it will work to use
// a simple array of codepoints with their IDs as the index
//
// private Vector codePoints = new Vector();
private NPCodePoint[] codePoints_ = new NPCodePoint[NPCodePoint.MAX_CODEPOINT_ID+1];
NPDataStream(NPDataStream ds)
{
super();
data_ = new byte[ds.data_.length];
System.arraycopy(ds.data_, 0,
data_, 0,
data_.length);
int i;
reqRepID_ = ds.reqRepID_;
correlation_ = ds.correlation_;
actionID_ = ds.actionID_;
// @B1D hostCCSID_ = ds.hostCCSID_;
converter_ = ds.converter_; // @B1A
for (i = 0; i< codePoints_.length; i++)
{
if (ds.codePoints_[i] != null)
{
codePoints_[i] = (NPCodePoint)ds.codePoints_[i].clone();
}
}
}
NPDataStream()
{
super();
reqRepID_ = 0;
correlation_ = 0;
actionID_ = 0;
// @B1D hostCCSID_ = 0;
converter_ = null; // @B1A
}
NPDataStream( int reqRepID )
{
super();
correlation_ = 0;
actionID_ = 0;
// @B1D hostCCSID_ = 0;
converter_ = null; // @B1A
setObject( reqRepID );
}
// This method is called when we need to copy this data stream into
// a new datastream. We (NP) have this method called when we are
// receiving a list of objects back from the server - each item in the
// the list has a new datastream created to catch it.
Object getNewDataStream()
{
return new NPDataStream(this);
}
public int hashCode()
{
return (reqRepID_ | 0x8000);
}
void setObject( int reqRepID )
{
this.reqRepID_ = reqRepID;
}
void setAction( int actionID )
{
this.actionID_ = actionID;
}
void setCorrelation( int correlation )
{
this.correlation_ = correlation;
}
int getCorrelationID()
{
return this.correlation_ ;
}
// @B1D void setHostCCSID(int ccsid )
// @B1D {
// @B1D hostCCSID_ = ccsid;
// @B1D // Look at any code points to determine how big this
// @B1D // data stream is. There does not have to be any code
// @B1D // points. The NP data stream always uses a fixed size
// @B1D // template, any variable data for the request/reply
// @B1D // is carried in code points.
// @B1D
// @B1D for( int i=0; i < codePoints_.length; i++ )
// @B1D {
// @B1D NPCodePoint cp = codePoints_[i];
// @B1D if (cp != null)
// @B1D {
// @B1D // Let the codepoint know the host ccsid encoding for any text data
// @B1D cp.setHostCCSID(hostCCSID_);
// @B1D }
// @B1D }
// @B1D
// @B1D
// @B1D }
void setConverter(ConverterImpl converter) // @B1A
{ // @B1A
converter_ = converter; // @B1A
// @B1A
// Look at any code points to determine how big this // @B1A
// data stream is. There does not have to be any code // @B1A
// points. The NP data stream always uses a fixed size // @B1A
// template, any variable data for the request/reply // @B1A
// is carried in code points. // @B1A
for( int i=0; i < codePoints_.length; i++ ) // @B1A
{ // @B1A
NPCodePoint cp = codePoints_[i]; // @B1A
if (cp != null) // @B1A
{ // @B1A
// Let the codepoint know the host ccsid encoding for any text data // @B1A
cp.setConverter(converter_); // @B1A
} // @B1A
} // @B1A
} // @B1A
void addCodePoint( NPCodePoint codePoint )
{
int ID = codePoint.getID();
if (validCPID(ID))
{
codePoints_[ID] = codePoint;
}
}
NPCodePoint getCodePoint(int ID)
{
NPCodePoint cp = null;
if (validCPID(ID))
{
cp = codePoints_[ID];
}
return (cp);
}
void resetCodePoints()
{
for (int i = 0; i 0x7FFF (negative number)
System.arraycopy(cpData, dataOffset, // source
data_, cpOffset, // dest
length);
cpOffset += length;
}
}
}
super.write(out);
}
// Most of the work for a reply data stream happens here.
int readAfterHeader( InputStream in )
throws IOException
{
// The header has already been read by ClientAccessDataStream().
int bytesAvailable = getLength() - HEADER_LENGTH;
// Read the required template.
int bytesRead = readFromStream(in, data_, HEADER_LENGTH, getTemplateLen() );
// Check the return code
// don't log an ERROR here - just info; this dumps a stack trace and
// some non-0 return codes are OK (empty list and some others).
if( 0 != getReturnCode() )
{
Trace.log(Trace.INFORMATION, " Netprint Datastream Return code was " + getReturnCode());
}
// Check for any code points in the data stream. Code points are optional.
// If there are codepoints coming back, they should be prestored in the the
// array of codepoints.
byte[] llcp = new byte[NPCodePoint.LEN_HEADER];
byte[] cpData = null;
NPCodePoint cp = null;
int temp;
int dataOffset;
while( bytesRead < bytesAvailable )
{
temp = readFromStream(in, llcp, 0, llcp.length);
if (temp != llcp.length)
{
Trace.log(Trace.ERROR, "Didn't read 6 bytes that we needed to!");
throw new IOException(Integer.toString(llcp.length));
}
bytesRead += temp;
int cpDataLength = BinaryConverter.byteArrayToInt(llcp, 0)-llcp.length; //sms
int cpID = BinaryConverter.byteArrayToUnsignedShort(llcp,4); //sms
// if the codepoint ID is not valid or is not prestored
// create a generic codepoint
if (!validCPID(cpID) || (codePoints_[cpID] == null))
{
Trace.log(Trace.ERROR, "NPDataStream receiving orphan code point ID =" + cpID);
cp = new NPCodePoint(cpID);
if (!validCPID(cpID))
{
codePoints_[0] = cp;
} else {
codePoints_[cpID] = cp;
}
} else {
cp = codePoints_[cpID];
}
// let the codepoint know the host ccsid encoding for any text data
// @B1D cp.setHostCCSID(hostCCSID_);
cp.setConverter(converter_); // @B1A
cpData = cp.getDataBuffer(cpDataLength); // ask codepoint for a receive buffer
dataOffset = cp.getOffset(); // find out where to start writing
temp = readFromStream(in, cpData, dataOffset, cpDataLength);
if (temp != cpDataLength)
{
Trace.log(Trace.ERROR, "Didn't read " + cpDataLength + " bytes that we needed!");
throw new IOException(Integer.toString(cpDataLength));
}
bytesRead += temp;
}
return( bytesRead );
}
private boolean validCPID(int ID)
{
if ((ID > 0) && (ID < codePoints_.length))
{
return true;
}
return false;
}
} // end of NPDataStream class
© 2015 - 2025 Weber Informatics LLC | Privacy Policy