mirror of
https://github.com/duplicati/duplicati.git
synced 2025-11-28 11:30:24 +08:00
This comes from suggestion on issue https://github.com/duplicati/duplicati/issues/6056 The chosen argument was --json which produces a json that wraps the result with extended properties such as in example: `` { "Timestamp": "2025-03-28T15:47:01.3368160-03:00", "UnixTimestamp": 1743187621, "Command": "import", "Success": true, "ExitCode": 0, "Messages": [ "Importing backup configuration from ../2-firstbacku.json", "Connecting to http://127.0.0.1:8200/...", "No database found in../data/", "Imported \"firstbackup (5)\" with ID 8" ], "Exceptions": [], "Imported": { "Id": "8", "Name": "firstbackup (5)" } } `` The documentation will reflect all commands and schemas as this makes its way into Canary
82 lines
3.7 KiB
C#
82 lines
3.7 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.CommandLine;
|
|
using System.CommandLine.NamingConventionBinder;
|
|
|
|
namespace Duplicati.CommandLine.ServerUtil.Commands;
|
|
|
|
public static class Import
|
|
{
|
|
public static Command Create() =>
|
|
new Command("import", "Import a backup configuration")
|
|
{
|
|
new Argument<FileInfo>("file", "The file to import, may be encrypted") {
|
|
Arity = ArgumentArity.ExactlyOne
|
|
},
|
|
new Argument<string>("passphrase", "The passphrase to use for decryption") {
|
|
Arity = ArgumentArity.ZeroOrOne
|
|
},
|
|
new Option<bool>(name: "--import-metadata", description: "Import metadata from the backup", getDefaultValue: () => false)
|
|
}
|
|
.WithHandler(CommandHandler.Create<Settings, OutputInterceptor, FileInfo, string, bool>(async (settings, output, file, passphrase, importMetadata) =>
|
|
{
|
|
if (!file.Exists)
|
|
throw new UserReportedException($"File {file.FullName} does not exist");
|
|
|
|
output.AppendConsoleMessage($"Importing backup configuration from {file.FullName}");
|
|
if (IsEncrypted(file))
|
|
{
|
|
if (output.JsonOutputMode)
|
|
throw new UserReportedException("No password provided with json mode.");
|
|
|
|
if (string.IsNullOrWhiteSpace(passphrase))
|
|
passphrase = HelperMethods.ReadPasswordFromConsole("The file is encrypted. Please provide the encryption password: ");
|
|
|
|
if (string.IsNullOrWhiteSpace(passphrase))
|
|
throw new UserReportedException("No password provided");
|
|
|
|
if (settings.SecretProvider != null)
|
|
{
|
|
var opts = new Dictionary<string, string?> { { "password", passphrase } };
|
|
await settings.ReplaceSecrets(opts).ConfigureAwait(false);
|
|
passphrase = opts["password"]!;
|
|
}
|
|
}
|
|
|
|
var connection = await settings.GetConnection(output);
|
|
var result = await connection.ImportBackup(file.FullName, passphrase, importMetadata);
|
|
|
|
output.AppendConsoleMessage($"Imported \"{result.Name}\" with ID {result.ID}");
|
|
output.AppendCustomObject( "Imported",new {Id = result.ID, Name = result.Name});
|
|
output.SetResult(true);
|
|
}));
|
|
|
|
private static bool IsEncrypted(FileInfo file)
|
|
{
|
|
using var fs = file.OpenRead();
|
|
var header = new byte[3].AsSpan();
|
|
if (fs.Read(header) != 3)
|
|
return false;
|
|
return header.SequenceEqual("AES"u8);
|
|
}
|
|
|
|
}
|