mirror of
https://github.com/duplicati/duplicati.git
synced 2025-11-28 03:20:25 +08:00
576 lines
26 KiB
C#
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;
|
|
}
|
|
}
|
|
}
|