using System; using System.Globalization; using System.IO; using System.Linq; using System.Reflection; using System.Threading; #if SMAPI_FOR_WINDOWS using System.Windows.Forms; #endif using Microsoft.Xna.Framework.Graphics; using Newtonsoft.Json; using StardewModdingAPI.Events; using StardewModdingAPI.Framework; using StardewModdingAPI.Inheritance; using StardewValley; using Monitor = StardewModdingAPI.Framework.Monitor; namespace StardewModdingAPI { /// The main entry point for SMAPI, responsible for hooking into and launching the game. public class Program { /********* ** Properties *********/ /// The full path to the Stardew Valley executable. #if SMAPI_FOR_WINDOWS private static readonly string GameExecutablePath = Path.Combine(Constants.ExecutionPath, "Stardew Valley.exe"); #else private static readonly string GameExecutablePath = Path.Combine(Constants.ExecutionPath, "StardewValley.exe"); #endif /// The full path to the folder containing mods. private static readonly string ModPath = Path.Combine(Constants.ExecutionPath, "Mods"); /// The log file to which to write messages. private static readonly LogFileManager LogFile = new LogFileManager(Constants.LogPath); /// The core logger for SMAPI. private static readonly Monitor Monitor = new Monitor("SMAPI", Program.LogFile); /// Whether SMAPI is running in developer mode. private static bool DeveloperMode; /********* ** Accessors *********/ /// The number of mods currently loaded by SMAPI. public static int ModsLoaded; /// The underlying game instance. public static SGame gamePtr; /// Whether the game is currently running. public static bool ready; /// The underlying game assembly. public static Assembly StardewAssembly; /// The underlying type. public static Type StardewProgramType; /// The field containing game's main instance. public static FieldInfo StardewGameInfo; // ReSharper disable once PossibleNullReferenceException /// The game's build type (i.e. GOG vs Steam). public static int BuildType => (int)Program.StardewProgramType.GetField("buildType", BindingFlags.Public | BindingFlags.Static).GetValue(null); /// Tracks the installed mods. internal static readonly ModRegistry ModRegistry = new ModRegistry(); /// Manages deprecation warnings. internal static readonly DeprecationManager DeprecationManager = new DeprecationManager(Program.Monitor, Program.ModRegistry); /********* ** Public methods *********/ /// The main entry point which hooks into and launches the game. private static void Main() { // set thread culture for consistent log formatting Thread.CurrentThread.CurrentCulture = CultureInfo.CreateSpecificCulture("en-GB"); // load user settings { string settingsFileName = $"{typeof(Program).Assembly.GetName().Name}-settings.json"; string settingsPath = Path.Combine(Constants.ExecutionPath, settingsFileName); if (File.Exists(settingsPath)) { string json = File.ReadAllText(settingsPath); UserSettings settings = JsonConvert.DeserializeObject(json); Program.DeveloperMode = settings?.DeveloperMode == true; if (Program.DeveloperMode) { Program.Monitor.ShowTraceInConsole = true; Program.Monitor.Log($"SMAPI is running in developer mode. The console may be much more verbose. You can disable developer mode by deleting the {settingsFileName} file in the game directory.", LogLevel.Alert); } } } // initialise legacy log Log.Monitor = new Monitor("legacy mod", Program.LogFile) { ShowTraceInConsole = Program.DeveloperMode }; Log.ModRegistry = Program.ModRegistry; // hook into & launch the game try { Log.SyncColour($"Launching SMAPI {Constants.Version} with Stardew Valley {Game1.version} on {Environment.OSVersion}", ConsoleColor.DarkGray); // make sure this is the first line, to simplify troubleshooting instructions // verify version if (String.Compare(Game1.version, Constants.MinimumGameVersion, StringComparison.InvariantCultureIgnoreCase) < 0) { Log.Error($"Oops! You're running Stardew Valley {Game1.version}, but the oldest supported version is {Constants.MinimumGameVersion}. Please update your game before using SMAPI. If you're on the Steam beta channel, note that the beta channel may not receive the latest updates."); return; } // initialise Log.Debug("Initialising..."); Console.Title = Constants.ConsoleTitle; Program.VerifyPath(Program.ModPath); Program.VerifyPath(Constants.LogDir); if (!File.Exists(Program.GameExecutablePath)) { Log.Error($"Couldn't find executable: {Program.GameExecutablePath}"); Console.ReadKey(); return; } // check for update Program.CheckForUpdateAsync(); // launch game Program.StartGame(); } catch (Exception ex) { Console.WriteLine(ex); Console.ReadKey(); Log.Error($"Critical error: {ex}"); } } /********* ** Private methods *********/ /// Asynchronously check for a new version of SMAPI, and print a message to the console if an update is available. private static void CheckForUpdateAsync() { new Thread(() => { try { GitRelease release = UpdateHelper.GetLatestVersionAsync(Constants.GitHubRepository).Result; Version latestVersion = new Version(release.Tag); if (latestVersion.CompareTo(Constants.Version) > 0) Log.AsyncColour($"You can update SMAPI from version {Constants.Version} to {latestVersion}", ConsoleColor.Magenta); } catch (Exception ex) { Log.Debug($"Couldn't check for a new version of SMAPI. This won't affect your game, but you may not be notified of new versions if this keeps happening.\n{ex}"); } }).Start(); } /// Hook into Stardew Valley and launch the game. private static void StartGame() { // load the game assembly (ignore security) Log.Debug("Preparing game..."); Program.StardewAssembly = Assembly.UnsafeLoadFrom(Program.GameExecutablePath); Program.StardewProgramType = Program.StardewAssembly.GetType("StardewValley.Program", true); Program.StardewGameInfo = Program.StardewProgramType.GetField("gamePtr"); // change the game's version Game1.version += $"-Z_MODDED | SMAPI {Constants.Version}"; // add error interceptors #if SMAPI_FOR_WINDOWS Application.ThreadException += Log.Application_ThreadException; Application.SetUnhandledExceptionMode(UnhandledExceptionMode.CatchException); #endif AppDomain.CurrentDomain.UnhandledException += Log.CurrentDomain_UnhandledException; // initialise game try { Log.Debug("Patching game..."); Program.gamePtr = new SGame(); // hook events Program.gamePtr.Exiting += (sender, e) => Program.ready = false; Program.gamePtr.Window.ClientSizeChanged += GraphicsEvents.InvokeResize; // patch graphics Game1.graphics.GraphicsProfile = GraphicsProfile.HiDef; // load mods Program.LoadMods(); // initialise Log.Debug("Tweaking game..."); Program.StardewGameInfo.SetValue(Program.StardewProgramType, Program.gamePtr); Program.gamePtr.IsMouseVisible = false; Program.gamePtr.Window.Title = $"Stardew Valley - Version {Game1.version}"; } catch (Exception ex) { Log.Error($"Game failed to initialise: {ex}"); return; } // initialise after game launches new Thread(() => { // wait for the game to load up while (!Program.ready) Thread.Sleep(1000); // register help command Command.RegisterCommand("help", "Lists all commands | 'help ' returns command description").CommandFired += Program.help_CommandFired; // raise game loaded event GameEvents.InvokeGameLoaded(); // listen for command line input Log.Debug("Starting console..."); Log.Info("Type 'help' for help, or 'help ' for a command's usage"); Thread consoleInputThread = new Thread(Program.ConsoleInputLoop); consoleInputThread.Start(); while (Program.ready) Thread.Sleep(1000 / 10); // Check if the game is still running 10 times a second // abort the console thread, we're closing if (consoleInputThread.ThreadState == ThreadState.Running) consoleInputThread.Abort(); Log.Info("Game has ended. Press any key to continue..."); Console.ReadKey(); Thread.Sleep(100); Environment.Exit(0); }).Start(); // start game loop Log.Debug("Starting Stardew Valley..."); try { Program.ready = true; Program.gamePtr.Run(); } catch (Exception ex) { Program.ready = false; Log.Error($"Game failed to start: {ex}"); } } /// Create a directory path if it doesn't exist. /// The directory path. private static void VerifyPath(string path) { try { if (!Directory.Exists(path)) Directory.CreateDirectory(path); } catch (Exception ex) { Log.Error($"Couldn't create a path: {path}\n\n{ex}"); } } /// Load and hook up all mods in the mod directory. private static void LoadMods() { Log.Debug("Loading mods..."); foreach (string directory in Directory.GetDirectories(Program.ModPath)) { foreach (string manifestPath in Directory.GetFiles(directory, "manifest.json")) { ModHelper helper = new ModHelper(directory); string errorPrefix = $"Couldn't load mod for manifest '{manifestPath}'"; // read manifest Manifest manifest = new Manifest(); try { // read manifest text string json = File.ReadAllText(manifestPath); if (string.IsNullOrEmpty(json)) { Log.Error($"{errorPrefix}: manifest is empty."); continue; } // deserialise manifest manifest = helper.ReadJsonFile("manifest.json"); if (manifest == null) { Log.Error($"{errorPrefix}: the manifest file does not exist."); continue; } if (string.IsNullOrEmpty(manifest.EntryDll)) { Log.Error($"{errorPrefix}: manifest doesn't specify an entry DLL."); continue; } // log deprecated fields if (manifest.UsedAuthourField) Program.DeprecationManager.Warn(manifest.Name, $"{nameof(Manifest)}.{nameof(Manifest.Authour)}", "1.0", DeprecationLevel.Notice); } catch (Exception ex) { Log.Error($"{errorPrefix}: manifest parsing failed.\n{ex}"); continue; } // create per-save directory if (manifest.PerSaveConfigs) { Program.DeprecationManager.Warn(manifest.Name, $"{nameof(Manifest)}.{nameof(Manifest.PerSaveConfigs)}", "1.0", DeprecationLevel.Notice); try { string psDir = Path.Combine(directory, "psconfigs"); Directory.CreateDirectory(psDir); if (!Directory.Exists(psDir)) { Log.Error($"{errorPrefix}: couldn't create the per-save configuration directory ('psconfigs') requested by this mod. The failure reason is unknown."); continue; } } catch (Exception ex) { Log.Error($"{errorPrefix}: couldm't create the per-save configuration directory ('psconfigs') requested by this mod.\n{ex}"); continue; } } // load DLL & hook up mod string targDll = string.Empty; try { targDll = Path.Combine(directory, manifest.EntryDll); if (!File.Exists(targDll)) { Log.Error($"{errorPrefix}: target DLL '{targDll}' does not exist."); continue; } Assembly modAssembly = Assembly.UnsafeLoadFrom(targDll); if (modAssembly.DefinedTypes.Count(x => x.BaseType == typeof(Mod)) > 0) { TypeInfo modEntryType = modAssembly.DefinedTypes.First(x => x.BaseType == typeof(Mod)); Mod modEntry = (Mod)modAssembly.CreateInstance(modEntryType.ToString()); if (modEntry != null) { // track mod Program.ModRegistry.Add(manifest, modAssembly); // hook up mod modEntry.Manifest = manifest; modEntry.Helper = helper; modEntry.Monitor = new Monitor(manifest.Name, Program.LogFile) { ShowTraceInConsole = Program.DeveloperMode }; modEntry.PathOnDisk = directory; Log.Info($"Loaded mod: {modEntry.Manifest.Name} by {modEntry.Manifest.Author}, v{modEntry.Manifest.Version} | {modEntry.Manifest.Description}"); Program.ModsLoaded += 1; modEntry.Entry(); // deprecated modEntry.Entry(modEntry.Helper); // raise deprecation warning for old Entry() method if (Program.DeprecationManager.IsVirtualMethodImplemented(modEntryType, typeof(Mod), nameof(Mod.Entry))) Program.DeprecationManager.Warn(manifest.Name, $"an old version of {nameof(Mod)}.{nameof(Mod.Entry)}", "1.0", DeprecationLevel.Notice); } } else Log.Error($"{errorPrefix}: the mod DLL does not contain an implementation of the 'Mod' class."); } catch (Exception ex) { Log.Error($"{errorPrefix}: an error occurred while loading the target DLL.\n{ex}"); } } } // print result Log.Debug($"Loaded {Program.ModsLoaded} mods."); Console.Title = Constants.ConsoleTitle; } /// Run a loop handling console input. private static void ConsoleInputLoop() { while (true) Command.CallCommand(Console.ReadLine()); } /// The method called when the user submits the help command in the console. /// The event sender. /// The event data. private static void help_CommandFired(object sender, EventArgsCommand e) { if (e.Command.CalledArgs.Length > 0) { var command = Command.FindCommand(e.Command.CalledArgs[0]); if (command == null) Log.Error("The specified command could't be found"); else Log.Info(command.CommandArgs.Length > 0 ? $"{command.CommandName}: {command.CommandDesc} - {string.Join(", ", command.CommandArgs)}" : $"{command.CommandName}: {command.CommandDesc}"); } else Log.Info("Commands: " + string.Join(", ", Command.RegisteredCommands.Select(x => x.CommandName))); } } }