@@ -31,11 +31,11 @@ private List<AppEntry> prevAppEntries
31
31
private List<AppEntry> prevAppEntries_;
32
32
*/
33
33
34
- public List < AppEntry > CurrentAppEntries
34
+ public IReadOnlyList < AppEntry > CurrentAppEntries
35
35
{
36
36
get
37
37
{
38
- return currentAppEntries ;
38
+ return currentAppEntries . AsReadOnly ( ) ;
39
39
}
40
40
}
41
41
private List < AppEntry > currentAppEntries ;
@@ -94,46 +94,54 @@ public AppEntrySaveHandler()
94
94
} finally { semaphoreSlim . Release ( ) ; }
95
95
96
96
97
- Logger . inst . Log ( currentAppEntries . Single ( x => x . AppPath == "F:\\ SteamLibrary\\ steamapps\\ common\\ Apex Legends\\ r5apex.exe" ) . ToString ( ) ) ;
98
- Logger . inst . Log ( prevAppEntries . Single ( x => x . AppPath == "F:\\ SteamLibrary\\ steamapps\\ common\\ Apex Legends\\ r5apex.exe" ) . ToString ( ) ) ;
97
+ // Logger.inst.Log(currentAppEntries.Single(x => x.AppPath == "F:\\SteamLibrary\\steamapps\\common\\Apex Legends\\r5apex.exe").ToString());
98
+ // Logger.inst.Log(prevAppEntries.Single(x => x.AppPath == "F:\\SteamLibrary\\steamapps\\common\\Apex Legends\\r5apex.exe").ToString());
99
99
}
100
100
101
- public void ChangeAppEntryByPath ( string path , AppEntry updatedAppEntry )
101
+ public void ChangeAppEntryByPathAndSave ( string path , AppEntry updatedAppEntry )
102
102
{
103
103
try
104
104
{
105
+ Logger . inst . Log ( $ "ChangeAppEntryByPath waiting on a semaphore ({ path } )") ;
105
106
semaphoreSlim . Wait ( ) ;
107
+ Logger . inst . Log ( $ "ChangeAppEntryByPath entered a semaphore ({ path } ).") ;
106
108
107
- Logger . inst . Log ( $ "prev: { CurrentAppEntries [ CurrentAppEntries . IndexOf ( CurrentAppEntries . Single ( x => x . AppPath == path ) ) ] } ") ;
109
+ Logger . inst . Log ( $ "Attempting to change \n { currentAppEntries . Single ( x=> x . AppPath == path ) } \n to \n { updatedAppEntry } ") ;
108
110
109
- int index = CurrentAppEntries . IndexOf ( CurrentAppEntries . Single ( x => x . AppPath == path ) ) ;
110
-
111
+ //Logger.inst.Log($"prev: {CurrentAppEntries[CurrentAppEntries.IndexOf(CurrentAppEntries.Single(x => x.AppPath == path))]}");
111
112
112
- /* //for-loop alternative, but the above should throw an error with an obvious enough meaning
113
- int index = -1;
114
- for (int i = 0; i < CurrentAppEntries.Count; i++)
115
- {
116
- AppEntry appEntry = CurrentAppEntries[i];
117
- if (appEntry.AppPath == path)
118
- {
119
- index = i; break;
120
- }
121
- }
113
+ int index = currentAppEntries . IndexOf ( CurrentAppEntries . Single ( x => x . AppPath == path ) ) ;
122
114
123
- if(index<0)
124
- throw new AppEntrySaverException($"UpdateAppEntry: No AppEntry with path {path} was found");
125
- */
115
+ currentAppEntries [ index ] = updatedAppEntry ;
126
116
127
- CurrentAppEntries [ index ] = updatedAppEntry ;
117
+ SaveAppEntryChanges_Internal ( ) ;
128
118
129
- Logger . inst . Log ( $ "new: { CurrentAppEntries [ CurrentAppEntries . IndexOf ( CurrentAppEntries . Single ( x => x . AppPath == path ) ) ] } ") ;
119
+ // Logger.inst.Log($"new: {CurrentAppEntries[CurrentAppEntries.IndexOf(CurrentAppEntries.Single(x => x.AppPath == path))]}");
130
120
} finally
131
121
{
132
- Logger . inst . Log ( "ChangeAppEntryByPath released a semaphore. " ) ;
122
+ Logger . inst . Log ( $ "ChangeAppEntryByPath released a semaphore ( { path } ) ") ;
133
123
semaphoreSlim . Release ( ) ;
134
124
}
135
125
}
136
126
127
+ public int RemoveAll ( Predicate < AppEntry > predicate )
128
+ {
129
+ return currentAppEntries . RemoveAll ( predicate ) ;
130
+ }
131
+
132
+ public void AddAppEntryAndSave ( AppEntry appEntry )
133
+ {
134
+ try
135
+ {
136
+ semaphoreSlim . Wait ( ) ;
137
+
138
+ currentAppEntries . Add ( appEntry ) ;
139
+
140
+ SaveAppEntryChanges_Internal ( ) ;
141
+
142
+ }
143
+ finally { semaphoreSlim . Release ( ) ; }
144
+ }
137
145
/*
138
146
* DECISION: Replacement or overwrite?
139
147
* Per-Entry Replacement (SELECTED):
@@ -147,71 +155,88 @@ public void ChangeAppEntryByPath(string path, AppEntry updatedAppEntry)
147
155
*
148
156
* UPDATE: Only per-entry eplacement is viable because the XML can still contain stuff that isn't in the registry!!
149
157
*/
158
+ /// <summary>
159
+ /// NOT THREAD SAFE
160
+ /// </summary>
150
161
public void SaveAppEntryChanges ( )
151
162
{
152
163
try
153
164
{
154
165
semaphoreSlim . Wait ( ) ;
155
166
156
- //Logger.inst.Log(currentAppEntries.Single(x => x.AppPath == "F:\\SteamLibrary\\steamapps\\common\\Apex Legends\\r5apex.exe").ToString());
157
- //Logger.inst.Log(prevAppEntries.Single(x => x.AppPath == "F:\\SteamLibrary\\steamapps\\common\\Apex Legends\\r5apex.exe").ToString());
158
- List < AppEntry > differences = new ( ) ;
159
- differences . AddRange ( currentAppEntries . Where ( entry => NotSameOrInPrevAppEntries ( entry ) ) ) ;
160
-
161
- List < string > existingAppPaths = new List < string > ( from appEntry in PreferencesXML . GetAppEntries ( ) select appEntry . AppPath ) ;
162
- List < AppEntry > needToAdd = new ( ) ;
163
- needToAdd . AddRange ( currentAppEntries . Where ( entry => ! existingAppPaths . Contains ( entry . AppPath ) ) ) ;
164
-
165
- //remove appentries whose *paths* exist in the prev but not the current
166
- List < AppEntry > needToRemoveFromXML = new ( ) ;
167
- foreach ( AppEntry entry in prevAppEntries )
168
- {
169
- if ( ! currentAppEntries . Exists ( a => a . AppPath == entry . AppPath ) )
170
- {
171
- needToRemoveFromXML . Add ( entry ) ;
172
- }
173
- }
167
+ SaveAppEntryChanges_Internal ( ) ;
168
+
169
+ } finally
170
+ {
171
+ semaphoreSlim . Release ( ) ;
172
+ }
173
+ }
174
174
175
- //new AppEntries should be added first so that ModifyAppEntry can actually find the AppEntry with the path
176
- foreach ( AppEntry appEntry in needToAdd )
175
+ private void SaveAppEntryChanges_Internal ( )
176
+ {
177
+ //Logger.inst.Log(currentAppEntries.Single(x => x.AppPath == "F:\\SteamLibrary\\steamapps\\common\\Apex Legends\\r5apex.exe").ToString());
178
+ //Logger.inst.Log(prevAppEntries.Single(x => x.AppPath == "F:\\SteamLibrary\\steamapps\\common\\Apex Legends\\r5apex.exe").ToString());
179
+ List < AppEntry > differences = new ( ) ;
180
+ differences . AddRange ( currentAppEntries . Where ( entry => NotSameOrInPrevAppEntries ( entry ) ) ) ;
181
+
182
+ List < string > existingAppPaths = new List < string > ( from appEntry in PreferencesXML . GetAppEntries ( ) select appEntry . AppPath ) ;
183
+ List < AppEntry > needToAdd = new ( ) ;
184
+ needToAdd . AddRange ( currentAppEntries . Where ( entry => ! existingAppPaths . Contains ( entry . AppPath ) ) ) ;
185
+
186
+ //remove appentries whose *paths* exist in the prev but not the current
187
+ List < AppEntry > needToRemoveFromXML = new ( ) ;
188
+ foreach ( AppEntry entry in prevAppEntries )
189
+ {
190
+ if ( ! currentAppEntries . Exists ( a => a . AppPath == entry . AppPath ) )
177
191
{
178
- PreferencesXML . AddAppEntryAndSave ( appEntry ) ;
192
+ needToRemoveFromXML . Add ( entry ) ;
179
193
}
194
+ }
180
195
181
- foreach ( AppEntry appEntry in differences )
182
- {
183
- PreferencesXML . ModifyAppEntryAndSave ( appEntry . AppPath , appEntry ) ;
184
- }
196
+ //new AppEntries should be added first so that ModifyAppEntry can actually find the AppEntry with the path
197
+ foreach ( AppEntry appEntry in needToAdd )
198
+ {
199
+ PreferencesXML . AddAppEntryAndSave ( appEntry ) ;
200
+ }
185
201
186
- foreach ( AppEntry appEntry in needToRemoveFromXML )
187
- {
188
- bool success = PreferencesXML . TryDeleteAppEntryAndSave ( appEntry . AppPath ) ;
189
- if ( ! success ) //this would probably only ever fail if AppEntries were deleted externally whilst the GUI or Service was still running
190
- throw new XMLHelperException ( $ "DeleteAppEntry: AppEntry with the specified path not found in data store: { appEntry . AppPath } ") ;
191
- }
202
+ foreach ( AppEntry appEntry in differences )
203
+ {
204
+ PreferencesXML . ModifyAppEntryAndSave ( appEntry . AppPath , appEntry ) ;
205
+ }
192
206
193
- prevAppEntries = DeepCopyAppEntries ( currentAppEntries ) ; //update the saved entries
207
+ foreach ( AppEntry appEntry in needToRemoveFromXML )
208
+ {
209
+ bool success = PreferencesXML . TryDeleteAppEntryAndSave ( appEntry . AppPath ) ;
210
+ if ( ! success ) //this would probably only ever fail if AppEntries were deleted externally whilst the GUI or Service was still running
211
+ throw new XMLHelperException ( $ "DeleteAppEntry: AppEntry with the specified path not found in data store: { appEntry . AppPath } ") ;
212
+ }
194
213
195
- //Logger.inst.Log(currentAppEntries.Single(x => x.AppPath == "F:\\SteamLibrary\\steamapps\\common\\Apex Legends\\r5apex.exe").ToString());
196
- //Logger.inst.Log(prevAppEntries.Single(x => x.AppPath == "F:\\SteamLibrary\\steamapps\\common\\Apex Legends\\r5apex.exe").ToString());
214
+ prevAppEntries = DeepCopyAppEntries ( currentAppEntries ) ; //update the saved entries
197
215
198
- Logger . inst . Log ( $ "Concluded saving. Differences: { differences . Count } Added: { needToAdd . Count } Removed: { needToRemoveFromXML . Count } ") ;
216
+ //Logger.inst.Log(currentAppEntries.Single(x => x.AppPath == "F:\\SteamLibrary\\steamapps\\common\\Apex Legends\\r5apex.exe").ToString());
217
+ //Logger.inst.Log(prevAppEntries.Single(x => x.AppPath == "F:\\SteamLibrary\\steamapps\\common\\Apex Legends\\r5apex.exe").ToString());
199
218
200
- bool NotSameOrInPrevAppEntries ( AppEntry appEntry )
219
+ Logger . inst . Log ( $ "Concluded saving. Differences: { differences . Count } Added: { needToAdd . Count } Removed: { needToRemoveFromXML . Count } ") ;
220
+ if ( differences . Count > 0 )
221
+ {
222
+ Logger . inst . Log ( "Changed:" ) ;
223
+ foreach ( AppEntry appEntry in differences )
201
224
{
202
- /*
203
- if (appEntry.AppPath == "F:\\SteamLibrary\\steamapps\\common\\Apex Legends\\r5apex.exe")
204
- {
205
- Logger.inst.Log(prevAppEntries.Single(x => x.AppPath == "F:\\SteamLibrary\\steamapps\\common\\Apex Legends\\r5apex.exe").ToString());
206
- Logger.inst.Log(CurrentAppEntries.Single(x => x.AppPath == "F:\\SteamLibrary\\steamapps\\common\\Apex Legends\\r5apex.exe").ToString());
207
- Logger.inst.Log($"{appEntry} not in prev app entries: {!prevAppEntries.Contains(appEntry)}");
208
- }
209
- */
210
- return ! prevAppEntries . Contains ( appEntry ) ; //Contains uses Equals() which is implemented in AppEntry
225
+ Logger . inst . Log ( appEntry . ToString ( ) ) ;
211
226
}
212
- } finally
227
+ }
228
+
229
+ bool NotSameOrInPrevAppEntries ( AppEntry appEntry )
213
230
{
214
- semaphoreSlim . Release ( ) ;
231
+ /*
232
+ if (appEntry.AppPath == "F:\\SteamLibrary\\steamapps\\common\\Apex Legends\\r5apex.exe")
233
+ {
234
+ Logger.inst.Log(prevAppEntries.Single(x => x.AppPath == "F:\\SteamLibrary\\steamapps\\common\\Apex Legends\\r5apex.exe").ToString());
235
+ Logger.inst.Log(CurrentAppEntries.Single(x => x.AppPath == "F:\\SteamLibrary\\steamapps\\common\\Apex Legends\\r5apex.exe").ToString());
236
+ Logger.inst.Log($"{appEntry} not in prev app entries: {!prevAppEntries.Contains(appEntry)}");
237
+ }
238
+ */
239
+ return ! prevAppEntries . Contains ( appEntry ) ; //Contains uses Equals() which is implemented in AppEntry
215
240
}
216
241
}
217
242
0 commit comments