duplicati/Duplicati/Library/Modules/Builtin/ReportHelper.cs
Kenneth Skovhede 151afe5d82 Add null checks to send-mail
This PR adds null checks to the email module to prevent crashes when sending email.

It also guards against improper reuse of the modules. This is related to #6343.

This fixes #6315
2025-06-12 22:17:02 +02:00

619 lines
25 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.Linq;
using System.Text;
using System.Text.RegularExpressions;
using Duplicati.Library.AutoUpdater;
using Duplicati.Library.Interface;
using Duplicati.Library.Modules.Builtin.ResultSerialization;
using Duplicati.Library.Utility;
namespace Duplicati.Library.Modules.Builtin
{
/// <summary>
/// A helper module that contains all shared code used in the various reporting modules
/// </summary>
public abstract class ReportHelper : IGenericCallbackModule
{
/// <summary>
/// The tag used for logging
/// </summary>
private static readonly string LOGTAG = Logging.Log.LogTagFromType<ReportHelper>();
/// <summary>
/// The salt used for calculating a backup Id from the remote URL
/// </summary>
private const string SALT = "DUPL";
/// <summary>
/// Name of the option used to specify subject
/// </summary>
protected abstract string SubjectOptionName { get; }
/// <summary>
/// Name of the option used to specify the body
/// </summary>
protected abstract string BodyOptionName { get; }
/// <summary>
/// Name of the option used to specify the level on which the operation is activated
/// </summary>
protected abstract string ActionLevelOptionName { get; }
/// <summary>
/// Name of the option used to specify if reports are sent for other operations than backups
/// </summary>
protected abstract string ActionOnAnyOperationOptionName { get; }
/// <summary>
/// Name of the option used to specify the log level
/// </summary>
protected abstract string LogLevelOptionName { get; }
/// <summary>
/// Name of the option used to specify the log filter
/// </summary>
protected abstract string LogFilterOptionName { get; }
/// <summary>
/// Name of the option used to specify the maximum number of log lines to include
/// </summary>
protected abstract string LogLinesOptionName { get; }
/// <summary>
/// Name of the option used to the output format
/// </summary>
protected abstract string ResultFormatOptionName { get; }
/// <summary>
/// Name of the option used to specify extra data to include in the report
/// </summary>
protected abstract string ExtraDataOptionName { get; }
/// <summary>
/// The default subject or title line
/// </summary>
protected virtual string DEFAULT_SUBJECT { get; } = "Duplicati %OPERATIONNAME% report for %backup-name%";
/// <summary>
/// The default report level
/// </summary>
protected virtual string DEFAULT_LEVEL { get; } = "all";
/// <summary>
/// The default report body
/// </summary>
protected virtual string DEFAULT_BODY { get; } = "%RESULT%";
/// <summary>
/// The default maximum number of log lines
/// </summary>
protected virtual int DEFAULT_LOGLINES { get; } = 100;
/// <summary>
/// The default log level
/// </summary>
protected virtual Logging.LogMessageType DEFAULT_LOG_LEVEL { get; } = Logging.LogMessageType.Warning;
/// <summary>
/// The default export format
/// </summary>
protected virtual ResultExportFormat DEFAULT_EXPORT_FORMAT { get; } = ResultExportFormat.Duplicati;
/// <summary>
/// The module key
/// </summary>
public abstract string Key { get; }
/// <summary>
/// The module display name
/// </summary>
public abstract string DisplayName { get; }
/// <summary>
/// The module description
/// </summary>
public abstract string Description { get; }
/// <summary>
/// The module default load setting
/// </summary>
public abstract bool LoadAsDefault { get; }
/// <summary>
/// The list of supported commands
/// </summary>
public abstract IList<ICommandLineArgument> SupportedCommands { get; }
/// <summary>
/// Returns the format used by the serializer
/// </summary>
protected ResultExportFormat ExportFormat => m_resultFormatSerializer.Format;
/// <summary>
/// The cached name of the operation
/// </summary>
protected string m_operationname;
/// <summary>
/// The cached remote url
/// </summary>
protected string m_remoteurl;
/// <summary>
/// The cached local path
/// </summary>
protected string[] m_localpath;
/// <summary>
/// The cached set of options
/// </summary>
protected IReadOnlyDictionary<string, string> m_options;
/// <summary>
/// The parsed result level
/// </summary>
protected string m_parsedresultlevel = string.Empty;
/// <summary>
/// The maximum number of log lines to include
/// </summary>
protected int m_maxmimumLogLines;
/// <summary>
/// A value indicating if this instance is configured
/// </summary>
private bool m_isConfigured;
/// <summary>
/// The mail subject
/// </summary>
private string m_subject;
/// <summary>
/// The mail body
/// </summary>
private string m_body;
/// <summary>
/// The mail send level
/// </summary>
private string[] m_levels;
/// <summary>
/// True to send all operations
/// </summary>
private bool m_sendAll;
/// <summary>
/// The extra values to include in the report
/// </summary>
private Dictionary<string, string> m_extraValues;
/// <summary>
/// The log scope that should be disposed
/// </summary>
private IDisposable m_logscope;
/// <summary>
/// The log storage
/// </summary>
private FileBackedStringList m_logstorage;
/// <summary>
/// Serializer to use when serializing the message.
/// </summary>
private IResultFormatSerializer m_resultFormatSerializer;
/// <summary>
/// Configures the module
/// </summary>
/// <returns><c>true</c>, if module should be used, <c>false</c> otherwise.</returns>
/// <param name="commandlineOptions">A set of commandline options passed to Duplicati</param>
protected abstract bool ConfigureModule(IDictionary<string, string> commandlineOptions);
/// <summary>
/// Sends the email message
/// </summary>
/// <param name="subject">The subject line.</param>
/// <param name="body">The message body.</param>
protected abstract void SendMessage(string subject, string body);
/// <summary>
/// This method is the interception where the module can interact with the execution environment and modify the settings.
/// </summary>
/// <param name="commandlineOptions">A set of commandline options passed to Duplicati</param>
public void Configure(IDictionary<string, string> commandlineOptions)
{
m_isConfigured = false;
if (!ConfigureModule(commandlineOptions))
return;
m_options = commandlineOptions.AsReadOnly();
m_isConfigured = true;
m_options.TryGetValue(SubjectOptionName, out m_subject);
m_options.TryGetValue(BodyOptionName, out m_body);
m_options.TryGetValue(ExtraDataOptionName, out var extraData);
if (!string.IsNullOrWhiteSpace(extraData))
{
if (extraData.Trim().StartsWith("{") && extraData.Trim().EndsWith("}"))
{
m_extraValues = System.Text.Json.JsonSerializer.Deserialize<Dictionary<string, string>>(extraData);
}
else
{
var values = Utility.Uri.ParseQueryString(extraData);
m_extraValues = values.AllKeys
.Where(x => !string.IsNullOrWhiteSpace(x))
.ToDictionary(key => key, key => values[key]);
}
}
string tmp;
m_options.TryGetValue(ActionLevelOptionName, out tmp);
if (!string.IsNullOrEmpty(tmp))
m_levels =
tmp
.Split(new string[] { "," }, StringSplitOptions.RemoveEmptyEntries)
.Where(x => !string.IsNullOrWhiteSpace(x))
.Select(x => x.Trim())
.ToArray();
if (m_levels == null || m_levels.Length == 0)
m_levels =
DEFAULT_LEVEL
.Split(new string[] { "," }, StringSplitOptions.RemoveEmptyEntries)
.Where(x => !string.IsNullOrWhiteSpace(x))
.Select(x => x.Trim())
.ToArray();
m_sendAll = Utility.Utility.ParseBoolOption(m_options, ActionOnAnyOperationOptionName);
ResultExportFormat resultFormat;
if (!m_options.TryGetValue(ResultFormatOptionName, out var tmpResultFormat))
resultFormat = DEFAULT_EXPORT_FORMAT;
else if (!Enum.TryParse(tmpResultFormat, true, out resultFormat))
resultFormat = DEFAULT_EXPORT_FORMAT;
m_resultFormatSerializer = ResultFormatSerializerProvider.GetSerializer(resultFormat);
m_options.TryGetValue(LogLinesOptionName, out var loglinestr);
if (!int.TryParse(loglinestr, out m_maxmimumLogLines))
m_maxmimumLogLines = DEFAULT_LOGLINES;
if (string.IsNullOrEmpty(m_subject))
m_subject = DEFAULT_SUBJECT;
if (string.IsNullOrEmpty(m_body))
m_body = DEFAULT_BODY;
m_options.TryGetValue(LogFilterOptionName, out var logfilterstring);
var filter = FilterExpression.ParseLogFilter(logfilterstring);
var logLevel = Utility.Utility.ParseEnumOption(m_options, LogLevelOptionName, DEFAULT_LOG_LEVEL);
m_logstorage = new FileBackedStringList();
m_logscope = Logging.Log.StartScope(m => m_logstorage.Add(m.AsString(true)), m =>
{
if (filter.Matches(m.FilterTag, out var result, out var match))
return result;
else if (m.Level < logLevel)
return false;
return true;
});
}
/// <summary>
/// Called when the operation starts
/// </summary>
/// <param name="operationname">The full name of the operation</param>
/// <param name="remoteurl">The remote backend url</param>
/// <param name="localpath">The local path, if required</param>
public virtual void OnStart(string operationname, ref string remoteurl, ref string[] localpath)
{
m_operationname = operationname;
m_remoteurl = remoteurl;
m_localpath = localpath;
}
/// <summary>
/// Helper method to perform template expansion
/// </summary>
/// <returns>The expanded template.</returns>
/// <param name="input">The input template.</param>
/// <param name="result">The result object.</param>
/// <param name="exception">An optional exception that has stopped the backup</param>
/// <param name="subjectline">If set to <c>true</c>, the result is intended for a subject or title line.</param>
protected virtual string ReplaceTemplate(string input, object result, Exception exception, bool subjectline)
=> ReplaceTemplate(input, result, exception, subjectline, m_resultFormatSerializer);
/// <summary>
/// Helper method to perform template expansion
/// </summary>
/// <returns>The expanded template.</returns>
/// <param name="input">The input template.</param>
/// <param name="result">The result object.</param>
/// <param name="exception">An optional exception that has stopped the backup</param>
/// <param name="subjectline">If set to <c>true</c>, the result is intended for a subject or title line.</param>
/// <param name="format">The format to use when serializing the result</param>
protected virtual string ReplaceTemplate(string input, object result, Exception exception, bool subjectline, ResultExportFormat format)
=> ReplaceTemplate(input, result, exception, subjectline, ResultFormatSerializerProvider.GetSerializer(format));
/// <summary>
/// The operation name template key
/// </summary>
private const string OPERATIONNAME = "OperationName";
/// <summary>
/// The remote url template key
/// </summary>
private const string REMOTEURL = "RemoteUrl";
/// <summary>
/// The local path template key
/// </summary>
private const string LOCALPATH = "LocalPath";
/// <summary>
/// The parsed result template key
/// </summary>
private const string PARSEDRESULT = "ParsedResult";
/// <summary>
/// The machine id template key
/// </summary>
private const string MACHINE_ID = "machine-id";
/// <summary>
/// The backup id template key
/// </summary>
private const string BACKUP_ID = "backup-id";
/// <summary>
/// The backup name template key
/// </summary>
private const string BACKUP_NAME = "backup-name";
/// <summary>
/// The machine name template key
/// </summary>
private const string MACHINE_NAME = "machine-name";
/// <summary>
/// The operating system template key
/// </summary>
private const string OPERATING_SYSTEM = "operating-system";
/// <summary>
/// The installation type template key
/// </summary>
private const string INSTALLATION_TYPE = "installation-type";
/// <summary>
/// The destination type template key
/// </summary>
private const string DESTINATION_TYPE = "destination-type";
/// <summary>
/// The next scheduled run template key
/// </summary>
private const string NEXT_SCHEDULED_RUN = "next-scheduled-run";
/// <summary>
/// The destination type template key
/// </summary>
private const string UPDATE_CHANNEL = "update-channel";
/// <summary>
/// The list of regular template keys
/// </summary>
private static readonly IReadOnlySet<string> OPERATION_TEMPLATE_KEYS = new HashSet<string>([
OPERATIONNAME, REMOTEURL, LOCALPATH, PARSEDRESULT
], StringComparer.OrdinalIgnoreCase);
/// <summary>
/// The list of extra template keys
/// </summary>
private static readonly IReadOnlySet<string> EXTRA_TEMPLATE_KEYS = new HashSet<string>([
MACHINE_ID, BACKUP_ID, BACKUP_NAME, MACHINE_NAME,
OPERATING_SYSTEM, INSTALLATION_TYPE, DESTINATION_TYPE, NEXT_SCHEDULED_RUN,
UPDATE_CHANNEL
], StringComparer.OrdinalIgnoreCase);
/// <summary>
/// Gets the default value for a template key
/// </summary>
/// <param name="name">The name of the template key</param>
/// <returns>The default value</returns>
private string GetDefaultValue(string name)
{
switch (name)
{
case OPERATIONNAME:
return m_operationname;
case REMOTEURL:
return m_remoteurl;
case LOCALPATH:
return m_localpath == null ? "" : string.Join(System.IO.Path.PathSeparator.ToString(), m_localpath);
case PARSEDRESULT:
return m_parsedresultlevel;
case MACHINE_ID:
return DataFolderManager.MachineID;
case BACKUP_ID:
return Utility.Utility.ByteArrayAsHexString(Utility.Utility.RepeatedHashWithSalt(m_remoteurl, SALT));
case BACKUP_NAME:
return System.IO.Path.GetFileNameWithoutExtension(Utility.Utility.getEntryAssembly().Location);
case MACHINE_NAME:
return DataFolderManager.MachineName;
case OPERATING_SYSTEM:
return UpdaterManager.OperatingSystemName;
case INSTALLATION_TYPE:
return UpdaterManager.PackageTypeId;
case DESTINATION_TYPE:
// Only return the url scheme, as the rest could contain sensitive information
var ix = m_remoteurl?.IndexOf("://", StringComparison.OrdinalIgnoreCase) ?? -1;
return Utility.Utility.GuessScheme(m_remoteurl) ?? "file";
case UPDATE_CHANNEL:
return UpdaterManager.CurrentChannel.ToString();
default:
return null;
}
}
/// <summary>
/// Helper method to perform template expansion
/// </summary>
/// <returns>The expanded template.</returns>
/// <param name="input">The input template.</param>
/// <param name="result">The result object.</param>
/// <param name="exception">An optional exception that has stopped the backup</param>
/// <param name="subjectline">If set to <c>true</c>, the result is intended for a subject or title line.</param>
/// <param name="resultFormatSerializer">The serializer to use when serializing the result</param>
protected virtual string ReplaceTemplate(string input, object result, Exception exception, bool subjectline, IResultFormatSerializer resultFormatSerializer)
{
// For JSON, ignore the template and just use the contents
if (resultFormatSerializer.Format == ResultExportFormat.Json && !subjectline)
{
var extra = new Dictionary<string, string>();
// Add the default values, if found in the template
foreach (var key in OPERATION_TEMPLATE_KEYS)
if (input.IndexOf($"%{key}%", StringComparison.OrdinalIgnoreCase) >= 0)
extra[key] = GetDefaultValue(key);
// Add any options that are whitelisted or used in the template
foreach (var kv in m_options)
if (EXTRA_TEMPLATE_KEYS.Contains(kv.Key) || input.IndexOf($"%{kv.Key}%", StringComparison.OrdinalIgnoreCase) >= 0)
extra[kv.Key] = kv.Value;
// Add any missing default values
foreach (var key in EXTRA_TEMPLATE_KEYS)
if (!extra.ContainsKey(key))
extra[key] = GetDefaultValue(key);
if (m_extraValues != null)
foreach (var v in m_extraValues)
extra[v.Key] = v.Value;
return resultFormatSerializer.Serialize(result, exception, LogLines, extra);
}
else
{
foreach (var key in OPERATION_TEMPLATE_KEYS)
input = Regex.Replace(input, $"%{key}%", GetDefaultValue(key) ?? "", RegexOptions.IgnoreCase | RegexOptions.CultureInvariant);
if (subjectline)
{
input = Regex.Replace(input, "\\%RESULT\\%", m_parsedresultlevel ?? "", RegexOptions.IgnoreCase | RegexOptions.CultureInvariant);
}
else
{
if (input.IndexOf("%RESULT%", StringComparison.OrdinalIgnoreCase) >= 0)
input = Regex.Replace(input, "\\%RESULT\\%", resultFormatSerializer.Serialize(result, exception, LogLines, null), RegexOptions.IgnoreCase | RegexOptions.CultureInvariant);
}
foreach (KeyValuePair<string, string> kv in m_options)
input = Regex.Replace(input, "\\%" + kv.Key + "\\%", kv.Value ?? "", RegexOptions.IgnoreCase | RegexOptions.CultureInvariant);
foreach (var key in EXTRA_TEMPLATE_KEYS)
if (!m_options.ContainsKey(key))
input = Regex.Replace(input, $"%{key}%", GetDefaultValue(key) ?? "", RegexOptions.IgnoreCase | RegexOptions.CultureInvariant);
if (m_extraValues != null)
foreach (var v in m_extraValues)
input = Regex.Replace(input, $"%{v.Key}%", v.Value, RegexOptions.IgnoreCase | RegexOptions.CultureInvariant);
// Remove any remaining template keys
input = Regex.Replace(input, "\\%[^\\%]+\\%", "");
return input;
}
}
/// <summary>
/// Gets the filtered set of log lines
/// </summary>
protected IEnumerable<string> LogLines
{
get
{
var logdata = m_logstorage.AsEnumerable();
if (m_maxmimumLogLines > 0)
{
logdata = logdata.Take(m_maxmimumLogLines);
if (m_logstorage.Count > m_maxmimumLogLines)
logdata = logdata.Concat(new string[] { $"... and {m_logstorage.Count - m_maxmimumLogLines} more" });
}
return logdata;
}
}
public void OnFinish(IBasicResults result, Exception exception)
{
// Dispose the current log scope
if (m_logscope != null)
{
try { m_logscope.Dispose(); }
catch { }
m_logscope = null;
}
if (!m_isConfigured)
return;
//If we do not report this action, then skip
if (!m_sendAll && !string.Equals(m_operationname, "Backup", StringComparison.OrdinalIgnoreCase))
return;
ParsedResultType level;
if (exception != null)
level = ParsedResultType.Fatal;
else if (result != null)
level = result.ParsedResult;
else
level = ParsedResultType.Error;
m_parsedresultlevel = level.ToString();
if (string.Equals(m_operationname, "Backup", StringComparison.OrdinalIgnoreCase))
{
if (!m_levels.Any(x => string.Equals(x, "all", StringComparison.OrdinalIgnoreCase)))
{
//Check if this level should send mail
if (!m_levels.Any(x => string.Equals(x, level.ToString(), StringComparison.OrdinalIgnoreCase)))
return;
}
}
try
{
string body = m_body;
string subject = m_subject;
if (body != DEFAULT_BODY)
{
try
{
if (System.IO.File.Exists(body) && System.IO.Path.IsPathRooted(body))
body = System.IO.File.ReadAllText(body);
}
catch (Exception ex)
{
Logging.Log.WriteWarningMessage(LOGTAG, "ReportSubmitError", ex, "Invalid path, or unable to read file given as body");
}
}
body = ReplaceTemplate(body, result, exception, false);
subject = ReplaceTemplate(subject, result, exception, true);
SendMessage(subject, body);
}
catch (Exception ex)
{
Exception top = ex;
var sb = new StringBuilder();
while (top != null)
{
if (sb.Length != 0)
sb.Append("--> ");
sb.AppendFormat("{0}: {1}{2}", top.GetType().FullName, top.Message, Environment.NewLine);
top = top.InnerException;
}
Logging.Log.WriteWarningMessage(LOGTAG, "ReportSubmitError", ex, Strings.ReportHelper.SendMessageFailedError(sb.ToString()));
}
}
}
}