123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602 |
- /* This file is part of SevenZipSharp.
- SevenZipSharp is free software: you can redistribute it and/or modify
- it under the terms of the GNU Lesser General Public License as published by
- the Free Software Foundation, either version 3 of the License, or
- (at your option) any later version.
- SevenZipSharp 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 Lesser General Public License for more details.
- You should have received a copy of the GNU Lesser General Public License
- along with SevenZipSharp. If not, see <http://www.gnu.org/licenses/>.
- */
- using System;
- using System.Collections.Generic;
- using System.Globalization;
- using System.IO;
- #if MONO
- using SevenZip.Mono.COM;
- using System.Runtime.InteropServices;
- #endif
- namespace SevenZip
- {
- #if UNMANAGED
- /// <summary>
- /// Archive extraction callback to handle the process of unpacking files
- /// </summary>
- internal sealed class ArchiveExtractCallback : CallbackBase, IArchiveExtractCallback, ICryptoGetTextPassword,
- IDisposable
- {
- private List<uint> _actualIndexes;
- private IInArchive _archive;
- /// <summary>
- /// For Compressing event.
- /// </summary>
- private long _bytesCount;
- private long _bytesWritten;
- private long _bytesWrittenOld;
- private string _directory;
- /// <summary>
- /// Rate of the done work from [0, 1].
- /// </summary>
- private float _doneRate;
- private SevenZipExtractor _extractor;
- private FakeOutStreamWrapper _fakeStream;
- private uint? _fileIndex;
- private int _filesCount;
- private OutStreamWrapper _fileStream;
- private bool _directoryStructure;
- private int _currentIndex;
- #if !WINCE
- const int MEMORY_PRESSURE = 64 * 1024 * 1024; //64mb seems to be the maximum value
- #endif
- #region Constructors
- /// <summary>
- /// Initializes a new instance of the ArchiveExtractCallback class
- /// </summary>
- /// <param name="archive">IInArchive interface for the archive</param>
- /// <param name="directory">Directory where files are to be unpacked to</param>
- /// <param name="filesCount">The archive files count</param>'
- /// <param name="extractor">The owner of the callback</param>
- /// <param name="actualIndexes">The list of actual indexes (solid archives support)</param>
- /// <param name="directoryStructure">The value indicating whether to preserve directory structure of extracted files.</param>
- public ArchiveExtractCallback(IInArchive archive, string directory, int filesCount, bool directoryStructure,
- List<uint> actualIndexes, SevenZipExtractor extractor)
- {
- Init(archive, directory, filesCount, directoryStructure, actualIndexes, extractor);
- }
- /// <summary>
- /// Initializes a new instance of the ArchiveExtractCallback class
- /// </summary>
- /// <param name="archive">IInArchive interface for the archive</param>
- /// <param name="directory">Directory where files are to be unpacked to</param>
- /// <param name="filesCount">The archive files count</param>
- /// <param name="password">Password for the archive</param>
- /// <param name="extractor">The owner of the callback</param>
- /// <param name="actualIndexes">The list of actual indexes (solid archives support)</param>
- /// <param name="directoryStructure">The value indicating whether to preserve directory structure of extracted files.</param>
- public ArchiveExtractCallback(IInArchive archive, string directory, int filesCount, bool directoryStructure,
- List<uint> actualIndexes, string password, SevenZipExtractor extractor)
- : base(password)
- {
- Init(archive, directory, filesCount, directoryStructure, actualIndexes, extractor);
- }
- /// <summary>
- /// Initializes a new instance of the ArchiveExtractCallback class
- /// </summary>
- /// <param name="archive">IInArchive interface for the archive</param>
- /// <param name="stream">The stream where files are to be unpacked to</param>
- /// <param name="filesCount">The archive files count</param>
- /// <param name="fileIndex">The file index for the stream</param>
- /// <param name="extractor">The owner of the callback</param>
- public ArchiveExtractCallback(IInArchive archive, Stream stream, int filesCount, uint fileIndex,
- SevenZipExtractor extractor)
- {
- Init(archive, stream, filesCount, fileIndex, extractor);
- }
- /// <summary>
- /// Initializes a new instance of the ArchiveExtractCallback class
- /// </summary>
- /// <param name="archive">IInArchive interface for the archive</param>
- /// <param name="stream">The stream where files are to be unpacked to</param>
- /// <param name="filesCount">The archive files count</param>
- /// <param name="fileIndex">The file index for the stream</param>
- /// <param name="password">Password for the archive</param>
- /// <param name="extractor">The owner of the callback</param>
- public ArchiveExtractCallback(IInArchive archive, Stream stream, int filesCount, uint fileIndex, string password,
- SevenZipExtractor extractor)
- : base(password)
- {
- Init(archive, stream, filesCount, fileIndex, extractor);
- }
- private void Init(IInArchive archive, string directory, int filesCount, bool directoryStructure,
- List<uint> actualIndexes, SevenZipExtractor extractor)
- {
- CommonInit(archive, filesCount, extractor);
- _directory = directory;
- _actualIndexes = actualIndexes;
- _directoryStructure = directoryStructure;
- if (!directory.EndsWith("" + Path.DirectorySeparatorChar, StringComparison.CurrentCulture))
- {
- _directory += Path.DirectorySeparatorChar;
- }
- }
- private void Init(IInArchive archive, Stream stream, int filesCount, uint fileIndex, SevenZipExtractor extractor)
- {
- CommonInit(archive, filesCount, extractor);
- _fileStream = new OutStreamWrapper(stream, false);
- _fileStream.BytesWritten += IntEventArgsHandler;
- _fileIndex = fileIndex;
- }
- private void CommonInit(IInArchive archive, int filesCount, SevenZipExtractor extractor)
- {
- _archive = archive;
- _filesCount = filesCount;
- _fakeStream = new FakeOutStreamWrapper();
- _fakeStream.BytesWritten += IntEventArgsHandler;
- _extractor = extractor;
- #if !WINCE
- GC.AddMemoryPressure(MEMORY_PRESSURE);
- #endif
- }
- #endregion
- #region Events
- /// <summary>
- /// Occurs when a new file is going to be unpacked
- /// </summary>
- /// <remarks>Occurs when 7-zip engine requests for an output stream for a new file to unpack in</remarks>
- public event EventHandler<FileInfoEventArgs> FileExtractionStarted;
- /// <summary>
- /// Occurs when a file has been successfully unpacked
- /// </summary>
- public event EventHandler<FileInfoEventArgs> FileExtractionFinished;
- /// <summary>
- /// Occurs when the archive is opened and 7-zip sends the size of unpacked data
- /// </summary>
- public event EventHandler<OpenEventArgs> Open;
- /// <summary>
- /// Occurs when the extraction is performed
- /// </summary>
- public event EventHandler<ProgressEventArgs> Extracting;
- /// <summary>
- /// Occurs during the extraction when a file already exists
- /// </summary>
- public event EventHandler<FileOverwriteEventArgs> FileExists;
- private void OnFileExists(FileOverwriteEventArgs e)
- {
- if (FileExists != null)
- {
- FileExists(this, e);
- }
- }
- private void OnOpen(OpenEventArgs e)
- {
- if (Open != null)
- {
- Open(this, e);
- }
- }
- private void OnFileExtractionStarted(FileInfoEventArgs e)
- {
- if (FileExtractionStarted != null)
- {
- FileExtractionStarted(this, e);
- }
- }
- private void OnFileExtractionFinished(FileInfoEventArgs e)
- {
- if (FileExtractionFinished != null)
- {
- FileExtractionFinished(this, e);
- }
- }
- private void OnExtracting(ProgressEventArgs e)
- {
- if (Extracting != null)
- {
- Extracting(this, e);
- }
- }
- private void IntEventArgsHandler(object sender, IntEventArgs e)
- {
- var pold = (int)((_bytesWrittenOld * 100) / _bytesCount);
- _bytesWritten += e.Value;
- var pnow = (int)((_bytesWritten * 100) / _bytesCount);
- if (pnow > pold)
- {
- if (pnow > 100)
- {
- pold = pnow = 0;
- }
- _bytesWrittenOld = _bytesWritten;
- OnExtracting(new ProgressEventArgs((byte)pnow, (byte)(pnow - pold)));
- }
- }
- #endregion
- #region IArchiveExtractCallback Members
- /// <summary>
- /// Gives the size of the unpacked archive files
- /// </summary>
- /// <param name="total">Size of the unpacked archive files (in bytes)</param>
- public void SetTotal(ulong total)
- {
- _bytesCount = (long)total;
- OnOpen(new OpenEventArgs(total));
- }
- public void SetCompleted(ref ulong completeValue) { }
- /// <summary>
- /// Sets output stream for writing unpacked data
- /// </summary>
- /// <param name="index">Current file index</param>
- /// <param name="outStream">Output stream pointer</param>
- /// <param name="askExtractMode">Extraction mode</param>
- /// <returns>0 if OK</returns>
- public int GetStream(uint index, out
- #if !MONO
- ISequentialOutStream
- #else
- HandleRef
- #endif
- outStream, AskMode askExtractMode)
- {
- #if !MONO
- outStream = null;
- #else
- outStream = new System.Runtime.InteropServices.HandleRef(null, IntPtr.Zero);
- #endif
- if (Canceled)
- {
- return -1;
- }
- _currentIndex = (int)index;
- if (askExtractMode == AskMode.Extract)
- {
- var fileName = _directory;
- if (!_fileIndex.HasValue)
- {
- #region Extraction to a file
- if (_actualIndexes == null || _actualIndexes.Contains(index))
- {
- var data = new PropVariant();
- _archive.GetProperty(index, ItemPropId.Path, ref data);
- string entryName = NativeMethods.SafeCast(data, "");
- #region Get entryName
- if (String.IsNullOrEmpty(entryName))
- {
- if (_filesCount == 1)
- {
- var archName = Path.GetFileName(_extractor.FileName);
- archName = archName.Substring(0, archName.LastIndexOf('.'));
- if (!archName.EndsWith(".tar", StringComparison.OrdinalIgnoreCase))
- {
- archName += ".tar";
- }
- entryName = archName;
- }
- else
- {
- entryName = "[no name] " + index.ToString(CultureInfo.InvariantCulture);
- }
- }
- #endregion
- fileName = Path.Combine(_directory, _directoryStructure? entryName : Path.GetFileName(entryName));
- _archive.GetProperty(index, ItemPropId.IsDirectory, ref data);
- try
- {
- fileName = ValidateFileName(fileName);
- }
- catch (Exception e)
- {
- AddException(e);
- goto FileExtractionStartedLabel;
- }
- if (!NativeMethods.SafeCast(data, false))
- {
- #region Branch
- _archive.GetProperty(index, ItemPropId.LastWriteTime, ref data);
- var time = NativeMethods.SafeCast(data, DateTime.MinValue);
- if (File.Exists(fileName))
- {
- var fnea = new FileOverwriteEventArgs(fileName);
- OnFileExists(fnea);
- if (fnea.Cancel)
- {
- Canceled = true;
- return -1;
- }
- if (String.IsNullOrEmpty(fnea.FileName))
- {
- #if !MONO
- outStream = _fakeStream;
- #else
- outStream = _fakeStream.Handle;
- #endif
- goto FileExtractionStartedLabel;
- }
- fileName = fnea.FileName;
- }
- try
- {
- _fileStream = new OutStreamWrapper(File.Create(fileName), fileName, time, true);
- }
- catch (Exception e)
- {
- if (e is FileNotFoundException)
- {
- AddException(
- new IOException("The file \"" + fileName +
- "\" was not extracted due to the File.Create fail."));
- }
- else
- {
- AddException(e);
- }
- outStream = _fakeStream;
- goto FileExtractionStartedLabel;
- }
- _fileStream.BytesWritten += IntEventArgsHandler;
- outStream = _fileStream;
- #endregion
- }
- else
- {
- #region Branch
- if (!Directory.Exists(fileName))
- {
- try
- {
- Directory.CreateDirectory(fileName);
- }
- catch (Exception e)
- {
- AddException(e);
- }
- outStream = _fakeStream;
- }
- #endregion
- }
- }
- else
- {
- outStream = _fakeStream;
- }
- #endregion
- }
- else
- {
- #region Extraction to a stream
- if (index == _fileIndex)
- {
- outStream = _fileStream;
- _fileIndex = null;
- }
- else
- {
- outStream = _fakeStream;
- }
- #endregion
- }
- FileExtractionStartedLabel:
- _doneRate += 1.0f / _filesCount;
- var iea = new FileInfoEventArgs(
- _extractor.ArchiveFileData[(int)index], PercentDoneEventArgs.ProducePercentDone(_doneRate));
- OnFileExtractionStarted(iea);
- if (iea.Cancel)
- {
- if (!String.IsNullOrEmpty(fileName))
- {
- _fileStream.Dispose();
- if (File.Exists(fileName))
- {
- try
- {
- File.Delete(fileName);
- }
- catch (Exception e)
- {
- AddException(e);
- }
- }
- }
- Canceled = true;
- return -1;
- }
- }
- return 0;
- }
- public void PrepareOperation(AskMode askExtractMode) { }
- /// <summary>
- /// Called when the archive was extracted
- /// </summary>
- /// <param name="operationResult"></param>
- public void SetOperationResult(OperationResult operationResult)
- {
- if (operationResult != OperationResult.Ok && ReportErrors)
- {
- switch (operationResult)
- {
- case OperationResult.CrcError:
- AddException(new ExtractionFailedException("File is corrupted. Crc check has failed."));
- break;
- case OperationResult.DataError:
- AddException(new ExtractionFailedException("File is corrupted. Data error has occured."));
- break;
- case OperationResult.UnsupportedMethod:
- AddException(new ExtractionFailedException("Unsupported method error has occured."));
- break;
- }
- }
- else
- {
- if (_fileStream != null && !_fileIndex.HasValue)
- {
- try
- {
- _fileStream.BytesWritten -= IntEventArgsHandler;
- _fileStream.Dispose();
- }
- catch (ObjectDisposedException) { }
- _fileStream = null;
- GC.Collect();
- GC.WaitForPendingFinalizers();
- }
- var iea = new FileInfoEventArgs(
- _extractor.ArchiveFileData[_currentIndex], PercentDoneEventArgs.ProducePercentDone(_doneRate));
- OnFileExtractionFinished(iea);
- if (iea.Cancel)
- {
- Canceled = true;
- }
- }
- }
- #endregion
- #region ICryptoGetTextPassword Members
- /// <summary>
- /// Sets password for the archive
- /// </summary>
- /// <param name="password">Password for the archive</param>
- /// <returns>Zero if everything is OK</returns>
- public int CryptoGetTextPassword(out string password)
- {
- password = Password;
- return 0;
- }
- #endregion
- #region IDisposable Members
- public void Dispose()
- {
- #if !WINCE
- GC.RemoveMemoryPressure(MEMORY_PRESSURE);
- #endif
- if (_fileStream != null)
- {
- try
- {
- _fileStream.Dispose();
- }
- catch (ObjectDisposedException) { }
- _fileStream = null;
- }
- if (_fakeStream != null)
- {
- try
- {
- _fakeStream.Dispose();
- }
- catch (ObjectDisposedException) { }
- _fakeStream = null;
- }
- }
- #endregion
- /// <summary>
- /// Validates the file name and ensures that the directory to the file name is valid and creates intermediate directories if necessary
- /// </summary>
- /// <param name="fileName">File name</param>
- /// <returns>The valid file name</returns>
- private static string ValidateFileName(string fileName)
- {
- if (String.IsNullOrEmpty(fileName))
- {
- throw new SevenZipArchiveException("some archive name is null or empty.");
- }
- var splittedFileName = new List<string>(fileName.Split(Path.DirectorySeparatorChar));
- #if !WINCE
- foreach (char chr in Path.GetInvalidFileNameChars())
- {
- for (int i = 0; i < splittedFileName.Count; i++)
- {
- if (chr == ':' && i == 0)
- {
- continue;
- }
- if (String.IsNullOrEmpty(splittedFileName[i]))
- {
- continue;
- }
- while (splittedFileName[i].IndexOf(chr) > -1)
- {
- splittedFileName[i] = splittedFileName[i].Replace(chr, '_');
- }
- }
- }
- #endif
- if (fileName.StartsWith(new string(Path.DirectorySeparatorChar, 2),
- StringComparison.CurrentCultureIgnoreCase))
- {
- splittedFileName.RemoveAt(0);
- splittedFileName.RemoveAt(0);
- splittedFileName[0] = new string(Path.DirectorySeparatorChar, 2) + splittedFileName[0];
- }
- if (splittedFileName.Count > 2)
- {
- string tfn = splittedFileName[0];
- for (int i = 1; i < splittedFileName.Count - 1; i++)
- {
- tfn += Path.DirectorySeparatorChar + splittedFileName[i];
- if (!Directory.Exists(tfn))
- {
- Directory.CreateDirectory(tfn);
- }
- }
- }
- return String.Join(new string(Path.DirectorySeparatorChar, 1), splittedFileName.ToArray());
- }
- }
- #endif
- }
|