duplicati/Duplicati/Library/RestAPI/LiveControls.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

448 lines
16 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 Duplicati.Library.Interface;
using Duplicati.Library.IO;
using Duplicati.Library.Snapshots;
using Duplicati.Server.Database;
namespace Duplicati.Server
{
/// <summary>
/// This class keeps track of the users modifications regarding
/// throttling and pause/resume
/// </summary>
public class LiveControls : ILiveControls
{
/// <summary>
/// Event that is activated the the live control state changes
/// </summary>
public sealed record LiveControlEvent
{
/// <summary>
/// The new state of the live control
/// </summary>
public required LiveControlState State { get; init; }
/// <summary>
/// A value that indicates if the transfers are paused
/// </summary>
public required bool TransfersPaused { get; init; }
/// <summary>
/// The time when processing will resume, or zero if paused indefinitely
/// </summary>
public required DateTime WaitTimeExpiration { get; init; }
}
/// <summary>
/// The tag used for logging
/// </summary>
private static readonly string LOGTAG = Duplicati.Library.Logging.Log.LogTagFromType<LiveControls>();
/// <summary>
/// An callback that is activated when the pause state changes
/// </summary>
public Action<LiveControlEvent> StateChanged;
/// <summary>
/// The possible states for the live control
/// </summary>
public enum LiveControlState
{
/// <summary>
/// Indicates that the backups are running
/// </summary>
Running,
/// <summary>
/// Indicates that the backups are currently suspended
/// </summary>
Paused
}
/// <summary>
/// The current control state
/// </summary>
private LiveControlState m_state;
/// <summary>
/// A value that indicates if the transfers are paused
/// </summary>
private bool m_transfersPaused = false;
/// <summary>
/// A value that indicates if the current pause state is caused by being suspended
/// </summary>
private bool m_pausedForSuspend = false;
/// <summary>
/// The time to pause for, used to ensure that a user set pause can override the suspend pause
/// </summary>
private DateTime m_suspendMinimumPause = new DateTime(0, DateTimeKind.Utc);
/// <summary>
/// Gets the current state for the control
/// </summary>
public LiveControlState State { get { return m_state; } }
/// <summary>
/// Gets a value that indicates if the backups are running
/// </summary>
public bool IsPaused => State == LiveControlState.Paused;
/// <summary>
/// Gets a value that indicates if the transfers are paused
/// </summary>
public bool TransfersPaused => m_transfersPaused;
/// <summary>
/// The object that ensures concurrent operations
/// </summary>
private readonly object m_lock = new object();
/// <summary>
/// The timer that is activated after a pause period.
/// </summary>
private System.Threading.Timer m_waitTimer;
/// <summary>
/// The time that the current pause is expected to expire
/// </summary>
private DateTime m_waitTimeExpiration = new DateTime(0, DateTimeKind.Utc);
/// <summary>
/// The connection to use
/// </summary>
private readonly Connection m_connection;
/// <summary>
/// The power mode provider, if any
/// </summary>
private IPowerModeProvider m_powerModeProvider;
/// <summary>
/// The current power mode provider
/// </summary>
private PowerModeProvider m_currentPowerModeProvider = PowerModeProvider.None;
/// <summary>
/// Constructs a new instance of the LiveControl
/// </summary>
/// <param name="connection">The connection to use</param>
public LiveControls(Connection connection)
{
m_connection = connection;
Init();
}
/// <summary>
/// Clamps the milliseconds to a valid range for the timer
/// </summary>
/// <param name="duration">The duration to clamp</param>
/// <returns>The clamped milliseconds</returns>
private static long ClampMilliseconds(TimeSpan duration)
{
if (duration.TotalMilliseconds < 100)
return 100;
if (duration > TimeSpan.FromHours(24))
return (long)TimeSpan.FromHours(24).TotalMilliseconds;
return (long)duration.TotalMilliseconds;
}
/// <summary>
/// Constructs a new instance of the LiveControl
/// </summary>
private void Init()
{
var settings = m_connection.ApplicationSettings;
m_state = LiveControlState.Running;
m_waitTimer = new System.Threading.Timer(m_waitTimer_Tick, this, System.Threading.Timeout.Infinite, System.Threading.Timeout.Infinite);
if (!string.IsNullOrEmpty(settings.StartupDelayDuration) && settings.StartupDelayDuration != "0")
{
var startupDelay = new TimeSpan(0);
try { startupDelay = Library.Utility.Timeparser.ParseTimeSpan(settings.StartupDelayDuration); }
catch { }
if (startupDelay.Ticks > 0)
{
m_waitTimeExpiration = DateTime.UtcNow.Add(startupDelay);
m_waitTimer.Change(ClampMilliseconds(startupDelay), System.Threading.Timeout.Infinite);
m_state = LiveControlState.Paused;
}
}
var pausedUntil = settings.PausedUntil;
if (pausedUntil != null)
{
if (pausedUntil.Value.Ticks == 0)
{
m_waitTimeExpiration = new DateTime(0, DateTimeKind.Utc);
m_waitTimer.Change(System.Threading.Timeout.Infinite, System.Threading.Timeout.Infinite);
m_state = LiveControlState.Paused;
}
else if (pausedUntil.Value > DateTime.UtcNow && pausedUntil.Value > m_waitTimeExpiration)
{
var period = pausedUntil.Value - DateTime.UtcNow;
if (period.TotalMilliseconds > 100)
{
m_waitTimeExpiration = pausedUntil.Value;
m_waitTimer.Change(ClampMilliseconds(period), System.Threading.Timeout.Infinite);
m_state = LiveControlState.Paused;
}
}
}
UpdatePowerModeProvider();
}
/// <summary>
/// Updates the current power mode provider, if changed
/// </summary>
public void UpdatePowerModeProvider()
{
try
{
var newProvider = m_connection.ApplicationSettings.PowerModeProvider;
if (newProvider == m_currentPowerModeProvider)
return;
if (m_powerModeProvider != null)
System.Threading.Interlocked.Exchange(ref m_powerModeProvider, null)?.Dispose();
m_powerModeProvider = PowerModeUtility.GetPowerModeProvider(newProvider);
m_currentPowerModeProvider = newProvider;
if (m_powerModeProvider != null)
{
m_powerModeProvider.OnResume = OnResume;
m_powerModeProvider.OnSuspend = OnSuspend;
}
}
catch { }
}
/// <summary>
/// Event that occurs when the timeout duration is exceeded
/// </summary>
/// <param name="sender">The sender of the event</param>
private void m_waitTimer_Tick(object sender)
{
lock (m_lock)
Resume();
}
/// <summary>
/// Creates a new event object
/// </summary>
/// <returns>A new event object</returns>
private LiveControlEvent CreateEvent()
{
lock (m_lock)
return new LiveControlEvent()
{
State = m_state,
TransfersPaused = m_transfersPaused,
WaitTimeExpiration = m_waitTimeExpiration
};
}
/// <summary>
/// Internal helper to reset the timeout timer
/// </summary>
/// <param name="timeout">The time to wait</param>
private void ResetTimer(string timeout)
{
lock (m_lock)
if (!string.IsNullOrEmpty(timeout))
{
var delay = Library.Utility.Timeparser.ParseTimeSpan(timeout);
m_waitTimeExpiration = DateTime.UtcNow.Add(delay);
m_waitTimer.Change(ClampMilliseconds(delay), System.Threading.Timeout.Infinite);
}
else
{
m_waitTimeExpiration = new DateTime(0, DateTimeKind.Utc);
m_waitTimer.Change(System.Threading.Timeout.Infinite, System.Threading.Timeout.Infinite);
}
}
/// <summary>
/// Internal helper to set the pause mode
/// </summary>
private void SetPauseMode()
{
LiveControlEvent ev = null;
lock (m_lock)
{
if (m_state == LiveControlState.Running)
{
m_state = LiveControlState.Paused;
if (StateChanged != null)
ev = CreateEvent();
}
}
if (ev != null)
StateChanged(ev);
}
/// <summary>
/// Pauses the backups until resumed
/// </summary>
public void Pause(bool alsoTransfers)
{
LiveControlEvent ev = null;
lock (m_lock)
{
m_transfersPaused = alsoTransfers;
m_waitTimeExpiration = new DateTime(0, DateTimeKind.Utc);
ResetTimer(null);
if (m_state == LiveControlState.Paused)
ev = StateChanged == null ? null : CreateEvent();
else
SetPauseMode();
}
if (ev != null)
StateChanged(ev);
}
/// <summary>
/// Resumes a backups to the running state
/// </summary>
public void Resume()
{
LiveControlEvent ev = null;
lock (m_lock)
{
if (m_state == LiveControlState.Paused)
{
//Make sure that the timer is cleared
ResetTimer(null);
m_transfersPaused = false;
m_waitTimeExpiration = new DateTime(0, DateTimeKind.Utc);
m_state = LiveControlState.Running;
if (StateChanged != null)
ev = CreateEvent();
}
}
if (ev != null)
StateChanged(ev);
}
/// <summary>
/// Suspends the backups for a given period
/// </summary>
/// <param name="timeout">The duration to wait</param>
/// <param name="alsoTransfers">If true, also pause the transfers</param>
public void Pause(string timeout, bool alsoTransfers)
{
Pause(Duplicati.Library.Utility.Timeparser.ParseTimeSpan(timeout), alsoTransfers);
}
/// <summary>
/// Suspends the backups for a given period
/// </summary>
/// <param name="timeout">The duration to wait</param>
/// <param name="alsoTransfers">If true, also pause the transfers</param>
public void Pause(TimeSpan timeout, bool alsoTransfers)
{
LiveControlEvent ev = null;
lock (m_lock)
{
m_waitTimeExpiration = DateTime.UtcNow.Add(timeout);
m_waitTimer.Change(ClampMilliseconds(timeout), System.Threading.Timeout.Infinite);
m_transfersPaused = alsoTransfers;
//We change the time, so we issue a new event
if (m_state == LiveControlState.Paused)
ev = StateChanged == null ? null : CreateEvent();
else
SetPauseMode();
}
if (ev != null)
StateChanged(ev);
}
/// <summary>
/// Gets the time the current pause is expected to end
/// </summary>
public DateTime EstimatedPauseEnd { get { return m_waitTimeExpiration; } }
/// <summary>
/// Method called when the power mode provider signals suspend
/// </summary>
private void OnSuspend()
{
//If we are running, register as being paused due to suspending
if (this.m_state == LiveControlState.Running)
{
this.SetPauseMode();
m_pausedForSuspend = true;
m_suspendMinimumPause = new DateTime(0, DateTimeKind.Utc);
}
else
{
if (m_waitTimeExpiration.Ticks != 0)
{
m_pausedForSuspend = true;
m_suspendMinimumPause = this.EstimatedPauseEnd;
ResetTimer(null);
}
}
}
/// <summary>
/// Method called when the power mode provider signals resume
/// </summary>
private void OnResume()
{
//If we have been been paused due to suspending, we un-pause now
if (m_pausedForSuspend)
{
long delayTicks = (m_suspendMinimumPause - DateTime.UtcNow).Ticks;
var appset = m_connection.ApplicationSettings;
if (!string.IsNullOrEmpty(appset.StartupDelayDuration) && appset.StartupDelayDuration != "0")
try { delayTicks = Math.Max(delayTicks, Library.Utility.Timeparser.ParseTimeSpan(appset.StartupDelayDuration).Ticks); }
catch { }
if (delayTicks > 0)
{
this.Pause(TimeSpan.FromTicks(delayTicks), true);
}
else
{
this.Resume();
}
}
m_pausedForSuspend = false;
m_suspendMinimumPause = new DateTime(0, DateTimeKind.Utc);
}
}
}