3
3
using CliWrap ;
4
4
using Microsoft . Extensions . Logging ;
5
5
using NexusMods . Common ;
6
+ using NexusMods . DataModel . Extensions ;
6
7
using NexusMods . DataModel . Loadouts ;
8
+ using NexusMods . DataModel . Loadouts . LoadoutSynchronizerDTOs ;
7
9
using NexusMods . Paths ;
8
10
9
11
namespace NexusMods . DataModel . Games ;
@@ -14,6 +16,7 @@ namespace NexusMods.DataModel.Games;
14
16
/// </summary>
15
17
public interface IRunGameTool : ITool
16
18
{
19
+
17
20
}
18
21
19
22
/// <summary>
@@ -47,11 +50,30 @@ public RunGameTool(ILogger<RunGameTool<T>> logger, T game, IProcessFactory proce
47
50
public string Name => $ "Run { _game . Name } ";
48
51
49
52
/// <inheritdoc />
50
- public async Task Execute ( Loadout loadout )
53
+ public async Task Execute ( Loadout loadout , ApplyPlan applyPlan , CancellationToken cancellationToken )
51
54
{
52
- var program = _game . GetPrimaryFile ( loadout . Installation . Store ) . Combine ( loadout . Installation . Locations [ GameFolderType . Game ] ) ;
55
+ var program = GetGamePath ( loadout , applyPlan ) ;
53
56
_logger . LogInformation ( "Running {Program}" , program ) ;
54
57
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
+
55
77
var stdOut = new StringBuilder ( ) ;
56
78
var stdErr = new StringBuilder ( ) ;
57
79
var command = new Command ( program . ToString ( ) )
@@ -60,11 +82,45 @@ public async Task Execute(Loadout loadout)
60
82
. WithValidation ( CommandResultValidation . None )
61
83
. WithWorkingDirectory ( program . Parent . ToString ( ) ) ;
62
84
63
-
64
- var result = await _processFactory . ExecuteAsync ( command ) ;
85
+
86
+ var result = await _processFactory . ExecuteAsync ( command , cancellationToken ) ;
65
87
if ( result . ExitCode != 0 )
66
88
_logger . LogError ( "While Running {Filename} : {Error} {Output}" , program , stdErr , stdOut ) ;
67
89
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
+
68
106
_logger . LogInformation ( "Finished running {Program}" , program ) ;
69
107
}
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
+ }
70
126
}
0 commit comments