duplicati/Duplicati/UnitTest/UtilityTests.cs
Kenneth Skovhede 8fca01f64f Updated NUnit
This closes #4939
2025-06-13 11:48:24 +02:00

576 lines
26 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 Duplicati.Library.Utility;
using Duplicati.Library.Common.IO;
using NUnit.Framework;
using System;
using System.Collections.Generic;
using System.Linq;
using System.Globalization;
using System.IO;
using System.Threading;
using System.Threading.Tasks;
using Duplicati.StreamUtil;
using Assert = NUnit.Framework.Legacy.ClassicAssert;
using NUnit.Framework.Legacy;
namespace Duplicati.UnitTest
{
public class UtilityTests : BasicSetupHelper
{
[Test]
[Category("Utility")]
public static void AppendDirSeparator()
{
const string noTrailingSlash = @"/a\b/c";
string hasTrailingSlash = noTrailingSlash + Util.DirectorySeparatorString;
string alternateSeparator = null;
if (String.Equals(Util.DirectorySeparatorString, "/", StringComparison.Ordinal))
{
alternateSeparator = @"\";
}
if (String.Equals(Util.DirectorySeparatorString, @"\", StringComparison.Ordinal))
{
alternateSeparator = "/";
}
Assert.AreEqual(hasTrailingSlash, Util.AppendDirSeparator(noTrailingSlash));
Assert.AreEqual(hasTrailingSlash, Util.AppendDirSeparator(hasTrailingSlash));
Assert.AreEqual(hasTrailingSlash, Util.AppendDirSeparator(noTrailingSlash), Util.DirectorySeparatorString);
Assert.AreEqual(hasTrailingSlash, Util.AppendDirSeparator(hasTrailingSlash), Util.DirectorySeparatorString);
Assert.AreEqual(noTrailingSlash + alternateSeparator, Util.AppendDirSeparator(noTrailingSlash, alternateSeparator));
Assert.AreEqual(noTrailingSlash + alternateSeparator, Util.AppendDirSeparator(noTrailingSlash + alternateSeparator, alternateSeparator));
Assert.AreEqual(hasTrailingSlash + alternateSeparator, Util.AppendDirSeparator(hasTrailingSlash, alternateSeparator));
}
[Test]
[Category("Utility")]
[TestCase("da-DK")]
[TestCase("en-US")]
[TestCase("hu-HU")]
[TestCase("tr-TR")]
public static void FilenameStringComparison(string cultureName)
{
Action<string, string> checkStringComparison = (x, y) => Assert.IsFalse(String.Equals(x, y, Utility.ClientFilenameStringComparison));
Action<string, string> checkStringComparer = (x, y) => Assert.IsFalse(new HashSet<string>(new[] { x }).Contains(y, Utility.ClientFilenameStringComparer));
CultureInfo originalCulture = CultureInfo.CurrentCulture;
try
{
CultureInfo.CurrentCulture = new CultureInfo(cultureName, false);
// These are equivalent with respect to hu-HU, but different with respect to en-US.
string ddzs = "ddzs";
string dzsdzs = "dzsdzs";
checkStringComparison(ddzs, dzsdzs);
checkStringComparer(ddzs, dzsdzs);
// Many cultures treat the following as equivalent.
string eAcuteOneCharacter = System.Text.Encoding.GetEncoding("iso-8859-1").GetString(new byte[] { 233 }); // 'é' as one character (ALT+0233).
string eAcuteTwoCharacters = "\u0065\u0301"; // 'e', combined with an acute accent (U+301).
checkStringComparison(eAcuteOneCharacter, eAcuteTwoCharacters);
checkStringComparer(eAcuteOneCharacter, eAcuteTwoCharacters);
// These are equivalent with respect to en-US, but different with respect to da-DK.
string aDiaeresisOneCharacter = "\u00C4"; // 'A' with a diaeresis.
string aDiaeresisTwoCharacters = "\u0041\u0308"; // 'A', combined with a diaeresis.
checkStringComparison(aDiaeresisOneCharacter, aDiaeresisTwoCharacters);
checkStringComparer(aDiaeresisOneCharacter, aDiaeresisTwoCharacters);
}
finally
{
CultureInfo.CurrentCulture = originalCulture;
}
}
[Test]
[Category("Utility")]
public static void ReadLimitLengthStreamWithGrowingFile()
{
var growingFilename = Path.GetTempFileName();
long fixedGrowingStreamLength, limitStreamLength, nextGrowingStreamLength;
using (CancellationTokenSource tokenSource = new CancellationTokenSource())
{
Task task = TestUtils.GrowingFile(growingFilename, tokenSource.Token);
Thread.Sleep(100);
using (FileStream growingStream = new FileStream(growingFilename, FileMode.Open, FileAccess.Read, FileShare.ReadWrite))
{
Thread.Sleep(100);
fixedGrowingStreamLength = growingStream.Length;
Thread.Sleep(100);
using (var limitStream = new ReadLimitLengthStream(growingStream, fixedGrowingStreamLength))
{
Thread.Sleep(100);
nextGrowingStreamLength = growingStream.Length;
limitStreamLength = limitStream.Length;
}
}
}
Console.WriteLine($"fixedGrowingStreamLength:{fixedGrowingStreamLength} limitStreamLength:{limitStreamLength} nextGrowingStreamLength:{nextGrowingStreamLength}");
Assert.IsTrue(fixedGrowingStreamLength > 0);
Assert.AreEqual(fixedGrowingStreamLength, limitStreamLength);
Assert.IsTrue(limitStreamLength < nextGrowingStreamLength);
}
[Test]
[Category("Utility")]
public static void ReadLimitLengthStream()
{
Action<IEnumerable<int>, IEnumerable<byte>> assertArray = (expected, actual) =>
{
List<int> expectedList = expected.ToList();
List<byte> actualList = actual.ToList();
Assert.AreEqual(expectedList.Count, actualList.Count, "Count");
for (int i = 0; i < expectedList.Count; i++)
{
Assert.AreEqual((byte)expectedList[i], actualList[i], "Index {0}", i);
}
};
byte[] readBuffer = new byte[10];
Action<Stream, int, int> testSeek = (stream, position, readByte) =>
{
stream.Position = position;
Assert.AreEqual(position, stream.Position);
Assert.AreEqual(readByte, stream.ReadByte());
Assert.AreEqual(position, stream.Seek(position, SeekOrigin.Begin));
Assert.AreEqual(position, stream.Position);
Array.Clear(readBuffer, 0, 10);
Assert.AreEqual(readByte >= 0 ? 1 : 0, stream.Read(readBuffer, 1, 1), "Read count");
if (readByte >= 0)
{
// Make sure nothing was read before or after the offset byte given as well.
Assert.AreEqual(readBuffer[0], 0);
Assert.AreEqual(readBuffer[1], readByte);
Assert.AreEqual(readBuffer[2], 0);
}
else
{
// Make sure nothing was read
Assert.AreEqual(readBuffer[0], 0);
Assert.AreEqual(readBuffer[1], 0);
Assert.AreEqual(readBuffer[2], 0);
}
Assert.AreEqual(position, stream.Seek(position - stream.Length, SeekOrigin.End));
Assert.AreEqual(position, stream.Position);
Assert.AreEqual(readByte, stream.ReadByte());
};
// Base stream is a sequence from 0..9
using (MemoryStream baseStream = new MemoryStream(Enumerable.Range(0, 10).Select(i => (byte)i).ToArray()))
{
using (ReadLimitLengthStream stream = new ReadLimitLengthStream(baseStream, 5))
{
Assert.AreEqual(5, stream.Length);
// Test reading past array bounds
Assert.AreEqual(5, stream.Read(readBuffer, 0, 10), "Read count");
assertArray(Enumerable.Range(0, 5), readBuffer.Take(5));
// Set the position directly and read a shorter range
stream.Position = 2;
Assert.AreEqual(2, stream.ReadAsync(readBuffer, 0, 2).Await(), "Read count");
assertArray(Enumerable.Range(2, 2), readBuffer.Take(2));
}
// Make sure the stream will be seeked if the start does not match the current position.
baseStream.Position = 0;
using (ReadLimitLengthStream stream = new ReadLimitLengthStream(baseStream, 2, 4))
{
// Make sure the position is updated when the stream is created
Assert.AreEqual(0, stream.Position);
// Test basic read
Assert.AreEqual(4, stream.Read(readBuffer, 0, 4), "Read count");
assertArray(Enumerable.Range(2, 4), readBuffer.Take(4));
// Test CopyTo
using (MemoryStream destination = new MemoryStream())
{
stream.Position = 0;
stream.CopyTo(destination);
assertArray(Enumerable.Range(2, 4), destination.ToArray());
}
// Test CopyToAsync
using (MemoryStream destination = new MemoryStream())
{
stream.Position = 0;
stream.CopyToAsync(destination).Await();
assertArray(Enumerable.Range(2, 4), destination.ToArray());
}
// Test seeking
testSeek(stream, 0, 2);
testSeek(stream, 1, 3);
testSeek(stream, 2, 4);
testSeek(stream, 3, 5);
testSeek(stream, 4, -1);
// Test seeking from current
stream.Position = 0;
Assert.AreEqual(2, stream.Seek(2, SeekOrigin.Current));
Assert.AreEqual(2, stream.Position);
// Test clamping of position
stream.Position = -2;
Assert.AreEqual(0, stream.Position);
Assert.AreEqual(2, baseStream.Position);
stream.Position = 9;
Assert.AreEqual(4, stream.Position);
Assert.AreEqual(6, baseStream.Position);
// Make sure changing the base stream still shows clamped positions
baseStream.Position = 0;
Assert.AreEqual(0, stream.Position);
baseStream.Position = 4;
Assert.AreEqual(2, stream.Position);
baseStream.Position = 10;
Assert.AreEqual(4, stream.Position);
// Reading when the baseStream is positioned before the start of the limit
// should still be possible, and should seek to return the correct values.
baseStream.Position = 0;
Assert.AreEqual(0, stream.Position);
Assert.AreEqual(2, stream.ReadByte());
}
// Check a stream with length 0
baseStream.Position = 0;
using (ReadLimitLengthStream stream = new ReadLimitLengthStream(baseStream, 2, 0))
{
// Make sure the position is updated when the stream is created
Assert.AreEqual(0, stream.Position);
// Test basic read
Assert.AreEqual(0, stream.Read(readBuffer, 0, 4), "Read count");
// Test seeking
testSeek(stream, 0, -1);
testSeek(stream, 0, -1);
// Test seeking from current
stream.Position = 0;
Assert.AreEqual(0, stream.Seek(2, SeekOrigin.Current));
Assert.AreEqual(0, stream.Position);
// Test clamping of position
stream.Position = -2;
Assert.AreEqual(0, stream.Position);
Assert.AreEqual(2, baseStream.Position);
stream.Position = 9;
Assert.AreEqual(0, stream.Position);
Assert.AreEqual(2, baseStream.Position);
// Make sure changing the base stream still shows clamped positions
baseStream.Position = 0;
Assert.AreEqual(0, stream.Position);
baseStream.Position = 4;
Assert.AreEqual(0, stream.Position);
baseStream.Position = 10;
Assert.AreEqual(0, stream.Position);
}
}
}
[Test]
[Category("Utility")]
public static void ForceStreamRead()
{
byte[] source = { 0x10, 0x20, 0x30, 0x40, 0x50 };
// Ensure that ReadOneByteStream returns one byte at a time.
byte[] buffer = new byte[source.Length];
ReadOneByteStream stream = new ReadOneByteStream(source);
Assert.AreEqual(1, stream.Read(buffer, 0, buffer.Length));
Assert.AreEqual(source.First(), buffer.First());
foreach (byte value in buffer.Skip(1))
{
Assert.AreEqual(default(byte), value);
}
// Buffer is larger than the length of the stream.
buffer = new byte[source.Length + 1];
int bytesRead = Utility.ForceStreamRead(new ReadOneByteStream(source), buffer, source.Length);
Assert.AreEqual(source.Length, bytesRead);
CollectionAssert.AreEqual(source, buffer.Take(source.Length));
Assert.AreEqual(default(byte), buffer.Last());
// Maximum number of bytes is larger than the length of the stream.
buffer = new byte[source.Length + 1];
bytesRead = Utility.ForceStreamRead(new ReadOneByteStream(source), buffer, source.Length + 1);
Assert.AreEqual(source.Length, bytesRead);
CollectionAssert.AreEqual(source, buffer.Take(bytesRead));
Assert.AreEqual(default(byte), buffer.Last());
// Maximum number of bytes is smaller than the length of the stream.
buffer = new byte[source.Length];
bytesRead = Utility.ForceStreamRead(new ReadOneByteStream(source), buffer, source.Length - 1);
Assert.AreEqual(source.Length - 1, bytesRead);
CollectionAssert.AreEqual(source.Take(bytesRead), buffer.Take(bytesRead));
Assert.AreEqual(default(byte), buffer.Last());
// Buffer is smaller than the length of the stream.
Assert.Throws<ArgumentException>(() => Utility.ForceStreamRead(new ReadOneByteStream(source), new byte[source.Length - 1], source.Length));
}
[Test]
[Category("Utility")]
public void GetUniqueItems()
{
string[] collection = { "A", "a", "A", "b", "c", "c" };
string[] uniqueItems = { "A", "a", "b", "c" };
string[] duplicateItems = { "A", "c" };
// Test with default comparer.
var actualUniqueItems = Utility.GetUniqueItems(collection, out var actualDuplicateItems);
CollectionAssert.AreEquivalent(uniqueItems, actualUniqueItems);
CollectionAssert.AreEquivalent(duplicateItems, actualDuplicateItems);
// Test with custom comparer.
IEqualityComparer<string> comparer = StringComparer.OrdinalIgnoreCase;
uniqueItems = new string[] { "a", "b", "c" };
duplicateItems = new string[] { "a", "c" };
actualDuplicateItems = null;
actualUniqueItems = Utility.GetUniqueItems(collection, comparer, out actualDuplicateItems);
Assert.That(actualUniqueItems, Is.EquivalentTo(uniqueItems).Using(comparer));
Assert.That(actualDuplicateItems, Is.EquivalentTo(duplicateItems).Using(comparer));
// Test with empty collection.
actualDuplicateItems = null;
actualUniqueItems = Utility.GetUniqueItems(new string[0], out actualDuplicateItems);
Assert.IsNotNull(actualUniqueItems);
Assert.IsNotNull(actualDuplicateItems);
}
[Test]
[Category("Utility")]
public void NormalizeDateTime()
{
DateTime baseDateTime = new DateTime(2000, 1, 2, 3, 4, 5);
DateTime baseDateTimeUTC = baseDateTime.ToUniversalTime();
Assert.AreEqual(baseDateTimeUTC, Utility.NormalizeDateTime(baseDateTime.AddMilliseconds(1)));
Assert.AreEqual(baseDateTimeUTC, Utility.NormalizeDateTime(baseDateTime.AddMilliseconds(500)));
Assert.AreEqual(baseDateTimeUTC, Utility.NormalizeDateTime(baseDateTime.AddMilliseconds(999)));
Assert.AreEqual(baseDateTimeUTC.AddSeconds(-1), Utility.NormalizeDateTime(baseDateTime.AddMilliseconds(-1)));
Assert.AreEqual(baseDateTimeUTC.AddSeconds(1), Utility.NormalizeDateTime(baseDateTime.AddSeconds(1.9)));
}
[Test]
[Category("Utility")]
public void NormalizeDateTimeToEpochSeconds()
{
DateTime baseDateTime = new DateTime(2000, 1, 2, 3, 4, 5);
long epochSeconds = (long)(baseDateTime.ToUniversalTime() - Utility.EPOCH).TotalSeconds;
Assert.AreEqual(epochSeconds, Utility.NormalizeDateTimeToEpochSeconds(baseDateTime.AddMilliseconds(1)));
Assert.AreEqual(epochSeconds, Utility.NormalizeDateTimeToEpochSeconds(baseDateTime.AddMilliseconds(500)));
Assert.AreEqual(epochSeconds, Utility.NormalizeDateTimeToEpochSeconds(baseDateTime.AddMilliseconds(999)));
Assert.AreEqual(epochSeconds - 1, Utility.NormalizeDateTimeToEpochSeconds(baseDateTime.AddMilliseconds(-1)));
Assert.AreEqual(epochSeconds + 1, Utility.NormalizeDateTimeToEpochSeconds(baseDateTime.AddSeconds(1.9)));
}
[Test]
[Category("Utility")]
public void ParseBool()
{
string[] expectTrue = { "1", "on", "true", "yes" };
string[] expectFalse = { "0", "off", "false", "no" };
string[] expectDefault = { null, "", "maybe" };
Func<bool> returnsTrue = () => true;
Func<bool> returnsFalse = () => false;
foreach (string value in expectTrue)
{
string message = $"{value} should be parsed to true.";
Assert.IsTrue(Utility.ParseBool(value, false), message);
Assert.IsTrue(Utility.ParseBool(value.ToUpper(CultureInfo.InvariantCulture), false), message);
Assert.IsTrue(Utility.ParseBool($" {value} ", false), message);
}
foreach (string value in expectFalse)
{
string message = $"{value} should be parsed to false.";
Assert.IsFalse(Utility.ParseBool(value, true), message);
Assert.IsFalse(Utility.ParseBool(value.ToUpper(CultureInfo.InvariantCulture), true), message);
Assert.IsFalse(Utility.ParseBool($" {value} ", true), message);
}
foreach (string value in expectDefault)
{
Assert.IsTrue(Utility.ParseBool(value, true));
Assert.IsTrue(Utility.ParseBool(value, returnsTrue));
Assert.IsFalse(Utility.ParseBool(value, false));
Assert.IsFalse(Utility.ParseBool(value, returnsFalse));
}
}
[Test]
[Category("Utility")]
public static void ThrottledStreamRead()
{
byte[] sourceBuffer = [0x10, 0x20, 0x30, 0x40, 0x50];
var destinationBuffer = new byte[sourceBuffer.Length + 1];
const int offset = 1;
const int bytesToRead = 3;
using (var baseStream = new MemoryStream(sourceBuffer))
{
const int readSpeed = 1;
const int writeSpeed = 1;
var throttledStream = new ThrottleEnabledStream(baseStream, readSpeed, writeSpeed);
var bytesRead = throttledStream.Read(destinationBuffer, offset, bytesToRead);
Assert.AreEqual(bytesToRead, bytesRead);
for (var k = 0; k < destinationBuffer.Length; k++)
{
if (offset <= k && k < offset + bytesToRead)
{
Assert.AreEqual(sourceBuffer[k - offset], destinationBuffer[k]);
}
else
{
Assert.AreEqual(default(byte), destinationBuffer[k]);
}
}
}
}
[Test]
[Category("Utility")]
public static void ThrottledStreamWrite()
{
byte[] initialBuffer = [0x10, 0x20, 0x30, 0x40, 0x50];
byte[] source = [0x60, 0x70, 0x80, 0x90];
const int offset = 1;
const int bytesToWrite = 3;
using (var baseStream = new MemoryStream(initialBuffer))
{
const int readSpeed = 1;
const int writeSpeed = 1;
var throttledStream = new ThrottleEnabledStream(baseStream, readSpeed, writeSpeed);
throttledStream.Write(source, offset, bytesToWrite);
byte[] result = baseStream.ToArray();
for (int k = 0; k < result.Length; k++)
{
if (k < bytesToWrite)
{
Assert.AreEqual(source[offset + k], result[k]);
}
else
{
Assert.AreEqual(initialBuffer[k], result[k]);
}
}
}
}
[Test]
[Category("Utility")]
public static void RetryDelayNoExponentialBackoff()
{
// test boundary values
TimeSpan baseDelay = TimeSpan.FromSeconds(1);
int[] testValues = { 1, 2, 11, 12, int.MaxValue };
double[] expect = { 1, 1, 1, 1, 1 };
for (int i = 0; i < testValues.Length; i++)
Assert.AreEqual(TimeSpan.FromSeconds(expect[i]), Utility.GetRetryDelay(baseDelay, testValues[i], false));
}
[Test]
[Category("Utility")]
public static void RetryDelayExponentialBackoff()
{
// test boundary values
TimeSpan baseDelay = TimeSpan.FromSeconds(1);
int[] testValues = { 1, 2, 11, 12, int.MaxValue };
double[] expect = { 1, 2, 1024, 1024, 1024 };
for (int i = 0; i < testValues.Length; i++)
Assert.AreEqual(TimeSpan.FromSeconds(expect[i]), Utility.GetRetryDelay(baseDelay, testValues[i], true));
}
}
/// <summary>
/// Mimic a Stream that can only read one byte at a time.
/// </summary>
class ReadOneByteStream : System.IO.MemoryStream
{
private readonly byte[] source;
public ReadOneByteStream(byte[] source)
{
this.source = source;
}
public override int Read(byte[] buffer, int offset, int count)
{
if (Object.ReferenceEquals(buffer, null))
{
throw new ArgumentNullException(nameof(buffer));
}
if (offset < 0)
{
throw new ArgumentOutOfRangeException(nameof(offset));
}
if (count < 0)
{
throw new ArgumentOutOfRangeException(nameof(count));
}
if (offset + count > buffer.Length)
{
throw new ArgumentException("The sum of offset and count must not be larger than the buffer size.");
}
if (offset < this.source.Length)
{
const int bytesRead = 1;
Array.Copy(this.source, offset, buffer, offset, bytesRead);
return bytesRead;
}
return 0;
}
}
}