using ICSharpCode.SharpZipLib.Core;
using System;
using System.IO;
namespace ICSharpCode.SharpZipLib.Zip
{
///
/// Basic implementation of
///
public class ZipEntryFactory : IEntryFactory
{
#region Enumerations
///
/// Defines the possible values to be used for the .
///
public enum TimeSetting
{
///
/// Use the recorded LastWriteTime value for the file.
///
LastWriteTime,
///
/// Use the recorded LastWriteTimeUtc value for the file
///
LastWriteTimeUtc,
///
/// Use the recorded CreateTime value for the file.
///
CreateTime,
///
/// Use the recorded CreateTimeUtc value for the file.
///
CreateTimeUtc,
///
/// Use the recorded LastAccessTime value for the file.
///
LastAccessTime,
///
/// Use the recorded LastAccessTimeUtc value for the file.
///
LastAccessTimeUtc,
///
/// Use a fixed value.
///
/// The actual value used can be
/// specified via the constructor or
/// using the with the setting set
/// to which will use the when this class was constructed.
/// The property can also be used to set this value.
Fixed,
}
#endregion Enumerations
#region Constructors
///
/// Initialise a new instance of the class.
///
/// A default , and the LastWriteTime for files is used.
public ZipEntryFactory()
{
nameTransform_ = new ZipNameTransform();
isUnicodeText_ = ZipStrings.UseUnicode;
}
///
/// Initialise a new instance of using the specified
///
/// The time setting to use when creating Zip entries.
public ZipEntryFactory(TimeSetting timeSetting) : this()
{
timeSetting_ = timeSetting;
}
///
/// Initialise a new instance of using the specified
///
/// The time to set all values to.
public ZipEntryFactory(DateTime time) : this()
{
timeSetting_ = TimeSetting.Fixed;
FixedDateTime = time;
}
#endregion Constructors
#region Properties
///
/// Get / set the to be used when creating new values.
///
///
/// Setting this property to null will cause a default name transform to be used.
///
public INameTransform NameTransform
{
get { return nameTransform_; }
set
{
if (value == null)
{
nameTransform_ = new ZipNameTransform();
}
else
{
nameTransform_ = value;
}
}
}
///
/// Get / set the in use.
///
public TimeSetting Setting
{
get { return timeSetting_; }
set { timeSetting_ = value; }
}
///
/// Get / set the value to use when is set to
///
public DateTime FixedDateTime
{
get { return fixedDateTime_; }
set
{
if (value.Year < 1970)
{
throw new ArgumentException("Value is too old to be valid", nameof(value));
}
fixedDateTime_ = value;
}
}
///
/// A bitmask defining the attributes to be retrieved from the actual file.
///
/// The default is to get all possible attributes from the actual file.
public int GetAttributes
{
get { return getAttributes_; }
set { getAttributes_ = value; }
}
///
/// A bitmask defining which attributes are to be set on.
///
/// By default no attributes are set on.
public int SetAttributes
{
get { return setAttributes_; }
set { setAttributes_ = value; }
}
///
/// Get set a value indicating whether unidoce text should be set on.
///
public bool IsUnicodeText
{
get { return isUnicodeText_; }
set { isUnicodeText_ = value; }
}
#endregion Properties
#region IEntryFactory Members
///
/// Make a new for a file.
///
/// The name of the file to create a new entry for.
/// Returns a new based on the .
public ZipEntry MakeFileEntry(string fileName)
{
return MakeFileEntry(fileName, null, true);
}
///
/// Make a new for a file.
///
/// The name of the file to create a new entry for.
/// If true entry detail is retrieved from the file system if the file exists.
/// Returns a new based on the .
public ZipEntry MakeFileEntry(string fileName, bool useFileSystem)
{
return MakeFileEntry(fileName, null, useFileSystem);
}
///
/// Make a new from a name.
///
/// The name of the file to create a new entry for.
/// An alternative name to be used for the new entry. Null if not applicable.
/// If true entry detail is retrieved from the file system if the file exists.
/// Returns a new based on the .
public ZipEntry MakeFileEntry(string fileName, string entryName, bool useFileSystem)
{
var result = new ZipEntry(nameTransform_.TransformFile(!string.IsNullOrEmpty(entryName) ? entryName : fileName));
result.IsUnicodeText = isUnicodeText_;
int externalAttributes = 0;
bool useAttributes = (setAttributes_ != 0);
FileInfo fi = null;
if (useFileSystem)
{
fi = new FileInfo(fileName);
}
if ((fi != null) && fi.Exists)
{
switch (timeSetting_)
{
case TimeSetting.CreateTime:
result.DateTime = fi.CreationTime;
break;
case TimeSetting.CreateTimeUtc:
result.DateTime = fi.CreationTimeUtc;
break;
case TimeSetting.LastAccessTime:
result.DateTime = fi.LastAccessTime;
break;
case TimeSetting.LastAccessTimeUtc:
result.DateTime = fi.LastAccessTimeUtc;
break;
case TimeSetting.LastWriteTime:
result.DateTime = fi.LastWriteTime;
break;
case TimeSetting.LastWriteTimeUtc:
result.DateTime = fi.LastWriteTimeUtc;
break;
case TimeSetting.Fixed:
result.DateTime = fixedDateTime_;
break;
default:
throw new ZipException("Unhandled time setting in MakeFileEntry");
}
result.Size = fi.Length;
useAttributes = true;
externalAttributes = ((int)fi.Attributes & getAttributes_);
}
else
{
if (timeSetting_ == TimeSetting.Fixed)
{
result.DateTime = fixedDateTime_;
}
}
if (useAttributes)
{
externalAttributes |= setAttributes_;
result.ExternalFileAttributes = externalAttributes;
}
return result;
}
///
/// Make a new for a directory.
///
/// The raw untransformed name for the new directory
/// Returns a new representing a directory.
public ZipEntry MakeDirectoryEntry(string directoryName)
{
return MakeDirectoryEntry(directoryName, true);
}
///
/// Make a new for a directory.
///
/// The raw untransformed name for the new directory
/// If true entry detail is retrieved from the file system if the file exists.
/// Returns a new representing a directory.
public ZipEntry MakeDirectoryEntry(string directoryName, bool useFileSystem)
{
var result = new ZipEntry(nameTransform_.TransformDirectory(directoryName));
result.IsUnicodeText = isUnicodeText_;
result.Size = 0;
int externalAttributes = 0;
DirectoryInfo di = null;
if (useFileSystem)
{
di = new DirectoryInfo(directoryName);
}
if ((di != null) && di.Exists)
{
switch (timeSetting_)
{
case TimeSetting.CreateTime:
result.DateTime = di.CreationTime;
break;
case TimeSetting.CreateTimeUtc:
result.DateTime = di.CreationTimeUtc;
break;
case TimeSetting.LastAccessTime:
result.DateTime = di.LastAccessTime;
break;
case TimeSetting.LastAccessTimeUtc:
result.DateTime = di.LastAccessTimeUtc;
break;
case TimeSetting.LastWriteTime:
result.DateTime = di.LastWriteTime;
break;
case TimeSetting.LastWriteTimeUtc:
result.DateTime = di.LastWriteTimeUtc;
break;
case TimeSetting.Fixed:
result.DateTime = fixedDateTime_;
break;
default:
throw new ZipException("Unhandled time setting in MakeDirectoryEntry");
}
externalAttributes = ((int)di.Attributes & getAttributes_);
}
else
{
if (timeSetting_ == TimeSetting.Fixed)
{
result.DateTime = fixedDateTime_;
}
}
// Always set directory attribute on.
externalAttributes |= (setAttributes_ | 16);
result.ExternalFileAttributes = externalAttributes;
return result;
}
#endregion IEntryFactory Members
#region Instance Fields
private INameTransform nameTransform_;
private DateTime fixedDateTime_ = DateTime.Now;
private TimeSetting timeSetting_ = TimeSetting.LastWriteTime;
private bool isUnicodeText_;
private int getAttributes_ = -1;
private int setAttributes_;
#endregion Instance Fields
}
}