mirror of
https://github.com/duplicati/duplicati.git
synced 2026-05-06 15:26:45 -04:00
301 lines
14 KiB
C#
301 lines
14 KiB
C#
// Copyright (C) 2024, 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 Duplicati.Library.Utility;
|
|
using System.Linq;
|
|
using Duplicati.Library.Logging;
|
|
using System.Reflection;
|
|
using System.Threading;
|
|
using System.Threading.Tasks;
|
|
using Duplicati.Library.Common.IO;
|
|
using NUnit.Framework;
|
|
using System.Runtime.InteropServices;
|
|
using Duplicati.Library.Common;
|
|
|
|
namespace Duplicati.UnitTest
|
|
{
|
|
public static class TestUtils
|
|
{
|
|
/// <summary>
|
|
/// The log tag
|
|
/// </summary>
|
|
private static readonly string LOGTAG = Library.Logging.Log.LogTagFromType(typeof(TestUtils));
|
|
|
|
public static Dictionary<string, string> DefaultOptions
|
|
{
|
|
get
|
|
{
|
|
var opts = new Dictionary<string, string>();
|
|
|
|
string auth_password = Path.Combine(Path.GetDirectoryName(Assembly.GetExecutingAssembly().Location), "unittest_authpassword.txt");
|
|
if (System.IO.File.Exists(auth_password))
|
|
opts["auth-password"] = File.ReadAllText(auth_password).Trim();
|
|
|
|
return opts;
|
|
}
|
|
}
|
|
|
|
public static async Task GrowingFile(string testFile, CancellationToken token)
|
|
{
|
|
try
|
|
{
|
|
var str = new string('*', 50);
|
|
while (true)
|
|
{
|
|
if (token.IsCancellationRequested)
|
|
{
|
|
continue;
|
|
}
|
|
File.AppendAllText(testFile, str);
|
|
await Task.Delay(18, token).ConfigureAwait(false);
|
|
}
|
|
}
|
|
finally
|
|
{
|
|
if (File.Exists(testFile))
|
|
{
|
|
File.Delete(testFile);
|
|
}
|
|
}
|
|
}
|
|
|
|
public static string GetDefaultTarget(string other = null)
|
|
{
|
|
string alttarget = System.IO.Path.Combine(Path.GetDirectoryName(Assembly.GetExecutingAssembly().Location), "unittest_target.txt");
|
|
|
|
if (File.Exists(alttarget))
|
|
return File.ReadAllText(alttarget).Trim();
|
|
else if (other != null)
|
|
return other;
|
|
else
|
|
using(var tf = new Library.Utility.TempFolder())
|
|
{
|
|
tf.Protected = true;
|
|
return "file://" + tf;
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Recursively copy a directory to another location.
|
|
/// </summary>
|
|
/// <param name="sourcefolder">Source directory path</param>
|
|
/// <param name="targetfolder">Destination directory path</param>
|
|
public static void CopyDirectoryRecursive(string sourcefolder, string targetfolder)
|
|
{
|
|
sourcefolder = Util.AppendDirSeparator(sourcefolder);
|
|
|
|
var work = new Queue<string>();
|
|
work.Enqueue(sourcefolder);
|
|
|
|
var timestampfailures = 0;
|
|
|
|
while (work.Count > 0)
|
|
{
|
|
var c = work.Dequeue();
|
|
|
|
var t = Path.Combine(targetfolder, c.Substring(sourcefolder.Length));
|
|
|
|
if (!Directory.Exists(t))
|
|
Directory.CreateDirectory(t);
|
|
|
|
try { Directory.SetCreationTimeUtc(t, Directory.GetCreationTimeUtc(c)); }
|
|
catch(Exception ex)
|
|
{
|
|
if (timestampfailures++ < 20)
|
|
Console.WriteLine("Failed to set creation time on dir {0}: {1}", t, ex.Message);
|
|
}
|
|
|
|
try { Directory.SetLastWriteTimeUtc(t, Directory.GetLastWriteTimeUtc(c)); }
|
|
catch(Exception ex)
|
|
{
|
|
if (timestampfailures++ < 20)
|
|
Console.WriteLine("Failed to set write time on dir {0}: {1}", t, ex.Message);
|
|
}
|
|
|
|
|
|
foreach(var n in Directory.EnumerateFiles(c))
|
|
{
|
|
var tf = Path.Combine(t, Path.GetFileName(n));
|
|
File.Copy(n, tf, true);
|
|
try { File.SetCreationTimeUtc(tf, System.IO.File.GetCreationTimeUtc(n)); }
|
|
catch(Exception ex)
|
|
{
|
|
if (timestampfailures++ < 20)
|
|
Console.WriteLine("Failed to set creation time on file {0}: {1}", n, ex.Message);
|
|
}
|
|
try { File.SetLastWriteTimeUtc(tf, System.IO.File.GetLastWriteTimeUtc(n)); }
|
|
catch(Exception ex)
|
|
{
|
|
if (timestampfailures++ < 20)
|
|
Console.WriteLine("Failed to set write time on file {0}: {1}", n, ex.Message);
|
|
}
|
|
}
|
|
|
|
foreach(var n in Directory.EnumerateDirectories(c))
|
|
work.Enqueue(n);
|
|
}
|
|
|
|
if (timestampfailures > 20)
|
|
Console.WriteLine("Encountered additional {0} timestamp errors!", timestampfailures);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Returns the index of a given string, using the file system case sensitivity
|
|
/// </summary>
|
|
/// <returns>The index of the entry or -1 if no entry was found</returns>
|
|
/// <param name='lst'>The list to search</param>
|
|
/// <param name='m'>The string to find</param>
|
|
private static int IndexOf(List<string> lst, string m)
|
|
{
|
|
StringComparison sc = Duplicati.Library.Utility.Utility.ClientFilenameStringComparison;
|
|
for(int i = 0; i < lst.Count; i++)
|
|
if (lst[i].Equals(m, sc))
|
|
return i;
|
|
|
|
return -1;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Asserts that the two directory trees are equivalent; i.e.,
|
|
/// that they they contain the same directories and files, recursively.
|
|
/// </summary>
|
|
/// <param name="expectedDir">The expected directory tree.</param>
|
|
/// <param name="actualDir">The actual directory tree.</param>
|
|
/// <param name="verifymetadata">True to also compare file metadata.</param>
|
|
/// <param name="contextMessage">Context information to include in an assert message.</param>
|
|
public static void AssertDirectoryTreesAreEquivalent(string expectedDir, string actualDir, bool verifymetadata, string contextMessage)
|
|
{
|
|
var localMessage = $"{contextMessage}, in directories {expectedDir} and {actualDir}";
|
|
// Assert that expectedDir and actualDir contain the same directories
|
|
var expectedSubdirs = SystemIO.IO_OS.EnumerateDirectories(expectedDir).OrderBy(SystemIO.IO_OS.PathGetFileName);
|
|
var actualSubdirs = SystemIO.IO_OS.EnumerateDirectories(actualDir).OrderBy(SystemIO.IO_OS.PathGetFileName);
|
|
Assert.That(expectedSubdirs.Select(SystemIO.IO_OS.PathGetFileName), Is.EquivalentTo(actualSubdirs.Select(SystemIO.IO_OS.PathGetFileName)), localMessage);
|
|
// Recursively compare the contained directories
|
|
var expectedSubdirsEnumerator = expectedSubdirs.GetEnumerator();
|
|
var actualSubdirsEnumerator = actualSubdirs.GetEnumerator();
|
|
while (expectedSubdirsEnumerator.MoveNext() && actualSubdirsEnumerator.MoveNext())
|
|
{
|
|
AssertDirectoryTreesAreEquivalent(expectedSubdirsEnumerator.Current, actualSubdirsEnumerator.Current, verifymetadata, contextMessage);
|
|
}
|
|
// Assert that expectedDir and actualDir contain the same files
|
|
var expectedFiles = SystemIO.IO_OS.EnumerateFiles(expectedDir).OrderBy(SystemIO.IO_OS.PathGetFileName);
|
|
var actualFiles = SystemIO.IO_OS.EnumerateFiles(actualDir).OrderBy(SystemIO.IO_OS.PathGetFileName);
|
|
Assert.That(expectedFiles.Select(SystemIO.IO_OS.PathGetFileName), Is.EquivalentTo(actualFiles.Select(SystemIO.IO_OS.PathGetFileName)), localMessage);
|
|
// Assert that the files are equal
|
|
var expectedFilesEnumerator = expectedFiles.GetEnumerator();
|
|
var actualFilesEnumerator = actualFiles.GetEnumerator();
|
|
while (expectedFilesEnumerator.MoveNext() && actualFilesEnumerator.MoveNext())
|
|
{
|
|
AssertFilesAreEqual(expectedFilesEnumerator.Current, actualFilesEnumerator.Current, verifymetadata, contextMessage);
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Asserts that two files are equal by comparing their length, contents, and, optionally, their metadata.
|
|
/// </summary>
|
|
/// <param name="expectedFile">The expected file.</param>
|
|
/// <param name="actualFile">The actual file.</param>
|
|
/// <param name="verifymetadata">True to also compare file metadata.</param>
|
|
/// <param name="contextMessage">Context information to include in an assert message.</param>
|
|
public static void AssertFilesAreEqual(string expectedFile, string actualFile, bool verifymetadata, string contextMessage)
|
|
{
|
|
using (var expectedFileStream = SystemIO.IO_OS.FileOpenRead(expectedFile))
|
|
using (var actualFileStream = SystemIO.IO_OS.FileOpenRead(actualFile))
|
|
{
|
|
// Compare file lengths
|
|
var expectedFileStreamLength = expectedFileStream.Length;
|
|
var actualFileStreamLength = actualFileStream.Length;
|
|
Assert.That(actualFileStreamLength, Is.EqualTo(expectedFileStreamLength), $"{contextMessage}, file size mismatch for {expectedFile} and {actualFile}");
|
|
// Compare file contents
|
|
// The byte-by-byte compare is dog-slow, so we use a fast(-er) check, and then report the first byte diff if required
|
|
if (!Utility.CompareStreams(expectedFileStream, actualFileStream, true))
|
|
{
|
|
// Reset stream positions
|
|
expectedFileStream.Position = 0;
|
|
actualFileStream.Position = 0;
|
|
for (long i = 0; i < expectedFileStreamLength; i++)
|
|
{
|
|
var expectedByte = expectedFileStream.ReadByte();
|
|
var actualByte = actualFileStream.ReadByte();
|
|
// For performance reasons, only exercise Assert mechanism and generate message if byte comparison fails
|
|
if (expectedByte != actualByte)
|
|
{
|
|
var message = $"{contextMessage}, file contents mismatch at position {i} for {expectedFile} and {actualFile}";
|
|
Assert.That(actualByte, Is.EqualTo(expectedByte), message);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
// Compare file metadata
|
|
if (verifymetadata)
|
|
{
|
|
// macOS seem to like to actually set the time to some value different than what you set by hundreds of milliseconds.
|
|
// Reading the time right after it is set gives the expected value but when read later it is slightly different.
|
|
// Maybe a bug in .net?
|
|
int granularity = OperatingSystem.IsMacOS() ? 2999 : 1;
|
|
Assert.That(
|
|
SystemIO.IO_OS.GetLastWriteTimeUtc(actualFile),
|
|
Is.EqualTo(SystemIO.IO_OS.GetLastWriteTimeUtc(expectedFile)).Within(granularity).Milliseconds,
|
|
$"{contextMessage}, last write time mismatch for {expectedFile} and {actualFile}");
|
|
Assert.That(
|
|
SystemIO.IO_OS.GetCreationTimeUtc(actualFile),
|
|
Is.EqualTo(SystemIO.IO_OS.GetCreationTimeUtc(expectedFile)).Within(granularity).Milliseconds,
|
|
$"{contextMessage}, creation time mismatch for {expectedFile} and {actualFile}");
|
|
}
|
|
}
|
|
|
|
public static Dictionary<string, string> Expand(this Dictionary<string, string> self, object extra)
|
|
{
|
|
var res = new Dictionary<string, string>(self);
|
|
foreach(var n in extra.GetType().GetFields())
|
|
{
|
|
var name = n.Name.Replace('_', '-');
|
|
var value = n.GetValue(extra);
|
|
res[name] = value == null ? "" : value.ToString();
|
|
}
|
|
|
|
foreach(var n in extra.GetType().GetProperties())
|
|
{
|
|
var name = n.Name.Replace('_', '-');
|
|
var value = n.GetValue(extra);
|
|
res[name] = value == null ? "" : value.ToString();
|
|
}
|
|
|
|
return res;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Write file <paramref name="path"/> with <paramref name="contents"/>.
|
|
/// </summary>
|
|
public static void WriteFile(string path, byte[] contents)
|
|
{
|
|
using (FileStream fileStream = SystemIO.IO_OS.FileOpenWrite(path))
|
|
{
|
|
Utility.CopyStream(new MemoryStream(contents), fileStream);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|