using IW4MAdmin.Application.API.Master; using IW4MAdmin.Application.EventParsers; using IW4MAdmin.Application.Factories; using IW4MAdmin.Application.Meta; using IW4MAdmin.Application.Migration; using IW4MAdmin.Application.Misc; using Microsoft.Extensions.DependencyInjection; using RestEase; using SharedLibraryCore; using SharedLibraryCore.Configuration; using SharedLibraryCore.Database.Models; using SharedLibraryCore.Dtos.Meta.Responses; using SharedLibraryCore.Exceptions; using SharedLibraryCore.Helpers; using SharedLibraryCore.Interfaces; using SharedLibraryCore.QueryHelper; using SharedLibraryCore.Repositories; using SharedLibraryCore.Services; using Stats.Dtos; using System; using System.IO; using System.Linq; using System.Net.Http; using System.Text; using System.Threading; using System.Threading.Tasks; using Data.Abstractions; using Data.Helpers; using Integrations.Source.Extensions; using IW4MAdmin.Application.Alerts; using IW4MAdmin.Application.Extensions; using IW4MAdmin.Application.IO; using IW4MAdmin.Application.Localization; using IW4MAdmin.Application.Plugin; using IW4MAdmin.Application.Plugin.Script; using IW4MAdmin.Application.QueryHelpers; using Microsoft.Extensions.Logging; using ILogger = Microsoft.Extensions.Logging.ILogger; using IW4MAdmin.Plugins.Stats.Client.Abstractions; using IW4MAdmin.Plugins.Stats.Client; using Microsoft.Extensions.Hosting; using Stats.Client.Abstractions; using Stats.Client; using Stats.Config; using Stats.Helpers; using WebfrontCore.QueryHelpers.Models; namespace IW4MAdmin.Application { public class Program { public static BuildNumber Version { get; } = BuildNumber.Parse(Utilities.GetVersionAsString()); private static ApplicationManager _serverManager; private static Task _applicationTask; private static IServiceProvider _serviceProvider; /// /// entrypoint of the application /// /// public static async Task Main(string[] args) { AppDomain.CurrentDomain.SetData("DataDirectory", Utilities.OperatingDirectory); AppDomain.CurrentDomain.AssemblyResolve += (sender, eventArgs) => { var libraryName = eventArgs.Name.Split(",").First(); var overrides = new[] { nameof(SharedLibraryCore), nameof(Stats) }; if (!overrides.Contains(libraryName)) { return AppDomain.CurrentDomain.GetAssemblies().FirstOrDefault(asm => asm.FullName == eventArgs.Name); } // added to be a bit more permissive with plugin references return AppDomain.CurrentDomain.GetAssemblies() .FirstOrDefault(asm => asm.FullName?.StartsWith(libraryName) ?? false); }; Console.OutputEncoding = Encoding.UTF8; Console.ForegroundColor = ConsoleColor.Gray; Console.CancelKeyPress += OnCancelKey; Console.WriteLine("====================================================="); Console.WriteLine(" IW4MAdmin"); Console.WriteLine(" by RaidMax "); Console.WriteLine($" Version {Utilities.GetVersionAsString()}"); Console.WriteLine("====================================================="); await LaunchAsync(args); } /// /// event callback executed when the control + c combination is detected /// gracefully stops the server manager and waits for all tasks to finish /// /// /// private static async void OnCancelKey(object sender, ConsoleCancelEventArgs e) { if (_serverManager is not null) { await _serverManager.Stop(); } if (_applicationTask is not null) { await _applicationTask; } } /// /// task that initializes application and starts the application monitoring and runtime tasks /// /// private static async Task LaunchAsync(string[] args) { restart: ITranslationLookup translationLookup = null; var logger = BuildDefaultLogger(new ApplicationConfiguration()); Utilities.DefaultLogger = logger; logger.LogInformation("Begin IW4MAdmin startup. Version is {Version} {@Args}", Version, args); try { // do any needed housekeeping file/folder migrations ConfigurationMigration.MoveConfigFolder10518(null); ConfigurationMigration.CheckDirectories(); ConfigurationMigration.RemoveObsoletePlugins20210322(); logger.LogDebug("Configuring services..."); var configHandler = new BaseConfigurationHandler("IW4MAdminSettings"); await configHandler.BuildAsync(); _serviceProvider = WebfrontCore.Program.InitializeServices(ConfigureServices, (configHandler.Configuration() ?? new ApplicationConfiguration()).WebfrontBindUrl); _serverManager = (ApplicationManager)_serviceProvider.GetRequiredService(); translationLookup = _serviceProvider.GetRequiredService(); await _serverManager.Init(); _applicationTask = RunApplicationTasksAsync(logger, _serverManager, _serviceProvider); await _applicationTask; logger.LogInformation("Shutdown completed successfully"); } catch (Exception e) { var failMessage = translationLookup == null ? "Failed to initialize IW4MAdmin" : translationLookup["MANAGER_INIT_FAIL"]; var exitMessage = translationLookup == null ? "Press enter to exit..." : translationLookup["MANAGER_EXIT"]; logger.LogCritical(e, "Failed to initialize IW4MAdmin"); Console.WriteLine(failMessage); while (e.InnerException != null) { e = e.InnerException; } if (e is ConfigurationException configException) { Console.WriteLine("{{fileName}} contains an error." .FormatExt(Path.GetFileName(configException.ConfigurationFileName))); foreach (var error in configException.Errors) { Console.WriteLine(error); } } else { Console.WriteLine(e.Message); } if (_serverManager is not null) { await _serverManager?.Stop(); } Console.WriteLine(exitMessage); await Console.In.ReadAsync(new char[1], 0, 1); return; } if (_serverManager.IsRestartRequested) { goto restart; } } /// /// runs the core application tasks /// /// private static Task RunApplicationTasksAsync(ILogger logger, ApplicationManager applicationManager, IServiceProvider serviceProvider) { var collectionService = serviceProvider.GetRequiredService(); var versionChecker = serviceProvider.GetRequiredService(); var masterCommunicator = serviceProvider.GetRequiredService(); var webfrontLifetime = serviceProvider.GetRequiredService(); using var onWebfrontErrored = new ManualResetEventSlim(); var webfrontTask = _serverManager.GetApplicationSettings().Configuration().EnableWebFront ? WebfrontCore.Program.GetWebHostTask(_serverManager.CancellationToken).ContinueWith(continuation => { if (!continuation.IsFaulted) { return; } logger.LogCritical("Unable to start webfront task. {Message}", continuation.Exception?.InnerException?.Message); logger.LogDebug(continuation.Exception, "Unable to start webfront task"); onWebfrontErrored.Set(); }) : Task.CompletedTask; if (_serverManager.GetApplicationSettings().Configuration().EnableWebFront) { try { onWebfrontErrored.Wait(webfrontLifetime.ApplicationStarted); } catch { // ignored when webfront successfully starts } if (onWebfrontErrored.IsSet) { return Task.CompletedTask; } } // we want to run this one on a manual thread instead of letting the thread pool handle it, // because we can't exit early from waiting on console input, and it prevents us from restarting async void ReadInput() => await ReadConsoleInput(logger); var inputThread = new Thread(ReadInput); inputThread.Start(); var tasks = new[] { applicationManager.Start(), versionChecker.CheckVersion(), webfrontTask, masterCommunicator.RunUploadStatus(_serverManager.CancellationToken), collectionService.BeginCollectionAsync(cancellationToken: _serverManager.CancellationToken) }; logger.LogDebug("Starting webfront and input tasks"); return Task.WhenAll(tasks); } /// /// reads input from the console and executes entered commands on the default server /// /// private static async Task ReadConsoleInput(ILogger logger) { if (Console.IsInputRedirected) { logger.LogInformation("Disabling console input as it has been redirected"); return; } EFClient origin = null; try { while (!_serverManager.CancellationToken.IsCancellationRequested) { if (!_serverManager.IsInitialized) { await Task.Delay(1000); continue; } var lastCommand = await Console.In.ReadLineAsync(); if (lastCommand == null) { continue; } if (!lastCommand.Any()) { continue; } var gameEvent = new GameEvent { Type = GameEvent.EventType.Command, Data = lastCommand, Origin = origin ??= Utilities.IW4MAdminClient(_serverManager.Servers.FirstOrDefault()), Owner = _serverManager.Servers[0] }; _serverManager.AddEvent(gameEvent); await gameEvent.WaitAsync(Utilities.DefaultCommandTimeout, _serverManager.CancellationToken); Console.Write('>'); } } catch (OperationCanceledException) { } } private static IServiceCollection HandlePluginRegistration(ApplicationConfiguration appConfig, IServiceCollection serviceCollection, IMasterApi masterApi) { var defaultLogger = BuildDefaultLogger(appConfig); var pluginServiceProvider = new ServiceCollection() .AddBaseLogger(appConfig) .AddSingleton(appConfig) .AddSingleton(masterApi) .AddSingleton() .AddSingleton() .BuildServiceProvider(); var pluginImporter = pluginServiceProvider.GetRequiredService(); // we need to register the rest client with regular collection serviceCollection.AddSingleton(masterApi); // register the native commands foreach (var commandType in typeof(SharedLibraryCore.Commands.QuitCommand).Assembly.GetTypes() .Concat(typeof(Program).Assembly.GetTypes().Where(type => type.Namespace?.StartsWith("IW4MAdmin.Application.Commands") ?? false)) .Where(command => command.BaseType == typeof(Command))) { defaultLogger.LogDebug("Registered native command type {Name}", commandType.Name); serviceCollection.AddSingleton(typeof(IManagerCommand), commandType); } // register the plugin implementations var (plugins, commands, configurations) = pluginImporter.DiscoverAssemblyPluginImplementations(); foreach (var pluginType in plugins) { var isV2 = pluginType.GetInterface(nameof(IPluginV2), false) != null; defaultLogger.LogDebug("Registering plugin type {Name}", pluginType.FullName); serviceCollection.AddSingleton(!isV2 ? typeof(IPlugin) : typeof(IPluginV2), pluginType); try { var registrationMethod = pluginType.GetMethod(nameof(IPluginV2.RegisterDependencies)); registrationMethod?.Invoke(null, new object[] { serviceCollection }); } catch (Exception ex) { defaultLogger.LogError(ex, "Could not register plugin of type {Type}", pluginType.Name); } } // register the plugin commands foreach (var commandType in commands) { defaultLogger.LogDebug("Registered plugin command type {Name}", commandType.FullName); serviceCollection.AddSingleton(typeof(IManagerCommand), commandType); } foreach (var configurationType in configurations) { defaultLogger.LogDebug("Registered plugin config type {Name}", configurationType.Name); var configInstance = (IBaseConfiguration) Activator.CreateInstance(configurationType); var handlerType = typeof(BaseConfigurationHandler<>).MakeGenericType(configurationType); var handlerInstance = Activator.CreateInstance(handlerType, configInstance.Name()); var genericInterfaceType = typeof(IConfigurationHandler<>).MakeGenericType(configurationType); serviceCollection.AddSingleton(genericInterfaceType, handlerInstance); } var scriptPlugins = pluginImporter.DiscoverScriptPlugins(); foreach (var scriptPlugin in scriptPlugins) { serviceCollection.AddSingleton(scriptPlugin.Item1, sp => sp.GetRequiredService() .CreateScriptPlugin(scriptPlugin.Item1, scriptPlugin.Item2)); } // register any eventable types foreach (var assemblyType in typeof(Program).Assembly.GetTypes() .Where(asmType => typeof(IRegisterEvent).IsAssignableFrom(asmType)) .Union(plugins.SelectMany(asm => asm.Assembly.GetTypes()) .Distinct() .Where(asmType => typeof(IRegisterEvent).IsAssignableFrom(asmType)))) { var instance = Activator.CreateInstance(assemblyType) as IRegisterEvent; serviceCollection.AddSingleton(instance); } return serviceCollection; } /// /// Configures the dependency injection services /// private static void ConfigureServices(IServiceCollection serviceCollection) { // todo: this is a quick fix AppContext.SetSwitch("Npgsql.EnableLegacyTimestampBehavior", true); serviceCollection.AddConfiguration("IW4MAdminSettings") .AddConfiguration() .AddConfiguration() .AddConfiguration("StatsPluginSettings"); // for legacy purposes. update at some point var appConfigHandler = new BaseConfigurationHandler("IW4MAdminSettings"); appConfigHandler.BuildAsync().GetAwaiter().GetResult(); var commandConfigHandler = new BaseConfigurationHandler("CommandConfiguration"); commandConfigHandler.BuildAsync().GetAwaiter().GetResult(); var appConfig = appConfigHandler.Configuration(); var masterUri = /*Utilities.IsDevelopment ? new Uri("http://127.0.0.1:8080") : appConfig?.MasterUrl ?? */new ApplicationConfiguration().MasterUrl; var httpClient = new HttpClient { BaseAddress = masterUri, Timeout = TimeSpan.FromSeconds(15) }; var masterRestClient = RestClient.For(httpClient); var translationLookup = Configure.Initialize(Utilities.DefaultLogger, masterRestClient, appConfig); if (appConfig == null) { appConfig = (ApplicationConfiguration) new ApplicationConfiguration().Generate(); appConfigHandler.Set(appConfig); appConfigHandler.Save().GetAwaiter().GetResult(); } // register override level names foreach (var (key, value) in appConfig.OverridePermissionLevelNames) { if (!Utilities.PermissionLevelOverrides.ContainsKey(key)) { Utilities.PermissionLevelOverrides.Add(key, value); } } // build the dependency list serviceCollection .AddBaseLogger(appConfig) .AddSingleton((IConfigurationHandler) appConfigHandler) .AddSingleton>(commandConfigHandler) .AddSingleton(serviceProvider => serviceProvider.GetRequiredService>() .Configuration() ?? new CommandConfiguration()) .AddSingleton() .AddSingleton() .AddSingleton() .AddSingleton() .AddSingleton() .AddSingleton() .AddSingleton() .AddSingleton() .AddSingleton() .AddSingleton() .AddSingleton, ClientService>() #pragma warning disable CS0618 .AddSingleton() #pragma warning restore CS0618 .AddSingleton() .AddSingleton() .AddSingleton() .AddSingleton() .AddSingleton() .AddSingleton() .AddSingleton, ReceivedPenaltyResourceQueryHelper>() .AddSingleton, AdministeredPenaltyResourceQueryHelper>() .AddSingleton, UpdatedAliasResourceQueryHelper>() .AddSingleton, ChatResourceQueryHelper>() .AddSingleton, ConnectionsResourceQueryHelper>() .AddSingleton, PermissionLevelChangedResourceQueryHelper>() .AddSingleton, ClientResourceQueryHelper>() .AddTransient() .AddSingleton() .AddSingleton() .AddSingleton() #pragma warning disable CS0612 .AddSingleton() #pragma warning restore CS0612 .AddSingleton() .AddSingleton() .AddSingleton() .AddSingleton() .AddSingleton() .AddSingleton(typeof(ILookupCache<>), typeof(LookupCache<>)) .AddSingleton(typeof(IDataValueCache<,>), typeof(DataValueCache<,>)) .AddSingleton() .AddSingleton() .AddSingleton(new GeoLocationService(Path.Join(".", "Resources", "GeoLite2-Country.mmdb"))) .AddSingleton() #pragma warning disable CS0618 .AddTransient() #pragma warning restore CS0618 .AddSingleton() .AddSingleton() .AddSingleton(new ConfigurationWatcher()) .AddSingleton(typeof(IConfigurationHandlerV2<>), typeof(BaseConfigurationHandlerV2<>)) .AddSingleton() .AddSingleton(translationLookup) .AddDatabaseContextOptions(appConfig); serviceCollection.AddSingleton(); serviceCollection.AddSource(); HandlePluginRegistration(appConfig, serviceCollection, masterRestClient); } private static ILogger BuildDefaultLogger(ApplicationConfiguration appConfig) { var collection = new ServiceCollection() .AddBaseLogger(appConfig) .BuildServiceProvider(); return collection.GetRequiredService>(); } } }