MosswartMassacre/MosswartMassacre/UpdateManager.cs

253 lines
No EOL
9.8 KiB
C#

using System;
using System.IO;
using System.Net.Http;
using System.Security.Cryptography;
using System.Threading.Tasks;
namespace MosswartMassacre
{
public static class UpdateManager
{
private const string UPDATE_URL = "https://git.snakedesert.se/SawatoMosswartsEnjoyersClub/MosswartMassacre/raw/branch/master/MosswartMassacre/bin/Release/MosswartMassacre.dll";
private static bool updateAvailable = false;
private static string remoteFileHash = string.Empty;
private static string localFileHash = string.Empty;
private static DateTime lastCheckTime = DateTime.MinValue;
public static bool IsUpdateAvailable => updateAvailable;
public static DateTime LastCheckTime => lastCheckTime;
/// <summary>
/// Calculate SHA256 hash of a file
/// </summary>
private static string CalculateFileHash(string filePath)
{
using (var sha256 = SHA256.Create())
{
using (var stream = File.OpenRead(filePath))
{
byte[] hashBytes = sha256.ComputeHash(stream);
return BitConverter.ToString(hashBytes).Replace("-", "").ToLowerInvariant();
}
}
}
/// <summary>
/// Calculate SHA256 hash of byte array
/// </summary>
private static string CalculateHash(byte[] data)
{
using (var sha256 = SHA256.Create())
{
byte[] hashBytes = sha256.ComputeHash(data);
return BitConverter.ToString(hashBytes).Replace("-", "").ToLowerInvariant();
}
}
public static async Task<bool> CheckForUpdateAsync()
{
try
{
PluginCore.WriteToChat("[Update] Checking for updates...");
// Get local file hash
string localPath = GetLocalDllPath();
if (!File.Exists(localPath))
{
PluginCore.WriteToChat("[Update] Error: Could not find local DLL file");
return false;
}
PluginCore.WriteToChat("[Update] Calculating local file hash...");
localFileHash = CalculateFileHash(localPath);
// Download remote file and calculate hash
using (var client = new HttpClient())
{
client.Timeout = TimeSpan.FromSeconds(30);
PluginCore.WriteToChat("[Update] Downloading remote file for comparison...");
var remoteData = await client.GetByteArrayAsync(UPDATE_URL);
if (remoteData == null || remoteData.Length == 0)
{
PluginCore.WriteToChat("[Update] Error: Could not download remote file");
return false;
}
PluginCore.WriteToChat("[Update] Calculating remote file hash...");
remoteFileHash = CalculateHash(remoteData);
}
// Compare hashes
updateAvailable = !string.Equals(localFileHash, remoteFileHash, StringComparison.OrdinalIgnoreCase);
lastCheckTime = DateTime.Now;
if (updateAvailable)
{
PluginCore.WriteToChat($"[Update] Update available!");
PluginCore.WriteToChat($"[Update] Local hash: {localFileHash}");
PluginCore.WriteToChat($"[Update] Remote hash: {remoteFileHash}");
}
else
{
PluginCore.WriteToChat("[Update] Up to date - hashes match");
}
return true;
}
catch (HttpRequestException ex)
{
PluginCore.WriteToChat($"[Update] Network error: {ex.Message}");
return false;
}
catch (TaskCanceledException)
{
PluginCore.WriteToChat("[Update] Request timed out");
return false;
}
catch (Exception ex)
{
PluginCore.WriteToChat($"[Update] Check failed: {ex.Message}");
return false;
}
}
public static async Task<bool> DownloadAndInstallUpdateAsync()
{
if (!updateAvailable)
{
PluginCore.WriteToChat("[Update] No update available. Run /mm checkforupdate first.");
return false;
}
try
{
PluginCore.WriteToChat("[Update] Downloading update...");
string localPath = GetLocalDllPath();
string tempPath = localPath + ".tmp";
string backupPath = localPath + ".bak";
// Download to temp file
using (var client = new HttpClient())
{
client.Timeout = TimeSpan.FromSeconds(30);
var response = await client.GetAsync(UPDATE_URL);
response.EnsureSuccessStatusCode();
using (var fileStream = File.Create(tempPath))
{
await response.Content.CopyToAsync(fileStream);
}
}
// Validate downloaded file by hash
PluginCore.WriteToChat("[Update] Validating downloaded file...");
var downloadedHash = CalculateFileHash(tempPath);
if (!string.Equals(downloadedHash, remoteFileHash, StringComparison.OrdinalIgnoreCase))
{
File.Delete(tempPath);
PluginCore.WriteToChat($"[Update] Download validation failed. Hash mismatch!");
PluginCore.WriteToChat($"[Update] Expected: {remoteFileHash}");
PluginCore.WriteToChat($"[Update] Got: {downloadedHash}");
return false;
}
PluginCore.WriteToChat("[Update] Download complete, installing...");
// Atomically replace current file with new version (creates backup automatically)
File.Replace(tempPath, localPath, backupPath);
// Clear update flag
updateAvailable = false;
PluginCore.WriteToChat("[Update] Update installed successfully!");
PluginCore.WriteToChat("[Update] Previous version backed up as MosswartMassacre.dll.bak");
// Wait a moment for file system to settle, then trigger hot reload
await System.Threading.Tasks.Task.Delay(1000);
try
{
// Touch the file to ensure FileSystemWatcher detects the change
File.SetLastWriteTime(localPath, DateTime.Now);
PluginCore.WriteToChat("[Update] Triggering hot reload...");
}
catch (Exception ex)
{
PluginCore.WriteToChat($"[Update] Could not trigger hot reload: {ex.Message}");
PluginCore.WriteToChat("[Update] Please use /mm gui to reload manually");
}
return true;
}
catch (HttpRequestException ex)
{
PluginCore.WriteToChat($"[Update] Download error: {ex.Message}");
return false;
}
catch (TaskCanceledException)
{
PluginCore.WriteToChat("[Update] Download timed out");
return false;
}
catch (UnauthorizedAccessException)
{
PluginCore.WriteToChat("[Update] File access denied. Make sure the plugin directory is writable.");
return false;
}
catch (Exception ex)
{
PluginCore.WriteToChat($"[Update] Install failed: {ex.Message}");
return false;
}
}
/// <summary>
/// Check for update and auto-install if available. Used by startup auto-update.
/// </summary>
public static async Task CheckAndInstallAsync()
{
try
{
bool checkOk = await CheckForUpdateAsync();
if (checkOk && updateAvailable)
{
PluginCore.WriteToChat("[Update] Auto-installing update...");
await DownloadAndInstallUpdateAsync();
}
}
catch (Exception ex)
{
PluginCore.WriteToChat($"[Update] Auto-update failed: {ex.Message}");
}
}
private static string GetLocalDllPath()
{
// Get the path to the current DLL
string assemblyPath = typeof(PluginCore).Assembly.Location;
// If empty (hot reload scenario), use AssemblyDirectory + filename
if (string.IsNullOrEmpty(assemblyPath))
{
return Path.Combine(PluginCore.AssemblyDirectory, "MosswartMassacre.dll");
}
return assemblyPath;
}
public static string GetUpdateStatus()
{
if (lastCheckTime == DateTime.MinValue)
{
return "Update Status: Not checked";
}
return updateAvailable ? "Update Status: Update available" : "Update Status: Up to date";
}
}
}