ArchiveExtractCallback.cs 22 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602
  1. /* This file is part of SevenZipSharp.
  2. SevenZipSharp is free software: you can redistribute it and/or modify
  3. it under the terms of the GNU Lesser General Public License as published by
  4. the Free Software Foundation, either version 3 of the License, or
  5. (at your option) any later version.
  6. SevenZipSharp is distributed in the hope that it will be useful,
  7. but WITHOUT ANY WARRANTY; without even the implied warranty of
  8. MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  9. GNU Lesser General Public License for more details.
  10. You should have received a copy of the GNU Lesser General Public License
  11. along with SevenZipSharp. If not, see <http://www.gnu.org/licenses/>.
  12. */
  13. using System;
  14. using System.Collections.Generic;
  15. using System.Globalization;
  16. using System.IO;
  17. #if MONO
  18. using SevenZip.Mono.COM;
  19. using System.Runtime.InteropServices;
  20. #endif
  21. namespace SevenZip
  22. {
  23. #if UNMANAGED
  24. /// <summary>
  25. /// Archive extraction callback to handle the process of unpacking files
  26. /// </summary>
  27. internal sealed class ArchiveExtractCallback : CallbackBase, IArchiveExtractCallback, ICryptoGetTextPassword,
  28. IDisposable
  29. {
  30. private List<uint> _actualIndexes;
  31. private IInArchive _archive;
  32. /// <summary>
  33. /// For Compressing event.
  34. /// </summary>
  35. private long _bytesCount;
  36. private long _bytesWritten;
  37. private long _bytesWrittenOld;
  38. private string _directory;
  39. /// <summary>
  40. /// Rate of the done work from [0, 1].
  41. /// </summary>
  42. private float _doneRate;
  43. private SevenZipExtractor _extractor;
  44. private FakeOutStreamWrapper _fakeStream;
  45. private uint? _fileIndex;
  46. private int _filesCount;
  47. private OutStreamWrapper _fileStream;
  48. private bool _directoryStructure;
  49. private int _currentIndex;
  50. #if !WINCE
  51. const int MEMORY_PRESSURE = 64 * 1024 * 1024; //64mb seems to be the maximum value
  52. #endif
  53. #region Constructors
  54. /// <summary>
  55. /// Initializes a new instance of the ArchiveExtractCallback class
  56. /// </summary>
  57. /// <param name="archive">IInArchive interface for the archive</param>
  58. /// <param name="directory">Directory where files are to be unpacked to</param>
  59. /// <param name="filesCount">The archive files count</param>'
  60. /// <param name="extractor">The owner of the callback</param>
  61. /// <param name="actualIndexes">The list of actual indexes (solid archives support)</param>
  62. /// <param name="directoryStructure">The value indicating whether to preserve directory structure of extracted files.</param>
  63. public ArchiveExtractCallback(IInArchive archive, string directory, int filesCount, bool directoryStructure,
  64. List<uint> actualIndexes, SevenZipExtractor extractor)
  65. {
  66. Init(archive, directory, filesCount, directoryStructure, actualIndexes, extractor);
  67. }
  68. /// <summary>
  69. /// Initializes a new instance of the ArchiveExtractCallback class
  70. /// </summary>
  71. /// <param name="archive">IInArchive interface for the archive</param>
  72. /// <param name="directory">Directory where files are to be unpacked to</param>
  73. /// <param name="filesCount">The archive files count</param>
  74. /// <param name="password">Password for the archive</param>
  75. /// <param name="extractor">The owner of the callback</param>
  76. /// <param name="actualIndexes">The list of actual indexes (solid archives support)</param>
  77. /// <param name="directoryStructure">The value indicating whether to preserve directory structure of extracted files.</param>
  78. public ArchiveExtractCallback(IInArchive archive, string directory, int filesCount, bool directoryStructure,
  79. List<uint> actualIndexes, string password, SevenZipExtractor extractor)
  80. : base(password)
  81. {
  82. Init(archive, directory, filesCount, directoryStructure, actualIndexes, extractor);
  83. }
  84. /// <summary>
  85. /// Initializes a new instance of the ArchiveExtractCallback class
  86. /// </summary>
  87. /// <param name="archive">IInArchive interface for the archive</param>
  88. /// <param name="stream">The stream where files are to be unpacked to</param>
  89. /// <param name="filesCount">The archive files count</param>
  90. /// <param name="fileIndex">The file index for the stream</param>
  91. /// <param name="extractor">The owner of the callback</param>
  92. public ArchiveExtractCallback(IInArchive archive, Stream stream, int filesCount, uint fileIndex,
  93. SevenZipExtractor extractor)
  94. {
  95. Init(archive, stream, filesCount, fileIndex, extractor);
  96. }
  97. /// <summary>
  98. /// Initializes a new instance of the ArchiveExtractCallback class
  99. /// </summary>
  100. /// <param name="archive">IInArchive interface for the archive</param>
  101. /// <param name="stream">The stream where files are to be unpacked to</param>
  102. /// <param name="filesCount">The archive files count</param>
  103. /// <param name="fileIndex">The file index for the stream</param>
  104. /// <param name="password">Password for the archive</param>
  105. /// <param name="extractor">The owner of the callback</param>
  106. public ArchiveExtractCallback(IInArchive archive, Stream stream, int filesCount, uint fileIndex, string password,
  107. SevenZipExtractor extractor)
  108. : base(password)
  109. {
  110. Init(archive, stream, filesCount, fileIndex, extractor);
  111. }
  112. private void Init(IInArchive archive, string directory, int filesCount, bool directoryStructure,
  113. List<uint> actualIndexes, SevenZipExtractor extractor)
  114. {
  115. CommonInit(archive, filesCount, extractor);
  116. _directory = directory;
  117. _actualIndexes = actualIndexes;
  118. _directoryStructure = directoryStructure;
  119. if (!directory.EndsWith("" + Path.DirectorySeparatorChar, StringComparison.CurrentCulture))
  120. {
  121. _directory += Path.DirectorySeparatorChar;
  122. }
  123. }
  124. private void Init(IInArchive archive, Stream stream, int filesCount, uint fileIndex, SevenZipExtractor extractor)
  125. {
  126. CommonInit(archive, filesCount, extractor);
  127. _fileStream = new OutStreamWrapper(stream, false);
  128. _fileStream.BytesWritten += IntEventArgsHandler;
  129. _fileIndex = fileIndex;
  130. }
  131. private void CommonInit(IInArchive archive, int filesCount, SevenZipExtractor extractor)
  132. {
  133. _archive = archive;
  134. _filesCount = filesCount;
  135. _fakeStream = new FakeOutStreamWrapper();
  136. _fakeStream.BytesWritten += IntEventArgsHandler;
  137. _extractor = extractor;
  138. #if !WINCE
  139. GC.AddMemoryPressure(MEMORY_PRESSURE);
  140. #endif
  141. }
  142. #endregion
  143. #region Events
  144. /// <summary>
  145. /// Occurs when a new file is going to be unpacked
  146. /// </summary>
  147. /// <remarks>Occurs when 7-zip engine requests for an output stream for a new file to unpack in</remarks>
  148. public event EventHandler<FileInfoEventArgs> FileExtractionStarted;
  149. /// <summary>
  150. /// Occurs when a file has been successfully unpacked
  151. /// </summary>
  152. public event EventHandler<FileInfoEventArgs> FileExtractionFinished;
  153. /// <summary>
  154. /// Occurs when the archive is opened and 7-zip sends the size of unpacked data
  155. /// </summary>
  156. public event EventHandler<OpenEventArgs> Open;
  157. /// <summary>
  158. /// Occurs when the extraction is performed
  159. /// </summary>
  160. public event EventHandler<ProgressEventArgs> Extracting;
  161. /// <summary>
  162. /// Occurs during the extraction when a file already exists
  163. /// </summary>
  164. public event EventHandler<FileOverwriteEventArgs> FileExists;
  165. private void OnFileExists(FileOverwriteEventArgs e)
  166. {
  167. if (FileExists != null)
  168. {
  169. FileExists(this, e);
  170. }
  171. }
  172. private void OnOpen(OpenEventArgs e)
  173. {
  174. if (Open != null)
  175. {
  176. Open(this, e);
  177. }
  178. }
  179. private void OnFileExtractionStarted(FileInfoEventArgs e)
  180. {
  181. if (FileExtractionStarted != null)
  182. {
  183. FileExtractionStarted(this, e);
  184. }
  185. }
  186. private void OnFileExtractionFinished(FileInfoEventArgs e)
  187. {
  188. if (FileExtractionFinished != null)
  189. {
  190. FileExtractionFinished(this, e);
  191. }
  192. }
  193. private void OnExtracting(ProgressEventArgs e)
  194. {
  195. if (Extracting != null)
  196. {
  197. Extracting(this, e);
  198. }
  199. }
  200. private void IntEventArgsHandler(object sender, IntEventArgs e)
  201. {
  202. var pold = (int)((_bytesWrittenOld * 100) / _bytesCount);
  203. _bytesWritten += e.Value;
  204. var pnow = (int)((_bytesWritten * 100) / _bytesCount);
  205. if (pnow > pold)
  206. {
  207. if (pnow > 100)
  208. {
  209. pold = pnow = 0;
  210. }
  211. _bytesWrittenOld = _bytesWritten;
  212. OnExtracting(new ProgressEventArgs((byte)pnow, (byte)(pnow - pold)));
  213. }
  214. }
  215. #endregion
  216. #region IArchiveExtractCallback Members
  217. /// <summary>
  218. /// Gives the size of the unpacked archive files
  219. /// </summary>
  220. /// <param name="total">Size of the unpacked archive files (in bytes)</param>
  221. public void SetTotal(ulong total)
  222. {
  223. _bytesCount = (long)total;
  224. OnOpen(new OpenEventArgs(total));
  225. }
  226. public void SetCompleted(ref ulong completeValue) { }
  227. /// <summary>
  228. /// Sets output stream for writing unpacked data
  229. /// </summary>
  230. /// <param name="index">Current file index</param>
  231. /// <param name="outStream">Output stream pointer</param>
  232. /// <param name="askExtractMode">Extraction mode</param>
  233. /// <returns>0 if OK</returns>
  234. public int GetStream(uint index, out
  235. #if !MONO
  236. ISequentialOutStream
  237. #else
  238. HandleRef
  239. #endif
  240. outStream, AskMode askExtractMode)
  241. {
  242. #if !MONO
  243. outStream = null;
  244. #else
  245. outStream = new System.Runtime.InteropServices.HandleRef(null, IntPtr.Zero);
  246. #endif
  247. if (Canceled)
  248. {
  249. return -1;
  250. }
  251. _currentIndex = (int)index;
  252. if (askExtractMode == AskMode.Extract)
  253. {
  254. var fileName = _directory;
  255. if (!_fileIndex.HasValue)
  256. {
  257. #region Extraction to a file
  258. if (_actualIndexes == null || _actualIndexes.Contains(index))
  259. {
  260. var data = new PropVariant();
  261. _archive.GetProperty(index, ItemPropId.Path, ref data);
  262. string entryName = NativeMethods.SafeCast(data, "");
  263. #region Get entryName
  264. if (String.IsNullOrEmpty(entryName))
  265. {
  266. if (_filesCount == 1)
  267. {
  268. var archName = Path.GetFileName(_extractor.FileName);
  269. archName = archName.Substring(0, archName.LastIndexOf('.'));
  270. if (!archName.EndsWith(".tar", StringComparison.OrdinalIgnoreCase))
  271. {
  272. archName += ".tar";
  273. }
  274. entryName = archName;
  275. }
  276. else
  277. {
  278. entryName = "[no name] " + index.ToString(CultureInfo.InvariantCulture);
  279. }
  280. }
  281. #endregion
  282. fileName = Path.Combine(_directory, _directoryStructure? entryName : Path.GetFileName(entryName));
  283. _archive.GetProperty(index, ItemPropId.IsDirectory, ref data);
  284. try
  285. {
  286. fileName = ValidateFileName(fileName);
  287. }
  288. catch (Exception e)
  289. {
  290. AddException(e);
  291. goto FileExtractionStartedLabel;
  292. }
  293. if (!NativeMethods.SafeCast(data, false))
  294. {
  295. #region Branch
  296. _archive.GetProperty(index, ItemPropId.LastWriteTime, ref data);
  297. var time = NativeMethods.SafeCast(data, DateTime.MinValue);
  298. if (File.Exists(fileName))
  299. {
  300. var fnea = new FileOverwriteEventArgs(fileName);
  301. OnFileExists(fnea);
  302. if (fnea.Cancel)
  303. {
  304. Canceled = true;
  305. return -1;
  306. }
  307. if (String.IsNullOrEmpty(fnea.FileName))
  308. {
  309. #if !MONO
  310. outStream = _fakeStream;
  311. #else
  312. outStream = _fakeStream.Handle;
  313. #endif
  314. goto FileExtractionStartedLabel;
  315. }
  316. fileName = fnea.FileName;
  317. }
  318. try
  319. {
  320. _fileStream = new OutStreamWrapper(File.Create(fileName), fileName, time, true);
  321. }
  322. catch (Exception e)
  323. {
  324. if (e is FileNotFoundException)
  325. {
  326. AddException(
  327. new IOException("The file \"" + fileName +
  328. "\" was not extracted due to the File.Create fail."));
  329. }
  330. else
  331. {
  332. AddException(e);
  333. }
  334. outStream = _fakeStream;
  335. goto FileExtractionStartedLabel;
  336. }
  337. _fileStream.BytesWritten += IntEventArgsHandler;
  338. outStream = _fileStream;
  339. #endregion
  340. }
  341. else
  342. {
  343. #region Branch
  344. if (!Directory.Exists(fileName))
  345. {
  346. try
  347. {
  348. Directory.CreateDirectory(fileName);
  349. }
  350. catch (Exception e)
  351. {
  352. AddException(e);
  353. }
  354. outStream = _fakeStream;
  355. }
  356. #endregion
  357. }
  358. }
  359. else
  360. {
  361. outStream = _fakeStream;
  362. }
  363. #endregion
  364. }
  365. else
  366. {
  367. #region Extraction to a stream
  368. if (index == _fileIndex)
  369. {
  370. outStream = _fileStream;
  371. _fileIndex = null;
  372. }
  373. else
  374. {
  375. outStream = _fakeStream;
  376. }
  377. #endregion
  378. }
  379. FileExtractionStartedLabel:
  380. _doneRate += 1.0f / _filesCount;
  381. var iea = new FileInfoEventArgs(
  382. _extractor.ArchiveFileData[(int)index], PercentDoneEventArgs.ProducePercentDone(_doneRate));
  383. OnFileExtractionStarted(iea);
  384. if (iea.Cancel)
  385. {
  386. if (!String.IsNullOrEmpty(fileName))
  387. {
  388. _fileStream.Dispose();
  389. if (File.Exists(fileName))
  390. {
  391. try
  392. {
  393. File.Delete(fileName);
  394. }
  395. catch (Exception e)
  396. {
  397. AddException(e);
  398. }
  399. }
  400. }
  401. Canceled = true;
  402. return -1;
  403. }
  404. }
  405. return 0;
  406. }
  407. public void PrepareOperation(AskMode askExtractMode) { }
  408. /// <summary>
  409. /// Called when the archive was extracted
  410. /// </summary>
  411. /// <param name="operationResult"></param>
  412. public void SetOperationResult(OperationResult operationResult)
  413. {
  414. if (operationResult != OperationResult.Ok && ReportErrors)
  415. {
  416. switch (operationResult)
  417. {
  418. case OperationResult.CrcError:
  419. AddException(new ExtractionFailedException("File is corrupted. Crc check has failed."));
  420. break;
  421. case OperationResult.DataError:
  422. AddException(new ExtractionFailedException("File is corrupted. Data error has occured."));
  423. break;
  424. case OperationResult.UnsupportedMethod:
  425. AddException(new ExtractionFailedException("Unsupported method error has occured."));
  426. break;
  427. }
  428. }
  429. else
  430. {
  431. if (_fileStream != null && !_fileIndex.HasValue)
  432. {
  433. try
  434. {
  435. _fileStream.BytesWritten -= IntEventArgsHandler;
  436. _fileStream.Dispose();
  437. }
  438. catch (ObjectDisposedException) { }
  439. _fileStream = null;
  440. GC.Collect();
  441. GC.WaitForPendingFinalizers();
  442. }
  443. var iea = new FileInfoEventArgs(
  444. _extractor.ArchiveFileData[_currentIndex], PercentDoneEventArgs.ProducePercentDone(_doneRate));
  445. OnFileExtractionFinished(iea);
  446. if (iea.Cancel)
  447. {
  448. Canceled = true;
  449. }
  450. }
  451. }
  452. #endregion
  453. #region ICryptoGetTextPassword Members
  454. /// <summary>
  455. /// Sets password for the archive
  456. /// </summary>
  457. /// <param name="password">Password for the archive</param>
  458. /// <returns>Zero if everything is OK</returns>
  459. public int CryptoGetTextPassword(out string password)
  460. {
  461. password = Password;
  462. return 0;
  463. }
  464. #endregion
  465. #region IDisposable Members
  466. public void Dispose()
  467. {
  468. #if !WINCE
  469. GC.RemoveMemoryPressure(MEMORY_PRESSURE);
  470. #endif
  471. if (_fileStream != null)
  472. {
  473. try
  474. {
  475. _fileStream.Dispose();
  476. }
  477. catch (ObjectDisposedException) { }
  478. _fileStream = null;
  479. }
  480. if (_fakeStream != null)
  481. {
  482. try
  483. {
  484. _fakeStream.Dispose();
  485. }
  486. catch (ObjectDisposedException) { }
  487. _fakeStream = null;
  488. }
  489. }
  490. #endregion
  491. /// <summary>
  492. /// Validates the file name and ensures that the directory to the file name is valid and creates intermediate directories if necessary
  493. /// </summary>
  494. /// <param name="fileName">File name</param>
  495. /// <returns>The valid file name</returns>
  496. private static string ValidateFileName(string fileName)
  497. {
  498. if (String.IsNullOrEmpty(fileName))
  499. {
  500. throw new SevenZipArchiveException("some archive name is null or empty.");
  501. }
  502. var splittedFileName = new List<string>(fileName.Split(Path.DirectorySeparatorChar));
  503. #if !WINCE
  504. foreach (char chr in Path.GetInvalidFileNameChars())
  505. {
  506. for (int i = 0; i < splittedFileName.Count; i++)
  507. {
  508. if (chr == ':' && i == 0)
  509. {
  510. continue;
  511. }
  512. if (String.IsNullOrEmpty(splittedFileName[i]))
  513. {
  514. continue;
  515. }
  516. while (splittedFileName[i].IndexOf(chr) > -1)
  517. {
  518. splittedFileName[i] = splittedFileName[i].Replace(chr, '_');
  519. }
  520. }
  521. }
  522. #endif
  523. if (fileName.StartsWith(new string(Path.DirectorySeparatorChar, 2),
  524. StringComparison.CurrentCultureIgnoreCase))
  525. {
  526. splittedFileName.RemoveAt(0);
  527. splittedFileName.RemoveAt(0);
  528. splittedFileName[0] = new string(Path.DirectorySeparatorChar, 2) + splittedFileName[0];
  529. }
  530. if (splittedFileName.Count > 2)
  531. {
  532. string tfn = splittedFileName[0];
  533. for (int i = 1; i < splittedFileName.Count - 1; i++)
  534. {
  535. tfn += Path.DirectorySeparatorChar + splittedFileName[i];
  536. if (!Directory.Exists(tfn))
  537. {
  538. Directory.CreateDirectory(tfn);
  539. }
  540. }
  541. }
  542. return String.Join(new string(Path.DirectorySeparatorChar, 1), splittedFileName.ToArray());
  543. }
  544. }
  545. #endif
  546. }