208 lines
6.4 KiB
C#
208 lines
6.4 KiB
C#
using System;
|
|
using System.Collections.Generic;
|
|
using System.Diagnostics;
|
|
using System.IO;
|
|
using System.IO.Compression;
|
|
using System.Linq;
|
|
using System.Net;
|
|
using System.Reflection;
|
|
using System.Text;
|
|
using System.Threading;
|
|
using System.Threading.Tasks;
|
|
using System.Windows;
|
|
using System.Windows.Threading;
|
|
using NLog;
|
|
using NLog.Targets;
|
|
using Torch.Utils;
|
|
|
|
namespace Torch.Server
|
|
{
|
|
public class Initializer
|
|
{
|
|
private static readonly Logger Log = LogManager.GetLogger(nameof(Initializer));
|
|
private bool _init;
|
|
private const string STEAMCMD_DIR = "steamcmd";
|
|
private const string STEAMCMD_ZIP = "temp.zip";
|
|
private static readonly string STEAMCMD_PATH = $"{STEAMCMD_DIR}\\steamcmd.exe";
|
|
private static readonly string RUNSCRIPT_PATH = $"{STEAMCMD_DIR}\\runscript.txt";
|
|
|
|
private const string RUNSCRIPT = @"force_install_dir ../
|
|
login anonymous
|
|
app_update 298740
|
|
quit";
|
|
|
|
private TorchConfig _config;
|
|
private TorchServer _server;
|
|
private string _basePath;
|
|
|
|
public TorchConfig Config => _config;
|
|
public TorchServer Server => _server;
|
|
|
|
public Initializer(string basePath)
|
|
{
|
|
_basePath = basePath;
|
|
}
|
|
|
|
public bool Initialize(string[] args)
|
|
{
|
|
if (_init)
|
|
return false;
|
|
|
|
#if !DEBUG
|
|
AppDomain.CurrentDomain.UnhandledException += HandleException;
|
|
#endif
|
|
|
|
if (!args.Contains("-noupdate"))
|
|
RunSteamCmd();
|
|
|
|
_config = InitConfig();
|
|
if (!_config.Parse(args))
|
|
return false;
|
|
|
|
if (!string.IsNullOrEmpty(_config.WaitForPID))
|
|
{
|
|
try
|
|
{
|
|
var pid = int.Parse(_config.WaitForPID);
|
|
var waitProc = Process.GetProcessById(pid);
|
|
Log.Info("Continuing in 5 seconds.");
|
|
Log.Warn($"Waiting for process {pid} to close");
|
|
while (!waitProc.HasExited)
|
|
{
|
|
Console.Write(".");
|
|
Thread.Sleep(1000);
|
|
}
|
|
}
|
|
catch
|
|
{
|
|
// ignored
|
|
}
|
|
}
|
|
|
|
_init = true;
|
|
return true;
|
|
}
|
|
|
|
public void Run()
|
|
{
|
|
_server = new TorchServer(_config);
|
|
var init = Task.Run(() => _server.Init());
|
|
if (!_config.NoGui)
|
|
{
|
|
if (_config.Autostart)
|
|
init.ContinueWith(x => _server.Start());
|
|
|
|
Log.Info("Showing UI");
|
|
Console.SetOut(TextWriter.Null);
|
|
NativeMethods.FreeConsole();
|
|
new TorchUI(_server).ShowDialog();
|
|
}
|
|
else
|
|
{
|
|
init.Wait();
|
|
_server.Start();
|
|
}
|
|
}
|
|
|
|
private TorchConfig InitConfig()
|
|
{
|
|
var configName = "Torch.cfg";
|
|
var configPath = Path.Combine(Directory.GetCurrentDirectory(), configName);
|
|
if (File.Exists(configName))
|
|
{
|
|
Log.Info($"Loading config {configPath}");
|
|
return TorchConfig.LoadFrom(configPath);
|
|
}
|
|
else
|
|
{
|
|
Log.Info($"Generating default config at {configPath}");
|
|
var config = new TorchConfig {InstancePath = Path.GetFullPath("Instance")};
|
|
config.Save(configPath);
|
|
return config;
|
|
}
|
|
}
|
|
|
|
private static void RunSteamCmd()
|
|
{
|
|
var log = LogManager.GetLogger("SteamCMD");
|
|
|
|
if (!Directory.Exists(STEAMCMD_DIR))
|
|
{
|
|
Directory.CreateDirectory(STEAMCMD_DIR);
|
|
}
|
|
|
|
if (!File.Exists(RUNSCRIPT_PATH))
|
|
File.WriteAllText(RUNSCRIPT_PATH, RUNSCRIPT);
|
|
|
|
if (!File.Exists(STEAMCMD_PATH))
|
|
{
|
|
try
|
|
{
|
|
log.Info("Downloading SteamCMD.");
|
|
using (var client = new WebClient())
|
|
client.DownloadFile("https://steamcdn-a.akamaihd.net/client/installer/steamcmd.zip", STEAMCMD_ZIP);
|
|
|
|
ZipFile.ExtractToDirectory(STEAMCMD_ZIP, STEAMCMD_DIR);
|
|
File.Delete(STEAMCMD_ZIP);
|
|
log.Info("SteamCMD downloaded successfully!");
|
|
}
|
|
catch
|
|
{
|
|
log.Error("Failed to download SteamCMD, unable to update the DS.");
|
|
return;
|
|
}
|
|
}
|
|
|
|
log.Info("Checking for DS updates.");
|
|
var steamCmdProc = new ProcessStartInfo(STEAMCMD_PATH, "+runscript runscript.txt")
|
|
{
|
|
WorkingDirectory = Path.Combine(Directory.GetCurrentDirectory(), STEAMCMD_DIR),
|
|
UseShellExecute = false,
|
|
RedirectStandardOutput = true,
|
|
StandardOutputEncoding = Encoding.ASCII
|
|
};
|
|
var cmd = Process.Start(steamCmdProc);
|
|
|
|
// ReSharper disable once PossibleNullReferenceException
|
|
while (!cmd.HasExited)
|
|
{
|
|
log.Info(cmd.StandardOutput.ReadLine());
|
|
Thread.Sleep(100);
|
|
}
|
|
}
|
|
|
|
private void LogException(Exception ex)
|
|
{
|
|
if (ex.InnerException != null)
|
|
{
|
|
LogException(ex.InnerException);
|
|
}
|
|
|
|
Log.Fatal(ex);
|
|
if (ex is ReflectionTypeLoadException exti)
|
|
foreach (Exception exl in exti.LoaderExceptions)
|
|
LogException(exl);
|
|
}
|
|
|
|
private void HandleException(object sender, UnhandledExceptionEventArgs e)
|
|
{
|
|
var ex = (Exception)e.ExceptionObject;
|
|
LogException(ex);
|
|
LogManager.Flush();
|
|
if (_config.RestartOnCrash)
|
|
{
|
|
Console.WriteLine("Restarting in 5 seconds.");
|
|
Thread.Sleep(5000);
|
|
var exe = typeof(Program).Assembly.Location;
|
|
_config.WaitForPID = Process.GetCurrentProcess().Id.ToString();
|
|
Process.Start(exe, _config.ToString());
|
|
}
|
|
else
|
|
{
|
|
MessageBox.Show("Torch encountered a fatal error and needs to close. Please check the logs for details.");
|
|
}
|
|
|
|
Process.GetCurrentProcess().Kill();
|
|
}
|
|
}
|
|
} |