using SharpCompress.Archives; using SharpCompress.Archives.Rar; using SharpCompress.Archives.Zip; using SharpCompress.Common; using SharpCompress.Readers; using SharpCompress.Writers; using System; using System.Collections.Concurrent; using System.Collections.Generic; using System.IO; using System.Linq; using System.Net.Http; using System.Text; using System.Threading.Tasks; using System.Web; namespace Masuit.Tools.Files { /// /// 7z压缩 /// public class SevenZipCompressor : ISevenZipCompressor { private readonly HttpClient _httpClient; /// /// /// /// public SevenZipCompressor(HttpClient httpClient) { _httpClient = httpClient; } /// /// 将多个文件压缩到一个文件流中,可保存为zip文件,方便于web方式下载 /// /// 多个文件路径,文件或文件夹,或网络路径http/https /// /// 文件流 public MemoryStream ZipStream(List files, string rootdir = "") { using var archive = CreateZipArchive(files, rootdir); var ms = new MemoryStream(); archive.SaveTo(ms, new WriterOptions(CompressionType.Deflate) { LeaveStreamOpen = true, ArchiveEncoding = new ArchiveEncoding() { Default = Encoding.UTF8 } }); return ms; } /// /// 压缩多个文件 /// /// 多个文件路径,文件或文件夹 /// 压缩到... /// 压缩包内部根文件夹 public void Zip(List files, string zipFile, string rootdir = "") { using var archive = CreateZipArchive(files, rootdir); archive.SaveTo(zipFile, new WriterOptions(CompressionType.Deflate) { LeaveStreamOpen = true, ArchiveEncoding = new ArchiveEncoding() { Default = Encoding.UTF8 } }); } /// /// 解压rar文件 /// /// rar文件 /// 解压到... /// 忽略空文件夹 public void UnRar(string rar, string dir = "", bool ignoreEmptyDir = true) { if (string.IsNullOrEmpty(dir)) { dir = Path.GetDirectoryName(rar); } using var archive = RarArchive.Open(rar); var entries = ignoreEmptyDir ? archive.Entries.Where(entry => !entry.IsDirectory) : archive.Entries; foreach (var entry in entries) { entry.WriteToDirectory(dir, new ExtractionOptions() { ExtractFullPath = true, Overwrite = true }); } } /// /// 解压文件,自动检测压缩包类型 /// /// rar文件 /// 解压到... /// 忽略空文件夹 public void Extract(string compressedFile, string dir = "", bool ignoreEmptyDir = true) => Decompress(compressedFile, dir, ignoreEmptyDir); /// /// 解压文件,自动检测压缩包类型 /// /// rar文件 /// 解压到... /// 忽略空文件夹 public void Decompress(string compressedFile, string dir = "", bool ignoreEmptyDir = true) { if (string.IsNullOrEmpty(dir)) { dir = Path.GetDirectoryName(compressedFile); } using Stream stream = File.OpenRead(compressedFile); using var reader = ReaderFactory.Open(stream); while (reader.MoveToNextEntry()) { if (ignoreEmptyDir) { reader.WriteEntryToDirectory(dir, new ExtractionOptions() { ExtractFullPath = true, Overwrite = true }); } else { if (!reader.Entry.IsDirectory) { reader.WriteEntryToDirectory(dir, new ExtractionOptions() { ExtractFullPath = true, Overwrite = true }); } } } } /// /// 创建zip包 /// /// /// /// private ZipArchive CreateZipArchive(List files, string rootdir) { var archive = ZipArchive.Create(); var dic = GetFileEntryMaps(files); var remoteUrls = files.Distinct().Where(s => s.StartsWith("http")).Select(s => { try { return new Uri(s); } catch (UriFormatException) { return null; } }).Where(u => u != null).ToList(); foreach (var pair in dic) { archive.AddEntry(Path.Combine(rootdir, pair.Value), pair.Key); } if (remoteUrls.Any()) { var streams = new ConcurrentDictionary(); Parallel.ForEach(remoteUrls, url => { _httpClient.GetAsync(url).ContinueWith(async t => { if (t.IsCompleted) { var res = await t; if (res.IsSuccessStatusCode) { Stream stream = await res.Content.ReadAsStreamAsync(); streams[Path.Combine(rootdir, Path.GetFileName(HttpUtility.UrlDecode(url.AbsolutePath)))] = stream; } } }).Wait(); }); foreach (var pair in streams) { archive.AddEntry(pair.Key, pair.Value); } } return archive; } /// /// 获取文件路径和zip-entry的映射 /// /// /// private Dictionary GetFileEntryMaps(List files) { var fileList = new List(); void GetFilesRecurs(string path) { //遍历目标文件夹的所有文件 fileList.AddRange(Directory.GetFiles(path)); //遍历目标文件夹的所有文件夹 foreach (var directory in Directory.GetDirectories(path)) { GetFilesRecurs(directory); } } files .Where(s => !s.StartsWith("http")) .ForEach(s => { if (Directory.Exists(s)) { GetFilesRecurs(s); } else { fileList.Add(s); } }); if (!fileList.Any()) { return new Dictionary(); } var dirname = new string(fileList.First().Substring(0, fileList.Min(s => s.Length)).TakeWhile((c, i) => fileList.All(s => s[i] == c)).ToArray()); if (!Directory.Exists(dirname)) { dirname = Directory.GetParent(dirname).FullName; } var dic = fileList.ToDictionary(s => s, s => s.Substring(dirname.Length)); return dic; } } }