RAGECOOP-V/Server/DownloadManager.cs

271 lines
8.7 KiB
C#
Raw Normal View History

2022-04-03 18:46:57 +02:00
using System.IO;
using System.Linq;
using System.Collections.Generic;
2022-04-03 02:27:30 +02:00
using Lidgren.Network;
namespace CoopServer
{
2022-04-02 23:02:49 +02:00
internal static class DownloadManager
{
private static readonly List<long> _clientsToDelete = new();
private static List<DownloadClient> _clients = new();
2022-04-02 23:02:49 +02:00
private static readonly List<DownloadFile> _files = new();
public static bool AnyFileExists = false;
2022-04-02 23:02:49 +02:00
public static void InsertClient(long nethandle)
{
lock (_clients)
{
_clients.Add(new DownloadClient(nethandle, new(_files)));
}
2022-04-02 23:02:49 +02:00
}
public static bool CheckForDirectoryAndFiles()
{
2022-04-03 02:27:30 +02:00
if (!Directory.Exists("clientside"))
{
return false;
}
2022-04-08 22:30:34 +02:00
string[] filePaths = Directory.GetFiles("clientside");
2022-04-03 02:27:30 +02:00
if (filePaths.Length == 0)
2022-04-02 23:02:49 +02:00
{
return false;
}
2022-04-03 02:27:30 +02:00
byte fileCount = 0;
2022-04-02 23:02:49 +02:00
foreach (string file in filePaths)
{
FileInfo fileInfo = new(file);
// ONLY JAVASCRIPT AND JSON FILES!
if (!new string[] { ".js", ".xml" }.Any(x => x == fileInfo.Extension))
{
Logging.Warning("Only files with \"*.js\" and \"*.xml\" can be sent!");
continue;
}
const int MAX_BUFFER = 5120; // 5KB
byte[] buffer = new byte[MAX_BUFFER];
ushort bytesRead = 0;
bool fileCreated = false;
DownloadFile newFile = null;
using (FileStream fs = File.Open(file, FileMode.Open, FileAccess.Read))
using (BufferedStream bs = new(fs))
{
while ((bytesRead = (ushort)bs.Read(buffer, 0, MAX_BUFFER)) != 0) // Reading 5KB chunks at time
{
2022-04-03 18:46:57 +02:00
if (!fileCreated && (fileCreated = true))
{
newFile = new()
{
FileID = fileCount,
FileName = fileInfo.Name,
FileType = fileInfo.Extension == ".json" ? Packets.DataFileType.Script : Packets.DataFileType.Map,
FileLength = fileInfo.Length,
FileChunks = new()
};
}
newFile.FileChunks.Add(buffer.Take(bytesRead).ToArray());
}
}
2022-04-02 23:02:49 +02:00
_files.Add(newFile);
2022-04-03 02:27:30 +02:00
fileCount++;
}
2022-04-02 23:02:49 +02:00
AnyFileExists = true;
return true;
}
public static void Tick()
{
lock (_clients)
{
lock (_clientsToDelete)
{
foreach (long nethandle in _clientsToDelete)
{
DownloadClient client = _clients.FirstOrDefault(x => x.NetHandle == nethandle);
if (client != null)
{
2022-04-06 06:52:02 +02:00
client.Finish();
_clients.Remove(client);
}
}
_clientsToDelete.Clear();
}
_clients.ForEach(client =>
2022-04-02 23:02:49 +02:00
{
if (client.SendFiles())
2022-04-02 23:02:49 +02:00
{
lock (Server.Clients)
{
Client x = Server.Clients.FirstOrDefault(x => x.NetHandle == client.NetHandle);
if (x != null)
{
x.FilesReceived = true;
}
}
AddClientToRemove(client.NetHandle);
}
});
}
}
2022-04-02 23:02:49 +02:00
public static void RemoveClient(long nethandle)
{
2022-04-06 06:52:02 +02:00
lock (_clients)
2022-04-02 23:02:49 +02:00
{
2022-04-06 06:52:02 +02:00
DownloadClient client = _clients.FirstOrDefault(x => x.NetHandle == nethandle);
if (client != null)
{
_clients.Remove(client);
}
2022-04-02 23:02:49 +02:00
}
}
2022-04-03 02:27:30 +02:00
/// <summary>
/// We try to remove the client when all files have been sent
/// </summary>
/// <param name="nethandle"></param>
2022-04-06 06:52:02 +02:00
/// <param name="id">Not currently used but maybe we can need this sometime</param>
2022-04-03 02:27:30 +02:00
public static void TryToRemoveClient(long nethandle, int id)
{
lock (_clients)
2022-04-03 02:27:30 +02:00
{
DownloadClient client = _clients.FirstOrDefault(x => x.NetHandle == nethandle);
if (client == null)
{
return;
}
2022-04-03 02:27:30 +02:00
client.FilePosition++;
2022-04-03 02:27:30 +02:00
if (client.DownloadComplete())
{
2022-04-06 06:52:02 +02:00
client.Finish();
_clients.Remove(client);
}
}
}
public static void AddClientToRemove(long nethandle)
{
lock (_clientsToDelete)
2022-04-03 02:27:30 +02:00
{
_clientsToDelete.Add(nethandle);
2022-04-03 02:27:30 +02:00
}
}
2022-04-02 23:02:49 +02:00
}
internal class DownloadClient
{
2022-04-03 02:27:30 +02:00
public long NetHandle = 0;
2022-04-03 18:46:57 +02:00
private readonly List<DownloadFile> _files = null;
2022-04-03 02:27:30 +02:00
public int FilePosition = 0;
2022-04-03 18:46:57 +02:00
private int _fileDataPosition = 0;
2022-04-03 02:27:30 +02:00
public DownloadClient(long nethandle, List<DownloadFile> files)
{
NetHandle = nethandle;
2022-04-03 18:46:57 +02:00
_files = files;
2022-04-03 02:27:30 +02:00
NetConnection conn = Server.MainNetServer.Connections.FirstOrDefault(x => x.RemoteUniqueIdentifier == NetHandle);
if (conn == null)
2022-04-03 02:27:30 +02:00
{
DownloadManager.AddClientToRemove(NetHandle);
return;
2022-04-03 02:27:30 +02:00
}
_files.ForEach(file =>
{
NetOutgoingMessage outgoingMessage = Server.MainNetServer.CreateMessage();
new Packets.FileTransferRequest()
{
ID = file.FileID,
FileType = (byte)file.FileType,
FileName = file.FileName,
FileLength = file.FileLength
}.PacketToNetOutGoingMessage(outgoingMessage);
Server.MainNetServer.SendMessage(outgoingMessage, conn, NetDeliveryMethod.ReliableOrdered, (byte)ConnectionChannel.File);
});
2022-04-03 02:27:30 +02:00
}
2022-04-02 23:02:49 +02:00
2022-04-06 06:52:02 +02:00
public void Finish()
{
NetConnection conn = Server.MainNetServer.Connections.FirstOrDefault(x => x.RemoteUniqueIdentifier == NetHandle);
if (conn == null)
{
return;
}
NetOutgoingMessage outgoingMessage = Server.MainNetServer.CreateMessage();
new Packets.FileTransferComplete()
{
ID = 0x0
}.PacketToNetOutGoingMessage(outgoingMessage);
Server.MainNetServer.SendMessage(outgoingMessage, conn, NetDeliveryMethod.ReliableOrdered, (byte)ConnectionChannel.File);
}
2022-04-02 23:02:49 +02:00
/// <summary>
///
/// </summary>
/// <returns>true if we are done otherwise false</returns>
2022-04-03 02:27:30 +02:00
public bool SendFiles()
2022-04-02 23:02:49 +02:00
{
2022-04-03 18:46:57 +02:00
DownloadFile file = _files[FilePosition];
2022-04-02 23:02:49 +02:00
2022-04-03 18:46:57 +02:00
Send(NetHandle, file);
2022-04-02 23:02:49 +02:00
if (_fileDataPosition >= file.FileChunks.Count)
2022-04-02 23:02:49 +02:00
{
2022-04-03 02:27:30 +02:00
FilePosition++;
2022-04-03 18:46:57 +02:00
_fileDataPosition = 0;
2022-04-02 23:02:49 +02:00
}
return DownloadComplete();
}
2022-04-03 02:27:30 +02:00
2022-04-03 18:46:57 +02:00
private void Send(long nethandle, DownloadFile file)
{
2022-04-03 02:27:30 +02:00
NetConnection conn = Server.MainNetServer.Connections.FirstOrDefault(x => x.RemoteUniqueIdentifier == nethandle);
if (conn == null)
{
DownloadManager.AddClientToRemove(NetHandle);
2022-04-03 02:27:30 +02:00
return;
}
NetOutgoingMessage outgoingMessage = Server.MainNetServer.CreateMessage();
new Packets.FileTransferTick() { ID = file.FileID, FileChunk = file.FileChunks[_fileDataPosition++] }.PacketToNetOutGoingMessage(outgoingMessage);
2022-04-03 02:27:30 +02:00
2022-04-06 06:52:02 +02:00
Server.MainNetServer.SendMessage(outgoingMessage, conn, NetDeliveryMethod.ReliableUnordered, (byte)ConnectionChannel.File);
}
2022-04-03 18:46:57 +02:00
public bool DownloadComplete()
{
2022-04-03 18:46:57 +02:00
return FilePosition >= _files.Count;
}
}
2022-04-03 18:46:57 +02:00
internal class DownloadFile
{
public byte FileID { get; set; } = 0;
public string FileName { get; set; } = string.Empty;
public Packets.DataFileType FileType { get; set; } = Packets.DataFileType.Script;
2022-04-03 18:46:57 +02:00
public long FileLength { get; set; } = 0;
public List<byte[]> FileChunks { get; set; } = null;
2022-04-03 18:46:57 +02:00
}
}