Skip to content

Commit 893efda

Browse files
authored
Support for SKSE launching when running Skyrim (#505)
1 parent 89ccdcd commit 893efda

File tree

9 files changed

+127
-10
lines changed

9 files changed

+127
-10
lines changed
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,28 @@
1+
using Microsoft.Extensions.Logging;
2+
using NexusMods.Common;
3+
using NexusMods.DataModel.Extensions;
4+
using NexusMods.DataModel.Games;
5+
using NexusMods.DataModel.Loadouts;
6+
using NexusMods.DataModel.Loadouts.LoadoutSynchronizerDTOs;
7+
using NexusMods.Paths;
8+
9+
namespace NexusMods.Games.BethesdaGameStudios;
10+
11+
/// <summary>
12+
/// A tool that allows running the game with a script extender.
13+
/// </summary>
14+
/// <typeparam name="T"></typeparam>
15+
public abstract class RunGameWithScriptExtender<T> : RunGameTool<T> where T : AGame {
16+
// ReSharper disable once ContextualLoggerProblem
17+
protected RunGameWithScriptExtender(ILogger<RunGameTool<T>> logger, T game, IProcessFactory processFactory)
18+
: base(logger, game, processFactory) { }
19+
20+
protected abstract GamePath ScriptLoaderPath { get; }
21+
22+
protected override AbsolutePath GetGamePath(Loadout loadout, ApplyPlan applyPlan)
23+
{
24+
return applyPlan.Flattened.ContainsKey(ScriptLoaderPath) ?
25+
ScriptLoaderPath.CombineChecked(loadout.Installation) :
26+
base.GetGamePath(loadout, applyPlan);
27+
}
28+
}

src/Games/NexusMods.Games.BethesdaGameStudios/Services.cs

+2-2
Original file line numberDiff line numberDiff line change
@@ -12,8 +12,8 @@ public static IServiceCollection AddBethesdaGameStudios(this IServiceCollection
1212
{
1313
services.AddAllSingleton<IGame, SkyrimSpecialEdition>();
1414
services.AddAllSingleton<IGame, SkyrimLegendaryEdition>();
15-
services.AddSingleton<ITool, RunGameTool<SkyrimLegendaryEdition>>();
16-
services.AddSingleton<ITool, RunGameTool<SkyrimSpecialEdition>>();
15+
services.AddSingleton<ITool, SkyrimLegendaryEditionGameTool>();
16+
services.AddSingleton<ITool, SkyrimSpecialEditionGameTool>();
1717
services.AddAllSingleton<IFileAnalyzer, PluginAnalyzer>();
1818
services.AddAllSingleton<ITypeFinder, TypeFinder>();
1919
return services;
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,14 @@
1+
using Microsoft.Extensions.Logging;
2+
using NexusMods.Common;
3+
using NexusMods.DataModel.Games;
4+
using NexusMods.Paths;
5+
6+
namespace NexusMods.Games.BethesdaGameStudios;
7+
8+
public class SkyrimLegendaryEditionGameTool : RunGameWithScriptExtender<SkyrimLegendaryEdition>
9+
{
10+
// ReSharper disable once ContextualLoggerProblem
11+
public SkyrimLegendaryEditionGameTool(ILogger<RunGameTool<SkyrimLegendaryEdition>> logger, SkyrimLegendaryEdition game, IProcessFactory processFactory)
12+
: base(logger, game, processFactory) { }
13+
protected override GamePath ScriptLoaderPath => new(GameFolderType.Game, "skse_loader.exe");
14+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,14 @@
1+
using Microsoft.Extensions.Logging;
2+
using NexusMods.Common;
3+
using NexusMods.DataModel.Games;
4+
using NexusMods.Paths;
5+
6+
namespace NexusMods.Games.BethesdaGameStudios;
7+
8+
public class SkyrimSpecialEditionGameTool : RunGameWithScriptExtender<SkyrimSpecialEdition>
9+
{
10+
// ReSharper disable once ContextualLoggerProblem
11+
public SkyrimSpecialEditionGameTool(ILogger<RunGameTool<SkyrimSpecialEdition>> logger, SkyrimSpecialEdition game, IProcessFactory processFactory)
12+
: base(logger, game, processFactory) { }
13+
protected override GamePath ScriptLoaderPath => new(GameFolderType.Game, "skse64_loader.exe");
14+
}

src/Games/NexusMods.Games.RedEngine/RedModDeployTool.cs

+2-1
Original file line numberDiff line numberDiff line change
@@ -4,6 +4,7 @@
44
using NexusMods.DataModel.Extensions;
55
using NexusMods.DataModel.Games;
66
using NexusMods.DataModel.Loadouts;
7+
using NexusMods.DataModel.Loadouts.LoadoutSynchronizerDTOs;
78
using NexusMods.Paths;
89

910
namespace NexusMods.Games.RedEngine;
@@ -21,7 +22,7 @@ public RedModDeployTool(ILogger<RedModDeployTool> logger)
2122

2223
public IEnumerable<GameDomain> Domains => new[] { Cyberpunk2077.StaticDomain };
2324

24-
public async Task Execute(Loadout loadout)
25+
public async Task Execute(Loadout loadout, ApplyPlan applyPlan, CancellationToken cancellationToken)
2526
{
2627
var exe = RedModPath.CombineChecked(loadout.Installation);
2728

src/NexusMods.DataModel/Games/ITool.cs

+4-1
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,5 @@
11
using NexusMods.DataModel.Loadouts;
2+
using NexusMods.DataModel.Loadouts.LoadoutSynchronizerDTOs;
23

34
namespace NexusMods.DataModel.Games;
45

@@ -22,5 +23,7 @@ public interface ITool
2223
/// Executes this tool.
2324
/// </summary>
2425
/// <param name="loadout">The collection of mods (loadout) to be used with this tool.</param>
25-
public Task Execute(Loadout loadout);
26+
/// <param name="applyPlan">The plan applied before this tool was executed</param>
27+
/// <param name="cancellationToken"></param>
28+
public Task Execute(Loadout loadout, ApplyPlan applyPlan, CancellationToken cancellationToken);
2629
}

src/NexusMods.DataModel/Games/RunGameTool.cs

+60-4
Original file line numberDiff line numberDiff line change
@@ -3,7 +3,9 @@
33
using CliWrap;
44
using Microsoft.Extensions.Logging;
55
using NexusMods.Common;
6+
using NexusMods.DataModel.Extensions;
67
using NexusMods.DataModel.Loadouts;
8+
using NexusMods.DataModel.Loadouts.LoadoutSynchronizerDTOs;
79
using NexusMods.Paths;
810

911
namespace NexusMods.DataModel.Games;
@@ -14,6 +16,7 @@ namespace NexusMods.DataModel.Games;
1416
/// </summary>
1517
public interface IRunGameTool : ITool
1618
{
19+
1720
}
1821

1922
/// <summary>
@@ -47,11 +50,30 @@ public RunGameTool(ILogger<RunGameTool<T>> logger, T game, IProcessFactory proce
4750
public string Name => $"Run {_game.Name}";
4851

4952
/// <inheritdoc />
50-
public async Task Execute(Loadout loadout)
53+
public async Task Execute(Loadout loadout, ApplyPlan applyPlan, CancellationToken cancellationToken)
5154
{
52-
var program = _game.GetPrimaryFile(loadout.Installation.Store).Combine(loadout.Installation.Locations[GameFolderType.Game]);
55+
var program = GetGamePath(loadout, applyPlan);
5356
_logger.LogInformation("Running {Program}", program);
5457

58+
var primaryFile = _game.GetPrimaryFile(loadout.Installation.Store).CombineChecked(loadout.Installation);
59+
var names = new HashSet<string>()
60+
{
61+
program.FileName,
62+
program.GetFileNameWithoutExtension(),
63+
primaryFile.FileName,
64+
primaryFile.GetFileNameWithoutExtension()
65+
};
66+
67+
// In the case of a preloader, we need to wait for the actual game file to exit
68+
// before we completely exit this routine. So get a list of all the processes with a give
69+
// name at the start, after the preloader finishes find any other processes with the same set of
70+
// names, and then we wait for those to exit.
71+
72+
// In the case of something like Skyrim this means we will start with loading skse64_loader.exe then
73+
// notice that SkyrimSE.exe is running and wait for that to exit.
74+
75+
var existing = FindMatchingProcesses(names).Select(p => p.Id).ToHashSet();
76+
5577
var stdOut = new StringBuilder();
5678
var stdErr = new StringBuilder();
5779
var command = new Command(program.ToString())
@@ -60,11 +82,45 @@ public async Task Execute(Loadout loadout)
6082
.WithValidation(CommandResultValidation.None)
6183
.WithWorkingDirectory(program.Parent.ToString());
6284

63-
64-
var result = await _processFactory.ExecuteAsync(command);
85+
86+
var result = await _processFactory.ExecuteAsync(command, cancellationToken);
6587
if (result.ExitCode != 0)
6688
_logger.LogError("While Running {Filename} : {Error} {Output}", program, stdErr, stdOut);
6789

90+
var newProcesses = FindMatchingProcesses(names)
91+
.Where(p => !existing.Contains(p.Id))
92+
.ToHashSet();
93+
94+
if (newProcesses.Count > 0)
95+
{
96+
_logger.LogInformation("Waiting for {Count} processes to exit", newProcesses.Count);
97+
while (true)
98+
{
99+
await Task.Delay(500, cancellationToken);
100+
if (newProcesses.All(p => p.HasExited))
101+
break;
102+
}
103+
_logger.LogInformation("All {Count} processes have exited", newProcesses.Count);
104+
}
105+
68106
_logger.LogInformation("Finished running {Program}", program);
69107
}
108+
109+
private static HashSet<Process> FindMatchingProcesses(HashSet<string> names)
110+
{
111+
return Process.GetProcesses()
112+
.Where(p => names.Contains(p.ProcessName))
113+
.ToHashSet();
114+
}
115+
116+
/// <summary>
117+
/// Returns the path to the main executable file for the game.
118+
/// </summary>
119+
/// <param name="loadout"></param>
120+
/// <param name="applyPlan"></param>
121+
/// <returns></returns>
122+
protected virtual AbsolutePath GetGamePath(Loadout loadout, ApplyPlan applyPlan)
123+
{
124+
return _game.GetPrimaryFile(loadout.Installation.Store).Combine(loadout.Installation.Locations[GameFolderType.Game]);
125+
}
70126
}

src/NexusMods.DataModel/ToolManager.cs

+1-1
Original file line numberDiff line numberDiff line change
@@ -47,7 +47,7 @@ public async Task<Loadout> RunTool(ITool tool, Loadout loadout, ModId? generated
4747
var plan = await _loadoutSynchronizer.MakeApplySteps(loadout, token);
4848
await _loadoutSynchronizer.Apply(plan, token);
4949

50-
await tool.Execute(loadout);
50+
await tool.Execute(loadout, plan, token);
5151
var modName = $"{tool.Name} Generated Files";
5252

5353
if (generatedFilesMod == null)

tests/NexusMods.StandardGameLocators.TestHelpers/ListFilesTool.cs

+2-1
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,6 @@
11
using NexusMods.DataModel.Games;
22
using NexusMods.DataModel.Loadouts;
3+
using NexusMods.DataModel.Loadouts.LoadoutSynchronizerDTOs;
34
using NexusMods.Paths;
45

56
namespace NexusMods.StandardGameLocators.TestHelpers;
@@ -8,7 +9,7 @@ public class ListFilesTool : ITool
89
{
910
public IEnumerable<GameDomain> Domains => new[] { GameDomain.From("stubbed-game") };
1011

11-
public async Task Execute(Loadout loadout)
12+
public async Task Execute(Loadout loadout, ApplyPlan applyPlan, CancellationToken cancellationToken)
1213
{
1314
var listPath = loadout.Installation.Locations[GameFolderType.Game];
1415
var outPath = GeneratedFilePath.Combine(listPath);

0 commit comments

Comments
 (0)