mirror of https://github.com/quasar/Quasar.git
578 lines
21 KiB
C#
578 lines
21 KiB
C#
using Quasar.Common.Enums;
|
|
using Quasar.Common.IO;
|
|
using Quasar.Common.Messages;
|
|
using Quasar.Common.Models;
|
|
using Quasar.Common.Networking;
|
|
using Quasar.Server.Enums;
|
|
using Quasar.Server.Models;
|
|
using Quasar.Server.Networking;
|
|
using System;
|
|
using System.Collections.Generic;
|
|
using System.IO;
|
|
using System.Linq;
|
|
using System.Threading;
|
|
|
|
namespace Quasar.Server.Messages
|
|
{
|
|
/// <summary>
|
|
/// Handles messages for the interaction with remote files and directories.
|
|
/// </summary>
|
|
public class FileManagerHandler : MessageProcessorBase<string>, IDisposable
|
|
{
|
|
/// <summary>
|
|
/// Represents the method that will handle drive changes.
|
|
/// </summary>
|
|
/// <param name="sender">The message processor which raised the event.</param>
|
|
/// <param name="drives">All currently available drives.</param>
|
|
public delegate void DrivesChangedEventHandler(object sender, Drive[] drives);
|
|
|
|
/// <summary>
|
|
/// Represents the method that will handle directory changes.
|
|
/// </summary>
|
|
/// <param name="sender">The message processor which raised the event.</param>
|
|
/// <param name="remotePath">The remote path of the directory.</param>
|
|
/// <param name="items">The directory content.</param>
|
|
public delegate void DirectoryChangedEventHandler(object sender, string remotePath, FileSystemEntry[] items);
|
|
|
|
/// <summary>
|
|
/// Represents the method that will handle file transfer updates.
|
|
/// </summary>
|
|
/// <param name="sender">The message processor which raised the event.</param>
|
|
/// <param name="transfer">The updated file transfer.</param>
|
|
public delegate void FileTransferUpdatedEventHandler(object sender, FileTransfer transfer);
|
|
|
|
/// <summary>
|
|
/// Raised when drives changed.
|
|
/// </summary>
|
|
/// <remarks>
|
|
/// Handlers registered with this event will be invoked on the
|
|
/// <see cref="System.Threading.SynchronizationContext"/> chosen when the instance was constructed.
|
|
/// </remarks>
|
|
public event DrivesChangedEventHandler DrivesChanged;
|
|
|
|
/// <summary>
|
|
/// Raised when a directory changed.
|
|
/// </summary>
|
|
/// <remarks>
|
|
/// Handlers registered with this event will be invoked on the
|
|
/// <see cref="System.Threading.SynchronizationContext"/> chosen when the instance was constructed.
|
|
/// </remarks>
|
|
public event DirectoryChangedEventHandler DirectoryChanged;
|
|
|
|
/// <summary>
|
|
/// Raised when a file transfer updated.
|
|
/// </summary>
|
|
/// <remarks>
|
|
/// Handlers registered with this event will be invoked on the
|
|
/// <see cref="System.Threading.SynchronizationContext"/> chosen when the instance was constructed.
|
|
/// </remarks>
|
|
public event FileTransferUpdatedEventHandler FileTransferUpdated;
|
|
|
|
/// <summary>
|
|
/// Reports changed remote drives.
|
|
/// </summary>
|
|
/// <param name="drives">The current remote drives.</param>
|
|
private void OnDrivesChanged(Drive[] drives)
|
|
{
|
|
SynchronizationContext.Post(d =>
|
|
{
|
|
var handler = DrivesChanged;
|
|
handler?.Invoke(this, (Drive[])d);
|
|
}, drives);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Reports a directory change.
|
|
/// </summary>
|
|
/// <param name="remotePath">The remote path of the directory.</param>
|
|
/// <param name="items">The directory content.</param>
|
|
private void OnDirectoryChanged(string remotePath, FileSystemEntry[] items)
|
|
{
|
|
SynchronizationContext.Post(i =>
|
|
{
|
|
var handler = DirectoryChanged;
|
|
handler?.Invoke(this, remotePath, (FileSystemEntry[])i);
|
|
}, items);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Reports updated file transfers.
|
|
/// </summary>
|
|
/// <param name="transfer">The updated file transfer.</param>
|
|
private void OnFileTransferUpdated(FileTransfer transfer)
|
|
{
|
|
SynchronizationContext.Post(t =>
|
|
{
|
|
var handler = FileTransferUpdated;
|
|
handler?.Invoke(this, (FileTransfer)t);
|
|
}, transfer.Clone());
|
|
}
|
|
|
|
/// <summary>
|
|
/// Keeps track of all active file transfers. Finished or canceled transfers get removed.
|
|
/// </summary>
|
|
private readonly List<FileTransfer> _activeFileTransfers = new List<FileTransfer>();
|
|
|
|
/// <summary>
|
|
/// Used in lock statements to synchronize access between UI thread and thread pool.
|
|
/// </summary>
|
|
private readonly object _syncLock = new object();
|
|
|
|
/// <summary>
|
|
/// The client which is associated with this file manager handler.
|
|
/// </summary>
|
|
private readonly Client _client;
|
|
|
|
/// <summary>
|
|
/// Used to only allow two simultaneous file uploads.
|
|
/// </summary>
|
|
private readonly Semaphore _limitThreads = new Semaphore(2, 2);
|
|
|
|
/// <summary>
|
|
/// Path to the base download directory of the client.
|
|
/// </summary>
|
|
private readonly string _baseDownloadPath;
|
|
|
|
private readonly TaskManagerHandler _taskManagerHandler;
|
|
|
|
/// <summary>
|
|
/// Initializes a new instance of the <see cref="FileManagerHandler"/> class using the given client.
|
|
/// </summary>
|
|
/// <param name="client">The associated client.</param>
|
|
/// <param name="subDirectory">Optional sub directory name.</param>
|
|
public FileManagerHandler(Client client, string subDirectory = "") : base(true)
|
|
{
|
|
_client = client;
|
|
_baseDownloadPath = Path.Combine(client.Value.DownloadDirectory, subDirectory);
|
|
_taskManagerHandler = new TaskManagerHandler(client);
|
|
_taskManagerHandler.ProcessActionPerformed += ProcessActionPerformed;
|
|
MessageHandler.Register(_taskManagerHandler);
|
|
}
|
|
|
|
/// <inheritdoc />
|
|
public override bool CanExecute(IMessage message) => message is FileTransferChunk ||
|
|
message is FileTransferCancel ||
|
|
message is FileTransferComplete ||
|
|
message is GetDrivesResponse ||
|
|
message is GetDirectoryResponse ||
|
|
message is SetStatusFileManager;
|
|
|
|
/// <inheritdoc />
|
|
public override bool CanExecuteFrom(ISender sender) => _client.Equals(sender);
|
|
|
|
/// <inheritdoc />
|
|
public override void Execute(ISender sender, IMessage message)
|
|
{
|
|
switch (message)
|
|
{
|
|
case FileTransferChunk file:
|
|
Execute(sender, file);
|
|
break;
|
|
case FileTransferCancel cancel:
|
|
Execute(sender, cancel);
|
|
break;
|
|
case FileTransferComplete complete:
|
|
Execute(sender, complete);
|
|
break;
|
|
case GetDrivesResponse drive:
|
|
Execute(sender, drive);
|
|
break;
|
|
case GetDirectoryResponse directory:
|
|
Execute(sender, directory);
|
|
break;
|
|
case SetStatusFileManager status:
|
|
Execute(sender, status);
|
|
break;
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Begins downloading a file from the client.
|
|
/// </summary>
|
|
/// <param name="remotePath">The remote path of the file to download.</param>
|
|
/// <param name="localFileName">The local file name.</param>
|
|
/// <param name="overwrite">Overwrite the local file with the newly downloaded.</param>
|
|
public void BeginDownloadFile(string remotePath, string localFileName = "", bool overwrite = false)
|
|
{
|
|
if (string.IsNullOrEmpty(remotePath))
|
|
return;
|
|
|
|
int id = GetUniqueFileTransferId();
|
|
|
|
if (!Directory.Exists(_baseDownloadPath))
|
|
Directory.CreateDirectory(_baseDownloadPath);
|
|
|
|
string fileName = string.IsNullOrEmpty(localFileName) ? Path.GetFileName(remotePath) : localFileName;
|
|
string localPath = Path.Combine(_baseDownloadPath, fileName);
|
|
|
|
int i = 1;
|
|
while (!overwrite && File.Exists(localPath))
|
|
{
|
|
// rename file if it exists already
|
|
var newFileName = string.Format("{0}({1}){2}", Path.GetFileNameWithoutExtension(localPath), i, Path.GetExtension(localPath));
|
|
localPath = Path.Combine(_baseDownloadPath, newFileName);
|
|
i++;
|
|
}
|
|
|
|
var transfer = new FileTransfer
|
|
{
|
|
Id = id,
|
|
Type = TransferType.Download,
|
|
LocalPath = localPath,
|
|
RemotePath = remotePath,
|
|
Status = "Pending...",
|
|
//Size = fileSize, TODO: Add file size here
|
|
TransferredSize = 0
|
|
};
|
|
|
|
try
|
|
{
|
|
transfer.FileSplit = new FileSplit(transfer.LocalPath, FileAccess.Write);
|
|
}
|
|
catch (Exception)
|
|
{
|
|
transfer.Status = "Error writing file";
|
|
OnFileTransferUpdated(transfer);
|
|
return;
|
|
}
|
|
|
|
lock (_syncLock)
|
|
{
|
|
_activeFileTransfers.Add(transfer);
|
|
}
|
|
|
|
OnFileTransferUpdated(transfer);
|
|
|
|
_client.Send(new FileTransferRequest {RemotePath = remotePath, Id = id});
|
|
}
|
|
|
|
/// <summary>
|
|
/// Begins uploading a file to the client.
|
|
/// </summary>
|
|
/// <param name="localPath">The local path of the file to upload.</param>
|
|
/// <param name="remotePath">Save the uploaded file to this remote path. If empty, generate a temporary file name.</param>
|
|
public void BeginUploadFile(string localPath, string remotePath = "")
|
|
{
|
|
new Thread(() =>
|
|
{
|
|
int id = GetUniqueFileTransferId();
|
|
|
|
FileTransfer transfer = new FileTransfer
|
|
{
|
|
Id = id,
|
|
Type = TransferType.Upload,
|
|
LocalPath = localPath,
|
|
RemotePath = remotePath,
|
|
Status = "Pending...",
|
|
TransferredSize = 0
|
|
};
|
|
|
|
try
|
|
{
|
|
transfer.FileSplit = new FileSplit(localPath, FileAccess.Read);
|
|
}
|
|
catch (Exception)
|
|
{
|
|
transfer.Status = "Error reading file";
|
|
OnFileTransferUpdated(transfer);
|
|
return;
|
|
}
|
|
|
|
transfer.Size = transfer.FileSplit.FileSize;
|
|
|
|
lock (_syncLock)
|
|
{
|
|
_activeFileTransfers.Add(transfer);
|
|
}
|
|
|
|
transfer.Size = transfer.FileSplit.FileSize;
|
|
OnFileTransferUpdated(transfer);
|
|
|
|
_limitThreads.WaitOne();
|
|
try
|
|
{
|
|
foreach (var chunk in transfer.FileSplit)
|
|
{
|
|
transfer.TransferredSize += chunk.Data.Length;
|
|
decimal progress = Math.Round((decimal) ((double) transfer.TransferredSize / (double) transfer.Size * 100.0), 2);
|
|
transfer.Status = $"Uploading...({progress}%)";
|
|
OnFileTransferUpdated(transfer);
|
|
|
|
bool transferCanceled;
|
|
lock (_syncLock)
|
|
{
|
|
transferCanceled = _activeFileTransfers.Count(f => f.Id == transfer.Id) == 0;
|
|
}
|
|
|
|
if (transferCanceled)
|
|
{
|
|
transfer.Status = "Canceled";
|
|
OnFileTransferUpdated(transfer);
|
|
_limitThreads.Release();
|
|
return;
|
|
}
|
|
|
|
// TODO: blocking sending might not be required, needs further testing
|
|
_client.SendBlocking(new FileTransferChunk
|
|
{
|
|
Id = id,
|
|
Chunk = chunk,
|
|
FilePath = remotePath,
|
|
FileSize = transfer.Size
|
|
});
|
|
}
|
|
}
|
|
catch (Exception)
|
|
{
|
|
lock (_syncLock)
|
|
{
|
|
// if transfer is already cancelled, just return
|
|
if (_activeFileTransfers.Count(f => f.Id == transfer.Id) == 0)
|
|
{
|
|
_limitThreads.Release();
|
|
return;
|
|
}
|
|
}
|
|
transfer.Status = "Error reading file";
|
|
OnFileTransferUpdated(transfer);
|
|
CancelFileTransfer(transfer.Id);
|
|
_limitThreads.Release();
|
|
return;
|
|
}
|
|
|
|
_limitThreads.Release();
|
|
}).Start();
|
|
}
|
|
|
|
/// <summary>
|
|
/// Cancels a file transfer.
|
|
/// </summary>
|
|
/// <param name="transferId">The id of the file transfer to cancel.</param>
|
|
public void CancelFileTransfer(int transferId)
|
|
{
|
|
_client.Send(new FileTransferCancel {Id = transferId});
|
|
}
|
|
|
|
/// <summary>
|
|
/// Renames a remote file or directory.
|
|
/// </summary>
|
|
/// <param name="remotePath">The remote file or directory path to rename.</param>
|
|
/// <param name="newPath">The new name of the remote file or directory path.</param>
|
|
/// <param name="type">The type of the file (file or directory).</param>
|
|
public void RenameFile(string remotePath, string newPath, FileType type)
|
|
{
|
|
_client.Send(new DoPathRename
|
|
{
|
|
Path = remotePath,
|
|
NewPath = newPath,
|
|
PathType = type
|
|
});
|
|
}
|
|
|
|
/// <summary>
|
|
/// Deletes a remote file or directory.
|
|
/// </summary>
|
|
/// <param name="remotePath">The remote file or directory path.</param>
|
|
/// <param name="type">The type of the file (file or directory).</param>
|
|
public void DeleteFile(string remotePath, FileType type)
|
|
{
|
|
_client.Send(new DoPathDelete {Path = remotePath, PathType = type});
|
|
}
|
|
|
|
/// <summary>
|
|
/// Starts a new process remotely.
|
|
/// </summary>
|
|
/// <param name="remotePath">The remote path used for starting the new process.</param>
|
|
public void StartProcess(string remotePath)
|
|
{
|
|
_taskManagerHandler.StartProcess(remotePath);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Adds an item to the startup of the client.
|
|
/// </summary>
|
|
/// <param name="item">The startup item to add.</param>
|
|
public void AddToStartup(StartupItem item)
|
|
{
|
|
_client.Send(new DoStartupItemAdd {StartupItem = item});
|
|
}
|
|
|
|
/// <summary>
|
|
/// Gets the directory contents for the remote path.
|
|
/// </summary>
|
|
/// <param name="remotePath">The remote path of the directory.</param>
|
|
public void GetDirectoryContents(string remotePath)
|
|
{
|
|
_client.Send(new GetDirectory {RemotePath = remotePath});
|
|
}
|
|
|
|
/// <summary>
|
|
/// Refreshes the remote drives.
|
|
/// </summary>
|
|
public void RefreshDrives()
|
|
{
|
|
_client.Send(new GetDrives());
|
|
}
|
|
|
|
private void Execute(ISender client, FileTransferChunk message)
|
|
{
|
|
FileTransfer transfer;
|
|
lock (_syncLock)
|
|
{
|
|
transfer = _activeFileTransfers.FirstOrDefault(t => t.Id == message.Id);
|
|
}
|
|
|
|
if (transfer == null)
|
|
return;
|
|
|
|
transfer.Size = message.FileSize;
|
|
transfer.TransferredSize += message.Chunk.Data.Length;
|
|
|
|
try
|
|
{
|
|
transfer.FileSplit.WriteChunk(message.Chunk);
|
|
}
|
|
catch (Exception)
|
|
{
|
|
transfer.Status = "Error writing file";
|
|
OnFileTransferUpdated(transfer);
|
|
CancelFileTransfer(transfer.Id);
|
|
return;
|
|
}
|
|
|
|
decimal progress = Math.Round((decimal) ((double) transfer.TransferredSize / (double) transfer.Size * 100.0), 2);
|
|
transfer.Status = $"Downloading...({progress}%)";
|
|
|
|
OnFileTransferUpdated(transfer);
|
|
}
|
|
|
|
private void Execute(ISender client, FileTransferCancel message)
|
|
{
|
|
FileTransfer transfer;
|
|
lock (_syncLock)
|
|
{
|
|
transfer = _activeFileTransfers.FirstOrDefault(t => t.Id == message.Id);
|
|
}
|
|
|
|
if (transfer != null)
|
|
{
|
|
transfer.Status = message.Reason;
|
|
OnFileTransferUpdated(transfer);
|
|
RemoveFileTransfer(transfer.Id);
|
|
// don't keep un-finished files
|
|
if (transfer.Type == TransferType.Download)
|
|
File.Delete(transfer.LocalPath);
|
|
}
|
|
}
|
|
|
|
private void Execute(ISender client, FileTransferComplete message)
|
|
{
|
|
FileTransfer transfer;
|
|
lock (_syncLock)
|
|
{
|
|
transfer = _activeFileTransfers.FirstOrDefault(t => t.Id == message.Id);
|
|
}
|
|
|
|
if (transfer != null)
|
|
{
|
|
transfer.RemotePath = message.FilePath; // required for temporary file names generated on the client
|
|
transfer.Status = "Completed";
|
|
RemoveFileTransfer(transfer.Id);
|
|
OnFileTransferUpdated(transfer);
|
|
}
|
|
}
|
|
|
|
private void Execute(ISender client, GetDrivesResponse message)
|
|
{
|
|
if (message.Drives?.Length == 0)
|
|
return;
|
|
|
|
OnDrivesChanged(message.Drives);
|
|
}
|
|
|
|
private void Execute(ISender client, GetDirectoryResponse message)
|
|
{
|
|
if (message.Items == null)
|
|
{
|
|
message.Items = new FileSystemEntry[0];
|
|
}
|
|
OnDirectoryChanged(message.RemotePath, message.Items);
|
|
}
|
|
|
|
private void Execute(ISender client, SetStatusFileManager message)
|
|
{
|
|
OnReport(message.Message);
|
|
}
|
|
|
|
private void ProcessActionPerformed(object sender, ProcessAction action, bool result)
|
|
{
|
|
if (action != ProcessAction.Start) return;
|
|
OnReport(result ? "Process started successfully" : "Process failed to start");
|
|
}
|
|
|
|
/// <summary>
|
|
/// Removes a file transfer given the transfer id.
|
|
/// </summary>
|
|
/// <param name="transferId">The file transfer id.</param>
|
|
private void RemoveFileTransfer(int transferId)
|
|
{
|
|
lock (_syncLock)
|
|
{
|
|
var transfer = _activeFileTransfers.FirstOrDefault(t => t.Id == transferId);
|
|
transfer?.FileSplit?.Dispose();
|
|
_activeFileTransfers.RemoveAll(s => s.Id == transferId);
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Generates a unique file transfer id.
|
|
/// </summary>
|
|
/// <returns>A unique file transfer id.</returns>
|
|
private int GetUniqueFileTransferId()
|
|
{
|
|
int id;
|
|
lock (_syncLock)
|
|
{
|
|
do
|
|
{
|
|
id = FileTransfer.GetRandomTransferId();
|
|
// generate new id until we have a unique one
|
|
} while (_activeFileTransfers.Any(f => f.Id == id));
|
|
}
|
|
|
|
return id;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Disposes all managed and unmanaged resources associated with this message processor.
|
|
/// </summary>
|
|
public void Dispose()
|
|
{
|
|
Dispose(true);
|
|
GC.SuppressFinalize(this);
|
|
}
|
|
|
|
protected virtual void Dispose(bool disposing)
|
|
{
|
|
if (disposing)
|
|
{
|
|
lock (_syncLock)
|
|
{
|
|
foreach (var transfer in _activeFileTransfers)
|
|
{
|
|
_client.Send(new FileTransferCancel {Id = transfer.Id});
|
|
transfer.FileSplit?.Dispose();
|
|
if (transfer.Type == TransferType.Download)
|
|
File.Delete(transfer.LocalPath);
|
|
}
|
|
|
|
_activeFileTransfers.Clear();
|
|
}
|
|
|
|
MessageHandler.Unregister(_taskManagerHandler);
|
|
_taskManagerHandler.ProcessActionPerformed -= ProcessActionPerformed;
|
|
}
|
|
}
|
|
}
|
|
}
|