mirror of
https://github.com/duplicati/duplicati.git
synced 2025-11-28 03:20:25 +08:00
226 lines
9 KiB
C#
226 lines
9 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.Linq;
|
|
using System.Threading;
|
|
using Duplicati.Library.AutoUpdater;
|
|
using Duplicati.Server.Database;
|
|
using Duplicati.Server.Serialization;
|
|
|
|
namespace Duplicati.Server
|
|
{
|
|
/// <summary>
|
|
/// The thread that checks on the update server if new versions are available
|
|
/// </summary>
|
|
public class UpdatePollThread(Connection connection, EventPollNotify eventPollNotify)
|
|
{
|
|
private Thread m_thread;
|
|
private volatile bool m_terminated = false;
|
|
private volatile bool m_forceCheck = false;
|
|
private readonly object m_lock = new object();
|
|
private AutoResetEvent m_waitSignal;
|
|
private double m_downloadProgress;
|
|
private bool m_disableChecks = false;
|
|
|
|
public bool IsUpdateRequested { get; private set; } = false;
|
|
|
|
public UpdatePollerStates ThreadState { get; private set; }
|
|
public double DownloadProgess
|
|
{
|
|
get { return m_downloadProgress; }
|
|
|
|
private set
|
|
{
|
|
var oldv = m_downloadProgress;
|
|
m_downloadProgress = value;
|
|
if ((int)(oldv * 100) != (int)(value * 100))
|
|
eventPollNotify.SignalNewEvent();
|
|
}
|
|
}
|
|
|
|
public void Init(bool disableChecks)
|
|
{
|
|
m_disableChecks = disableChecks;
|
|
m_waitSignal = new AutoResetEvent(false);
|
|
ThreadState = UpdatePollerStates.Waiting;
|
|
m_thread = new Thread(Run)
|
|
{
|
|
IsBackground = true,
|
|
Name = "UpdatePollThread"
|
|
};
|
|
m_thread.Start();
|
|
}
|
|
|
|
public void CheckNow()
|
|
{
|
|
lock (m_lock)
|
|
{
|
|
m_forceCheck = true;
|
|
m_waitSignal.Set();
|
|
}
|
|
}
|
|
|
|
public void Terminate()
|
|
{
|
|
lock (m_lock)
|
|
{
|
|
m_terminated = true;
|
|
m_waitSignal.Set();
|
|
}
|
|
}
|
|
|
|
public void Reschedule()
|
|
{
|
|
m_waitSignal.Set();
|
|
}
|
|
|
|
private void Run()
|
|
{
|
|
// Wait on startup
|
|
m_waitSignal.WaitOne(TimeSpan.FromMinutes(1), true);
|
|
|
|
while (!m_terminated)
|
|
{
|
|
var nextCheck = connection.ApplicationSettings.NextUpdateCheck;
|
|
|
|
var maxcheck = TimeSpan.FromDays(7);
|
|
try
|
|
{
|
|
maxcheck = Library.Utility.Timeparser.ParseTimeSpan(connection.ApplicationSettings.UpdateCheckInterval);
|
|
}
|
|
catch
|
|
{
|
|
}
|
|
|
|
// If we have some weirdness, just check now
|
|
if (nextCheck - DateTime.UtcNow > maxcheck)
|
|
nextCheck = DateTime.UtcNow - TimeSpan.FromSeconds(1);
|
|
|
|
if (nextCheck < DateTime.UtcNow || m_forceCheck)
|
|
{
|
|
DateTime started = DateTime.UtcNow;
|
|
connection.ApplicationSettings.LastUpdateCheck = started;
|
|
nextCheck = connection.ApplicationSettings.NextUpdateCheck;
|
|
|
|
// If this is not forced, and we have disabled checks, we just update the next check time
|
|
if (m_disableChecks && !m_forceCheck)
|
|
continue;
|
|
|
|
lock (m_lock)
|
|
m_forceCheck = false;
|
|
|
|
ThreadState = UpdatePollerStates.Checking;
|
|
eventPollNotify.SignalNewEvent();
|
|
|
|
if (!Enum.TryParse<ReleaseType>(connection.ApplicationSettings.UpdateChannel, true, out var rt))
|
|
rt = ReleaseType.Unknown;
|
|
|
|
// Choose the default channel in case we have unknown
|
|
rt = rt == ReleaseType.Unknown ? AutoUpdateSettings.DefaultUpdateChannel : rt;
|
|
|
|
try
|
|
{
|
|
var update = UpdaterManager.CheckForUpdate(rt);
|
|
if (update != null)
|
|
connection.ApplicationSettings.UpdatedVersion = update;
|
|
}
|
|
catch
|
|
{
|
|
}
|
|
|
|
// It could be that we have registered an update from a more unstable channel,
|
|
// but the user has switched to a more stable channel.
|
|
// In that case we discard the old update to avoid offering it.
|
|
if (connection.ApplicationSettings.UpdatedVersion != null)
|
|
{
|
|
Library.AutoUpdater.ReleaseType updatert;
|
|
var updatertstring = connection.ApplicationSettings.UpdatedVersion.ReleaseType;
|
|
if (string.Equals(updatertstring, "preview", StringComparison.OrdinalIgnoreCase))
|
|
updatertstring = Library.AutoUpdater.ReleaseType.Experimental.ToString();
|
|
|
|
if (!Enum.TryParse<Library.AutoUpdater.ReleaseType>(updatertstring, true, out updatert))
|
|
updatert = Duplicati.Library.AutoUpdater.ReleaseType.Nightly;
|
|
|
|
if (updatert == Duplicati.Library.AutoUpdater.ReleaseType.Unknown)
|
|
updatert = Duplicati.Library.AutoUpdater.ReleaseType.Nightly;
|
|
|
|
if (updatert > rt)
|
|
connection.ApplicationSettings.UpdatedVersion = null;
|
|
}
|
|
|
|
// If the update is the same or older than the current version, we discard it
|
|
// NOTE: This check is also inside the UpdaterManager.CheckForUpdate method,
|
|
// but we may have a stale version recorded, so we force-clear it here
|
|
if (connection.ApplicationSettings.UpdatedVersion != null)
|
|
{
|
|
if (UpdaterManager.TryParseVersion(connection.ApplicationSettings.UpdatedVersion.Version) <= UpdaterManager.TryParseVersion(UpdaterManager.SelfVersion.Version))
|
|
connection.ApplicationSettings.UpdatedVersion = null;
|
|
}
|
|
|
|
var updatedinfo = connection.ApplicationSettings.UpdatedVersion;
|
|
if (updatedinfo != null && UpdaterManager.TryParseVersion(updatedinfo.Version) > UpdaterManager.TryParseVersion(UpdaterManager.SelfVersion.Version))
|
|
{
|
|
var package = updatedinfo.FindPackage();
|
|
|
|
connection.RegisterNotification(
|
|
NotificationType.Information,
|
|
"Found update",
|
|
updatedinfo.Displayname,
|
|
null,
|
|
null,
|
|
"update:new",
|
|
null,
|
|
"NewUpdateFound",
|
|
null,
|
|
(self, all) =>
|
|
{
|
|
return all.FirstOrDefault(x => x.Action == "update:new") ?? self;
|
|
}
|
|
);
|
|
}
|
|
}
|
|
|
|
DownloadProgess = 0;
|
|
|
|
if (ThreadState != UpdatePollerStates.Waiting)
|
|
{
|
|
ThreadState = UpdatePollerStates.Waiting;
|
|
eventPollNotify.SignalNewEvent();
|
|
}
|
|
|
|
var waitTime = nextCheck - DateTime.UtcNow;
|
|
|
|
// Guard against spin-loop
|
|
if (waitTime.TotalSeconds < 5)
|
|
waitTime = TimeSpan.FromSeconds(5);
|
|
|
|
// Guard against year-long waits
|
|
// A re-check does not cause an update check
|
|
if (waitTime.TotalDays > 1)
|
|
waitTime = TimeSpan.FromDays(1);
|
|
|
|
m_waitSignal.WaitOne(waitTime, true);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|