001: /*
002: ** Authored by Timothy Gerard Endres
003: ** <mailto:time@gjt.org> <http://www.trustice.com>
004: **
005: ** This work has been placed into the public domain.
006: ** You may use this work in any way and for any purpose you wish.
007: **
008: ** THIS SOFTWARE IS PROVIDED AS-IS WITHOUT WARRANTY OF ANY KIND,
009: ** NOT EVEN THE IMPLIED WARRANTY OF MERCHANTABILITY. THE AUTHOR
010: ** OF THIS SOFTWARE, ASSUMES _NO_ RESPONSIBILITY FOR ANY
011: ** CONSEQUENCE RESULTING FROM THE USE, MODIFICATION, OR
012: ** REDISTRIBUTION OF THIS SOFTWARE.
013: **
014: */
015:
016: package installer;
017:
018: import java.io.*;
019:
020: /**
021: * The TarOutputStream writes a UNIX tar archive as an OutputStream.
022: * Methods are provided to put entries, and then write their contents
023: * by writing to this stream using write().
024: *
025: *
026: * @version $Revision: 4480 $
027: * @author Timothy Gerard Endres,
028: * <a href="mailto:time@gjt.org">time@trustice.com</a>.
029: * @see TarBuffer
030: * @see TarHeader
031: * @see TarEntry
032: */
033:
034: public class TarOutputStream extends FilterOutputStream {
035: protected boolean debug;
036: protected int currSize;
037: protected int currBytes;
038: protected byte[] oneBuf;
039: protected byte[] recordBuf;
040: protected int assemLen;
041: protected byte[] assemBuf;
042: protected TarBuffer buffer;
043:
044: public TarOutputStream(OutputStream os) {
045: this (os, TarBuffer.DEFAULT_BLKSIZE, TarBuffer.DEFAULT_RCDSIZE);
046: }
047:
048: public TarOutputStream(OutputStream os, int blockSize) {
049: this (os, blockSize, TarBuffer.DEFAULT_RCDSIZE);
050: }
051:
052: public TarOutputStream(OutputStream os, int blockSize,
053: int recordSize) {
054: super (os);
055:
056: this .buffer = new TarBuffer(os, blockSize, recordSize);
057:
058: this .debug = false;
059: this .assemLen = 0;
060: this .assemBuf = new byte[recordSize];
061: this .recordBuf = new byte[recordSize];
062: this .oneBuf = new byte[1];
063: }
064:
065: /**
066: * Sets the debugging flag.
067: *
068: * @param debugF True to turn on debugging.
069: */
070: public void setDebug(boolean debugF) {
071: this .debug = debugF;
072: }
073:
074: /**
075: * Sets the debugging flag in this stream's TarBuffer.
076: *
077: * @param debugF True to turn on debugging.
078: */
079: public void setBufferDebug(boolean debug) {
080: this .buffer.setDebug(debug);
081: }
082:
083: /**
084: * Ends the TAR archive without closing the underlying OutputStream.
085: * The result is that the EOF record of nulls is written.
086: */
087:
088: public void finish() throws IOException {
089: this .writeEOFRecord();
090: }
091:
092: /**
093: * Ends the TAR archive and closes the underlying OutputStream.
094: * This means that finish() is called followed by calling the
095: * TarBuffer's close().
096: */
097:
098: public void close() throws IOException {
099: this .finish();
100: this .buffer.close();
101: }
102:
103: /**
104: * Get the record size being used by this stream's TarBuffer.
105: *
106: * @return The TarBuffer record size.
107: */
108: public int getRecordSize() {
109: return this .buffer.getRecordSize();
110: }
111:
112: /**
113: * Put an entry on the output stream. This writes the entry's
114: * header record and positions the output stream for writing
115: * the contents of the entry. Once this method is called, the
116: * stream is ready for calls to write() to write the entry's
117: * contents. Once the contents are written, closeEntry()
118: * <B>MUST</B> be called to ensure that all buffered data
119: * is completely written to the output stream.
120: *
121: * @param entry The TarEntry to be written to the archive.
122: */
123: public void putNextEntry(TarEntry entry) throws IOException {
124: if (entry.getHeader().name.length() > TarHeader.NAMELEN)
125: throw new InvalidHeaderException("file name '"
126: + entry.getHeader().name + "' is too long ( > "
127: + TarHeader.NAMELEN + " bytes )");
128:
129: entry.writeEntryHeader(this .recordBuf);
130: this .buffer.writeRecord(this .recordBuf);
131:
132: this .currBytes = 0;
133:
134: if (entry.isDirectory())
135: this .currSize = 0;
136: else
137: this .currSize = (int) entry.getSize();
138: }
139:
140: /**
141: * Close an entry. This method MUST be called for all file
142: * entries that contain data. The reason is that we must
143: * buffer data written to the stream in order to satisfy
144: * the buffer's record based writes. Thus, there may be
145: * data fragments still being assembled that must be written
146: * to the output stream before this entry is closed and the
147: * next entry written.
148: */
149: public void closeEntry() throws IOException {
150: if (this .assemLen > 0) {
151: for (int i = this .assemLen; i < this .assemBuf.length; ++i)
152: this .assemBuf[i] = 0;
153:
154: this .buffer.writeRecord(this .assemBuf);
155:
156: this .currBytes += this .assemLen;
157: this .assemLen = 0;
158: }
159:
160: if (this .currBytes < this .currSize)
161: throw new IOException("entry closed at '" + this .currBytes
162: + "' before the '" + this .currSize
163: + "' bytes specified in the header were written");
164: }
165:
166: /**
167: * Writes a byte to the current tar archive entry.
168: *
169: * This method simply calls read( byte[], int, int ).
170: *
171: * @param b The byte written.
172: */
173: public void write(int b) throws IOException {
174: this .oneBuf[0] = (byte) b;
175: this .write(this .oneBuf, 0, 1);
176: }
177:
178: /**
179: * Writes bytes to the current tar archive entry.
180: *
181: * This method simply calls read( byte[], int, int ).
182: *
183: * @param wBuf The buffer to write to the archive.
184: * @return The number of bytes read, or -1 at EOF.
185: */
186: public void write(byte[] wBuf) throws IOException {
187: this .write(wBuf, 0, wBuf.length);
188: }
189:
190: /**
191: * Writes bytes to the current tar archive entry. This method
192: * is aware of the current entry and will throw an exception if
193: * you attempt to write bytes past the length specified for the
194: * current entry. The method is also (painfully) aware of the
195: * record buffering required by TarBuffer, and manages buffers
196: * that are not a multiple of recordsize in length, including
197: * assembling records from small buffers.
198: *
199: * This method simply calls read( byte[], int, int ).
200: *
201: * @param wBuf The buffer to write to the archive.
202: * @param wOffset The offset in the buffer from which to get bytes.
203: * @param numToWrite The number of bytes to write.
204: */
205: public void write(byte[] wBuf, int wOffset, int numToWrite)
206: throws IOException {
207: if ((this .currBytes + numToWrite) > this .currSize)
208: throw new IOException("request to write '" + numToWrite
209: + "' bytes exceeds size in header of '"
210: + this .currSize + "' bytes");
211:
212: //
213: // We have to deal with assembly!!!
214: // The programmer can be writing little 32 byte chunks for all
215: // we know, and we must assemble complete records for writing.
216: // REVIEW Maybe this should be in TarBuffer? Could that help to
217: // eliminate some of the buffer copying.
218: //
219: if (this .assemLen > 0) {
220: if ((this .assemLen + numToWrite) >= this .recordBuf.length) {
221: int aLen = this .recordBuf.length - this .assemLen;
222:
223: System.arraycopy(this .assemBuf, 0, this .recordBuf, 0,
224: this .assemLen);
225:
226: System.arraycopy(wBuf, wOffset, this .recordBuf,
227: this .assemLen, aLen);
228:
229: this .buffer.writeRecord(this .recordBuf);
230:
231: this .currBytes += this .recordBuf.length;
232:
233: wOffset += aLen;
234: numToWrite -= aLen;
235: this .assemLen = 0;
236: } else // ( (this.assemLen + numToWrite ) < this.recordBuf.length )
237: {
238: System.arraycopy(wBuf, wOffset, this .assemBuf,
239: this .assemLen, numToWrite);
240: wOffset += numToWrite;
241: this .assemLen += numToWrite;
242: numToWrite -= numToWrite;
243: }
244: }
245:
246: //
247: // When we get here we have EITHER:
248: // o An empty "assemble" buffer.
249: // o No bytes to write (numToWrite == 0)
250: //
251:
252: for (; numToWrite > 0;) {
253: if (numToWrite < this .recordBuf.length) {
254: System.arraycopy(wBuf, wOffset, this .assemBuf,
255: this .assemLen, numToWrite);
256: this .assemLen += numToWrite;
257: break;
258: }
259:
260: this .buffer.writeRecord(wBuf, wOffset);
261:
262: int num = this .recordBuf.length;
263: this .currBytes += num;
264: numToWrite -= num;
265: wOffset += num;
266: }
267: }
268:
269: /**
270: * Write an EOF (end of archive) record to the tar archive.
271: * An EOF record consists of a record of all zeros.
272: */
273: private void writeEOFRecord() throws IOException {
274: for (int i = 0; i < this .recordBuf.length; ++i)
275: this .recordBuf[i] = 0;
276: this.buffer.writeRecord(this.recordBuf);
277: }
278:
279: }
|