mirror of
https://github.com/duplicati/duplicati.git
synced 2025-11-28 19:40:25 +08:00
250 lines
No EOL
9 KiB
C#
250 lines
No EOL
9 KiB
C#
// Copyright (C) 2025, The Duplicati Team
|
|
// https://duplicati.com, hello@duplicati.com
|
|
//
|
|
// Permission is hereby granted, free of charge, to any person obtaining a
|
|
// copy of this software and associated documentation files (the "Software"),
|
|
// to deal in the Software without restriction, including without limitation
|
|
// the rights to use, copy, modify, merge, publish, distribute, sublicense,
|
|
// and/or sell copies of the Software, and to permit persons to whom the
|
|
// Software is furnished to do so, subject to the following conditions:
|
|
//
|
|
// The above copyright notice and this permission notice shall be included in
|
|
// all copies or substantial portions of the Software.
|
|
//
|
|
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS
|
|
// OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
|
|
// FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
|
|
// DEALINGS IN THE SOFTWARE.
|
|
|
|
using System;
|
|
using System.IO;
|
|
using System.Collections.Generic;
|
|
using System.Linq;
|
|
|
|
namespace Duplicati.CommandLine.RecoveryTool
|
|
{
|
|
public static class FileIndex
|
|
{
|
|
public static int Run(List<string> args, Dictionary<string, string> options, Library.Utility.IFilter filter)
|
|
{
|
|
if (args.Count != 2)
|
|
{
|
|
Console.WriteLine("Invalid argument count ({0} expected 2): {1}{2}", args.Count, Environment.NewLine, string.Join(Environment.NewLine, args));
|
|
return 100;
|
|
}
|
|
|
|
var folder = Path.GetFullPath(args[1]);
|
|
|
|
if (!Directory.Exists(folder))
|
|
{
|
|
Console.WriteLine("Folder not found: {0}", folder);
|
|
return 100;
|
|
}
|
|
|
|
Directory.SetCurrentDirectory(folder);
|
|
|
|
string ixfile;
|
|
options.TryGetValue("indexfile", out ixfile);
|
|
if (string.IsNullOrWhiteSpace(ixfile))
|
|
ixfile = "index.txt";
|
|
|
|
ixfile = Path.GetFullPath(ixfile);
|
|
if (!File.Exists(ixfile))
|
|
{
|
|
using (File.Create(ixfile))
|
|
{
|
|
}
|
|
}
|
|
else
|
|
{
|
|
Console.WriteLine("Sorting existing index file");
|
|
SortFile(ixfile, ixfile);
|
|
}
|
|
|
|
var filecount = Directory.EnumerateFiles(folder).Count();
|
|
|
|
Console.WriteLine("Processing {0} files", filecount);
|
|
|
|
var i = 0;
|
|
var errors = 0;
|
|
var totalblocks = 0L;
|
|
var files = 0;
|
|
foreach (var file in Directory.EnumerateFiles(folder))
|
|
{
|
|
Console.Write("{0}: {1}", i, file);
|
|
|
|
try
|
|
{
|
|
var p = Duplicati.Library.Main.Volumes.VolumeBase.ParseFilename(Path.GetFileName(file));
|
|
if (p == null)
|
|
{
|
|
Console.WriteLine(" - Not a Duplicati file, ignoring");
|
|
continue;
|
|
}
|
|
|
|
if (p.FileType != Duplicati.Library.Main.RemoteVolumeType.Blocks)
|
|
{
|
|
Console.WriteLine(" - Filetype {0}, skipping", p.FileType);
|
|
continue;
|
|
}
|
|
|
|
if (!string.IsNullOrWhiteSpace(p.EncryptionModule))
|
|
{
|
|
Console.WriteLine(" - Encrypted, skipping");
|
|
continue;
|
|
}
|
|
|
|
var filekey = Path.GetFileName(file);
|
|
|
|
var blocks = 0;
|
|
using (var stream = new System.IO.FileStream(file, FileMode.Open, FileAccess.Read, FileShare.Read))
|
|
using (var cp = Library.DynamicLoader.CompressionLoader.GetModule(p.CompressionModule, stream, Library.Interface.ArchiveMode.Read, options))
|
|
using (var tf = new Library.Utility.TempFile())
|
|
{
|
|
using (var sw = new StreamWriter(tf))
|
|
foreach (var f in cp.ListFiles(null))
|
|
{
|
|
sw.WriteLine("{0}, {1}", Library.Utility.Utility.Base64UrlToBase64Plain(f), filekey);
|
|
blocks++;
|
|
}
|
|
|
|
files++;
|
|
totalblocks += blocks;
|
|
|
|
Console.Write(" {0} hashes found, sorting ...", blocks);
|
|
|
|
SortFile(tf, tf);
|
|
|
|
Console.WriteLine(" done!");
|
|
|
|
Console.Write("Merging {0} hashes ...", totalblocks);
|
|
|
|
MergeFiles(ixfile, tf, ixfile);
|
|
|
|
Console.WriteLine(" done!");
|
|
}
|
|
}
|
|
catch (Exception ex)
|
|
{
|
|
Console.WriteLine(" error: {0}", ex);
|
|
errors++;
|
|
}
|
|
|
|
i++;
|
|
}
|
|
|
|
Console.Write("Sorting index file ...");
|
|
SortFile(ixfile, ixfile);
|
|
Console.WriteLine(" done!");
|
|
|
|
Console.WriteLine("Processed {0} files and found {1} hashes", files, totalblocks);
|
|
if (errors > 0)
|
|
Console.WriteLine("Experienced {0} errors", errors);
|
|
|
|
return 0;
|
|
}
|
|
|
|
public static void SortFile(string filein, string fileout)
|
|
{
|
|
try
|
|
{
|
|
// If the file can fit into memory, this is MUCH faster
|
|
using (var tfout = new Library.Utility.TempFile())
|
|
{
|
|
var data = File.ReadAllLines(filein);
|
|
Array.Sort(data, StringComparer.Ordinal);
|
|
File.WriteAllLines(tfout, data);
|
|
|
|
File.Copy(tfout, fileout, true);
|
|
return;
|
|
}
|
|
}
|
|
catch
|
|
{
|
|
}
|
|
|
|
using (var tfin = new Library.Utility.TempFile())
|
|
using (var tfout = new Library.Utility.TempFile())
|
|
{
|
|
long swaps;
|
|
|
|
File.Copy(filein, tfin, true);
|
|
|
|
do
|
|
{
|
|
swaps = 0L;
|
|
|
|
using (var sw = new System.IO.StreamWriter(tfout))
|
|
using (var sr = new System.IO.StreamReader(tfin))
|
|
{
|
|
var c1 = sr.ReadLine();
|
|
var c2 = sr.ReadLine();
|
|
|
|
while (c1 != null || c2 != null)
|
|
{
|
|
if (c1 != null && c1.StartsWith("a", StringComparison.Ordinal))
|
|
Console.Write("");
|
|
var cmp = StringComparer.Ordinal.Compare(c1, c2);
|
|
|
|
if (c2 == null || (c1 != null && cmp < 0))
|
|
{
|
|
sw.WriteLine(c1);
|
|
|
|
c1 = c2;
|
|
c2 = sr.ReadLine();
|
|
}
|
|
else
|
|
{
|
|
if (cmp != 0)
|
|
sw.WriteLine(c2);
|
|
c2 = sr.ReadLine();
|
|
swaps++;
|
|
}
|
|
}
|
|
}
|
|
|
|
File.Copy(tfout, tfin, true);
|
|
} while (swaps > 0);
|
|
|
|
File.Copy(tfout, fileout, true);
|
|
}
|
|
}
|
|
|
|
private static void MergeFiles(string file1, string file2, string fileout)
|
|
{
|
|
using (var tf = new Library.Utility.TempFile())
|
|
{
|
|
using (var sw = new System.IO.StreamWriter(tf))
|
|
using (var sr1 = new System.IO.StreamReader(file1))
|
|
using (var sr2 = new System.IO.StreamReader(file2))
|
|
{
|
|
var c1 = sr1.ReadLine();
|
|
var c2 = sr2.ReadLine();
|
|
|
|
while (c1 != null || c2 != null)
|
|
{
|
|
var cmp = StringComparer.Ordinal.Compare(c1, c2);
|
|
|
|
if (c2 == null || (c1 != null && cmp < 0))
|
|
{
|
|
sw.WriteLine(c1);
|
|
c1 = sr1.ReadLine();
|
|
}
|
|
else
|
|
{
|
|
if (cmp != 0)
|
|
sw.WriteLine(c2);
|
|
|
|
c2 = sr2.ReadLine();
|
|
}
|
|
}
|
|
}
|
|
|
|
File.Copy(tf, fileout, true);
|
|
}
|
|
}
|
|
}
|
|
} |