-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathImportTask.cs
295 lines (237 loc) · 9.84 KB
/
ImportTask.cs
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
using Blake3;
using gfoidl.Base64;
using Heliosphere.Exceptions;
using Heliosphere.Model;
using Heliosphere.Util;
using StrawberryShake;
namespace Heliosphere;
internal class ImportTask : IDisposable {
internal ImportTaskState State { get; private set; } = ImportTaskState.NotRunning;
internal uint StateCurrent { get; private set; }
internal uint StateMax { get; private set; }
internal FirstHalfData? Data { get; private set; }
private Plugin Plugin { get; }
private string DirectoryName { get; }
private string ModName { get; }
private Guid PackageId { get; }
private Guid VariantId { get; }
private Guid VersionId { get; }
private string Version { get; }
private string? DownloadKey { get; }
private string? _penumbraPath;
private string? _fullDirectory;
internal ImportTask(
Plugin plugin,
string directoryName,
string modName,
Guid packageId,
Guid variantId,
Guid versionId,
string version,
string? downloadKey
) {
this.Plugin = plugin;
this.DirectoryName = directoryName;
this.ModName = modName;
this.PackageId = packageId;
this.VariantId = variantId;
this.VersionId = versionId;
this.Version = version;
this.DownloadKey = downloadKey;
}
/// <inheritdoc />
public void Dispose() {
}
internal void Start() {
Task.Factory.StartNew(async () => {
try {
this.State = ImportTaskState.Hashing;
var hashes = await this.Hash();
this.State = ImportTaskState.GettingFileList;
var files = await this.GetFiles();
this.State = ImportTaskState.Checking;
var fileCounts = await this.Check(hashes, files);
this.Data = new FirstHalfData {
Files = fileCounts,
HashedFiles = hashes,
NeededFiles = files,
};
this.State = ImportTaskState.WaitingForConfirmation;
this.StateCurrent = this.StateMax = 0;
} catch (Exception ex) {
Plugin.Log.Error(ex, "Exception when running import task");
this.State = ImportTaskState.Errored;
}
});
}
internal void Continue() {
Task.Factory.StartNew(async () => {
if (this.Data == null) {
throw new InvalidOperationException("called Continue but Start was never called/did not complete successfully");
}
this.State = ImportTaskState.Renaming;
this.Rename();
this.State = ImportTaskState.Deleting;
this.Delete();
this.State = ImportTaskState.StartingDownload;
await this.StartDownload();
});
}
private async Task<Dictionary<string, List<string>>> Hash(CancellationToken token = default) {
this.StateCurrent = this.StateMax = 0;
if (!this.Plugin.Penumbra.TryGetModDirectory(out this._penumbraPath)) {
throw new Exception("Penumbra is not set up or is not loaded");
}
using var semaphore = new SemaphoreSlim(Environment.ProcessorCount, Environment.ProcessorCount);
this._fullDirectory = Path.Join(this._penumbraPath, this.DirectoryName);
var tasks = Directory.EnumerateFiles(this._fullDirectory, "*", SearchOption.AllDirectories)
// ReSharper disable once AccessToDisposedClosure
// disposed after this task has completed, so it's fine
.Select(filePath => this.HashFile(semaphore, filePath, token));
var rawHashes = await Task.WhenAll(tasks);
return rawHashes
.GroupBy(tuple => tuple.hash)
.ToDictionary(
g => g.Key,
g => g.Select(tuple => tuple.filePath).ToList()
);
}
private async Task<(string hash, string filePath)> HashFile(SemaphoreSlim semaphore, string filePath, CancellationToken token = default) {
using var guard = await SemaphoreGuard.WaitAsync(semaphore, token);
using var hasher = new Blake3HashAlgorithm();
hasher.Initialize();
await using var file = FileHelper.OpenRead(filePath);
var hashBytes = await hasher.ComputeHashAsync(file, token);
var hash = Base64.Url.Encode(hashBytes);
this.StateCurrent += 1;
return (hash, filePath);
}
private async Task<Dictionary<string, List<NeededFile>>> GetFiles(CancellationToken token = default) {
this.StateCurrent = this.StateMax = 0;
var result = await Plugin.GraphQl.Importer.ExecuteAsync(this.VersionId, this.DownloadKey, token);
result.EnsureNoErrors();
var files = result.Data?.GetVersion?.NeededFiles.Files ?? throw new MissingVersionException(this.VersionId);
// NOTE: meta files will always have to be redownloaded, since penumbra
// deletes them after import, so there's no reason to check for
// them
var filtered = new Dictionary<string, List<NeededFile>>();
foreach (var (hash, list) in files.Files) {
foreach (var file in list.Values) {
var filteredList = file
.Where(item => !item.GamePath.EndsWith(".meta"))
.ToList();
if (filteredList.Count > 0) {
if (!filtered.TryGetValue(hash, out var needed)) {
needed = [];
filtered[hash] = needed;
}
needed.AddRange(file);
}
}
}
return filtered;
}
private Task<(uint Have, uint Needed)> Check(
IReadOnlyDictionary<string, List<string>> hashes,
Dictionary<string, List<NeededFile>> files
) {
var needed = (uint) files.Count;
var have = 0u;
this.StateCurrent = 0;
this.StateMax = needed;
foreach (var (hash, _) in files) {
if (hashes.ContainsKey(hash)) {
have += 1;
}
this.StateCurrent += 1;
}
return Task.FromResult((have, needed));
}
private void Rename() {
this.StateCurrent = 0;
this.StateMax = this.Data!.Files.Have;
// first create the files directory
var filesPath = Path.GetFullPath(Path.Join(this._fullDirectory!, "files"));
Directory.CreateDirectory(filesPath);
// rename all the files we have and need to their hashes
foreach (var (hash, files) in this.Data.NeededFiles) {
if (!this.Data.HashedFiles.TryGetValue(hash, out var paths)) {
continue;
}
// note that the DownloadTask will duplicate the files for us, so we
// only need to rename once to one extension
var ext = files
.Select(item => Path.GetExtension(item.GamePath))
.FirstOrDefault(item => !string.IsNullOrWhiteSpace(item));
if (ext == null) {
Plugin.Log.Warning($"file with no extension: {hash}");
continue;
}
var newPath = Path.ChangeExtension(Path.Join(filesPath, hash), ext);
File.Move(paths[0], newPath);
this.StateCurrent += 1;
}
// lastly, rename the directory itself
var newDirName = HeliosphereMeta.ModDirectoryName(this.PackageId, this.ModName, this.Version, this.VariantId);
var newDirPath = Path.Join(this._penumbraPath!, newDirName);
Directory.Move(this._fullDirectory!, newDirPath);
this._fullDirectory = newDirPath;
}
private void Delete() {
// the DownloadTask will create all the necessary metadata for us, so
// we can delete everything outside the files directory - the
// DownloadTask will delete anything extra inside the files directory
this.StateCurrent = 0;
this.StateMax = 0;
// delete all non-"files" directories
foreach (var dirPath in Directory.EnumerateDirectories(this._fullDirectory!)) {
if (Path.GetFileName(dirPath) == "files") {
continue;
}
Directory.Delete(dirPath, true);
}
// delete all top-level files
foreach (var filePath in Directory.EnumerateFiles(this._fullDirectory!)) {
FileHelper.Delete(filePath);
}
// delete the old mod from penumbra
this.Plugin.Penumbra.DeleteMod(this.DirectoryName);
// copy the settings from the old mod to the new one
this.Plugin.Penumbra.CopyModSettings(this.DirectoryName, Path.GetFileName(this._fullDirectory!));
}
private async Task StartDownload(CancellationToken token = default) {
this.StateCurrent = 0;
this.StateMax = 1;
await this.Plugin.AddDownloadAsync(new DownloadTask {
Plugin = this.Plugin,
ModDirectory = this._penumbraPath!,
PackageId = this.PackageId,
VariantId = this.VariantId,
VersionId = this.VersionId,
IncludeTags = this.Plugin.Config.IncludeTags,
OpenInPenumbra = this.Plugin.Config.OpenPenumbraAfterInstall,
PenumbraCollection = this.Plugin.Config.OneClickCollectionId,
DownloadKey = this.DownloadKey,
Full = true,
Options = [],
Notification = null,
}, token);
this.StateCurrent += 1;
}
}
internal class FirstHalfData {
internal required (uint Have, uint Needed) Files { get; init; }
internal required Dictionary<string, List<NeededFile>> NeededFiles { get; init; }
internal required Dictionary<string, List<string>> HashedFiles { get; init; }
}
internal enum ImportTaskState {
NotRunning,
Hashing,
GettingFileList,
Checking,
WaitingForConfirmation,
Renaming,
Deleting,
StartingDownload,
Errored,
}