mirror of
https://github.com/duplicati/duplicati.git
synced 2026-05-07 15:49:35 -04:00
294 lines
11 KiB
C#
294 lines
11 KiB
C#
// Copyright (C) 2024, 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 Duplicati.Library.RestAPI;
|
|
using System;
|
|
using System.Collections.Generic;
|
|
using System.IO;
|
|
using System.Linq;
|
|
using System.Text;
|
|
using Duplicati.Library.RestAPI;
|
|
|
|
namespace Duplicati.Server.WebServer.RESTMethods
|
|
{
|
|
public class CommandLine : IRESTMethodGET, IRESTMethodPOST
|
|
{
|
|
private static readonly string LOGTAG = Library.Logging.Log.LogTagFromType<CommandLine>();
|
|
|
|
private class LogWriter : System.IO.TextWriter
|
|
{
|
|
private readonly ActiveRun m_target;
|
|
private readonly StringBuilder m_sb = new StringBuilder();
|
|
private int m_newlinechars = 0;
|
|
|
|
public LogWriter(ActiveRun target)
|
|
{
|
|
m_target = target;
|
|
}
|
|
|
|
public override Encoding Encoding { get { return System.Text.Encoding.UTF8; } }
|
|
|
|
public override void Write(char value)
|
|
{
|
|
lock(m_target.Lock)
|
|
{
|
|
m_sb.Append(value);
|
|
if (NewLine[m_newlinechars] == value)
|
|
{
|
|
m_newlinechars++;
|
|
if (m_newlinechars == NewLine.Length)
|
|
WriteLine(string.Empty);
|
|
}
|
|
else
|
|
m_newlinechars = 0;
|
|
}
|
|
}
|
|
|
|
public override void WriteLine(string value)
|
|
{
|
|
value = value ?? string.Empty;
|
|
lock(m_target.Lock)
|
|
{
|
|
m_target.LastAccess = DateTime.Now;
|
|
|
|
//Avoid writing the log if it does not exist
|
|
if (m_target.IsLogDisposed)
|
|
{
|
|
FIXMEGlobal.LogHandler.WriteMessage(new Library.Logging.LogEntry("Attempted to write message after closing: {0}", new object[] { value }, Library.Logging.LogMessageType.Warning, LOGTAG, "CommandLineOutputAfterLogClosed", null));
|
|
return;
|
|
}
|
|
|
|
try
|
|
{
|
|
if (m_sb.Length != 0)
|
|
{
|
|
m_target.Log.Add(m_sb + value);
|
|
m_sb.Length = 0;
|
|
m_newlinechars = 0;
|
|
}
|
|
else
|
|
{
|
|
m_target.Log.Add(value);
|
|
}
|
|
}
|
|
catch (Exception ex)
|
|
{
|
|
// This can happen on a very unlucky race where IsLogDisposed is set right after the check
|
|
FIXMEGlobal.LogHandler.WriteMessage(new Library.Logging.LogEntry("Failed to forward commandline message: {0}", new object[] { value }, Library.Logging.LogMessageType.Warning, LOGTAG, "CommandLineOutputAfterLogClosed", ex));
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
private class ActiveRun
|
|
{
|
|
public readonly string ID = Guid.NewGuid().ToString();
|
|
public DateTime LastAccess = DateTime.Now;
|
|
public readonly Library.Utility.FileBackedStringList Log = new Library.Utility.FileBackedStringList();
|
|
public Runner.IRunnerData Task;
|
|
public LogWriter Writer;
|
|
public readonly object Lock = new object();
|
|
public bool Finished = false;
|
|
public bool Started = false;
|
|
public bool IsLogDisposed = false;
|
|
public System.Threading.Thread Thread;
|
|
}
|
|
|
|
private readonly Dictionary<string, ActiveRun> m_activeItems = new Dictionary<string, ActiveRun>();
|
|
private System.Threading.Tasks.Task m_cleanupTask;
|
|
|
|
public void POST(string key, RequestInfo info)
|
|
{
|
|
if (string.IsNullOrWhiteSpace(key))
|
|
{
|
|
string[] args;
|
|
var str = System.Threading.Tasks.Task.Run(async () =>
|
|
{
|
|
using (var sr = new System.IO.StreamReader(info.Request.Body, System.Text.Encoding.UTF8, true))
|
|
|
|
return await sr.ReadToEndAsync();
|
|
}).GetAwaiter().GetResult();
|
|
args = Newtonsoft.Json.JsonConvert.DeserializeObject<string[]>(str);
|
|
|
|
var k = new ActiveRun();
|
|
k.Writer = new LogWriter(k);
|
|
|
|
m_activeItems[k.ID] = k;
|
|
StartCleanupTask();
|
|
|
|
k.Task = Runner.CreateCustomTask((sink) =>
|
|
{
|
|
try
|
|
{
|
|
k.Thread = System.Threading.Thread.CurrentThread;
|
|
k.Started = true;
|
|
|
|
var code = Duplicati.CommandLine.Program.RunCommandLine(k.Writer, k.Writer, c => {
|
|
k.Task.SetController(c);
|
|
c.AppendSink(sink);
|
|
}, args);
|
|
k.Writer.WriteLine("Return code: {0}", code);
|
|
}
|
|
catch (Exception ex)
|
|
{
|
|
var rx = ex;
|
|
if (rx is System.Reflection.TargetInvocationException)
|
|
rx = rx.InnerException;
|
|
|
|
if (rx is Library.Interface.UserInformationException)
|
|
k.Log.Add(rx.Message);
|
|
else
|
|
k.Log.Add(rx.ToString());
|
|
|
|
throw rx;
|
|
}
|
|
finally
|
|
{
|
|
k.Finished = true;
|
|
k.Thread = null;
|
|
}
|
|
});
|
|
|
|
FIXMEGlobal.WorkThread.AddTask(k.Task);
|
|
|
|
info.OutputOK(new
|
|
{
|
|
ID = k.ID
|
|
});
|
|
}
|
|
else
|
|
{
|
|
if (!key.EndsWith("/abort", StringComparison.OrdinalIgnoreCase))
|
|
{
|
|
info.ReportClientError("Only abort commands are allowed", System.Net.HttpStatusCode.BadRequest);
|
|
return;
|
|
}
|
|
|
|
key = key.Substring(0, key.Length - "/abort".Length);
|
|
if (string.IsNullOrWhiteSpace(key))
|
|
{
|
|
info.ReportClientError("No task key found", System.Net.HttpStatusCode.BadRequest);
|
|
return;
|
|
}
|
|
|
|
ActiveRun t;
|
|
if (!m_activeItems.TryGetValue(key, out t))
|
|
{
|
|
info.OutputError(code: System.Net.HttpStatusCode.NotFound);
|
|
return;
|
|
}
|
|
|
|
var tt = t.Task;
|
|
if (tt != null)
|
|
tt.Abort();
|
|
|
|
var tr = t.Thread;
|
|
if (tr != null)
|
|
tr.Interrupt();
|
|
|
|
info.OutputOK();
|
|
}
|
|
}
|
|
|
|
private void StartCleanupTask()
|
|
{
|
|
if (m_cleanupTask == null || m_cleanupTask.IsCompleted || m_cleanupTask.IsFaulted || m_cleanupTask.IsCanceled)
|
|
m_cleanupTask = RunCleanupAsync();
|
|
}
|
|
|
|
private async System.Threading.Tasks.Task RunCleanupAsync()
|
|
{
|
|
while (m_activeItems.Count > 0)
|
|
{
|
|
var oldest = m_activeItems.Values
|
|
.OrderBy(x => x.LastAccess)
|
|
.FirstOrDefault();
|
|
|
|
if (oldest != null)
|
|
{
|
|
// If the task has finished, we just wait a little to allow the UI to pick it up
|
|
var timeout = oldest.Finished ? TimeSpan.FromMinutes(5) : TimeSpan.FromDays(1);
|
|
if (DateTime.Now - oldest.LastAccess > timeout)
|
|
{
|
|
oldest.IsLogDisposed = true;
|
|
m_activeItems.Remove(oldest.ID);
|
|
oldest.Log.Dispose();
|
|
|
|
// Fix all expired, or stop running
|
|
continue;
|
|
}
|
|
}
|
|
|
|
await System.Threading.Tasks.Task.Delay(TimeSpan.FromMinutes(1)).ConfigureAwait(false);
|
|
}
|
|
}
|
|
|
|
public void GET(string key, RequestInfo info)
|
|
{
|
|
if (string.IsNullOrWhiteSpace(key))
|
|
{
|
|
info.OutputOK(
|
|
Duplicati.CommandLine.Program.SupportedCommands
|
|
);
|
|
}
|
|
else
|
|
{
|
|
ActiveRun t;
|
|
if (!m_activeItems.TryGetValue(key, out t))
|
|
{
|
|
info.OutputError(code: System.Net.HttpStatusCode.NotFound);
|
|
return;
|
|
}
|
|
|
|
int pagesize;
|
|
int offset;
|
|
|
|
int.TryParse(info.Request.QueryString["pagesize"].Value, out pagesize);
|
|
int.TryParse(info.Request.QueryString["offset"].Value, out offset);
|
|
pagesize = Math.Max(10, Math.Min(500, pagesize));
|
|
offset = Math.Max(0, offset);
|
|
var items = new List<string>();
|
|
long count;
|
|
bool started;
|
|
bool finished;
|
|
|
|
lock(t.Lock)
|
|
{
|
|
t.LastAccess = DateTime.Now;
|
|
count = t.Log.Count;
|
|
offset = Math.Min((int)count, offset);
|
|
items.AddRange(t.Log.Skip(offset).Take(pagesize));
|
|
finished = t.Finished;
|
|
started = t.Started;
|
|
}
|
|
|
|
info.OutputOK(new
|
|
{
|
|
Pagesize = pagesize,
|
|
Offset = offset,
|
|
Count = count,
|
|
Items = items,
|
|
Finished = finished,
|
|
Started = started
|
|
});
|
|
}
|
|
}
|
|
}
|
|
}
|