duplicati/Duplicati/WebserverCore/Services/SystemInfoProvider.cs
Kenneth Skovhede 3269c87329 Add PowerMode provider selection
This PR adds settings to choose a Power Mode provider, which will detect if the system is in a paused/suspended state and avoid starting tasks while being suspended.

This currently only works for Windows, where the previous inplementation is named `NET` (for .NET) and the new mode is named `Native` and using the Windows documented approach with a hidden window that listens for `WM_POWERBROADCAST` messages.

The default is `Native` but can now also be disabled by choosing the `None` provider.
2025-10-15 17:58:24 +02:00

378 lines
15 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.Globalization;
using Duplicati.Library.AutoUpdater;
using Duplicati.Library.Localization;
using Duplicati.Library.Snapshots;
using Duplicati.Library.Utility.Options;
using Duplicati.Server;
using Duplicati.Server.Database;
using Duplicati.Server.Serialization.Interface;
using Duplicati.WebserverCore.Abstractions;
using Duplicati.WebserverCore.Dto;
namespace Duplicati.WebserverCore.Services;
/// <summary>
/// Produces system information.
/// </summary>
public class SystemInfoProvider(IApplicationSettings applicationSettings, Connection connection) : ISystemInfoProvider
{
/// <summary>
/// The API extensions that are available
/// </summary>
public static readonly string[] SupportedAPIExtensions = [
"v2:backup:list-filesets",
"v2:backup:list-folder",
"v2:backup:list-versions",
"v2:backup:search",
"v2:destination:test",
"v1:websocket",
"v1:gettask:taskstarted",
"v1:gettask:taskfinished",
"v1:websocket:authenticate",
"v1:subscribe:legacystatus",
"v1:subscribe:backuplist",
"v1:subscribe:serversettings",
"v1:subscribe:progress",
"v1:subscribe:taskqueue",
"v1:subscribe:taskcompleted",
"v1:subscribe:notifications",
// "v1:subscribe:scheduler",
];
/// <summary>
/// The API scopes that are available
/// </summary>
private static readonly string[] APIScopes = ["*"];
/// <summary>
/// System information that does not change during runtime.
/// </summary>
private sealed record StaticSystemInformation
{
/// <summary>
/// Gets or sets the API version.
/// </summary>
public required int APIVersion { get; init; }
/// <summary>
/// Gets or sets the password placeholder.
/// </summary>
public required string PasswordPlaceholder { get; init; }
/// <summary>
/// Gets or sets the server version.
/// </summary>
public required string? ServerVersion { get; init; }
/// <summary>
/// Gets or sets the server version name.
/// </summary>
public required string ServerVersionName { get; init; }
/// <summary>
/// Gets or sets the server version type.
/// </summary>
public required string? ServerVersionType { get; init; }
/// <summary>
/// The default URL to present for remote control registration
/// </summary>
public required string RemoteControlRegistrationUrl { get; init; }
/// <summary>
/// Gets or sets the default update channel.
/// </summary>
public required string DefaultUpdateChannel { get; init; }
/// <summary>
/// Gets or sets the default usage report level.
/// </summary>
public required string DefaultUsageReportLevel { get; init; }
/// <summary>
/// Gets or sets the OS type.
/// </summary>
public required string OSType { get; init; }
/// <summary>
/// Gets or sets the OS version.
/// </summary>
public required string OSVersion { get; init; }
/// <summary>
/// Gets or sets the directory separator.
/// </summary>
public required char DirectorySeparator { get; init; }
/// <summary>
/// Gets or sets the path separator.
/// </summary>
public required char PathSeparator { get; init; }
/// <summary>
/// Gets or sets a value indicating whether the filesystem is case sensitive.
/// </summary>
public required bool CaseSensitiveFilesystem { get; init; }
/// <summary>
/// Gets or sets the machine name.
/// </summary>
public required string MachineName { get; init; }
/// <summary>
/// Gets or sets the package type ID.
/// </summary>
public required string PackageTypeId { get; init; }
/// <summary>
/// Gets or sets the user name.
/// </summary>
public required string UserName { get; init; }
/// <summary>
/// Gets or sets the new line character.
/// </summary>
public required string NewLine { get; init; }
/// <summary>
/// Gets or sets the CLR version.
/// </summary>
public required string CLRVersion { get; init; }
/// <summary>
/// Gets or sets the options.
/// </summary>
public required Library.Interface.ICommandLineArgument[] Options { get; init; }
/// <summary>
/// Gets or sets the compression modules.
/// </summary>
public required IDynamicModule[] CompressionModules { get; init; }
/// <summary>
/// Gets or sets the encryption modules.
/// </summary>
public required IDynamicModule[] EncryptionModules { get; init; }
/// <summary>
/// Gets or sets the backend modules.
/// </summary>
public required IDynamicModule[] BackendModules { get; init; }
/// <summary>
/// Gets or sets the generic modules.
/// </summary>
public required IDynamicModule[] GenericModules { get; init; }
/// <summary>
/// Gets or sets the web modules.
/// </summary>
public required IDynamicModule[] WebModules { get; init; }
/// <summary>
/// Gets or sets the connection modules.
/// </summary>
public required IDynamicModule[] ConnectionModules { get; init; }
/// <summary>
/// Gets or sets the server modules.
/// </summary>
public required object[] ServerModules { get; init; }
/// <summary>
/// Gets or sets the secret provider modules.
/// </summary>
public required IDynamicModule[] SecretProviderModules { get; init; }
/// <summary>
/// Gets or sets a value indicating whether alternate update URLs are being used.
/// </summary>
public required bool UsingAlternateUpdateURLs { get; init; }
/// <summary>
/// Gets or sets the log levels.
/// </summary>
public required string[] LogLevels { get; init; }
/// <summary>
/// Gets or sets the special folders.
/// </summary>
public required SystemInfoDto.SpecialFolderDto[] SpecialFolders { get; init; }
/// <summary>
/// Gets or sets the supported locales.
/// </summary>
public required SystemInfoDto.LocaleDto[] SupportedLocales { get; init; }
/// <summary>
/// The timezones available on the system
/// </summary>
public required IEnumerable<SystemInfoDto.TimeZoneDto> TimeZones { get; init; }
/// <summary>
/// The power mode providers supported
/// </summary>
public required string[] PowerModeProviders { get; init; }
}
/// <summary>
/// Creates a new instance of the <see cref="StaticSystemInformation"/> class.
/// </summary>
/// <returns>The new instance.</returns>
private static StaticSystemInformation CreateStaticSystemInformation()
=> new StaticSystemInformation
{
APIVersion = 1,
PasswordPlaceholder = Connection.PASSWORD_PLACEHOLDER,
ServerVersion = UpdaterManager.SelfVersion.Version,
ServerVersionName = License.VersionNumbers.VERSION_NAME,
ServerVersionType = UpdaterManager.SelfVersion.ReleaseType,
RemoteControlRegistrationUrl = Library.RemoteControl.RegisterForRemote.DefaultRegisterationUrl,
DefaultUpdateChannel = AutoUpdateSettings.DefaultUpdateChannel.ToString(),
DefaultUsageReportLevel = Library.UsageReporter.Reporter.DefaultReportLevel,
OSType = UpdaterManager.OperatingSystemName,
OSVersion = Library.UsageReporter.OSInfoHelper.PlatformString,
DirectorySeparator = Path.DirectorySeparatorChar,
PathSeparator = Path.PathSeparator,
CaseSensitiveFilesystem = Library.Utility.Utility.IsFSCaseSensitive,
MachineName = Environment.MachineName,
PackageTypeId = UpdaterManager.PackageTypeId,
UserName = OperatingSystem.IsWindows() ? System.Security.Principal.WindowsIdentity.GetCurrent().Name : Environment.UserName,
NewLine = Environment.NewLine,
CLRVersion = Environment.Version.ToString(),
Options = Server.Serializable.ServerSettings.Options,
CompressionModules = Server.Serializable.ServerSettings.CompressionModules,
EncryptionModules = Server.Serializable.ServerSettings.EncryptionModules,
BackendModules = Server.Serializable.ServerSettings.BackendModules,
GenericModules = Server.Serializable.ServerSettings.GenericModules,
WebModules = Server.Serializable.ServerSettings.WebModules,
ConnectionModules = Server.Serializable.ServerSettings.ConnectionModules,
ServerModules = Server.Serializable.ServerSettings.ServerModules,
SecretProviderModules = Server.Serializable.ServerSettings.SecretProviderModules,
UsingAlternateUpdateURLs = AutoUpdateSettings.UsesAlternateURLs,
LogLevels = Enum.GetNames(typeof(Library.Logging.LogMessageType)),
PowerModeProviders = OperatingSystem.IsWindows()
? [string.Empty, PowerModeProvider.None.ToString(), PowerModeProvider.Net.ToString(), PowerModeProvider.Native.ToString()]
: [string.Empty, PowerModeProvider.None.ToString()],
SpecialFolders = SpecialFolders.Nodes.Select(n => new Dto.SystemInfoDto.SpecialFolderDto { ID = n.id, Path = n.resolvedpath }).ToArray(),
SupportedLocales = Library.Localization.LocalizationService.SupportedCultures
.Select(x => new Dto.SystemInfoDto.LocaleDto
{
Code = x,
EnglishName = new CultureInfo(x).EnglishName,
DisplayName = new CultureInfo(x).NativeName
}).ToArray(),
TimeZones = Library.Utility.TimeZoneHelper.GetTimeZones()
.Select(x => new Dto.SystemInfoDto.TimeZoneDto
{
ID = x.Id,
DisplayName = x.DisplayName,
CurrentUTCOffset = x.CurrentUtcOffset.ToString()
}),
};
/// <summary>
/// The lock guarding the _systemInfoBases cache.
/// </summary>
private readonly object _lock = new();
/// <summary>
/// Cached static system information, the key is the language code.
/// </summary>
private Dictionary<string, StaticSystemInformation> _systemInfoBases = new(StringComparer.OrdinalIgnoreCase);
/// <inheritdoc />
public SystemInfoDto GetSystemInfo(CultureInfo? browserlanguage)
{
browserlanguage ??= CultureInfo.InvariantCulture;
var currentLanguage = LocalizationService.CurrentUI?.Culture.Name;
if (string.IsNullOrWhiteSpace(currentLanguage))
currentLanguage = CultureInfo.CurrentUICulture.Name;
StaticSystemInformation? systeminfo;
lock (_lock)
{
if (!_systemInfoBases.TryGetValue(currentLanguage, out systeminfo))
{
systeminfo = CreateStaticSystemInformation();
_systemInfoBases[currentLanguage] = systeminfo;
}
}
var disabledAPIExtensions = connection.ApplicationSettings.DisabledAPIExtensions;
// Return the system information, patch in dynamic values
return new SystemInfoDto()
{
APIVersion = systeminfo.APIVersion,
PasswordPlaceholder = systeminfo.PasswordPlaceholder,
ServerVersion = systeminfo.ServerVersion,
ServerVersionName = systeminfo.ServerVersionName,
ServerVersionType = systeminfo.ServerVersionType,
RemoteControlRegistrationUrl = systeminfo.RemoteControlRegistrationUrl,
StartedBy = applicationSettings.Origin,
DefaultUpdateChannel = systeminfo.DefaultUpdateChannel,
DefaultUsageReportLevel = systeminfo.DefaultUsageReportLevel,
ServerTime = DateTime.Now,
ServerTimeZone = TimeZoneInfo.Local.Id,
OSType = systeminfo.OSType,
OSVersion = systeminfo.OSVersion,
DirectorySeparator = systeminfo.DirectorySeparator,
PathSeparator = systeminfo.PathSeparator,
CaseSensitiveFilesystem = systeminfo.CaseSensitiveFilesystem,
MachineName = systeminfo.MachineName,
PackageTypeId = systeminfo.PackageTypeId,
UserName = systeminfo.UserName,
NewLine = systeminfo.NewLine,
CLRVersion = systeminfo.CLRVersion,
Options = systeminfo.Options,
CompressionModules = systeminfo.CompressionModules,
EncryptionModules = systeminfo.EncryptionModules,
BackendModules = systeminfo.BackendModules,
GenericModules = systeminfo.GenericModules,
WebModules = systeminfo.WebModules,
ConnectionModules = systeminfo.ConnectionModules,
ServerModules = systeminfo.ServerModules,
SecretProviderModules = systeminfo.SecretProviderModules,
UsingAlternateUpdateURLs = systeminfo.UsingAlternateUpdateURLs,
LogLevels = systeminfo.LogLevels,
SpecialFolders = systeminfo.SpecialFolders,
APIExtensions = SupportedAPIExtensions.Where(ext => !disabledAPIExtensions.Contains(ext)).ToArray(),
APIScopes = APIScopes,
BrowserLocale = new SystemInfoDto.LocaleDto()
{
Code = browserlanguage.Name,
EnglishName = browserlanguage.EnglishName,
DisplayName = browserlanguage.NativeName
},
SupportedLocales = systeminfo.SupportedLocales,
BrowserLocaleSupported = LocalizationService.IsCultureSupported(browserlanguage),
TimeZones = systeminfo.TimeZones,
DefaultOAuthURL = AuthIdOptionsHelper.DUPLICATI_OAUTH_SERVICE,
DefaultOAuthURLv2 = AuthIdOptionsHelper.DUPLICATI_OAUTH_SERVICE_NEW,
PowerModeProviders = systeminfo.PowerModeProviders,
};
}
}