duplicati/Duplicati/CommandLine/BackendTester/Program.cs
Kenneth Skovhede 012aeed7de Add dynamic streaming toggle
This PR adds a dynamic property so a backend can signal if it supports streaming, based on the settings.

This is currently used for the File backend, so that toggling `--use-move-for-put` will disable streaming on the backend instead of relying on the `--disable-streaming-transfers` flag.
2025-11-03 12:48:37 +01:00

707 lines
38 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.Collections.Generic;
using System.Diagnostics;
using System.Text;
using Duplicati.Library.Interface;
using System.Linq;
using System.Globalization;
using System.Threading;
using Duplicati.Library.Utility;
using Duplicati.StreamUtil;
using System.Threading.Tasks;
namespace Duplicati.CommandLine.BackendTester
{
public class Program
{
class TempFile
{
public readonly string remotefilename;
public readonly string localfilename;
public readonly byte[] hash;
public readonly long length;
public bool found = false;
public TempFile(string remotefilename, string localfilename, byte[] hash, long length)
{
this.remotefilename = remotefilename;
this.localfilename = localfilename;
this.hash = hash;
this.length = length;
}
}
private const string ValidFilenameChars = "abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ123456789";
private const string ExtendedChars = "-_',=)(&%$#@! +";
/// <summary>
/// The main entry point for the application.
/// </summary>
[STAThread]
public static int Main(string[] _args)
{
try
{
Library.AutoUpdater.PreloadSettingsLoader.ConfigurePreloadSettings(ref _args, Library.AutoUpdater.PackageHelper.NamedExecutable.BackendTester);
if (_args.Length == 1)
{
try
{
var p = Environment.ExpandEnvironmentVariables(_args[0]);
if (System.IO.File.Exists(p))
_args = (from x in System.IO.File.ReadLines(p)
where !string.IsNullOrWhiteSpace(x) && !x.Trim().StartsWith("#", StringComparison.Ordinal)
select x.Trim()
).ToArray();
}
catch
{
}
}
List<string> args = new List<string>(_args);
Dictionary<string, string> options = Library.Utility.CommandLineParser.ExtractOptions(args);
if (args.Count != 1 || HelpOptionExtensions.IsArgumentAnyHelpString(args))
{
Console.WriteLine("Usage: <protocol>://<username>:<password>@<path>");
Console.WriteLine("Example: ftp://user:pass@server/folder");
Console.WriteLine();
Console.WriteLine($"Supported backends: {string.Join(",", Duplicati.Library.DynamicLoader.BackendLoader.Keys)}");
Console.WriteLine();
List<string> lines = new List<string>();
foreach (Library.Interface.ICommandLineArgument arg in SupportedCommands)
Library.Interface.CommandLineArgument.PrintArgument(lines, arg);
foreach (string s in lines)
Console.WriteLine(s);
return 0;
}
if (options.ContainsKey("tempdir") && !string.IsNullOrEmpty(options["tempdir"]))
Library.Utility.SystemContextSettings.DefaultTempPath = options["tempdir"];
using var SystemSettings = Duplicati.Library.Utility.SystemContextSettings.StartSession();
if (!options.ContainsKey("auth_password") && !string.IsNullOrEmpty(System.Environment.GetEnvironmentVariable("AUTH_PASSWORD")))
options["auth_password"] = System.Environment.GetEnvironmentVariable("AUTH_PASSWORD");
if (!options.ContainsKey("auth_username") && !string.IsNullOrEmpty(System.Environment.GetEnvironmentVariable("AUTH_USERNAME")))
options["auth_username"] = System.Environment.GetEnvironmentVariable("AUTH_USERNAME");
int reruns = 5;
if (options.ContainsKey("reruns"))
reruns = int.Parse(options["reruns"]);
for (int i = 0; i < reruns; i++)
{
Console.WriteLine($"{DateTime.Now.ToShortDateString()} {DateTime.Now.ToLongTimeString()}");
Console.WriteLine($"{LogTimeStamp}Starting run no {i}");
if (!Run(args, options, i == 0))
return 1;
}
Console.WriteLine($"{LogTimeStamp}Unittest complete!");
return 0;
}
catch (Exception ex)
{
Console.WriteLine($"{LogTimeStamp}Unittest failed: {ex}");
}
return 1;
}
static bool Run(List<string> args, Dictionary<string, string> options, bool first)
{
bool failAfterFinished = false;
var backend = Library.DynamicLoader.BackendLoader.GetBackend(args[0], options);
if (backend == null)
{
Console.WriteLine("Unsupported backend");
Console.WriteLine();
Console.WriteLine($"Supported backends: {string.Join(",", Duplicati.Library.DynamicLoader.BackendLoader.Keys)}");
return false;
}
var allowedChars = ValidFilenameChars;
if (options.ContainsKey("extended-chars"))
{
allowedChars += String.IsNullOrEmpty(options["extended-chars"]) ? ExtendedChars : options["extended-chars"];
}
var autoCreateFolders = Library.Utility.Utility.ParseBoolOption(options, "auto-create-folder");
var retries = 0;
if (options.ContainsKey("failure-retries"))
retries = int.Parse(options["failure-retries"]);
options.TryGetValue("enable-module", out var enabledModulesValue);
options.TryGetValue("disable-module", out var disabledModulesValue);
var enabledModules = enabledModulesValue == null ? new string[0] : enabledModulesValue.Trim().ToLower(CultureInfo.InvariantCulture).Split(',');
var disabledModules = disabledModulesValue == null ? new string[0] : disabledModulesValue.Trim().ToLower(CultureInfo.InvariantCulture).Split(',');
var loadedModules = new List<IGenericModule>();
foreach (var m in Library.DynamicLoader.GenericLoader.Modules)
if (!disabledModules.Contains(m.Key, StringComparer.OrdinalIgnoreCase) && (m.LoadAsDefault || enabledModules.Contains(m.Key, StringComparer.OrdinalIgnoreCase)))
{
m.Configure(options);
loadedModules.Add(m);
}
try
{
IEnumerable<Library.Interface.IFileEntry> curlist = null;
try
{
Retry(() => backend.TestAsync(CancellationToken.None), retries).Await();
curlist = Retry(() => backend.ListAsync(CancellationToken.None).ToBlockingEnumerable().ToList(), retries);
}
catch (FolderMissingException)
{
if (autoCreateFolders)
{
try
{
Retry(() => backend.CreateFolderAsync(CancellationToken.None), retries).Await();
curlist = Retry(() => backend.ListAsync(CancellationToken.None).ToBlockingEnumerable().ToList(), retries);
}
catch (Exception ex)
{
Console.WriteLine($"{LogTimeStamp}Autocreate folder failed with message: {ex.Message}");
}
}
if (curlist == null)
throw;
}
foreach (Library.Interface.IFileEntry fe in curlist)
if (!fe.IsFolder)
{
if (Library.Utility.Utility.ParseBoolOption(options, "auto-clean") && first)
if (Library.Utility.Utility.ParseBoolOption(options, "force"))
{
Console.WriteLine($"{LogTimeStamp}Auto clean, removing file: {fe.Name}");
Retry(() => backend.DeleteAsync(fe.Name, CancellationToken.None), retries).Await();
continue;
}
else
Console.WriteLine("Specify the --force flag to actually delete files");
var fileCount = curlist.Where(x => !x.IsFolder).Count();
var filenames = curlist.Where(x => !x.IsFolder).Select(x => x.Name).Take(10).ToList();
Console.WriteLine($"{LogTimeStamp}*** Remote folder contains {fileCount} file(s), aborting");
Console.WriteLine($"{LogTimeStamp}*** First {filenames.Count} file(s): {Environment.NewLine}{string.Join(Environment.NewLine, filenames)}");
if (fileCount > filenames.Count)
Console.WriteLine($"{LogTimeStamp}*** ... and {fileCount - filenames.Count} more file(s)");
return false;
}
var number_of_files = 10;
var min_file_size = 1024;
var max_file_size = 1024 * 1024 * 50;
var min_filename_size = 5;
var max_filename_size = 80;
var disableStreaming = Library.Utility.Utility.ParseBoolOption(options, "disable-streaming-transfers");
var skipOverwriteTest = Library.Utility.Utility.ParseBoolOption(options, "skip-overwrite-test");
var trimFilenameSpaces = Library.Utility.Utility.ParseBoolOption(options, "trim-filename-spaces");
var waitAfterUpload = TimeSpan.Zero;
var waitAfterDelete = TimeSpan.Zero;
var useStreaming = backend is IStreamingBackend streamingBackend && streamingBackend.SupportsStreaming && !disableStreaming;
var throttleUpload = 0L;
if (options.TryGetValue("throttle-upload", out var throttleUploadString))
{
if (!useStreaming)
Console.WriteLine($"{LogTimeStamp}Warning: Throttling is only supported in this tool on streaming backends");
throttleUpload = Duplicati.Library.Utility.Sizeparser.ParseSize(throttleUploadString, "kb");
}
var throttleDownload = 0L;
if (options.TryGetValue("throttle-download", out var throttleDownloadString))
{
if (!useStreaming)
Console.WriteLine($"{LogTimeStamp}Warning: Throttling is only supported in this tool on streaming backends");
throttleDownload = Duplicati.Library.Utility.Sizeparser.ParseSize(throttleDownloadString, "kb");
}
int readWriteTimeout = (Environment.GetEnvironmentVariable("READ_WRITE_TIMEOUT_SECONDS") is { } timeout
&& int.TryParse(timeout, out var seconds)
&& seconds == -1)
? Timeout.Infinite
: Environment.GetEnvironmentVariable("READ_WRITE_TIMEOUT_SECONDS") is { } timeoutRetry
&& int.TryParse(timeoutRetry, out var secondsRetry)
? (int)TimeSpan.FromSeconds(secondsRetry).TotalMilliseconds
: (int)TimeSpan.FromSeconds(60).TotalMilliseconds;
// Allow overriding the timeout for the backend here, even if timeouts are disabled
if (options.TryGetValue("read-write-timeout", out var readWriteTimeoutString))
readWriteTimeout = (int)Timeparser.ParseTimeSpan(readWriteTimeoutString).TotalMilliseconds;
if (readWriteTimeout <= 0)
readWriteTimeout = Timeout.Infinite;
Console.WriteLine($"{LogTimeStamp}Read Write Timeout set to {(readWriteTimeout == Timeout.Infinite ? "infinite" : readWriteTimeout + " ms")}");
if (options.ContainsKey("number-of-files"))
number_of_files = int.Parse(options["number-of-files"]);
if (options.ContainsKey("min-file-size"))
min_file_size = (int)Duplicati.Library.Utility.Sizeparser.ParseSize(options["min-file-size"], "mb");
if (options.ContainsKey("max-file-size"))
max_file_size = (int)Duplicati.Library.Utility.Sizeparser.ParseSize(options["max-file-size"], "mb");
if (options.ContainsKey("min-filename-length"))
min_filename_size = int.Parse(options["min-filename-length"]);
if (options.ContainsKey("max-filename-length"))
max_filename_size = int.Parse(options["max-filename-length"]);
if (options.ContainsKey("wait-after-upload"))
waitAfterUpload = Timeparser.ParseTimeSpan(options["wait-after-upload"]);
if (options.ContainsKey("wait-after-delete"))
waitAfterDelete = Timeparser.ParseTimeSpan(options["wait-after-delete"]);
var rnd = new Random();
var sha = System.Security.Cryptography.SHA256.Create();
//Create random files
using (var tf = new Duplicati.Library.Utility.TempFolder())
{
var files = new List<TempFile>();
for (int i = 0; i < number_of_files; i++)
{
var filename = CreateRandomRemoteFileName(min_filename_size, max_filename_size, allowedChars, trimFilenameSpaces, rnd);
var localfilename = CreateRandomFile(tf, i, min_file_size, max_file_size, rnd);
//Calculate local hash and length
using (var fs = new System.IO.FileStream(localfilename, System.IO.FileMode.Open, System.IO.FileAccess.Read))
files.Add(new TempFile(filename, localfilename, sha.ComputeHash(fs), fs.Length));
}
byte[] dummyFileHash = null;
if (!skipOverwriteTest)
{
Console.WriteLine($"{LogTimeStamp}Uploading wrong files ...");
using (var dummy = Library.Utility.TempFile.WrapExistingFile(CreateRandomFile(tf, files.Count, 1024, 2048, rnd)))
{
using (var fs = new System.IO.FileStream(dummy, System.IO.FileMode.Open, System.IO.FileAccess.Read))
dummyFileHash = sha.ComputeHash(fs);
//Upload a dummy file for entry 0 and the last one, they will be replaced by the real files afterwards
//We upload entry 0 twice just to try to freak any internal cache list
Retry(() => Uploadfile(dummy, 0, files[0].remotefilename, backend, disableStreaming, throttleUpload, readWriteTimeout), retries);
Retry(() => Uploadfile(dummy, 0, files[0].remotefilename, backend, disableStreaming, throttleUpload, readWriteTimeout), retries);
Retry(() => Uploadfile(dummy, files.Count - 1, files[files.Count - 1].remotefilename, backend, disableStreaming, throttleUpload, readWriteTimeout), retries);
}
}
Console.WriteLine($"{LogTimeStamp}Uploading files ...");
for (int i = 0; i < files.Count; i++)
Retry(() => Uploadfile(files[i].localfilename, i, files[i].remotefilename, backend, disableStreaming, throttleUpload, readWriteTimeout), retries);
TempFile originalRenamedFile = null;
string renamedFileNewName = null;
if (backend is IRenameEnabledBackend renameEnabledBackend)
{
// Rename the second file in the list, if there are more than one. If not, just do the first one.
int renameIndex = files.Count > 1 ? 1 : 0;
originalRenamedFile = files[renameIndex];
renamedFileNewName = CreateRandomRemoteFileName(min_filename_size, max_filename_size, allowedChars, trimFilenameSpaces, rnd);
Console.WriteLine($"{LogTimeStamp}Renaming file {renameIndex} from {originalRenamedFile.remotefilename} to {renamedFileNewName}");
Retry(() => renameEnabledBackend.RenameAsync(originalRenamedFile.remotefilename, renamedFileNewName, CancellationToken.None), retries).Await();
files[renameIndex] = new TempFile(renamedFileNewName, originalRenamedFile.localfilename, originalRenamedFile.hash, originalRenamedFile.length);
}
if (waitAfterUpload > TimeSpan.Zero)
{
Console.WriteLine($"{LogTimeStamp}Waiting {waitAfterUpload} after upload");
Thread.Sleep(waitAfterUpload);
}
Console.WriteLine($"{LogTimeStamp}Verifying file list ...");
curlist = Retry(() => backend.ListAsync(CancellationToken.None).ToBlockingEnumerable().ToList(), retries);
foreach (var fe in curlist)
if (!fe.IsFolder)
{
bool found = false;
for (var i = 0; i < files.Count; i++)
{
var tx = files[i];
if (tx.remotefilename == fe.Name)
{
if (tx.found)
Console.WriteLine($"{LogTimeStamp}*** File {i}) with name {tx.remotefilename} was found more than once");
found = true;
tx.found = true;
if (fe.Size > 0 && tx.length != fe.Size)
Console.WriteLine($"{LogTimeStamp}*** File {i} with name {tx.remotefilename} has size {tx.length} but the size was reported as {fe.Size}");
break;
}
}
if (!found)
if (originalRenamedFile != null && renamedFileNewName != null && originalRenamedFile.remotefilename == fe.Name)
{
Console.WriteLine($"{LogTimeStamp}*** File with name {fe.Name} was found on server but was supposed to have been renamed to {renamedFileNewName}!");
}
else
{
Console.WriteLine($"{LogTimeStamp}*** File with name {fe.Name} was found on server but not uploaded!");
}
}
for (var i = 0; i < files.Count; i++)
if (!files[i].found)
Console.WriteLine($"{LogTimeStamp}*** File {i} with name {files[i].remotefilename} was uploaded but not found afterwards");
Console.WriteLine($"{LogTimeStamp}Downloading files");
for (int i = 0; i < files.Count; i++)
{
using (var cf = new Duplicati.Library.Utility.TempFile())
{
Exception e = null;
Console.Write($"{LogTimeStamp}Downloading file {i} ... ");
var sw = Stopwatch.StartNew();
var throttleManager = new ThrottleManager() { Limit = throttleDownload };
try
{
Retry(async () =>
{
if (backend is IStreamingBackend streamingBackend && streamingBackend.SupportsStreaming && !disableStreaming)
{
using (var fs = new System.IO.FileStream(cf, System.IO.FileMode.Create, System.IO.FileAccess.Write, System.IO.FileShare.None))
using (var timeoutStream = new TimeoutObservingStream(fs) { WriteTimeout = readWriteTimeout })
using (var ts = new ThrottleEnabledStream(timeoutStream, throttleManager))
using (var nss = new NonSeekableStream(ts))
await streamingBackend.GetAsync(files[i].remotefilename, nss, timeoutStream.TimeoutToken);
}
else
await backend.GetAsync(files[i].remotefilename, cf, CancellationToken.None);
}, retries).Await();
e = null;
}
catch (Exception ex)
{
e = ex;
}
sw.Stop();
if (e != null)
{
failAfterFinished = true;
Console.WriteLine($"{LogTimeStamp}failed\n*** Error: {e} after {sw.ElapsedMilliseconds} ms");
}
else
Console.WriteLine($" done in {sw.ElapsedMilliseconds} ms (~{Utility.FormatSizeString(files[i].length / sw.Elapsed.TotalSeconds)}/s)");
Console.Write($"{LogTimeStamp}Checking hash ... ");
using (var fs = new System.IO.FileStream(cf, System.IO.FileMode.Open, System.IO.FileAccess.Read))
if (Convert.ToBase64String(sha.ComputeHash(fs)) != Convert.ToBase64String(files[i].hash))
{
if (dummyFileHash != null && Convert.ToBase64String(sha.ComputeHash(fs)) == Convert.ToBase64String(dummyFileHash))
Console.WriteLine($"{LogTimeStamp}failed\n*** Downloaded file was the dummy file"); // Should this be failed?
else
{
failAfterFinished = true;
Console.WriteLine($"{LogTimeStamp}failed\n*** Downloaded file was corrupt");
}
}
else
Console.WriteLine("done");
}
}
Console.WriteLine($"{LogTimeStamp}Deleting files...");
for (int i = 0; i < files.Count; i++)
try
{
Console.WriteLine($"{LogTimeStamp}Deleting file {i}");
Retry(() => backend.DeleteAsync(files[i].remotefilename, CancellationToken.None), retries).Await();
}
catch (Exception ex)
{
failAfterFinished = true;
Console.WriteLine($"{LogTimeStamp}*** Failed to delete file {files[i].remotefilename}, message: {ex}");
}
if (waitAfterDelete > TimeSpan.Zero)
{
Console.WriteLine($"{LogTimeStamp}Waiting {waitAfterDelete} after delete");
Thread.Sleep(waitAfterDelete);
}
curlist = Retry(() => backend.ListAsync(CancellationToken.None).ToBlockingEnumerable().ToList(), retries);
foreach (var fe in curlist)
if (!fe.IsFolder)
{
Console.WriteLine($"{LogTimeStamp}*** Remote folder contains {fe.Name} after cleanup");
}
// Test some error cases
Console.WriteLine($"{LogTimeStamp}Checking retrieval of non-existent file...");
var caughtExpectedException = false;
try
{
using (var tempFile = new Duplicati.Library.Utility.TempFile())
{
backend.GetAsync($"NonExistentFile-{Guid.NewGuid()}", tempFile.Name, CancellationToken.None).Await();
}
}
catch (FileMissingException)
{
Console.WriteLine($"{LogTimeStamp}Caught expected FileMissingException");
caughtExpectedException = true;
}
catch (Exception ex)
{
Console.WriteLine($"{LogTimeStamp}*** Retrieval of non-existent file failed: {ex}");
}
if (!caughtExpectedException)
{
failAfterFinished = true;
Console.WriteLine($"{LogTimeStamp}*** Retrieval of non-existent file should have failed with FileMissingException");
}
}
// Test quota retrieval
if (backend is IQuotaEnabledBackend quotaEnabledBackend)
{
Console.WriteLine($"{LogTimeStamp}Checking quota...");
IQuotaInfo quota = null;
bool noException;
try
{
quota = Retry(() => quotaEnabledBackend.GetQuotaInfoAsync(CancellationToken.None), retries).Await();
noException = true;
}
catch (Exception ex)
{
Console.WriteLine($"{LogTimeStamp}*** Checking quota information failed: {ex}");
noException = false;
}
if (noException)
{
if (quota != null)
{
Console.WriteLine($"{LogTimeStamp}Free Space: {Library.Utility.Utility.FormatSizeString(quota.FreeQuotaSpace)}");
Console.WriteLine($"{LogTimeStamp}Total Space: {Library.Utility.Utility.FormatSizeString(quota.TotalQuotaSpace)}");
}
else
{
Console.WriteLine($"{LogTimeStamp}Unable to retrieve quota information");
}
}
}
// Test DNSName lookup
Console.WriteLine($"{LogTimeStamp}Checking DNS names used by this backend...");
try
{
var dnsNames = Retry(() => backend.GetDNSNamesAsync(CancellationToken.None), retries).Await();
if (dnsNames != null)
{
foreach (string dnsName in dnsNames)
{
Console.WriteLine(dnsName);
}
}
else
{
Console.WriteLine($"{LogTimeStamp}No DNS names reported");
}
}
catch (Exception ex)
{
Console.WriteLine($"{LogTimeStamp}*** Checking DNSName failed: {ex}");
}
}
finally
{
foreach (var m in loadedModules)
if (m is IDisposable disposable)
disposable.Dispose();
}
return !failAfterFinished;
}
private static void Uploadfile(string localfilename, int i, string remotefilename, IBackend backend, bool disableStreaming, long throttle, int readWriteTimeout)
{
var filesize = new System.IO.FileInfo(localfilename).Length;
Console.Write($"{LogTimeStamp}Uploading file {i}, {Utility.FormatSizeString(filesize)} ... ");
Exception e = null;
var sw = Stopwatch.StartNew();
var throttleManager = new ThrottleManager() { Limit = throttle };
try
{
if (backend is IStreamingBackend streamingBackend && streamingBackend.SupportsStreaming && !disableStreaming)
{
using (var fs = new System.IO.FileStream(localfilename, System.IO.FileMode.Open, System.IO.FileAccess.Read, System.IO.FileShare.Read))
using (var timeoutStream = new TimeoutObservingStream(fs) { ReadTimeout = readWriteTimeout })
using (var ts = new ThrottleEnabledStream(timeoutStream, throttleManager))
using (var nss = new NonSeekableStream(ts))
streamingBackend.PutAsync(remotefilename, nss, timeoutStream.TimeoutToken).Await();
}
else
backend.PutAsync(remotefilename, localfilename, CancellationToken.None).Await();
e = null;
}
catch (Exception ex)
{
e = ex;
}
sw.Stop();
if (e != null)
{
Console.WriteLine($"{LogTimeStamp}Failed to upload file {i}, error message: {e}, remote name: {remotefilename} after {sw.ElapsedMilliseconds} ms");
while (e.InnerException != null)
{
e = e.InnerException;
Console.WriteLine($"{LogTimeStamp} Inner exception: {e}");
}
}
else
{
Console.WriteLine($" done! in {sw.ElapsedMilliseconds} ms (~{Utility.FormatSizeString(filesize / sw.Elapsed.TotalSeconds)}/s)");
}
}
private static string CreateRandomRemoteFileName(int min_filename_size, int max_filename_size, string allowedChars, bool trimFilenameSpaces, Random rnd)
{
StringBuilder filenameBuilder = new StringBuilder();
int filenamelen = rnd.Next(min_filename_size, max_filename_size);
for (int j = 0; j < filenamelen; j++)
filenameBuilder.Append(allowedChars[rnd.Next(0, allowedChars.Length)]);
string filename = filenameBuilder.ToString();
if (trimFilenameSpaces)
filename = filename.Trim();
return filename;
}
private static string CreateRandomFile(Library.Utility.TempFolder tf, int i, int min_file_size, int max_file_size, Random rnd)
{
Console.Write($"{LogTimeStamp}Generating file {i}");
string filename = System.IO.Path.Combine(tf, i.ToString());
using (System.IO.FileStream fs = new System.IO.FileStream(filename, System.IO.FileMode.CreateNew, System.IO.FileAccess.Write))
{
//Random size
byte[] buf = new byte[1024];
int size = rnd.Next(min_file_size, max_file_size);
Console.WriteLine($" ({Duplicati.Library.Utility.Utility.FormatSizeString(size)})");
while (size > 0)
{
rnd.NextBytes(buf);
fs.Write(buf, 0, Math.Min(buf.Length, size));
size -= buf.Length;
}
}
return filename;
}
private static async Task<T> Retry<T>(Func<Task<T>> action, int retries)
{
var total = retries;
while (true)
{
retries--;
try
{
return await action();
}
catch
{
if (retries <= 0)
throw;
var delay = Utility.GetRetryDelay(TimeSpan.FromSeconds(1), total - retries, true);
Console.WriteLine($"{LogTimeStamp}Operation failed with exception, {retries} retries left, delaying retry for {delay.TotalMilliseconds} ms");
await Task.Delay(delay);
Console.WriteLine($"{LogTimeStamp}Retrying after error, {retries} retries left");
}
}
}
private static async Task Retry(Func<Task> action, int retries)
=> await Retry(async () => { await action(); return true; }, retries);
private static void Retry(Action action, int retries)
=> Retry(() => { action(); return true; }, retries);
private static T Retry<T>(Func<T> action, int retries)
=> Retry(() => Task.FromResult(action()), retries).Result;
private static string LogTimeStamp => $"[{DateTime.Now:HH:mm:ss fff}] ";
public static IList<ICommandLineArgument> SupportedCommands => [
new CommandLineArgument("reruns", CommandLineArgument.ArgumentType.Integer, "The number of test runs to perform", "A number that describes how many times the test is performed", "5"),
new CommandLineArgument("tempdir", CommandLineArgument.ArgumentType.Path, "The path used to store temporary files", "The backend tester will use the system default temp path. You can set this option to choose another path."),
new CommandLineArgument("extended-chars", CommandLineArgument.ArgumentType.String, "A list of allowed extended filename chars", "A list of characters besides {a-z, A-Z, 0-9} to use when generating filenames", ExtendedChars),
new CommandLineArgument("number-of-files", CommandLineArgument.ArgumentType.Integer, "The number of files to test with", "An integer describing how many files to upload during a test run", "10"),
new CommandLineArgument("min-file-size", CommandLineArgument.ArgumentType.Size, "The minimum allowed file size", "File sizes are chosen at random, this value is the lower bound", "1kb"),
new CommandLineArgument("max-file-size", CommandLineArgument.ArgumentType.Size, "The maximum allowed file size", "File sizes are chosen at random, this value is the upper bound", "50mb"),
new CommandLineArgument("min-filename-length", CommandLineArgument.ArgumentType.Integer, "The minimum allowed filename length", "File name lengths are chosen at random, this value is the lower bound", "5"),
new CommandLineArgument("max-filename-length", CommandLineArgument.ArgumentType.Integer, "The minimum allowed filename length", "File name lengths are chosen at random, this value is the upper bound", "80"),
new CommandLineArgument("trim-filename-spaces", CommandLineArgument.ArgumentType.Boolean, "Trim whitespace from filenames", "A value that indicates if whitespace should be trimmed from the ends of randomly generated filenames", "false"),
new CommandLineArgument("auto-create-folder", CommandLineArgument.ArgumentType.Boolean, "Allow automatic folder creation", "A value that indicates if missing folders are created automatically", "false"),
new CommandLineArgument("skip-overwrite-test", CommandLineArgument.ArgumentType.Boolean, "Bypass the overwrite test", "A value that indicates if dummy files should be uploaded prior to uploading the real files", "false"),
new CommandLineArgument("auto-clean", CommandLineArgument.ArgumentType.Boolean, "Remove any files found in target folder", "A value that indicates if all files in the target folder should be deleted before starting the first test", "false"),
new CommandLineArgument("force", CommandLineArgument.ArgumentType.Boolean, "Activate file deletion", "A value that indicates if existing files should really be deleted when using auto-clean", "false"),
new CommandLineArgument("wait-after-upload", CommandLineArgument.ArgumentType.Timespan, "Wait after all uploads", "A value that indicates how long to wait after all files are uploaded, to account for the backends eventual consistency", "0s"),
new CommandLineArgument("wait-after-delete", CommandLineArgument.ArgumentType.Timespan, "Wait after all deletes", "A value that indicates how long to wait after each delete operation, to account for the backends eventual consistency", "0s"),
new CommandLineArgument("failure-retries", CommandLineArgument.ArgumentType.Integer, "The number of retries for each operation", "An integer that indicates how many times an operation should be retried before giving up", "0"),
];
}
}