Xamarin Public Jenkins (auto-signing) 0b4a830db1 Imported Upstream version 4.4.2.4
Former-commit-id: 92904c9c5915c37244316e42ba99e7b934ed7ee2
2016-07-21 09:40:10 +00:00

267 lines
7.8 KiB
C#

//
// ZipArchiveEntry.cs
//
// Author:
// Joao Matos <joao.matos@xamarin.com>
// Martin Baulig <martin.baulig@xamarin.com>
//
// Copyright (c) 2013 Xamarin Inc. (http://www.xamarin.com)
//
// Permission is hereby granted, free of charge, to any person obtaining a copy
// of this software and associated documentation files (the "Software"), to deal
// in the Software without restriction, including without limitation the rights
// to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
// copies of the Software, and to permit persons to whom the Software is
// furnished to do so, subject to the following conditions:
//
// The above copyright notice and this permission notice shall be included in
// all copies or substantial portions of the Software.
//
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
// THE SOFTWARE.
using System.Collections.Generic;
using System.Collections.ObjectModel;
using System.Linq;
using System.Text;
using SharpCompress.Common;
namespace System.IO.Compression
{
public class ZipArchive : IDisposable
{
internal Stream stream;
internal readonly bool leaveStreamOpen;
internal readonly ZipArchiveMode mode;
internal Encoding entryNameEncoding;
internal bool disposed;
internal List<ZipArchiveEntry> entries;
internal SharpCompress.Archive.Zip.ZipArchive zipFile;
public ZipArchive (Stream stream)
{
if (stream == null)
throw new ArgumentNullException("stream");
this.stream = stream;
mode = ZipArchiveMode.Read;
CreateZip(mode);
}
public ZipArchive (Stream stream, ZipArchiveMode mode)
{
if (stream == null)
throw new ArgumentNullException("stream");
this.stream = stream;
this.mode = mode;
CreateZip(mode);
}
public ZipArchive (Stream stream, ZipArchiveMode mode, bool leaveOpen)
{
if (stream == null)
throw new ArgumentNullException("stream");
this.stream = stream;
this.mode = mode;
leaveStreamOpen = leaveOpen;
CreateZip(mode);
}
public ZipArchive (Stream stream, ZipArchiveMode mode, bool leaveOpen, Encoding entryNameEncoding)
{
if (stream == null)
throw new ArgumentNullException("stream");
this.stream = stream;
this.mode = mode;
leaveStreamOpen = leaveOpen;
this.entryNameEncoding = entryNameEncoding;
CreateZip(mode);
}
private void CreateZip(ZipArchiveMode mode)
{
try {
if (mode != ZipArchiveMode.Read && mode != ZipArchiveMode.Create && mode != ZipArchiveMode.Update)
throw new ArgumentOutOfRangeException("mode");
// If the mode parameter is set to Read, the stream must support reading.
if (mode == ZipArchiveMode.Read && !stream.CanRead)
throw new ArgumentException("Stream must support reading for Read archive mode");
// If the mode parameter is set to Create, the stream must support writing.
if (mode == ZipArchiveMode.Create && !stream.CanWrite)
throw new ArgumentException("Stream must support writing for Create archive mode");
// If the mode parameter is set to Update, the stream must support reading, writing, and seeking.
if (mode == ZipArchiveMode.Update && (!stream.CanRead || !stream.CanWrite || !stream.CanSeek))
throw new ArgumentException("Stream must support reading, writing and seeking for Update archive mode");
// If the stream is not seekable, then buffer it into memory (same behavior as .NET).
if (mode == ZipArchiveMode.Read && !stream.CanSeek)
{
var memoryStream = new MemoryStream();
stream.CopyTo(memoryStream);
if (!leaveStreamOpen)
stream.Dispose();
this.stream = memoryStream;
}
try {
zipFile = mode != ZipArchiveMode.Create && stream.Length != 0
? SharpCompress.Archive.Zip.ZipArchive.Open(stream)
: SharpCompress.Archive.Zip.ZipArchive.Create();
} catch (Exception e) {
throw new InvalidDataException("The contents of the stream are not in the zip archive format.", e);
}
entries = new List<ZipArchiveEntry>();
if (Mode != ZipArchiveMode.Create) {
foreach (var entry in zipFile.Entries) {
var zipEntry = new ZipArchiveEntry(this, entry);
entries.Add(zipEntry);
}
}
}
catch {
if (!leaveStreamOpen)
stream.Dispose();
throw;
}
}
public ReadOnlyCollection<ZipArchiveEntry> Entries {
get {
if (disposed)
throw new ObjectDisposedException("The zip archive has been disposed.");
if (Mode == ZipArchiveMode.Create)
throw new NotSupportedException("Cannot access entries in Create mode.");
if (zipFile == null)
throw new InvalidDataException("The zip archive is corrupt, and its entries cannot be retrieved.");
if (entries == null)
return new ReadOnlyCollection<ZipArchiveEntry>(new List<ZipArchiveEntry>());
return new ReadOnlyCollection<ZipArchiveEntry>(entries);
}
}
public ZipArchiveMode Mode {
get {
if (disposed)
throw new ObjectDisposedException("The zip archive has been disposed.");
return mode;
}
}
public ZipArchiveEntry CreateEntry (string entryName)
{
if (disposed)
throw new ObjectDisposedException("The zip archive has been disposed.");
return CreateEntry(entryName, CompressionLevel.Optimal);
}
internal SharpCompress.Archive.Zip.ZipArchiveEntry CreateEntryInternal(string entryName)
{
var memoryStream = new MemoryStream();
var entry = zipFile.AddEntry(entryName, memoryStream);
return entry;
}
public ZipArchiveEntry CreateEntry (string entryName, CompressionLevel compressionLevel)
{
if (disposed)
throw new ObjectDisposedException("The zip archive has been disposed.");
if (entryName == string.Empty)
throw new ArgumentException("Entry name cannot be empty.");
if (entryName == null)
throw new ArgumentNullException("entryName");
if (mode != ZipArchiveMode.Create && mode != ZipArchiveMode.Update)
throw new NotSupportedException("The zip archive does not support writing.");
if (zipFile == null)
throw new InvalidDataException("The zip archive is corrupt, and its entries cannot be retrieved.");
var internalEntry = CreateEntryInternal(entryName);
var archiveEntry = new ZipArchiveEntry(this, internalEntry);
entries.Add(archiveEntry);
return archiveEntry;
}
public ZipArchiveEntry GetEntry (string entryName)
{
if (disposed)
throw new ObjectDisposedException("The zip archive has been disposed.");
if (entryName == string.Empty)
throw new ArgumentException("Entry name cannot be empty.");
if (entryName == null)
throw new ArgumentNullException("entryName");
if (mode != ZipArchiveMode.Read && mode != ZipArchiveMode.Update)
throw new NotSupportedException("The zip archive does not support reading.");
if (zipFile == null)
throw new InvalidDataException("The zip archive is corrupt, and its entries cannot be retrieved.");
return entries.FirstOrDefault(e => e.FullName == entryName);
}
private void Save()
{
using (var newZip = new MemoryStream()) {
zipFile.SaveTo(newZip, CompressionType.Deflate, entryNameEncoding ?? Encoding.UTF8);
stream.SetLength(0);
stream.Position = 0;
newZip.Position = 0;
newZip.CopyTo(stream);
}
}
protected virtual void Dispose (bool disposing)
{
if (disposed)
return;
if (mode != ZipArchiveMode.Read)
Save();
disposed = true;
if (leaveStreamOpen)
return;
if (stream != null) {
stream.Dispose();
stream = null;
}
}
public void Dispose ()
{
Dispose(true);
GC.SuppressFinalize(this);
}
}
}