src.com.ibm.as400.access.NPDataStream Maven / Gradle / Ivy
///////////////////////////////////////////////////////////////////////////////
//
// 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