| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488 |
- // ZipInputStream.cs
- // Copyright (C) 2001 Mike Krueger
- //
- // This file was translated from java, it was part of the GNU Classpath
- // Copyright (C) 2001 Free Software Foundation, Inc.
- //
- // This program is free software; you can redistribute it and/or
- // modify it under the terms of the GNU General Public License
- // as published by the Free Software Foundation; either version 2
- // of the License, or (at your option) any later version.
- //
- // This program is distributed in the hope that it will be useful,
- // but WITHOUT ANY WARRANTY; without even the implied warranty of
- // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
- // GNU General Public License for more details.
- //
- // You should have received a copy of the GNU General Public License
- // along with this program; if not, write to the Free Software
- // Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
- //
- // Linking this library statically or dynamically with other modules is
- // making a combined work based on this library. Thus, the terms and
- // conditions of the GNU General Public License cover the whole
- // combination.
- //
- // As a special exception, the copyright holders of this library give you
- // permission to link this library with independent modules to produce an
- // executable, regardless of the license terms of these independent
- // modules, and to copy and distribute the resulting executable under
- // terms of your choice, provided that you also meet, for each linked
- // independent module, the terms and conditions of the license of that
- // module. An independent module is a module which is not derived from
- // or based on this library. If you modify this library, you may extend
- // this exception to your version of the library, but you are not
- // obligated to do so. If you do not wish to do so, delete this
- // exception statement from your version.
- using System;
- using System.Text;
- using System.IO;
- using ICSharpCode.SharpZipLib.Checksums;
- using ICSharpCode.SharpZipLib.Zip.Compression;
- using ICSharpCode.SharpZipLib.Zip.Compression.Streams;
- namespace ICSharpCode.SharpZipLib.Zip
- {
- /// <summary>
- /// This is a FilterInputStream that reads the files baseInputStream an zip archive
- /// one after another. It has a special method to get the zip entry of
- /// the next file. The zip entry contains information about the file name
- /// size, compressed size, CRC, etc.
- /// It includes support for STORED and DEFLATED entries.
- ///
- /// author of the original java version : Jochen Hoenicke
- /// </summary>
- /// <example> This sample shows how to read a zip file
- /// <code lang="C#">
- /// using System;
- /// using System.Text;
- /// using System.IO;
- ///
- /// using NZlib.Zip;
- ///
- /// class MainClass
- /// {
- /// public static void Main(string[] args)
- /// {
- /// ZipInputStream s = new ZipInputStream(File.OpenRead(args[0]));
- ///
- /// ZipEntry theEntry;
- /// while ((theEntry = s.GetNextEntry()) != null) {
- /// int size = 2048;
- /// byte[] data = new byte[2048];
- ///
- /// Console.Write("Show contents (y/n) ?");
- /// if (Console.ReadLine() == "y") {
- /// while (true) {
- /// size = s.Read(data, 0, data.Length);
- /// if (size > 0) {
- /// Console.Write(new ASCIIEncoding().GetString(data, 0, size));
- /// } else {
- /// break;
- /// }
- /// }
- /// }
- /// }
- /// s.Close();
- /// }
- /// }
- /// </code>
- /// </example>
- public class ZipInputStream : InflaterInputStream
- {
- Crc32 crc = new Crc32();
- ZipEntry entry = null;
-
- long size;
- int method;
- int flags;
- long avail;
- string password = null;
-
- public string Password {
- get {
- return password;
- }
- set {
- password = value;
- }
- }
-
- /// <summary>
- /// Creates a new Zip input stream, reading a zip archive.
- /// </summary>
- public ZipInputStream(Stream baseInputStream) : base(baseInputStream, new Inflater(true))
- {
- }
-
- void FillBuf()
- {
- avail = len = baseInputStream.Read(buf, 0, buf.Length);
- }
-
- int ReadBuf(byte[] outBuf, int offset, int length)
- {
- if (avail <= 0) {
- FillBuf();
- if (avail <= 0) {
- return 0;
- }
- }
- if (length > avail) {
- length = (int)avail;
- }
- System.Array.Copy(buf, len - (int)avail, outBuf, offset, length);
- avail -= length;
- return length;
- }
-
- void ReadFully(byte[] outBuf)
- {
- int off = 0;
- int len = outBuf.Length;
- while (len > 0) {
- int count = ReadBuf(outBuf, off, len);
- if (count == -1) {
- throw new Exception();
- }
- off += count;
- len -= count;
- }
- }
-
- int ReadLeByte()
- {
- if (avail <= 0) {
- FillBuf();
- if (avail <= 0) {
- throw new ZipException("EOF in header");
- }
- }
- return buf[len - avail--] & 0xff;
- }
-
- /// <summary>
- /// Read an unsigned short baseInputStream little endian byte order.
- /// </summary>
- int ReadLeShort()
- {
- return ReadLeByte() | (ReadLeByte() << 8);
- }
-
- /// <summary>
- /// Read an int baseInputStream little endian byte order.
- /// </summary>
- int ReadLeInt()
- {
- return ReadLeShort() | (ReadLeShort() << 16);
- }
-
- /// <summary>
- /// Read an int baseInputStream little endian byte order.
- /// </summary>
- long ReadLeLong()
- {
- return ReadLeInt() | (ReadLeInt() << 32);
- }
-
- /// <summary>
- /// Open the next entry from the zip archive, and return its description.
- /// If the previous entry wasn't closed, this method will close it.
- /// </summary>
- public ZipEntry GetNextEntry()
- {
- if (crc == null) {
- throw new InvalidOperationException("Closed.");
- }
- if (entry != null) {
- CloseEntry();
- }
-
- if (this.cryptbuffer != null) {
- if (avail == 0 && inf.RemainingInput != 0) {
- avail = inf.RemainingInput - 16;
- inf.Reset();
- }
- baseInputStream.Position -= this.len;
- baseInputStream.Read(this.buf, 0, this.len);
- }
-
- int header = ReadLeInt();
-
- // -jr- added end sig for empty zip files, Zip64 end sig and digital sig for files that have them...
- if (header == ZipConstants.CENSIG ||
- header == ZipConstants.ENDSIG ||
- header == ZipConstants.CENDIGITALSIG ||
- header == ZipConstants.CENSIG64) {
- // Central Header reached or end of empty zip file
- Close();
- return null;
- }
- // -jr- 07-Dec-2003 ignore spanning temporary signatures if found
- // SPANNINGSIG is same as descriptor signature and is untested as yet.
- if (header == ZipConstants.SPANTEMPSIG || header == ZipConstants.SPANNINGSIG) {
- header = ReadLeInt();
- }
-
- if (header != ZipConstants.LOCSIG) {
- throw new ZipException("Wrong Local header signature: 0x" + String.Format("{0:X}", header));
- }
-
- short version = (short)ReadLeShort();
-
- flags = ReadLeShort();
- method = ReadLeShort();
- uint dostime = (uint)ReadLeInt();
- int crc2 = ReadLeInt();
- csize = ReadLeInt();
- size = ReadLeInt();
- int nameLen = ReadLeShort();
- int extraLen = ReadLeShort();
- bool isCrypted = (flags & 1) == 1;
- if (method == ZipOutputStream.STORED && (!isCrypted && csize != size || (isCrypted && csize - 12 != size))) {
- throw new ZipException("Stored, but compressed != uncompressed");
- }
-
- byte[] buffer = new byte[nameLen];
- ReadFully(buffer);
-
- string name = ZipConstants.ConvertToString(buffer);
-
- entry = new ZipEntry(name);
- entry.IsCrypted = isCrypted;
- entry.Version = (ushort)version;
- if (method != 0 && method != 8) {
- throw new ZipException("unknown compression method " + method);
- }
- entry.CompressionMethod = (CompressionMethod)method;
-
- if ((flags & 8) == 0) {
- entry.Crc = crc2 & 0xFFFFFFFFL;
- entry.Size = size & 0xFFFFFFFFL;
- entry.CompressedSize = csize & 0xFFFFFFFFL;
- }
-
- entry.DosTime = dostime;
-
- if (extraLen > 0) {
- byte[] extra = new byte[extraLen];
- ReadFully(extra);
- entry.ExtraData = extra;
- }
-
- // test for encryption
- if (isCrypted) {
- if (password == null) {
- throw new ZipException("No password set.");
- }
- InitializePassword(password);
- cryptbuffer = new byte[12];
- ReadFully(cryptbuffer);
- DecryptBlock(cryptbuffer, 0, cryptbuffer.Length);
- if ((flags & 8) == 0) {// -jr- 10-Feb-2004 Dont yet know correct size here....
- csize -= 12;
- }
- } else {
- cryptbuffer = null;
- }
-
- if (method == ZipOutputStream.DEFLATED && avail > 0) {
- System.Array.Copy(buf, len - (int)avail, buf, 0, (int)avail);
- len = (int)avail;
- avail = 0;
- if (isCrypted) {
- DecryptBlock(buf, 0, Math.Min((int)csize, len));
- }
- inf.SetInput(buf, 0, len);
- }
-
- return entry;
- }
- private void ReadDataDescr()
- {
- if (ReadLeInt() != ZipConstants.EXTSIG) {
- throw new ZipException("Data descriptor signature not found");
- }
- entry.Crc = ReadLeInt() & 0xFFFFFFFFL;
- csize = ReadLeInt();
- size = ReadLeInt();
- entry.Size = size & 0xFFFFFFFFL;
- entry.CompressedSize = csize & 0xFFFFFFFFL;
- }
-
- /// <summary>
- /// Closes the current zip entry and moves to the next one.
- /// </summary>
- public void CloseEntry()
- {
- if (crc == null) {
- throw new InvalidOperationException("Closed.");
- }
-
- if (entry == null) {
- return;
- }
-
- if (method == ZipOutputStream.DEFLATED) {
- if ((flags & 8) != 0) {
- /* We don't know how much we must skip, read until end. */
- byte[] tmp = new byte[2048];
- while (Read(tmp, 0, tmp.Length) > 0)
- ;
- /* read will close this entry */
- return;
- }
- csize -= inf.TotalIn;
- avail = inf.RemainingInput;
- }
- if (avail > csize && csize >= 0) {
- avail -= csize;
- } else {
- csize -= avail;
- avail = 0;
- while (csize != 0) {
- int skipped = (int)base.Skip(csize & 0xFFFFFFFFL);
-
- if (skipped <= 0) {
- throw new ZipException("zip archive ends early.");
- }
-
- csize -= skipped;
- }
- }
-
- size = 0;
- crc.Reset();
- if (method == ZipOutputStream.DEFLATED) {
- inf.Reset();
- }
- entry = null;
- }
-
- public override int Available {
- get {
- return entry != null ? 1 : 0;
- }
- }
-
- /// <summary>
- /// Reads a byte from the current zip entry.
- /// </summary>
- /// <returns>
- /// the byte or -1 on EOF.
- /// </returns>
- /// <exception name="System.IO.IOException">
- /// IOException if a i/o error occured.
- /// </exception>
- /// <exception name="ICSharpCode.SharpZipLib.ZipException">
- /// ZipException if the deflated stream is corrupted.
- /// </exception>
- public override int ReadByte()
- {
- byte[] b = new byte[1];
- if (Read(b, 0, 1) <= 0) {
- return -1; // ok
- }
- return b[0] & 0xff;
- }
-
- /// <summary>
- /// Reads a block of bytes from the current zip entry.
- /// </summary>
- /// <returns>
- /// the number of bytes read (may be smaller, even before EOF), or -1 on EOF.
- /// </returns>
- /// <exception name="Exception">
- /// IOException if a i/o error occured.
- /// ZipException if the deflated stream is corrupted.
- /// </exception>
- public override int Read(byte[] b, int off, int len)
- {
- if (crc == null) {
- throw new InvalidOperationException("Closed.");
- }
-
- if (entry == null) {
- return 0;
- }
- bool finished = false;
-
- switch (method) {
- case ZipOutputStream.DEFLATED:
- len = base.Read(b, off, len);
- if (len <= 0) { // TODO BUG1 -jr- Check this was < 0 but avail was not adjusted causing failure in later calls
- if (!inf.IsFinished) {
- throw new ZipException("Inflater not finished!?");
- }
- avail = inf.RemainingInput;
-
- // BUG1 -jr- With bit 3 set you dont yet know the size
- if ((flags & 8) == 0 && (inf.TotalIn != csize || inf.TotalOut != size)) {
- throw new ZipException("size mismatch: " + csize + ";" + size + " <-> " + inf.TotalIn + ";" + inf.TotalOut);
- }
- inf.Reset();
- finished = true;
- }
- break;
-
- case ZipOutputStream.STORED:
- if (len > csize && csize >= 0) {
- len = (int)csize;
- }
- len = ReadBuf(b, off, len);
- if (len > 0) {
- csize -= len;
- size -= len;
- }
-
- if (csize == 0) {
- finished = true;
- } else {
- if (len < 0) {
- throw new ZipException("EOF in stored block");
- }
- }
-
- // decrypting crypted data
- if (cryptbuffer != null) {
- DecryptBlock(b, off, len);
- }
-
- break;
- }
-
- if (len > 0) {
- crc.Update(b, off, len);
- }
-
- if (finished) {
- if ((flags & 8) != 0) {
- ReadDataDescr();
- }
-
- if ((crc.Value & 0xFFFFFFFFL) != entry.Crc && entry.Crc != -1) {
- throw new ZipException("CRC mismatch");
- }
- crc.Reset();
- entry = null;
- }
- return len;
- }
-
- /// <summary>
- /// Closes the zip file.
- /// </summary>
- /// <exception name="Exception">
- /// if a i/o error occured.
- /// </exception>
- public override void Close()
- {
- base.Close();
- crc = null;
- entry = null;
- }
-
- }
- }
|