SevenZipExtractor.cs 51 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376
  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.Collections.ObjectModel;
  16. using System.Diagnostics;
  17. using System.Globalization;
  18. using System.IO;
  19. #if DOTNET20
  20. using System.Threading;
  21. #else
  22. using System.Linq;
  23. #endif
  24. using SevenZip.Sdk.Compression.Lzma;
  25. #if MONO
  26. using SevenZip.Mono.COM;
  27. #endif
  28. namespace SevenZip
  29. {
  30. /// <summary>
  31. /// Class for extracting and getting information about 7-zip archives
  32. /// </summary>
  33. public sealed partial class SevenZipExtractor
  34. #if UNMANAGED
  35. : SevenZipBase, IDisposable
  36. #endif
  37. {
  38. #if UNMANAGED
  39. private List<ArchiveFileInfo> _archiveFileData;
  40. private IInArchive _archive;
  41. private IInStream _archiveStream;
  42. private int _offset;
  43. private ArchiveOpenCallback _openCallback;
  44. private string _fileName;
  45. private Stream _inStream;
  46. private long? _packedSize;
  47. private long? _unpackedSize;
  48. private uint? _filesCount;
  49. private bool? _isSolid;
  50. private bool _opened;
  51. private bool _disposed;
  52. private InArchiveFormat _format;
  53. private ReadOnlyCollection<ArchiveFileInfo> _archiveFileInfoCollection;
  54. private ReadOnlyCollection<ArchiveProperty> _archiveProperties;
  55. private ReadOnlyCollection<string> _volumeFileNames;
  56. #region Constructors
  57. /// <summary>
  58. /// General initialization function.
  59. /// </summary>
  60. /// <param name="archiveFullName">The archive file name.</param>
  61. private void Init(string archiveFullName)
  62. {
  63. _fileName = archiveFullName;
  64. bool isExecutable;
  65. _format = FileChecker.CheckSignature(archiveFullName, out _offset, out isExecutable);
  66. PreserveDirectoryStructure = true;
  67. SevenZipLibraryManager.LoadLibrary(this, _format);
  68. try
  69. {
  70. _archive = SevenZipLibraryManager.InArchive(_format, this);
  71. }
  72. catch (SevenZipLibraryException)
  73. {
  74. SevenZipLibraryManager.FreeLibrary(this, _format);
  75. throw;
  76. }
  77. if (isExecutable && _format != InArchiveFormat.PE)
  78. {
  79. if (!Check())
  80. {
  81. CommonDispose();
  82. _format = InArchiveFormat.PE;
  83. SevenZipLibraryManager.LoadLibrary(this, _format);
  84. try
  85. {
  86. _archive = SevenZipLibraryManager.InArchive(_format, this);
  87. }
  88. catch (SevenZipLibraryException)
  89. {
  90. SevenZipLibraryManager.FreeLibrary(this, _format);
  91. throw;
  92. }
  93. }
  94. }
  95. }
  96. /// <summary>
  97. /// General initialization function.
  98. /// </summary>
  99. /// <param name="stream">The stream to read the archive from.</param>
  100. private void Init(Stream stream)
  101. {
  102. ValidateStream(stream);
  103. bool isExecutable;
  104. _format = FileChecker.CheckSignature(stream, out _offset, out isExecutable);
  105. PreserveDirectoryStructure = true;
  106. SevenZipLibraryManager.LoadLibrary(this, _format);
  107. try
  108. {
  109. _inStream = new ArchiveEmulationStreamProxy(stream, _offset);
  110. _packedSize = stream.Length;
  111. _archive = SevenZipLibraryManager.InArchive(_format, this);
  112. }
  113. catch (SevenZipLibraryException)
  114. {
  115. SevenZipLibraryManager.FreeLibrary(this, _format);
  116. throw;
  117. }
  118. if (isExecutable && _format != InArchiveFormat.PE)
  119. {
  120. if (!Check())
  121. {
  122. CommonDispose();
  123. _format = InArchiveFormat.PE;
  124. try
  125. {
  126. _inStream = new ArchiveEmulationStreamProxy(stream, _offset);
  127. _packedSize = stream.Length;
  128. _archive = SevenZipLibraryManager.InArchive(_format, this);
  129. }
  130. catch (SevenZipLibraryException)
  131. {
  132. SevenZipLibraryManager.FreeLibrary(this, _format);
  133. throw;
  134. }
  135. }
  136. }
  137. }
  138. /// <summary>
  139. /// Initializes a new instance of SevenZipExtractor class.
  140. /// </summary>
  141. /// <param name="archiveStream">The stream to read the archive from.
  142. /// Use SevenZipExtractor(string) to extract from disk, though it is not necessary.</param>
  143. /// <remarks>The archive format is guessed by the signature.</remarks>
  144. public SevenZipExtractor(Stream archiveStream)
  145. {
  146. Init(archiveStream);
  147. }
  148. /// <summary>
  149. /// Initializes a new instance of SevenZipExtractor class
  150. /// </summary>
  151. /// <param name="archiveFullName">The archive full file name</param>
  152. public SevenZipExtractor(string archiveFullName)
  153. {
  154. Init(archiveFullName);
  155. }
  156. /// <summary>
  157. /// Initializes a new instance of SevenZipExtractor class.
  158. /// </summary>
  159. /// <param name="archiveFullName">The archive full file name.</param>
  160. /// <param name="password">Password for an encrypted archive.</param>
  161. public SevenZipExtractor(string archiveFullName, string password)
  162. : base(password)
  163. {
  164. Init(archiveFullName);
  165. }
  166. /// <summary>
  167. /// Initializes a new instance of SevenZipExtractor class.
  168. /// </summary>
  169. /// <param name="archiveStream">The stream to read the archive from.</param>
  170. /// <param name="password">Password for an encrypted archive.</param>
  171. /// <remarks>The archive format is guessed by the signature.</remarks>
  172. public SevenZipExtractor(
  173. Stream archiveStream, string password)
  174. : base(password)
  175. {
  176. Init(archiveStream);
  177. }
  178. #endregion
  179. #region Properties
  180. /// <summary>
  181. /// Gets or sets archive full file name
  182. /// </summary>
  183. public string FileName
  184. {
  185. get
  186. {
  187. DisposedCheck();
  188. return _fileName;
  189. }
  190. }
  191. /// <summary>
  192. /// Gets the size of the archive file
  193. /// </summary>
  194. public long PackedSize
  195. {
  196. get
  197. {
  198. DisposedCheck();
  199. return _packedSize.HasValue
  200. ?
  201. _packedSize.Value
  202. :
  203. _fileName != null
  204. ?
  205. (new FileInfo(_fileName)).Length
  206. :
  207. -1;
  208. }
  209. }
  210. /// <summary>
  211. /// Gets the size of unpacked archive data
  212. /// </summary>
  213. public long UnpackedSize
  214. {
  215. get
  216. {
  217. DisposedCheck();
  218. if (!_unpackedSize.HasValue)
  219. {
  220. return -1;
  221. }
  222. return _unpackedSize.Value;
  223. }
  224. }
  225. /// <summary>
  226. /// Gets a value indicating whether the archive is solid
  227. /// </summary>
  228. public bool IsSolid
  229. {
  230. get
  231. {
  232. DisposedCheck();
  233. if (!_isSolid.HasValue)
  234. {
  235. GetArchiveInfo(true);
  236. }
  237. if (_isSolid != null)
  238. {
  239. return _isSolid.Value;
  240. }
  241. throw new SevenZipException("_isSolid == null");
  242. }
  243. }
  244. /// <summary>
  245. /// Gets the number of files in the archive
  246. /// </summary>
  247. [CLSCompliant(false)]
  248. public uint FilesCount
  249. {
  250. get
  251. {
  252. DisposedCheck();
  253. if (!_filesCount.HasValue)
  254. {
  255. GetArchiveInfo(true);
  256. }
  257. if (_filesCount != null)
  258. {
  259. return _filesCount.Value;
  260. }
  261. throw new SevenZipException("_filesCount == null");
  262. }
  263. }
  264. /// <summary>
  265. /// Gets archive format
  266. /// </summary>
  267. public InArchiveFormat Format
  268. {
  269. get
  270. {
  271. DisposedCheck();
  272. return _format;
  273. }
  274. }
  275. /// <summary>
  276. /// Gets or sets the value indicatin whether to preserve the directory structure of extracted files.
  277. /// </summary>
  278. public bool PreserveDirectoryStructure { get; set; }
  279. #endregion
  280. /// <summary>
  281. /// Checked whether the class was disposed.
  282. /// </summary>
  283. /// <exception cref="System.ObjectDisposedException" />
  284. private void DisposedCheck()
  285. {
  286. if (_disposed)
  287. {
  288. throw new ObjectDisposedException("SevenZipExtractor");
  289. }
  290. #if !WINCE
  291. RecreateInstanceIfNeeded();
  292. #endif
  293. }
  294. #region Core private functions
  295. private ArchiveOpenCallback GetArchiveOpenCallback()
  296. {
  297. return _openCallback ?? (_openCallback = String.IsNullOrEmpty(Password)
  298. ? new ArchiveOpenCallback(_fileName)
  299. : new ArchiveOpenCallback(_fileName, Password));
  300. }
  301. /// <summary>
  302. /// Gets the archive input stream.
  303. /// </summary>
  304. /// <returns>The archive input wrapper stream.</returns>
  305. private IInStream GetArchiveStream(bool dispose)
  306. {
  307. if (_archiveStream != null)
  308. {
  309. if (_archiveStream is DisposeVariableWrapper)
  310. {
  311. (_archiveStream as DisposeVariableWrapper).DisposeStream = dispose;
  312. }
  313. return _archiveStream;
  314. }
  315. if (_inStream != null)
  316. {
  317. _inStream.Seek(0, SeekOrigin.Begin);
  318. _archiveStream = new InStreamWrapper(_inStream, false);
  319. }
  320. else
  321. {
  322. if (!_fileName.EndsWith(".001", StringComparison.OrdinalIgnoreCase))
  323. {
  324. _archiveStream = new InStreamWrapper(
  325. new ArchiveEmulationStreamProxy(new FileStream(
  326. _fileName, FileMode.Open, FileAccess.Read, FileShare.ReadWrite),
  327. _offset),
  328. dispose);
  329. }
  330. else
  331. {
  332. _archiveStream = new InMultiStreamWrapper(_fileName, dispose);
  333. _packedSize = (_archiveStream as InMultiStreamWrapper).Length;
  334. }
  335. }
  336. return _archiveStream;
  337. }
  338. /// <summary>
  339. /// Opens the archive and throws exceptions or returns OperationResult.DataError if any error occurs.
  340. /// </summary>
  341. /// <param name="archiveStream">The IInStream compliant class instance, that is, the input stream.</param>
  342. /// <param name="openCallback">The ArchiveOpenCallback instance.</param>
  343. /// <returns>OperationResult.Ok if Open() succeeds.</returns>
  344. private OperationResult OpenArchiveInner(IInStream archiveStream,
  345. IArchiveOpenCallback openCallback)
  346. {
  347. ulong checkPos = 1 << 15;
  348. int res = _archive.Open(archiveStream, ref checkPos, openCallback);
  349. return (OperationResult)res;
  350. }
  351. /// <summary>
  352. /// Opens the archive and throws exceptions or returns OperationResult.DataError if any error occurs.
  353. /// </summary>
  354. /// <param name="archiveStream">The IInStream compliant class instance, that is, the input stream.</param>
  355. /// <param name="openCallback">The ArchiveOpenCallback instance.</param>
  356. /// <returns>True if Open() succeeds; otherwise, false.</returns>
  357. private bool OpenArchive(IInStream archiveStream,
  358. ArchiveOpenCallback openCallback)
  359. {
  360. if (!_opened)
  361. {
  362. if (OpenArchiveInner(archiveStream, openCallback) != OperationResult.Ok)
  363. {
  364. if (!ThrowException(null, new SevenZipArchiveException()))
  365. {
  366. return false;
  367. }
  368. }
  369. _volumeFileNames = new ReadOnlyCollection<string>(openCallback.VolumeFileNames);
  370. _opened = true;
  371. }
  372. return true;
  373. }
  374. /// <summary>
  375. /// Retrieves all information about the archive.
  376. /// </summary>
  377. /// <exception cref="SevenZip.SevenZipArchiveException"/>
  378. private void GetArchiveInfo(bool disposeStream)
  379. {
  380. if (_archive == null)
  381. {
  382. if (!ThrowException(null, new SevenZipArchiveException()))
  383. {
  384. return;
  385. }
  386. }
  387. else
  388. {
  389. IInStream archiveStream;
  390. using ((archiveStream = GetArchiveStream(disposeStream)) as IDisposable)
  391. {
  392. var openCallback = GetArchiveOpenCallback();
  393. if (!_opened)
  394. {
  395. if (!OpenArchive(archiveStream, openCallback))
  396. {
  397. return;
  398. }
  399. _opened = !disposeStream;
  400. }
  401. _filesCount = _archive.GetNumberOfItems();
  402. _archiveFileData = new List<ArchiveFileInfo>((int)_filesCount);
  403. if (_filesCount != 0)
  404. {
  405. var data = new PropVariant();
  406. try
  407. {
  408. #region Getting archive items data
  409. for (uint i = 0; i < _filesCount; i++)
  410. {
  411. try
  412. {
  413. var fileInfo = new ArchiveFileInfo { Index = (int)i };
  414. _archive.GetProperty(i, ItemPropId.Path, ref data);
  415. fileInfo.FileName = NativeMethods.SafeCast(data, "[no name]");
  416. _archive.GetProperty(i, ItemPropId.LastWriteTime, ref data);
  417. fileInfo.LastWriteTime = NativeMethods.SafeCast(data, DateTime.Now);
  418. _archive.GetProperty(i, ItemPropId.CreationTime, ref data);
  419. fileInfo.CreationTime = NativeMethods.SafeCast(data, DateTime.Now);
  420. _archive.GetProperty(i, ItemPropId.LastAccessTime, ref data);
  421. fileInfo.LastAccessTime = NativeMethods.SafeCast(data, DateTime.Now);
  422. _archive.GetProperty(i, ItemPropId.Size, ref data);
  423. fileInfo.Size = NativeMethods.SafeCast<ulong>(data, 0);
  424. if (fileInfo.Size == 0)
  425. {
  426. fileInfo.Size = NativeMethods.SafeCast<uint>(data, 0);
  427. }
  428. _archive.GetProperty(i, ItemPropId.Attributes, ref data);
  429. fileInfo.Attributes = NativeMethods.SafeCast<uint>(data, 0);
  430. _archive.GetProperty(i, ItemPropId.IsDirectory, ref data);
  431. fileInfo.IsDirectory = NativeMethods.SafeCast(data, false);
  432. _archive.GetProperty(i, ItemPropId.Encrypted, ref data);
  433. fileInfo.Encrypted = NativeMethods.SafeCast(data, false);
  434. _archive.GetProperty(i, ItemPropId.Crc, ref data);
  435. fileInfo.Crc = NativeMethods.SafeCast<uint>(data, 0);
  436. _archive.GetProperty(i, ItemPropId.Comment, ref data);
  437. fileInfo.Comment = NativeMethods.SafeCast(data, "");
  438. _archiveFileData.Add(fileInfo);
  439. }
  440. catch (InvalidCastException)
  441. {
  442. ThrowException(null, new SevenZipArchiveException("probably archive is corrupted."));
  443. }
  444. }
  445. #endregion
  446. #region Getting archive properties
  447. uint numProps = _archive.GetNumberOfArchiveProperties();
  448. var archProps = new List<ArchiveProperty>((int)numProps);
  449. for (uint i = 0; i < numProps; i++)
  450. {
  451. string propName;
  452. ItemPropId propId;
  453. ushort varType;
  454. _archive.GetArchivePropertyInfo(i, out propName, out propId, out varType);
  455. _archive.GetArchiveProperty(propId, ref data);
  456. if (propId == ItemPropId.Solid)
  457. {
  458. _isSolid = NativeMethods.SafeCast(data, true);
  459. }
  460. // TODO Add more archive properties
  461. if (PropIdToName.PropIdNames.ContainsKey(propId))
  462. {
  463. archProps.Add(new ArchiveProperty
  464. {
  465. Name = PropIdToName.PropIdNames[propId],
  466. Value = data.Object
  467. });
  468. }
  469. else
  470. {
  471. Debug.WriteLine(
  472. "An unknown archive property encountered (code " +
  473. ((int)propId).ToString(CultureInfo.InvariantCulture) + ')');
  474. }
  475. }
  476. _archiveProperties = new ReadOnlyCollection<ArchiveProperty>(archProps);
  477. if (!_isSolid.HasValue && _format == InArchiveFormat.Zip)
  478. {
  479. _isSolid = false;
  480. }
  481. if (!_isSolid.HasValue)
  482. {
  483. _isSolid = true;
  484. }
  485. #endregion
  486. }
  487. catch (Exception)
  488. {
  489. if (openCallback.ThrowException())
  490. {
  491. throw;
  492. }
  493. }
  494. }
  495. }
  496. if (disposeStream)
  497. {
  498. _archive.Close();
  499. _archiveStream = null;
  500. }
  501. _archiveFileInfoCollection = new ReadOnlyCollection<ArchiveFileInfo>(_archiveFileData);
  502. }
  503. }
  504. /// <summary>
  505. /// Ensure that _archiveFileData is loaded.
  506. /// </summary>
  507. /// <param name="disposeStream">Dispose the archive stream after this operation.</param>
  508. private void InitArchiveFileData(bool disposeStream)
  509. {
  510. if (_archiveFileData == null)
  511. {
  512. GetArchiveInfo(disposeStream);
  513. }
  514. }
  515. /// <summary>
  516. /// Produces an array of indexes from 0 to the maximum value in the specified array
  517. /// </summary>
  518. /// <param name="indexes">The source array</param>
  519. /// <returns>The array of indexes from 0 to the maximum value in the specified array</returns>
  520. private static uint[] SolidIndexes(uint[] indexes)
  521. {
  522. #if CS4
  523. int max = indexes.Aggregate(0, (current, i) => Math.Max(current, (int) i));
  524. #else
  525. int max = 0;
  526. foreach (uint i in indexes)
  527. {
  528. max = Math.Max(max, (int)i);
  529. }
  530. #endif
  531. if (max > 0)
  532. {
  533. max++;
  534. var res = new uint[max];
  535. for (int i = 0; i < max; i++)
  536. {
  537. res[i] = (uint)i;
  538. }
  539. return res;
  540. }
  541. return indexes;
  542. }
  543. /// <summary>
  544. /// Checkes whether all the indexes are valid.
  545. /// </summary>
  546. /// <param name="indexes">The indexes to check.</param>
  547. /// <returns>True is valid; otherwise, false.</returns>
  548. private static bool CheckIndexes(params int[] indexes)
  549. {
  550. #if CS4 // Wow, C# 4 is great!
  551. return indexes.All(i => i >= 0);
  552. #else
  553. bool res = true;
  554. foreach (int i in indexes)
  555. {
  556. if (i < 0)
  557. {
  558. res = false;
  559. break;
  560. }
  561. }
  562. return res;
  563. #endif
  564. }
  565. private void ArchiveExtractCallbackCommonInit(ArchiveExtractCallback aec)
  566. {
  567. aec.Open += ((s, e) => { _unpackedSize = (long)e.TotalSize; });
  568. aec.FileExtractionStarted += FileExtractionStartedEventProxy;
  569. aec.FileExtractionFinished += FileExtractionFinishedEventProxy;
  570. aec.Extracting += ExtractingEventProxy;
  571. aec.FileExists += FileExistsEventProxy;
  572. }
  573. /// <summary>
  574. /// Gets the IArchiveExtractCallback callback
  575. /// </summary>
  576. /// <param name="directory">The directory where extract the files</param>
  577. /// <param name="filesCount">The number of files to be extracted</param>
  578. /// <param name="actualIndexes">The list of actual indexes (solid archives support)</param>
  579. /// <returns>The ArchiveExtractCallback callback</returns>
  580. private ArchiveExtractCallback GetArchiveExtractCallback(string directory, int filesCount,
  581. List<uint> actualIndexes)
  582. {
  583. var aec = String.IsNullOrEmpty(Password)
  584. ? new ArchiveExtractCallback(_archive, directory, filesCount, PreserveDirectoryStructure, actualIndexes, this)
  585. : new ArchiveExtractCallback(_archive, directory, filesCount, PreserveDirectoryStructure, actualIndexes, Password, this);
  586. ArchiveExtractCallbackCommonInit(aec);
  587. return aec;
  588. }
  589. /// <summary>
  590. /// Gets the IArchiveExtractCallback callback
  591. /// </summary>
  592. /// <param name="stream">The stream where extract the file</param>
  593. /// <param name="index">The file index</param>
  594. /// <param name="filesCount">The number of files to be extracted</param>
  595. /// <returns>The ArchiveExtractCallback callback</returns>
  596. private ArchiveExtractCallback GetArchiveExtractCallback(Stream stream, uint index, int filesCount)
  597. {
  598. var aec = String.IsNullOrEmpty(Password)
  599. ? new ArchiveExtractCallback(_archive, stream, filesCount, index, this)
  600. : new ArchiveExtractCallback(_archive, stream, filesCount, index, Password, this);
  601. ArchiveExtractCallbackCommonInit(aec);
  602. return aec;
  603. }
  604. private void FreeArchiveExtractCallback(ArchiveExtractCallback callback)
  605. {
  606. callback.Open -= ((s, e) => { _unpackedSize = (long)e.TotalSize; });
  607. callback.FileExtractionStarted -= FileExtractionStartedEventProxy;
  608. callback.FileExtractionFinished -= FileExtractionFinishedEventProxy;
  609. callback.Extracting -= ExtractingEventProxy;
  610. callback.FileExists -= FileExistsEventProxy;
  611. }
  612. #endregion
  613. #endif
  614. /// <summary>
  615. /// Checks if the specified stream supports extraction.
  616. /// </summary>
  617. /// <param name="stream">The stream to check.</param>
  618. private static void ValidateStream(Stream stream)
  619. {
  620. if (stream == null)
  621. {
  622. throw new ArgumentNullException("stream");
  623. }
  624. if (!stream.CanSeek || !stream.CanRead)
  625. {
  626. throw new ArgumentException("The specified stream can not seek or read.", "stream");
  627. }
  628. if (stream.Length == 0)
  629. {
  630. throw new ArgumentException("The specified stream has zero length.", "stream");
  631. }
  632. }
  633. #if UNMANAGED
  634. #region IDisposable Members
  635. private void CommonDispose()
  636. {
  637. if (_opened)
  638. {
  639. try
  640. {
  641. if (_archive != null)
  642. {
  643. _archive.Close();
  644. }
  645. }
  646. catch (Exception) { }
  647. }
  648. _archive = null;
  649. _archiveFileData = null;
  650. _archiveProperties = null;
  651. _archiveFileInfoCollection = null;
  652. _inStream = null;
  653. if (_openCallback != null)
  654. {
  655. try
  656. {
  657. _openCallback.Dispose();
  658. }
  659. catch (ObjectDisposedException) { }
  660. _openCallback = null;
  661. }
  662. if (_archiveStream != null)
  663. {
  664. if (_archiveStream is IDisposable)
  665. {
  666. try
  667. {
  668. if (_archiveStream is DisposeVariableWrapper)
  669. {
  670. (_archiveStream as DisposeVariableWrapper).DisposeStream = true;
  671. }
  672. (_archiveStream as IDisposable).Dispose();
  673. }
  674. catch (ObjectDisposedException) { }
  675. _archiveStream = null;
  676. }
  677. }
  678. SevenZipLibraryManager.FreeLibrary(this, _format);
  679. }
  680. /// <summary>
  681. /// Releases the unmanaged resources used by SevenZipExtractor.
  682. /// </summary>
  683. public void Dispose()
  684. {
  685. if (!_disposed)
  686. {
  687. CommonDispose();
  688. }
  689. _disposed = true;
  690. GC.SuppressFinalize(this);
  691. }
  692. #endregion
  693. #region Core public Members
  694. #region Events
  695. /// <summary>
  696. /// Occurs when a new file is going to be unpacked.
  697. /// </summary>
  698. /// <remarks>Occurs when 7-zip engine requests for an output stream for a new file to unpack in.</remarks>
  699. public event EventHandler<FileInfoEventArgs> FileExtractionStarted;
  700. /// <summary>
  701. /// Occurs when a file has been successfully unpacked.
  702. /// </summary>
  703. public event EventHandler<FileInfoEventArgs> FileExtractionFinished;
  704. /// <summary>
  705. /// Occurs when the archive has been unpacked.
  706. /// </summary>
  707. public event EventHandler<EventArgs> ExtractionFinished;
  708. /// <summary>
  709. /// Occurs when data are being extracted.
  710. /// </summary>
  711. /// <remarks>Use this event for accurate progress handling and various ProgressBar.StepBy(e.PercentDelta) routines.</remarks>
  712. public event EventHandler<ProgressEventArgs> Extracting;
  713. /// <summary>
  714. /// Occurs during the extraction when a file already exists.
  715. /// </summary>
  716. public event EventHandler<FileOverwriteEventArgs> FileExists;
  717. #region Event proxies
  718. /// <summary>
  719. /// Event proxy for FileExtractionStarted.
  720. /// </summary>
  721. /// <param name="sender">The sender of the event.</param>
  722. /// <param name="e">The event arguments.</param>
  723. private void FileExtractionStartedEventProxy(object sender, FileInfoEventArgs e)
  724. {
  725. OnEvent(FileExtractionStarted, e, true);
  726. }
  727. /// <summary>
  728. /// Event proxy for FileExtractionFinished.
  729. /// </summary>
  730. /// <param name="sender">The sender of the event.</param>
  731. /// <param name="e">The event arguments.</param>
  732. private void FileExtractionFinishedEventProxy(object sender, FileInfoEventArgs e)
  733. {
  734. OnEvent(FileExtractionFinished, e, true);
  735. }
  736. /// <summary>
  737. /// Event proxy for Extractng.
  738. /// </summary>
  739. /// <param name="sender">The sender of the event.</param>
  740. /// <param name="e">The event arguments.</param>
  741. private void ExtractingEventProxy(object sender, ProgressEventArgs e)
  742. {
  743. OnEvent(Extracting, e, false);
  744. }
  745. /// <summary>
  746. /// Event proxy for FileExists.
  747. /// </summary>
  748. /// <param name="sender">The sender of the event.</param>
  749. /// <param name="e">The event arguments.</param>
  750. private void FileExistsEventProxy(object sender, FileOverwriteEventArgs e)
  751. {
  752. OnEvent(FileExists, e, true);
  753. }
  754. #endregion
  755. #endregion
  756. #region Properties
  757. /// <summary>
  758. /// Gets the collection of ArchiveFileInfo with all information about files in the archive
  759. /// </summary>
  760. public ReadOnlyCollection<ArchiveFileInfo> ArchiveFileData
  761. {
  762. get
  763. {
  764. DisposedCheck();
  765. InitArchiveFileData(true);
  766. return _archiveFileInfoCollection;
  767. }
  768. }
  769. /// <summary>
  770. /// Gets the properties for the current archive
  771. /// </summary>
  772. public ReadOnlyCollection<ArchiveProperty> ArchiveProperties
  773. {
  774. get
  775. {
  776. DisposedCheck();
  777. InitArchiveFileData(true);
  778. return _archiveProperties;
  779. }
  780. }
  781. /// <summary>
  782. /// Gets the collection of all file names contained in the archive.
  783. /// </summary>
  784. /// <remarks>
  785. /// Each get recreates the collection
  786. /// </remarks>
  787. public ReadOnlyCollection<string> ArchiveFileNames
  788. {
  789. get
  790. {
  791. DisposedCheck();
  792. InitArchiveFileData(true);
  793. var fileNames = new List<string>(_archiveFileData.Count);
  794. #if CS4
  795. fileNames.AddRange(_archiveFileData.Select(afi => afi.FileName));
  796. #else
  797. foreach (var afi in _archiveFileData)
  798. {
  799. fileNames.Add(afi.FileName);
  800. }
  801. #endif
  802. return new ReadOnlyCollection<string>(fileNames);
  803. }
  804. }
  805. /// <summary>
  806. /// Gets the list of archive volume file names.
  807. /// </summary>
  808. public ReadOnlyCollection<string> VolumeFileNames
  809. {
  810. get
  811. {
  812. DisposedCheck();
  813. InitArchiveFileData(true);
  814. return _volumeFileNames;
  815. }
  816. }
  817. #endregion
  818. /// <summary>
  819. /// Performs the archive integrity test.
  820. /// </summary>
  821. /// <returns>True is the archive is ok; otherwise, false.</returns>
  822. public bool Check()
  823. {
  824. DisposedCheck();
  825. try
  826. {
  827. InitArchiveFileData(false);
  828. var archiveStream = GetArchiveStream(true);
  829. var openCallback = GetArchiveOpenCallback();
  830. if (!OpenArchive(archiveStream, openCallback))
  831. {
  832. return false;
  833. }
  834. using (var aec = GetArchiveExtractCallback("", (int)_filesCount, null))
  835. {
  836. try
  837. {
  838. CheckedExecute(
  839. _archive.Extract(null, UInt32.MaxValue, 1, aec),
  840. SevenZipExtractionFailedException.DEFAULT_MESSAGE, aec);
  841. }
  842. finally
  843. {
  844. FreeArchiveExtractCallback(aec);
  845. }
  846. }
  847. }
  848. catch (Exception)
  849. {
  850. return false;
  851. }
  852. finally
  853. {
  854. if (_archive != null)
  855. {
  856. _archive.Close();
  857. }
  858. _archiveStream = null;
  859. _opened = false;
  860. }
  861. return true;
  862. }
  863. #region ExtractFile overloads
  864. /// <summary>
  865. /// Unpacks the file by its name to the specified stream.
  866. /// </summary>
  867. /// <param name="fileName">The file full name in the archive file table.</param>
  868. /// <param name="stream">The stream where the file is to be unpacked.</param>
  869. public void ExtractFile(string fileName, Stream stream)
  870. {
  871. DisposedCheck();
  872. InitArchiveFileData(false);
  873. int index = -1;
  874. foreach (ArchiveFileInfo afi in _archiveFileData)
  875. {
  876. if (afi.FileName == fileName && !afi.IsDirectory)
  877. {
  878. index = afi.Index;
  879. break;
  880. }
  881. }
  882. if (index == -1)
  883. {
  884. if (!ThrowException(null, new ArgumentOutOfRangeException(
  885. "fileName",
  886. "The specified file name was not found in the archive file table.")))
  887. {
  888. return;
  889. }
  890. }
  891. else
  892. {
  893. ExtractFile(index, stream);
  894. }
  895. }
  896. /// <summary>
  897. /// Unpacks the file by its index to the specified stream.
  898. /// </summary>
  899. /// <param name="index">Index in the archive file table.</param>
  900. /// <param name="stream">The stream where the file is to be unpacked.</param>
  901. public void ExtractFile(int index, Stream stream)
  902. {
  903. DisposedCheck();
  904. ClearExceptions();
  905. if (!CheckIndexes(index))
  906. {
  907. if (!ThrowException(null, new ArgumentException("The index must be more or equal to zero.", "index")))
  908. {
  909. return;
  910. }
  911. }
  912. if (!stream.CanWrite)
  913. {
  914. if (!ThrowException(null, new ArgumentException("The specified stream can not be written.", "stream")))
  915. {
  916. return;
  917. }
  918. }
  919. InitArchiveFileData(false);
  920. if (index > _filesCount - 1)
  921. {
  922. if (!ThrowException(null, new ArgumentOutOfRangeException(
  923. "index", "The specified index is greater than the archive files count.")))
  924. {
  925. return;
  926. }
  927. }
  928. var indexes = new[] {(uint) index};
  929. if (_isSolid.Value)
  930. {
  931. indexes = SolidIndexes(indexes);
  932. }
  933. var archiveStream = GetArchiveStream(false);
  934. var openCallback = GetArchiveOpenCallback();
  935. if (!OpenArchive(archiveStream, openCallback))
  936. {
  937. return;
  938. }
  939. try
  940. {
  941. using (var aec = GetArchiveExtractCallback(stream, (uint) index, indexes.Length))
  942. {
  943. try
  944. {
  945. CheckedExecute(
  946. _archive.Extract(indexes, (uint) indexes.Length, 0, aec),
  947. SevenZipExtractionFailedException.DEFAULT_MESSAGE, aec);
  948. }
  949. finally
  950. {
  951. FreeArchiveExtractCallback(aec);
  952. }
  953. }
  954. }
  955. catch (Exception)
  956. {
  957. if (openCallback.ThrowException())
  958. {
  959. throw;
  960. }
  961. }
  962. OnEvent(ExtractionFinished, EventArgs.Empty, false);
  963. ThrowUserException();
  964. }
  965. #endregion
  966. #region ExtractFiles overloads
  967. /// <summary>
  968. /// Unpacks files by their indices to the specified directory.
  969. /// </summary>
  970. /// <param name="indexes">indexes of the files in the archive file table.</param>
  971. /// <param name="directory">Directory where the files are to be unpacked.</param>
  972. public void ExtractFiles(string directory, params int[] indexes)
  973. {
  974. DisposedCheck();
  975. ClearExceptions();
  976. if (!CheckIndexes(indexes))
  977. {
  978. if (
  979. !ThrowException(null, new ArgumentException("The indexes must be more or equal to zero.", "indexes")))
  980. {
  981. return;
  982. }
  983. }
  984. InitArchiveFileData(false);
  985. #region Indexes stuff
  986. var uindexes = new uint[indexes.Length];
  987. for (int i = 0; i < indexes.Length; i++)
  988. {
  989. uindexes[i] = (uint) indexes[i];
  990. }
  991. #if CS4
  992. if (uindexes.Where(i => i >= _filesCount).Any(
  993. i => !ThrowException(null,
  994. new ArgumentOutOfRangeException("indexes",
  995. "Index must be less than " +
  996. _filesCount.Value.ToString(
  997. CultureInfo.InvariantCulture) + "!"))))
  998. {
  999. return;
  1000. }
  1001. #else
  1002. foreach (uint i in uindexes)
  1003. {
  1004. if (i >= _filesCount)
  1005. {
  1006. if (!ThrowException(null,
  1007. new ArgumentOutOfRangeException("indexes",
  1008. "Index must be less than " +
  1009. _filesCount.Value.ToString(
  1010. CultureInfo.InvariantCulture) + "!")))
  1011. {
  1012. return;
  1013. }
  1014. }
  1015. }
  1016. #endif
  1017. var origIndexes = new List<uint>(uindexes);
  1018. origIndexes.Sort();
  1019. uindexes = origIndexes.ToArray();
  1020. if (_isSolid.Value)
  1021. {
  1022. uindexes = SolidIndexes(uindexes);
  1023. }
  1024. #endregion
  1025. try
  1026. {
  1027. IInStream archiveStream;
  1028. using ((archiveStream = GetArchiveStream(origIndexes.Count != 1)) as IDisposable)
  1029. {
  1030. var openCallback = GetArchiveOpenCallback();
  1031. if (!OpenArchive(archiveStream, openCallback))
  1032. {
  1033. return;
  1034. }
  1035. try
  1036. {
  1037. using (var aec = GetArchiveExtractCallback(directory, (int) _filesCount, origIndexes))
  1038. {
  1039. try
  1040. {
  1041. CheckedExecute(
  1042. _archive.Extract(uindexes, (uint) uindexes.Length, 0, aec),
  1043. SevenZipExtractionFailedException.DEFAULT_MESSAGE, aec);
  1044. }
  1045. finally
  1046. {
  1047. FreeArchiveExtractCallback(aec);
  1048. }
  1049. }
  1050. }
  1051. catch (Exception)
  1052. {
  1053. if (openCallback.ThrowException())
  1054. {
  1055. throw;
  1056. }
  1057. }
  1058. }
  1059. OnEvent(ExtractionFinished, EventArgs.Empty, false);
  1060. }
  1061. finally
  1062. {
  1063. if (origIndexes.Count > 1)
  1064. {
  1065. if (_archive != null)
  1066. {
  1067. _archive.Close();
  1068. }
  1069. _archiveStream = null;
  1070. _opened = false;
  1071. }
  1072. }
  1073. ThrowUserException();
  1074. }
  1075. /// <summary>
  1076. /// Unpacks files by their full names to the specified directory.
  1077. /// </summary>
  1078. /// <param name="fileNames">Full file names in the archive file table.</param>
  1079. /// <param name="directory">Directory where the files are to be unpacked.</param>
  1080. public void ExtractFiles(string directory, params string[] fileNames)
  1081. {
  1082. DisposedCheck();
  1083. InitArchiveFileData(false);
  1084. var indexes = new List<int>(fileNames.Length);
  1085. var archiveFileNames = new List<string>(ArchiveFileNames);
  1086. foreach (string fn in fileNames)
  1087. {
  1088. if (!archiveFileNames.Contains(fn))
  1089. {
  1090. if (
  1091. !ThrowException(null,
  1092. new ArgumentOutOfRangeException("fileNames",
  1093. "File \"" + fn +
  1094. "\" was not found in the archive file table.")))
  1095. {
  1096. return;
  1097. }
  1098. }
  1099. else
  1100. {
  1101. foreach (ArchiveFileInfo afi in _archiveFileData)
  1102. {
  1103. if (afi.FileName == fn && !afi.IsDirectory)
  1104. {
  1105. indexes.Add(afi.Index);
  1106. break;
  1107. }
  1108. }
  1109. }
  1110. }
  1111. ExtractFiles(directory, indexes.ToArray());
  1112. }
  1113. /// <summary>
  1114. /// Extracts files from the archive, giving a callback the choice what
  1115. /// to do with each file. The order of the files is given by the archive.
  1116. /// 7-Zip (and any other solid) archives are NOT supported.
  1117. /// </summary>
  1118. /// <param name="extractFileCallback">The callback to call for each file in the archive.</param>
  1119. public void ExtractFiles(ExtractFileCallback extractFileCallback)
  1120. {
  1121. DisposedCheck();
  1122. InitArchiveFileData(false);
  1123. if (IsSolid)
  1124. {
  1125. // solid strategy
  1126. }
  1127. else
  1128. {
  1129. foreach (ArchiveFileInfo archiveFileInfo in ArchiveFileData)
  1130. {
  1131. var extractFileCallbackArgs = new ExtractFileCallbackArgs(archiveFileInfo);
  1132. extractFileCallback(extractFileCallbackArgs);
  1133. if (extractFileCallbackArgs.CancelExtraction)
  1134. {
  1135. break;
  1136. }
  1137. if (extractFileCallbackArgs.ExtractToStream != null || extractFileCallbackArgs.ExtractToFile != null)
  1138. {
  1139. bool callDone = false;
  1140. try
  1141. {
  1142. if (extractFileCallbackArgs.ExtractToStream != null)
  1143. {
  1144. ExtractFile(archiveFileInfo.Index, extractFileCallbackArgs.ExtractToStream);
  1145. }
  1146. else
  1147. {
  1148. using (var file = new FileStream(extractFileCallbackArgs.ExtractToFile, FileMode.CreateNew,
  1149. FileAccess.Write, FileShare.None, 8192))
  1150. {
  1151. ExtractFile(archiveFileInfo.Index, file);
  1152. }
  1153. }
  1154. callDone = true;
  1155. }
  1156. catch (Exception ex)
  1157. {
  1158. extractFileCallbackArgs.Exception = ex;
  1159. extractFileCallbackArgs.Reason = ExtractFileCallbackReason.Failure;
  1160. extractFileCallback(extractFileCallbackArgs);
  1161. if (!ThrowException(null, ex))
  1162. {
  1163. return;
  1164. }
  1165. }
  1166. if (callDone)
  1167. {
  1168. extractFileCallbackArgs.Reason = ExtractFileCallbackReason.Done;
  1169. extractFileCallback(extractFileCallbackArgs);
  1170. }
  1171. }
  1172. }
  1173. }
  1174. }
  1175. #endregion
  1176. /// <summary>
  1177. /// Unpacks the whole archive to the specified directory.
  1178. /// </summary>
  1179. /// <param name="directory">The directory where the files are to be unpacked.</param>
  1180. public void ExtractArchive(string directory)
  1181. {
  1182. DisposedCheck();
  1183. ClearExceptions();
  1184. InitArchiveFileData(false);
  1185. try
  1186. {
  1187. IInStream archiveStream;
  1188. using ((archiveStream = GetArchiveStream(true)) as IDisposable)
  1189. {
  1190. var openCallback = GetArchiveOpenCallback();
  1191. if (!OpenArchive(archiveStream, openCallback))
  1192. {
  1193. return;
  1194. }
  1195. try
  1196. {
  1197. using (var aec = GetArchiveExtractCallback(directory, (int) _filesCount, null))
  1198. {
  1199. try
  1200. {
  1201. CheckedExecute(
  1202. _archive.Extract(null, UInt32.MaxValue, 0, aec),
  1203. SevenZipExtractionFailedException.DEFAULT_MESSAGE, aec);
  1204. OnEvent(ExtractionFinished, EventArgs.Empty, false);
  1205. }
  1206. finally
  1207. {
  1208. FreeArchiveExtractCallback(aec);
  1209. }
  1210. }
  1211. }
  1212. catch (Exception ex)
  1213. {
  1214. if (openCallback.ThrowException())
  1215. {
  1216. throw ex;
  1217. }
  1218. }
  1219. }
  1220. }
  1221. finally
  1222. {
  1223. if (_archive != null)
  1224. {
  1225. _archive.Close();
  1226. }
  1227. _archiveStream = null;
  1228. _opened = false;
  1229. }
  1230. ThrowUserException();
  1231. }
  1232. #endregion
  1233. #endif
  1234. #region LZMA SDK functions
  1235. internal static byte[] GetLzmaProperties(Stream inStream, out long outSize)
  1236. {
  1237. var lzmAproperties = new byte[5];
  1238. if (inStream.Read(lzmAproperties, 0, 5) != 5)
  1239. {
  1240. throw new LzmaException();
  1241. }
  1242. outSize = 0;
  1243. for (int i = 0; i < 8; i++)
  1244. {
  1245. int b = inStream.ReadByte();
  1246. if (b < 0)
  1247. {
  1248. throw new LzmaException();
  1249. }
  1250. outSize |= ((long) (byte) b) << (i << 3);
  1251. }
  1252. return lzmAproperties;
  1253. }
  1254. /// <summary>
  1255. /// Decompress the specified stream (C# inside)
  1256. /// </summary>
  1257. /// <param name="inStream">The source compressed stream</param>
  1258. /// <param name="outStream">The destination uncompressed stream</param>
  1259. /// <param name="inLength">The length of compressed data (null for inStream.Length)</param>
  1260. /// <param name="codeProgressEvent">The event for handling the code progress</param>
  1261. public static void DecompressStream(Stream inStream, Stream outStream, int? inLength,
  1262. EventHandler<ProgressEventArgs> codeProgressEvent)
  1263. {
  1264. if (!inStream.CanRead || !outStream.CanWrite)
  1265. {
  1266. throw new ArgumentException("The specified streams are invalid.");
  1267. }
  1268. var decoder = new Decoder();
  1269. long outSize, inSize = (inLength.HasValue ? inLength.Value : inStream.Length) - inStream.Position;
  1270. decoder.SetDecoderProperties(GetLzmaProperties(inStream, out outSize));
  1271. decoder.Code(
  1272. inStream, outStream, inSize, outSize,
  1273. new LzmaProgressCallback(inSize, codeProgressEvent));
  1274. }
  1275. /// <summary>
  1276. /// Decompress byte array compressed with LZMA algorithm (C# inside)
  1277. /// </summary>
  1278. /// <param name="data">Byte array to decompress</param>
  1279. /// <returns>Decompressed byte array</returns>
  1280. public static byte[] ExtractBytes(byte[] data)
  1281. {
  1282. using (var inStream = new MemoryStream(data))
  1283. {
  1284. var decoder = new Decoder();
  1285. inStream.Seek(0, 0);
  1286. using (var outStream = new MemoryStream())
  1287. {
  1288. long outSize;
  1289. decoder.SetDecoderProperties(GetLzmaProperties(inStream, out outSize));
  1290. decoder.Code(inStream, outStream, inStream.Length - inStream.Position, outSize, null);
  1291. return outStream.ToArray();
  1292. }
  1293. }
  1294. }
  1295. #endregion
  1296. }
  1297. }