Book a Demo!
CoCalc Logo Icon
StoreFeaturesDocsShareSupportNewsAboutPoliciesSign UpSign In
PojavLauncherTeam
GitHub Repository: PojavLauncherTeam/PojavLauncher_iOS
Path: blob/main/Natives/MinecraftResourceDownloadTask.m
589 views
1
#include <CommonCrypto/CommonDigest.h>
2
3
#import "authenticator/BaseAuthenticator.h"
4
#import "installer/modpack/ModpackAPI.h"
5
#import "AFNetworking.h"
6
#import "LauncherNavigationController.h"
7
#import "LauncherPreferences.h"
8
#import "MinecraftResourceDownloadTask.h"
9
#import "MinecraftResourceUtils.h"
10
#import "ios_uikit_bridge.h"
11
#import "utils.h"
12
13
@interface MinecraftResourceDownloadTask ()
14
@property AFURLSessionManager* manager;
15
@end
16
17
@implementation MinecraftResourceDownloadTask
18
19
- (instancetype)init {
20
self = [super init];
21
// TODO: implement background download
22
NSURLSessionConfiguration *configuration = [NSURLSessionConfiguration defaultSessionConfiguration];
23
configuration.timeoutIntervalForRequest = 86400;
24
//backgroundSessionConfigurationWithIdentifier:@"net.kdt.pojavlauncher.downloadtask"];
25
self.manager = [[AFURLSessionManager alloc] initWithSessionConfiguration:configuration];
26
self.fileList = [NSMutableArray new];
27
self.progressList = [NSMutableArray new];
28
return self;
29
}
30
31
// Add file to the queue
32
- (NSURLSessionDownloadTask *)createDownloadTask:(NSString *)url size:(NSUInteger)size sha:(NSString *)sha altName:(NSString *)altName toPath:(NSString *)path success:(void (^)())success {
33
BOOL fileExists = [NSFileManager.defaultManager fileExistsAtPath:path];
34
// logSuccess?
35
if (fileExists && [self checkSHA:sha forFile:path altName:altName]) {
36
if (success) success();
37
return nil;
38
} else if (![self checkAccessWithDialog:YES]) {
39
return nil;
40
}
41
42
NSString *name = altName ?: path.lastPathComponent;
43
NSURLRequest *request = [NSURLRequest requestWithURL:[NSURL URLWithString:url]];
44
__block NSProgress *progress;
45
__block NSURLSessionDownloadTask *task = [self.manager downloadTaskWithRequest:request progress:nil
46
destination:^NSURL * _Nonnull(NSURL * _Nonnull targetPath, NSURLResponse * _Nonnull response) {
47
NSLog(@"[MCDL] Downloading %@", name);
48
progress = [self.manager downloadProgressForTask:task];
49
if (!size && task) {
50
[self addDownloadTaskToProgress:task size:response.expectedContentLength];
51
[self.fileList addObject:name];
52
}
53
[NSFileManager.defaultManager createDirectoryAtPath:path.stringByDeletingLastPathComponent withIntermediateDirectories:YES attributes:nil error:nil];
54
[NSFileManager.defaultManager removeItemAtPath:path error:nil];
55
return [NSURL fileURLWithPath:path];
56
} completionHandler:^(NSURLResponse * _Nonnull response, NSURL * _Nullable filePath, NSError * _Nullable error) {
57
if (self.progress.cancelled) {
58
// Ignore any further errors
59
} else if (error != nil) {
60
[self finishDownloadWithError:error file:name];
61
} else if (![self checkSHA:sha forFile:path altName:altName]) {
62
[self finishDownloadWithErrorString:[NSString stringWithFormat:@"Failed to verify file %@: SHA1 mismatch", path.lastPathComponent]];
63
} else {
64
progress.totalUnitCount = progress.completedUnitCount;
65
if (success) success();
66
}
67
}];
68
69
if (size && task) {
70
[self addDownloadTaskToProgress:task size:size];
71
[self.fileList addObject:name];
72
}
73
74
return task;
75
}
76
77
- (NSURLSessionDownloadTask *)createDownloadTask:(NSString *)url size:(NSUInteger)size sha:(NSString *)sha altName:(NSString *)altName toPath:(NSString *)path {
78
return [self createDownloadTask:url size:size sha:sha altName:altName toPath:path success:nil];
79
}
80
81
- (void)addDownloadTaskToProgress:(NSURLSessionDownloadTask *)task size:(NSInteger)size {
82
NSProgress *progress = [self.manager downloadProgressForTask:task];
83
NSUInteger fileSize = size>0 ? size : 1;
84
progress.kind = NSProgressKindFile;
85
if (size > 0) {
86
progress.totalUnitCount = fileSize;
87
}
88
[self.progressList addObject:progress];
89
[self.progress addChild:progress withPendingUnitCount:fileSize];
90
self.progress.totalUnitCount += fileSize;
91
self.textProgress.totalUnitCount = self.progress.totalUnitCount;
92
}
93
94
- (void)downloadVersionMetadata:(NSDictionary *)version success:(void (^)())success {
95
// Download base json
96
NSString *versionStr = version[@"id"];
97
if ([versionStr isEqualToString:@"latest-release"]) {
98
versionStr = getPrefObject(@"internal.latest_version.release");
99
} else if ([versionStr isEqualToString:@"latest-snapshot"]) {
100
versionStr = getPrefObject(@"internal.latest_version.snapshot");
101
}
102
103
NSString *path = [NSString stringWithFormat:@"%1$s/versions/%2$@/%[email protected]", getenv("POJAV_GAME_DIR"), versionStr];
104
// Find it again to resolve latest-*
105
version = (id)[MinecraftResourceUtils findVersion:versionStr inList:remoteVersionList];
106
107
void(^completionBlock)(void) = ^{
108
self.metadata = parseJSONFromFile(path);
109
if (self.metadata[@"NSErrorObject"]) {
110
[self finishDownloadWithErrorString:[self.metadata[@"NSErrorObject"] localizedDescription]];
111
return;
112
}
113
if (self.metadata[@"inheritsFrom"]) {
114
NSMutableDictionary *inheritsFromDict = parseJSONFromFile([NSString stringWithFormat:@"%1$s/versions/%2$@/%[email protected]", getenv("POJAV_GAME_DIR"), self.metadata[@"inheritsFrom"]]);
115
if (inheritsFromDict) {
116
[MinecraftResourceUtils processVersion:self.metadata inheritsFrom:inheritsFromDict];
117
self.metadata = inheritsFromDict;
118
}
119
}
120
[MinecraftResourceUtils tweakVersionJson:self.metadata];
121
success();
122
};
123
124
if (!version) {
125
// This is likely local version, check if json exists and has inheritsFrom
126
NSMutableDictionary *json = parseJSONFromFile(path);
127
if (json[@"NSErrorObject"]) {
128
[self finishDownloadWithErrorString:[json[@"NSErrorObject"] localizedDescription]];
129
return;
130
} else if (json[@"inheritsFrom"]) {
131
version = (id)[MinecraftResourceUtils findVersion:json[@"inheritsFrom"] inList:remoteVersionList];
132
path = [NSString stringWithFormat:@"%1$s/versions/%2$@/%[email protected]", getenv("POJAV_GAME_DIR"), json[@"inheritsFrom"]];
133
} else {
134
completionBlock();
135
return;
136
}
137
}
138
139
versionStr = version[@"id"];
140
NSString *url = version[@"url"];
141
NSString *sha = url.stringByDeletingLastPathComponent.lastPathComponent;
142
NSUInteger size = [version[@"size"] unsignedLongLongValue];
143
144
NSURLSessionDownloadTask *task = [self createDownloadTask:url size:size sha:sha altName:nil toPath:path success:completionBlock];
145
[task resume];
146
}
147
148
#pragma mark - Minecraft installation
149
150
- (void)downloadAssetMetadataWithSuccess:(void (^)())success {
151
NSDictionary *assetIndex = self.metadata[@"assetIndex"];
152
if (!assetIndex) {
153
success();
154
return;
155
}
156
NSString *name = [NSString stringWithFormat:@"assets/indexes/%@.json", assetIndex[@"id"]];
157
NSString *path = [@(getenv("POJAV_GAME_DIR")) stringByAppendingPathComponent:name];
158
NSString *url = assetIndex[@"url"];
159
NSString *sha = url.stringByDeletingLastPathComponent.lastPathComponent;
160
NSUInteger size = [assetIndex[@"size"] unsignedLongLongValue];
161
NSURLSessionDownloadTask *task = [self createDownloadTask:url size:size sha:sha altName:name toPath:path success:^{
162
self.metadata[@"assetIndexObj"] = parseJSONFromFile(path);
163
success();
164
}];
165
[task resume];
166
}
167
168
- (NSArray *)downloadClientLibraries {
169
NSMutableArray *tasks = [NSMutableArray new];
170
for (NSDictionary *library in self.metadata[@"libraries"]) {
171
NSString *name = library[@"name"];
172
173
NSMutableDictionary *artifact = library[@"downloads"][@"artifact"];
174
if (artifact == nil && [name containsString:@":"]) {
175
NSLog(@"[MCDL] Unknown artifact object for %@, attempting to generate one", name);
176
artifact = [[NSMutableDictionary alloc] init];
177
NSString *prefix = library[@"url"] == nil ? @"https://libraries.minecraft.net/" : [library[@"url"] stringByReplacingOccurrencesOfString:@"http://" withString:@"https://"];
178
NSArray *libParts = [name componentsSeparatedByString:@":"];
179
artifact[@"path"] = [NSString stringWithFormat:@"%1$@/%2$@/%3$@/%2$@-%[email protected]", [libParts[0] stringByReplacingOccurrencesOfString:@"." withString:@"/"], libParts[1], libParts[2]];
180
artifact[@"url"] = [NSString stringWithFormat:@"%@%@", prefix, artifact[@"path"]];
181
artifact[@"sha1"] = library[@"checksums"][0];
182
}
183
184
NSString *path = [NSString stringWithFormat:@"%s/libraries/%@", getenv("POJAV_GAME_DIR"), artifact[@"path"]];
185
NSString *sha = artifact[@"sha1"];
186
NSUInteger size = [artifact[@"size"] unsignedLongLongValue];
187
NSString *url = artifact[@"url"];
188
if ([library[@"skip"] boolValue]) {
189
NSLog(@"[MDCL] Skipped library %@", name);
190
continue;
191
}
192
193
NSURLSessionDownloadTask *task = [self createDownloadTask:url size:size sha:sha altName:name toPath:path success:nil];
194
if (task) {
195
[tasks addObject:task];
196
} else if (self.progress.cancelled) {
197
return nil;
198
}
199
}
200
return tasks;
201
}
202
203
- (NSArray *)downloadClientAssets {
204
NSMutableArray *tasks = [NSMutableArray new];
205
NSDictionary *assets = self.metadata[@"assetIndexObj"];
206
if (!assets) {
207
return @[];
208
}
209
for (NSString *name in assets[@"objects"]) {
210
NSDictionary *object = assets[@"objects"][name];
211
NSString *hash = object[@"hash"];
212
NSString *pathname = [NSString stringWithFormat:@"%@/%@", [hash substringToIndex:2], hash];
213
NSUInteger size = [object[@"size"] unsignedLongLongValue];
214
215
NSString *path;
216
if ([assets[@"map_to_resources"] boolValue]) {
217
path = [NSString stringWithFormat:@"%s/resources/%@", getenv("POJAV_GAME_DIR"), name];
218
} else {
219
path = [NSString stringWithFormat:@"%s/assets/objects/%@", getenv("POJAV_GAME_DIR"), pathname];
220
}
221
222
/* Special case for 1.19+
223
* Since 1.19-pre1, setting the window icon on macOS invokes ObjC.
224
* However, if an IOException occurs, it won't try to set.
225
* We skip downloading the icon file to workaround this. */
226
if ([name hasSuffix:@"/minecraft.icns"]) {
227
[NSFileManager.defaultManager removeItemAtPath:path error:nil];
228
continue;
229
}
230
231
NSString *url = [NSString stringWithFormat:@"https://resources.download.minecraft.net/%@", pathname];
232
NSURLSessionDownloadTask *task = [self createDownloadTask:url size:size sha:hash altName:name toPath:path success:nil];
233
if (task) {
234
[tasks addObject:task];
235
} else if (self.progress.cancelled) {
236
return nil;
237
}
238
}
239
return tasks;
240
}
241
242
- (void)downloadVersion:(NSDictionary *)version {
243
[self prepareForDownload];
244
[self downloadVersionMetadata:version success:^{
245
[self downloadAssetMetadataWithSuccess:^{
246
NSArray *libTasks = [self downloadClientLibraries];
247
NSArray *assetTasks = [self downloadClientAssets];
248
// Drop the 1 byte we set initially
249
self.progress.totalUnitCount--;
250
self.textProgress.totalUnitCount--;
251
if (self.progress.totalUnitCount == 0) {
252
// We have nothing to download, invoke completion observer
253
self.progress.totalUnitCount = 1;
254
self.progress.completedUnitCount = 1;
255
self.textProgress.totalUnitCount = 1;
256
self.textProgress.completedUnitCount = 1;
257
return;
258
}
259
[libTasks makeObjectsPerformSelector:@selector(resume)];
260
[assetTasks makeObjectsPerformSelector:@selector(resume)];
261
[self.metadata removeObjectForKey:@"assetIndexObj"];
262
}];
263
}];
264
}
265
266
#pragma mark - Modpack installation
267
268
- (void)downloadModpackFromAPI:(ModpackAPI *)api detail:(NSDictionary *)modDetail atIndex:(NSUInteger)selectedVersion {
269
[self prepareForDownload];
270
271
NSString *url = modDetail[@"versionUrls"][selectedVersion];
272
NSUInteger size = [modDetail[@"versionSizes"][selectedVersion] unsignedLongLongValue];
273
NSString *sha = modDetail[@"versionHashes"][selectedVersion];
274
NSString *name = [[modDetail[@"title"] lowercaseString] stringByTrimmingCharactersInSet:NSCharacterSet.whitespaceCharacterSet];
275
name = [name stringByReplacingOccurrencesOfString:@" " withString:@"_"];
276
NSString *packagePath = [NSTemporaryDirectory() stringByAppendingFormat:@"/%@.zip", name];
277
278
NSURLSessionDownloadTask *task = [self createDownloadTask:url size:size sha:sha altName:nil toPath:packagePath success:^{
279
NSString *path = [NSString stringWithFormat:@"%s/custom_gamedir/%@", getenv("POJAV_GAME_DIR"), name];
280
[api downloader:self submitDownloadTasksFromPackage:packagePath toPath:path];
281
}];
282
[task resume];
283
}
284
285
#pragma mark - Utilities
286
287
- (void)prepareForDownload {
288
// Create a fake progress which is used to update completedUnitCount properly
289
// (completedUnitCount does not update unless subprogress completes)
290
self.textProgress = [NSProgress new];
291
self.textProgress.kind = NSProgressKindFile;
292
self.textProgress.fileOperationKind = NSProgressFileOperationKindDownloading;
293
self.textProgress.totalUnitCount = -1;
294
295
self.progress = [NSProgress new];
296
// Push 1 byte so it won't accidentally finish after downloading assets index
297
self.progress.totalUnitCount = 1;
298
[self.fileList removeAllObjects];
299
[self.progressList removeAllObjects];
300
}
301
302
- (void)finishDownloadWithErrorString:(NSString *)error {
303
[self.progress cancel];
304
[self.manager invalidateSessionCancelingTasks:YES resetSession:YES];
305
showDialog(localize(@"Error", nil), error);
306
self.handleError();
307
}
308
309
- (void)finishDownloadWithError:(NSError *)error file:(NSString *)file {
310
NSString *errorStr = [NSString stringWithFormat:localize(@"launcher.mcl.error_download", NULL), file, error.localizedDescription];
311
NSLog(@"[MCDL] Error: %@ %@", errorStr, NSThread.callStackSymbols);
312
[self finishDownloadWithErrorString:errorStr];
313
}
314
315
// Check if the account has permission to download
316
- (BOOL)checkAccessWithDialog:(BOOL)show {
317
// for now
318
BOOL accessible = [BaseAuthenticator.current.authData[@"username"] hasPrefix:@"Demo."] || BaseAuthenticator.current.authData[@"xboxGamertag"] != nil;
319
if (!accessible) {
320
[self.progress cancel];
321
if (show) {
322
[self finishDownloadWithErrorString:@"Minecraft can't be legally installed when logged in with a local account. Please switch to an online account to continue."];
323
}
324
}
325
return accessible;
326
}
327
328
// Check SHA of the file
329
- (BOOL)checkSHAIgnorePref:(NSString *)sha forFile:(NSString *)path altName:(NSString *)altName logSuccess:(BOOL)logSuccess {
330
if (sha.length == 0) {
331
// When sha = skip, only check for file existence
332
BOOL existence = [NSFileManager.defaultManager fileExistsAtPath:path];
333
if (existence) {
334
NSLog(@"[MCDL] Warning: couldn't find SHA for %@, have to assume it's good.", path);
335
}
336
return existence;
337
}
338
339
NSData *data = [NSData dataWithContentsOfFile:path];
340
if (data == nil) {
341
NSLog(@"[MCDL] SHA1 checker: file doesn't exist: %@", altName ? altName : path.lastPathComponent);
342
return NO;
343
}
344
345
unsigned char digest[CC_SHA1_DIGEST_LENGTH];
346
CC_SHA1(data.bytes, (CC_LONG)data.length, digest);
347
NSMutableString *localSHA = [NSMutableString stringWithCapacity:CC_SHA1_DIGEST_LENGTH * 2];
348
for(int i = 0; i < CC_SHA1_DIGEST_LENGTH; i++) {
349
[localSHA appendFormat:@"%02x", digest[i]];
350
}
351
352
BOOL check = [sha isEqualToString:localSHA];
353
if (!check || (getPrefBool(@"general.debug_logging") && logSuccess)) {
354
NSLog(@"[MCDL] SHA1 %@ for %@%@",
355
(check ? @"passed" : @"failed"),
356
(altName ? altName : path.lastPathComponent),
357
(check ? @"" : [NSString stringWithFormat:@" (expected: %@, got: %@)", sha, localSHA]));
358
}
359
return check;
360
}
361
362
- (BOOL)checkSHA:(NSString *)sha forFile:(NSString *)path altName:(NSString *)altName logSuccess:(BOOL)logSuccess {
363
if (getPrefBool(@"general.check_sha")) {
364
return [self checkSHAIgnorePref:sha forFile:path altName:altName logSuccess:logSuccess];
365
} else {
366
return [NSFileManager.defaultManager fileExistsAtPath:path];
367
}
368
}
369
370
- (BOOL)checkSHA:(NSString *)sha forFile:(NSString *)path altName:(NSString *)altName {
371
return [self checkSHA:sha forFile:path altName:altName logSuccess:altName==nil];
372
}
373
374
@end
375
376