jtopenlite.com.ibm.jtopenlite.file.FileConnection Maven / Gradle / Ivy
Show all versions of jt400 Show documentation
///////////////////////////////////////////////////////////////////////////////
//
// JTOpenLite
//
// Filename: FileConnection.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) 2011-2012 International Business Machines Corporation and
// others. All rights reserved.
//
///////////////////////////////////////////////////////////////////////////////
package com.ibm.jtopenlite.file;
import com.ibm.jtopenlite.*;
import java.io.*;
import java.net.*;
import java.util.ArrayList;
import java.util.List;
import javax.net.ssl.SSLSocketFactory;
/**
* Represents a TCP/IP socket connection to the System i File host server
* (QSERVER/QPWFSERVSO prestart jobs).
*
For an example, see {@link com.ibm.jtopenlite.samples.AccessIfsFile}.
**/
public class FileConnection extends HostServerConnection //implements Connection
{
public static final int DEFAULT_FILE_SERVER_PORT = 8473;
public static final int DEFAULT_SSL_FILE_SERVER_PORT = 9473;
private int ccsid_;
private int datastreamLevel_;
private int maxDataBlockSize_;
private int correlationID_ = 1;
private int newCorrelationID()
{
if (correlationID_ == 0x7FFFFFFF) correlationID_ = 0;
return ++correlationID_;
}
private FileConnection(SystemInfo info, Socket socket, HostInputStream in, HostOutputStream out, int ccsid, int datastreamLevel, int maxDataBlockSize, String user, String jobName)
{
super(info, user, jobName, socket, in, out);
ccsid_ = ccsid;
datastreamLevel_ = datastreamLevel;
maxDataBlockSize_ = maxDataBlockSize;
}
protected void sendEndJobRequest() throws IOException
{
}
public static FileConnection getConnection(String system, String user, String password) throws IOException
{
return getConnection(false, system, user, password);
}
public static FileConnection getConnection(final boolean isSSL, String system, String user, String password) throws IOException
{
SignonConnection conn = SignonConnection.getConnection(isSSL, system, user, password);
try
{
return getConnection(isSSL, conn.getInfo(), user, password);
}
finally
{
conn.close();
}
}
public static FileConnection getConnection(SystemInfo info, String user, String password) throws IOException
{
return getConnection(false, info, user, password);
}
public static FileConnection getConnection(final boolean isSSL, SystemInfo info, String user, String password) throws IOException
{
return getConnection(isSSL, info, user, password, isSSL ? DEFAULT_SSL_FILE_SERVER_PORT : DEFAULT_FILE_SERVER_PORT);
}
public static FileConnection getConnection(SystemInfo info, String user, String password, int filePort) throws IOException
{
return getConnection(false, info, user, password, filePort);
}
public static FileConnection getConnection(final boolean isSSL, SystemInfo info, String user, String password, int filePort) throws IOException
{
if (filePort < 0 || filePort > 65535)
{
throw new IOException("Bad file port: "+filePort);
}
FileConnection conn = null;
Socket fileServer = isSSL? SSLSocketFactory.getDefault().createSocket(info.getSystem(), filePort) : new Socket(info.getSystem(), filePort);
InputStream in = fileServer.getInputStream();
OutputStream out = fileServer.getOutputStream();
try
{
HostOutputStream dout = new HostOutputStream(new BufferedOutputStream(out));
HostInputStream din = new HostInputStream(new BufferedInputStream(in));
String jobName = connect(info, dout, din, 0xE002, user, password);
sendExchangeAttributesRequest(dout);
dout.flush();
int length = din.readInt();
if (length < 22)
{
throw DataStreamException.badLength("fileExchangeAttributes", length);
}
din.skipBytes(16);
int rc = din.readShort();
if (rc != 0)
{
throw DataStreamException.badReturnCode("fileExchangeAttributes", rc);
}
int datastreamLevel = din.readShort();
din.skipBytes(2);
int maxDataBlockSize = din.readInt();
//int ccsidLL = din.readInt();
//int ccsidCP = din.readShort();
din.skipBytes(6);
int ccsid = din.readShort();
int remaining = length-38;
din.skipBytes(remaining);
conn = new FileConnection(info, fileServer, din, dout, ccsid, datastreamLevel, maxDataBlockSize, user, jobName);
return conn;
}
finally
{
if (conn == null)
{
in.close();
out.close();
fileServer.close();
}
}
}
/**
* Returns a list of files under the specified directory or file spec.
* For example, these all return the same file listing:
*
* - list("/home/smith")
* - list("/home/smith/*")
* - list("/home/smith/../smith")
*
**/
public List listFiles(String dir) throws IOException
{
if (dir.indexOf("*") < 0)
{
if (!dir.endsWith("/"))
{
dir = dir + "/*";
}
else
{
dir = dir + "*";
}
}
String parent = dir;
int slash = parent.lastIndexOf("/");
if (slash < 0)
{
parent = "";
}
else if (slash > 0)
{
parent = parent.substring(0,slash+1);
}
sendListFilesRequest(Conv.stringToUnicodeByteArray(dir));
out_.flush();
ArrayList files = new ArrayList();
int chain = 1;
while (chain != 0)
{
int length = in_.readInt();
if (length < 24)
{
throw DataStreamException.badLength("listFiles", length);
}
int headerID = in_.readShort();
int serverID = in_.readShort();
int csInstance = in_.readInt();
int correlationID = in_.readInt();
int templateLength = in_.readShort();
int reqRepID = in_.readShort();
chain = in_.readShort();
if (reqRepID == 0x8001)
{
int rc = in_.readShort();
int remaining = length-24;
in_.skipBytes(remaining);
if (rc != FileConstants.RC_NO_MORE_FILES)
{
throw DataStreamException.badReturnCode("listFiles", rc);
}
}
else if (reqRepID == 0x8005)
{
int numRead = 22;
long createDate = convertDate(in_);
long modifyDate = convertDate(in_);
long accessDate = convertDate(in_);
int fileSize = in_.readInt();
int fixedAttributes = in_.readInt();
int objectType = in_.readShort();
int numExtAttrs = in_.readShort();
int bytesEANames = in_.readInt();
int bytesEAValues = in_.readInt();
int version = in_.readInt();
int amountAccessed = in_.readShort();
int accessHistory = in_.readByte();
int fileCCSID = in_.readShort();
int checkoutCCSID = in_.readShort();
int restartID = in_.readInt();
long largeFileSize = in_.readLong();
in_.skipBytes(2);
int symlink = in_.readByte(); // 91
numRead = 92;
int fileNameLLOffset = 20 + templateLength;
int toSkip = fileNameLLOffset-numRead;
in_.skipBytes(toSkip);
numRead += toSkip;
int fileNameLength = in_.readInt()-6;
in_.skipBytes(2); // 0x0002 -- Name CP.
numRead += 6;
byte[] nameBuf = new byte[fileNameLength];
in_.readFully(nameBuf);
numRead += fileNameLength;
String name = Conv.unicodeByteArrayToString(nameBuf, 0, nameBuf.length);
if (!name.equals(".") && !name.equals(".."))
{
FileHandle h = FileHandle.createEmptyHandle();
h.setName(name);
h.setPath(parent+name);
h.setDataCCSID(fileCCSID);
h.setCreateDate(createDate);
h.setModifyDate(modifyDate);
h.setAccessDate(accessDate);
h.setSize(largeFileSize);
h.setVersion(version);
h.setSymlink(symlink == 1);
h.setDirectory(objectType == 2);
files.add(h);
}
int remaining = length-numRead;
in_.skipBytes(remaining);
}
else
{
int remaining = length-22;
in_.skipBytes(remaining);
throw DataStreamException.badReply("listFiles", reqRepID);
}
}
return files;
}
/**
* Returns a list of filenames under the specified directory or file spec.
* For example, these all return the same file listing:
*
* - list("/home/smith")
* - list("/home/smith/*")
* - list("/home/smith/../smith")
*
* To return more than just the names of files, use {@link #listFiles listFiles()}.
**/
public List list(String dir) throws IOException
{
// System.out.println("Server CCSID: "+getInfo().getServerCCSID());
if (dir.indexOf("*") < 0)
{
if (!dir.endsWith("/"))
{
dir = dir + "/*";
}
else
{
dir = dir + "*";
}
}
sendListFilesRequest(Conv.stringToUnicodeByteArray(dir));
out_.flush();
ArrayList files = new ArrayList();
int chain = 1;
while (chain != 0)
{
int length = in_.readInt();
if (length < 24)
{
throw DataStreamException.badLength("listFiles", length);
}
int headerID = in_.readShort();
int serverID = in_.readShort();
int csInstance = in_.readInt();
int correlationID = in_.readInt();
int templateLength = in_.readShort();
int reqRepID = in_.readShort();
chain = in_.readShort();
if (reqRepID == 0x8001)
{
int rc = in_.readShort();
int remaining = length-24;
in_.skipBytes(remaining);
if (rc != FileConstants.RC_NO_MORE_FILES)
{
throw DataStreamException.badReturnCode("listFiles", rc);
}
}
else if (reqRepID == 0x8005)
{
int numRead = 22;
/* long createDate = in_.readLong();
long modifyDate = in_.readLong();
long actual = ((modifyDate >> 32) & 0x00FFFFFFFFL)*1000L + ((modifyDate & 0x00FFFFFFFFL)/1000);
System.out.println(new java.util.Date(actual));
long accessDate = in_.readLong();
int fileSize = in_.readInt();
int fixedAttributes = in_.readInt();
int objectType = in_.readShort();
int numExtAttrs = in_.readShort();
int bytesEANames = in_.readInt();
int bytesEAValues = in_.readInt();
int version = in_.readInt();
int amountAccessed = in_.readShort();
int accessHistory = in_.readByte();
int fileCCSID = in_.readShort();
int checkoutCCSID = in_.readShort();
int restartID = in_.readInt();
long largeFileSize = in_.readLong();
in_.skipBytes(2);
int symlink = in_.readByte(); // 91
*/
in_.skipBytes(70);
numRead = 92;
int fileNameLLOffset = 20 + templateLength;
int toSkip = fileNameLLOffset-numRead;
in_.skipBytes(toSkip);
numRead += toSkip;
int fileNameLength = in_.readInt()-6;
in_.skipBytes(2); // 0x0002 -- Name CP.
numRead += 6;
byte[] nameBuf = new byte[fileNameLength];
in_.readFully(nameBuf);
numRead += fileNameLength;
String name = Conv.unicodeByteArrayToString(nameBuf, 0, nameBuf.length);
if (!name.equals(".") && !name.equals(".."))
{
files.add(name);
}
int remaining = length-numRead;
in_.skipBytes(remaining);
}
else
{
int remaining = length-22;
in_.skipBytes(remaining);
throw DataStreamException.badReply("listFiles", reqRepID);
}
}
return files;
}
private void sendListFilesRequest(byte[] dirnameUnicode) throws IOException
{
// 20 + 20 + 6 +
out_.writeInt(46+dirnameUnicode.length); // Length.
out_.writeShort(0); // Header ID.
out_.writeShort(0xE002); // Server ID.
out_.writeInt(0); // CS instance.
out_.writeInt(newCorrelationID()); // Correlation ID.
out_.writeShort(20); // Template length.
out_.writeShort(0x000A); // ReqRep ID.
out_.writeShort(0); // Chain indicator.
out_.writeInt(0); // File handle.
out_.writeShort(1200); // CCSID.
out_.writeInt(1); // Working dir handle.
out_.writeShort(0); // Check authority. 0=none required, 1=read required, 2=write requried, 4=exec required.
out_.writeShort(0xFFFF); // Maximum get count (-1 = no max).
out_.writeShort(0x0101); // File attribute list level (0x0101 = return 8-byte file size; 0x0001 = reserved flag).
out_.writeShort(1); // Pattern matching (POSIX=0, POSIX-all=1, OS/2=2).
out_.writeInt(dirnameUnicode.length+6); // Filename LL.
out_.writeShort(2); // Filename CP.
out_.write(dirnameUnicode);
}
/**
* Deletes the specified file.
**/
public int deleteFile(String filename) throws IOException
{
sendDeleteFileRequest(out_, Conv.stringToUnicodeByteArray(filename));
out_.flush();
int length = in_.readInt();
if (length < 24)
{
throw DataStreamException.badLength("deleteFile", length);
}
int headerID = in_.readShort();
int serverID = in_.readShort();
int csInstance = in_.readInt();
int correlationID = in_.readInt();
int templateLength = in_.readShort();
int reqRepID = in_.readShort();
in_.skipBytes(2);
if (reqRepID == 0x8001)
{
int rc = in_.readShort();
int remaining = length-24;
in_.skipBytes(remaining);
return rc;
}
else
{
int remaining = length-22;
in_.skipBytes(remaining);
throw DataStreamException.badReply("deleteFile", reqRepID);
}
}
/**
* Opens a file for read-write access, a share option of SHARE_ALL, and a data CCSID of 1208 (UTF-8).
**/
public int openFile(String filename, FileHandle buffer) throws IOException
{
return openFile(filename, buffer, FileHandle.OPEN_READ_WRITE, FileHandle.SHARE_ALL, true, 1208);
}
public int openFile(String filename, FileHandle buffer, int openType, int shareOption, boolean createIfNotExist, int dataCCSID) throws IOException
{
if (buffer.isOpen())
{
// Someone is re-using their buffer without closing the file!
//TODO closeFile(buffer);
}
sendOpenFileRequest(Conv.stringToUnicodeByteArray(filename), openType, shareOption, createIfNotExist, dataCCSID);
out_.flush();
int length = in_.readInt();
if (length < 24)
{
throw DataStreamException.badLength("openFile", length);
}
int headerID = in_.readShort();
int serverID = in_.readShort();
int csInstance = in_.readInt();
int correlationID = in_.readInt();
int templateLength = in_.readShort();
int reqRepID = in_.readShort();
in_.skipBytes(2);
int remaining = length-22;
int rc = 0;
if (reqRepID == 0x8001)
{
rc = in_.readShort();
remaining -= 2;
}
else if (reqRepID == 0x8002)
{
int handle = in_.readInt();
long id = in_.readLong();
dataCCSID = in_.readShort();
int actionTaken = in_.readShort();
long createDate = convertDate(in_);
long modifyDate = convertDate(in_);
long accessDate = convertDate(in_);
int fileSize = in_.readInt();
long actualFileSize = fileSize;
int fixedAttribs = in_.readInt();
int needExtAttribs = in_.readShort();
int numExtAttribs = in_.readShort();
int charsExtAttribsNames = in_.readInt();
int bytesExtAttribsValues = in_.readInt();
int version = in_.readInt();
int amountAccessed = in_.readShort();
int accessHistory = in_.readByte();
remaining -= 67;
if (length >= 97)
{
long largeFileSize = in_.readLong();
remaining -= 8;
actualFileSize = largeFileSize;
}
buffer.setOpen(true);
buffer.setOpenType(openType);
buffer.setName(filename);
buffer.setHandle(handle);
buffer.setID(id);
buffer.setDataCCSID(dataCCSID);
buffer.setCreateDate(createDate);
buffer.setModifyDate(modifyDate);
buffer.setAccessDate(accessDate);
buffer.setSize(actualFileSize);
buffer.setVersion(version);
}
else
{
in_.skipBytes(remaining);
throw DataStreamException.badReply("openFile", reqRepID);
}
in_.skipBytes(remaining-2);
return rc;
}
public int closeFile(FileHandle handle) throws IOException
{
if (!handle.isOpen())
{
// Someone is trying to close an already-closed handle!
//TODO
return -1;
}
sendCloseFileRequest(handle.getHandle());
out_.flush();
int length = in_.readInt();
if (length < 24)
{
throw DataStreamException.badLength("closeFile", length);
}
int headerID = in_.readShort();
int serverID = in_.readShort();
int csInstance = in_.readInt();
int correlationID = in_.readInt();
int templateLength = in_.readShort();
int reqRepID = in_.readShort();
in_.skipBytes(2);
int remaining = length-22;
if (reqRepID == 0x8001 || reqRepID == 0x8004)
{
int rc = in_.readShort();
in_.skipBytes(remaining-2);
return rc;
}
else
{
in_.skipBytes(remaining);
throw DataStreamException.badReply("closeFile", reqRepID);
}
}
public int readFile(FileHandle handle, byte[] buffer, int bufferOffset, int bufferLength) throws IOException
{
long currentOffset = handle.getOffset();
sendReadRequest(handle.getHandle(), currentOffset, bufferLength);
out_.flush();
int length = in_.readInt();
if (length < 24)
{
throw DataStreamException.badLength("readFile", length);
}
int headerID = in_.readShort();
int serverID = in_.readShort();
int csInstance = in_.readInt();
int correlationID = in_.readInt();
int templateLength = in_.readShort();
int reqRepID = in_.readShort();
int chain = in_.readShort(); //in_.skipBytes(2);
int numRead = 22;
int remaining = length-22;
int rc = 0;
int numToRead = -1;
if (reqRepID == 0x8001)
{
rc = in_.readShort();
remaining -= 2;
numRead += 2;
}
else if (reqRepID == 0x8003)
{
int ccsid = in_.readShort();
int dataLength = in_.readInt();
in_.skipBytes(2);
remaining -= 8;
int numBytes = dataLength-6;
numToRead = numBytes > length ? length : numBytes;
int numToSkip = numToRead >= numBytes ? 0 : numBytes-length;
in_.readFully(buffer, bufferOffset, numToRead);
in_.skipBytes(numToSkip);
handle.setOffset(currentOffset + numToRead);
remaining -= numBytes;
numRead += 6;
numRead += numToRead;
numRead += numToSkip;
}
else
{
in_.skipBytes(remaining);
numRead += remaining;
throw DataStreamException.badReply("readFile", reqRepID);
}
in_.skipBytes(remaining-2);
handle.setLastStatus(rc);
return numToRead;
}
/**
* Writes data to the file starting at the current file offset in the handle, and optionally sync-ing the data to disk;
* upon a successful write, the current file offset and size are incremented.
**/
public void writeFile(FileHandle handle, byte[] buffer, int bufferOffset, int bufferLength, boolean sync) throws IOException
{
long currentOffset = handle.getOffset();
sendWriteRequest(handle.getHandle(), currentOffset, buffer, bufferOffset, bufferLength, handle.getDataCCSID(), sync);
out_.flush();
int length = in_.readInt();
if (length < 24)
{
throw DataStreamException.badLength("writeFile", length);
}
int headerID = in_.readShort();
int serverID = in_.readShort();
int csInstance = in_.readInt();
int correlationID = in_.readInt();
int templateLength = in_.readShort();
int reqRepID = in_.readShort();
int chain = in_.readShort(); //in_.skipBytes(2);
int numRead = 22;
int rc = 0;
if (reqRepID == 0x8001)
{
rc = in_.readShort();
numRead += 2;
}
else if (reqRepID == 0x800B)
{
rc = in_.readShort();
int previousFileSize = in_.readInt();
int bytesNotWritten = in_.readInt();
numRead += 10;
handle.setOffset(currentOffset + (bufferLength-bytesNotWritten));
handle.setSize(handle.getSize() + (bufferLength-bytesNotWritten));
}
else
{
in_.skipBytes(length-numRead);
numRead = length;
throw DataStreamException.badReply("writeFile", reqRepID);
}
in_.skipBytes(length-numRead);
handle.setLastStatus(rc);
}
private void sendWriteRequest(int handle, long fileOffset, byte[] data, int dataOffset, int dataLength, int dataCCSID, boolean sync) throws IOException
{
if (datastreamLevel_ < 16 && fileOffset > 0x007FFFFFFFL)
{
throw new IOException("File offset too large: "+fileOffset);
}
final int templateLength = datastreamLevel_ < 16 ? 18 : 34;
out_.writeInt(26+templateLength+dataLength); // Length.
out_.writeShort(0); // Header ID.
out_.writeShort(0xE002); // Server ID.
out_.writeInt(0); // CS instance.
out_.writeInt(newCorrelationID()); // Correlation ID.
out_.writeShort(templateLength); // Template length.
out_.writeShort(0x0004); // ReqRep ID.
out_.writeShort(0); // Chain indicator.
out_.writeInt(handle); // File handle.
if (datastreamLevel_ < 16)
{
out_.writeInt(0); // Base offset.
out_.writeInt((int)fileOffset); // Relative offset.
}
else
{
out_.writeInt(0); // Base offset.
out_.writeInt(0); // Relative offset.
}
out_.writeShort(sync ? 3 : 2); // Data flags. 3=forceToStorage(sync),2=return immediately without sync.
out_.writeShort(dataCCSID); // Data CCSID.
if (datastreamLevel_ >= 16)
{
out_.writeLong(0); // Large base offset.
out_.writeLong(fileOffset); // Large relative offset.
}
out_.writeInt(dataLength+6); // Data LL.
out_.writeShort(0x0020); // Data CP.
out_.write(data, dataOffset, dataLength); // Data.
}
private void sendReadRequest(int handle, long fileOffset, int length) throws IOException
{
if (datastreamLevel_ < 16 && fileOffset > 0x007FFFFFFFL)
{
throw new IOException("File offset too large: "+fileOffset);
}
final int templateLength = datastreamLevel_ < 16 ? 22 : 38;
out_.writeInt(20+templateLength); // Length.
out_.writeShort(0); // Header ID.
out_.writeShort(0xE002); // Server ID.
out_.writeInt(0); // CS instance.
out_.writeInt(newCorrelationID()); // Correlation ID.
out_.writeShort(templateLength); // Template length.
out_.writeShort(0x0003); // ReqRep ID.
out_.writeShort(0); // Chain indicator.
out_.writeInt(handle); // File handle.
if (datastreamLevel_ < 16)
{
out_.writeInt(0); // Base offset.
out_.writeInt((int)fileOffset); // Relative offset.
}
else
{
out_.writeInt(0); // Base offset.
out_.writeInt(0); // Relative offset.
}
out_.writeInt(length); // Read length.
out_.writeInt(0); // Pre-read length.
if (datastreamLevel_ >= 16)
{
out_.writeLong(0); // Large base offset.
out_.writeLong(fileOffset); // Large relative offset.
}
}
// Only used to get the owner of a file. Boooo.
private void sendListAttributesOA1Request(int handle, int flags1, int flags2) throws IOException
{
out_.writeInt(54); // Length.
out_.writeShort(0); // Header ID.
out_.writeShort(0xE002); // Server ID.
out_.writeInt(0); // CS instance.
out_.writeInt(newCorrelationID()); // Correlation ID.
out_.writeShort(20); // Template length.
out_.writeShort(0x000A); // ReqRep ID.
out_.writeShort(0); // Chain indicator.
out_.writeInt(handle); // File handle.
out_.writeShort(0); // CCSID.
out_.writeInt(1); // Working dir handle.
out_.writeShort(0); // Check authority: 0=no auth required, 1=read required, 2=write required, 4=exec required.
out_.writeShort(0xFFFF); // Max get count.
out_.writeShort(0x42); // Use OA1 with open file handle.
out_.writeShort(0); // Pattern matching. 0=posix
out_.writeInt(14); // OA1 flags LL.
out_.writeShort(0x0010); // OA1 flags CP.
out_.writeInt(flags1); // OA1 flags1.
out_.writeInt(flags2); // OA1 flags2.
}
private void sendCloseFileRequest(int handle) throws IOException
{
out_.writeInt(41); // Length.
out_.writeShort(0); // Header ID.
out_.writeShort(0xE002); // Server ID.
out_.writeInt(0); // CS instance.
out_.writeInt(newCorrelationID()); // Correlation ID.
out_.writeShort(21); // Template length.
out_.writeShort(0x0009); // ReqRep ID.
out_.writeShort(0); // Chain indicator.
out_.writeInt(handle); // File handle.
out_.writeShort(2); // Data flags.
out_.writeShort(0xFFFF); // CCSID.
out_.writeShort(100); // Amount accessed.
out_.writeByte(0); // Access history.
out_.writeLong(0); // Modify date.
}
private void sendOpenFileRequest(byte[] filenameUnicode, int openType, int share, boolean create, int dataCCSID) throws IOException
{
final int templateLength = datastreamLevel_ < 16 ? 36 : 44;
out_.writeInt(26+templateLength+filenameUnicode.length); // Length.
out_.writeShort(0); // Header ID.
out_.writeShort(0xE002); // Server ID.
out_.writeInt(0); // CS instance.
out_.writeInt(newCorrelationID()); // Correlation ID.
out_.writeShort(templateLength); // Template length.
out_.writeShort(0x0002); // ReqRep ID.
out_.writeShort(0); // Chain indicator.
out_.writeShort(1200); // Filename CCSID.
out_.writeInt(1); // Working dir handle.
out_.writeShort(dataCCSID); // File data CCSID.
out_.writeShort(openType); // Access intent. 1=read, 2=write, 4=program load
out_.writeShort(share); // Share option. 0=all, 3=none, 2=read, 1=write.
out_.writeShort(0); // Data conversion. 0=none, 1=use client CCSID, 2=use server CCSID.
out_.writeShort(create ? 1 : 8); // Duplicate file option. 16=failIfNoExist+replaceIfExist, 8=failIfNoExist+openIfExist, 4=createOpenIfNoExist+failIfExist, 2=createOpenIfNoExist+replaceIfExist, 1=createOpenIfNoExist+openIfExist
out_.writeInt(0); // Create size.
out_.writeInt(0); // Fixed attributes.
out_.writeShort(1); // Attribute list level.
out_.writeInt(0); // Pre-read offset.
out_.writeInt(0); // Pre-read length.
if (datastreamLevel_ >= 16)
{
out_.writeLong(0); // Large create size.
}
out_.writeInt(filenameUnicode.length+6); // Filename LL.
out_.writeShort(2); // Filename CP.
out_.write(filenameUnicode);
}
private void sendDeleteFileRequest(HostOutputStream out, byte[] filenameUnicode) throws IOException
{
out.writeInt(34+filenameUnicode.length); // Length.
out.writeShort(0); // Header ID.
out.writeShort(0xE002); // Server ID.
out.writeInt(0); // CS instance.
out.writeInt(newCorrelationID()); // Correlation ID.
out.writeShort(8); // Template length.
out.writeShort(0x000C); // ReqRep ID.
out.writeShort(0); // Chain indicator.
out.writeShort(1200); // CCSID.
out.writeInt(1); // Working dir handle.
out.writeInt(filenameUnicode.length+6); // Filename LL.
out.writeShort(2); // Filename CP.
out.write(filenameUnicode);
}
private static void sendExchangeAttributesRequest(HostOutputStream out) throws IOException
{
out.writeInt(42); // Length.
out.writeShort(0); // Header ID.
out.writeShort(0xE002); // Server ID.
out.writeInt(0); // CS instance.
out.writeInt(0); // Correlation ID.
out.writeShort(10); // Template length.
out.writeShort(0x0016); // ReqRep ID.
out.writeShort(0); // Chain indicator.
out.writeShort(8); // Datastream level. V5R3 or higher.
out.writeShort(6); // Don't use posix return codes (8), use GMT (4), use PC pattern match (2).
out.writeInt(0xFFFFFFFF); // Max data block.
out.writeInt(12); // CCSID LL.
out.writeShort(10); // CCSID CP.
out.writeShort(1200); // Preferred CCSID UTF-16.
out.writeShort(13488); // Preferred CCSID UnicodeBig.
out.writeShort(61952); // Preferred CCSID old IFS UnicodeBig.
}
private static long convertDate(HostInputStream in) throws IOException
{
// The IFS format is 32-bit seconds, 32-bit milliseconds.
int seconds = in.readInt();
int microseconds = in.readInt();
long milliseconds = (seconds * 1000L) + (microseconds/1000);
return milliseconds;
}
}