duplicati/Duplicati/Library/RestAPI/LogWriteHandler.cs
Kenneth Skovhede ef631a1199 Added feature to suppress warnings
This PR adds the option to suppress warnings by their log id. Warning IDs that are supplied to `--suppress-warnings` will be converted to information messages before being logged.

If a specific warning is disabled, such as `CompressionReadErrorFallback`, this will then no longer count as a warning for the job, and the log file will see the warning as an information message.

This PR also adds two simpler filter options that makes it possible to filter log messages by supplying the log IDs. This can already be achieved with log filters, but the ID filter is a bit simpler to apply, as you only need to know the ID.

Finally, this PR also adds common logging for errors in the categories:
- Permission denied, id = `PermissionDenied`
- File locked, id = `FileLocked`
- Path not found, id = `PathNotFound`
- Path too long, id = `PathTooLong`

These new log ids makes it simpler to ignore warnings about locked or inaccesible files.
2025-05-20 14:26:10 +02:00

260 lines
8.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;
using System.Linq;
using System.Collections.Generic;
using Duplicati.Library.Main;
using Duplicati.WebserverCore.Abstractions;
using Duplicati.Library.Logging;
namespace Duplicati.Server
{
/// <summary>
/// Class that handles logging from the server,
/// and provides an entry point for the runner
/// to redirect log output to a file
/// </summary>
public class LogWriteHandler : ILogWriteHandler
{
/// <summary>
/// The number of messages to keep when inactive
/// </summary>
private const int INACTIVE_SIZE = 30;
/// <summary>
/// The number of messages to keep when active
/// </summary>
private const int ACTIVE_SIZE = 5000;
/// <summary>
/// Basic implementation of a ring-buffer
/// </summary>
private class RingBuffer<T> : IEnumerable<T>
{
private readonly T[] m_buffer;
private int m_head;
private int m_tail;
private int m_length;
private int m_key;
private readonly object m_lock = new object();
public RingBuffer(int size, IEnumerable<T> initial = null)
{
m_buffer = new T[size];
if (initial != null)
foreach (var t in initial)
this.Enqueue(t);
}
public int Length { get { return m_length; } }
public void Enqueue(T item)
{
lock (m_lock)
{
m_key++;
m_buffer[m_head] = item;
m_head = (m_head + 1) % m_buffer.Length;
if (m_length == m_buffer.Length)
m_tail = (m_tail + 1) % m_buffer.Length;
else
m_length++;
}
}
#region IEnumerable implementation
public IEnumerator<T> GetEnumerator()
{
var k = m_key;
for (var i = 0; i < m_length; i++)
if (m_key != k)
throw new InvalidOperationException("Buffer was modified while reading");
else
yield return m_buffer[(m_tail + i) % m_buffer.Length];
}
#endregion
#region IEnumerable implementation
System.Collections.IEnumerator System.Collections.IEnumerable.GetEnumerator()
{
return GetEnumerator();
}
#endregion
public T[] FlatArray(Func<T, bool> filter = null)
{
lock (m_lock)
if (filter == null)
return this.ToArray();
else
return this.Where(filter).ToArray();
}
public int Size { get { return m_buffer.Length; } }
}
private readonly DateTime[] m_timeouts;
private readonly object m_lock = new object();
private volatile bool m_anytimeouts = false;
private RingBuffer<ILogWriteHandler.LiveLogEntry> m_buffer;
private readonly ControllerMultiLogTarget m_target = new ControllerMultiLogTarget(null, LogMessageType.Warning, null, null);
private LogMessageType m_logLevel;
public LogWriteHandler()
{
var fields = Enum.GetValues(typeof(LogMessageType));
m_timeouts = new DateTime[fields.Length];
m_buffer = new RingBuffer<ILogWriteHandler.LiveLogEntry>(INACTIVE_SIZE);
}
public void RenewTimeout(LogMessageType type)
{
lock (m_lock)
{
m_timeouts[(int)type] = DateTime.Now.AddSeconds(30);
m_anytimeouts = true;
if (m_buffer == null || m_buffer.Size == INACTIVE_SIZE)
m_buffer = new RingBuffer<ILogWriteHandler.LiveLogEntry>(ACTIVE_SIZE, m_buffer);
}
}
public void SetServerFile(string path, LogMessageType level)
{
var dir = System.IO.Path.GetDirectoryName(System.IO.Path.GetFullPath(path));
if (!System.IO.Directory.Exists(dir))
System.IO.Directory.CreateDirectory(dir);
m_target.AddTarget(new StreamLogDestination(path), level, null);
UpdateLogLevel();
}
public void AppendLogDestination(ILogDestination destination, LogMessageType level)
{
m_target.AddTarget(destination, level, null);
UpdateLogLevel();
}
public ILogWriteHandler.LiveLogEntry[] AfterTime(DateTime offset, LogMessageType level)
{
RenewTimeout(level);
UpdateLogLevel();
offset = offset.ToUniversalTime();
lock (m_lock)
{
if (m_buffer == null)
return new ILogWriteHandler.LiveLogEntry[0];
return m_buffer.FlatArray((x) => x.When > offset && x.Type >= level);
}
}
public ILogWriteHandler.LiveLogEntry[] AfterID(long id, LogMessageType level, int pagesize)
{
RenewTimeout(level);
UpdateLogLevel();
lock (m_lock)
{
if (m_buffer == null)
return [];
var buffer = m_buffer.FlatArray((x) => x.ID > id && x.Type >= level);
// Return the <page_size> newest entries
if (buffer.Length > pagesize)
{
var index = buffer.Length - pagesize;
return buffer.Skip(index).Take(pagesize).ToArray();
}
else
{
return buffer;
}
}
}
private int[] GetActiveTimeouts()
{
var i = 0;
return (from n in m_timeouts
let ix = i++
where n > DateTime.Now
select ix).ToArray();
}
private void UpdateLogLevel()
{
m_logLevel =
(LogMessageType)GetActiveTimeouts().Append((int)m_target.MinimumLevel).Min();
}
#region ILog implementation
public void WriteMessage(LogEntry entry)
{
if (entry.Level < m_logLevel)
return;
try
{
m_target.WriteMessage(entry);
}
catch
{
}
lock (m_lock)
{
if (m_anytimeouts)
{
var q = GetActiveTimeouts();
if (q.Length == 0)
{
UpdateLogLevel();
m_anytimeouts = false;
if (m_buffer == null || m_buffer.Size != INACTIVE_SIZE)
m_buffer = new RingBuffer<ILogWriteHandler.LiveLogEntry>(INACTIVE_SIZE, m_buffer);
}
}
if (m_buffer != null)
m_buffer.Enqueue(new ILogWriteHandler.LiveLogEntry(entry));
}
}
#endregion
#region IDisposable implementation
public void Dispose()
{
m_target.Dispose();
}
#endregion
}
}