292 lines
		
	
	
	
		
			12 KiB
		
	
	
	
		
			Objective-C
		
	
	
	
	
	
			
		
		
	
	
			292 lines
		
	
	
	
		
			12 KiB
		
	
	
	
		
			Objective-C
		
	
	
	
	
	
| /*
 | |
| This file is part of Telegram Desktop,
 | |
| the official desktop application for the Telegram messaging service.
 | |
| 
 | |
| For license and copyright information please follow this link:
 | |
| https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
 | |
| */
 | |
| #import <Cocoa/Cocoa.h>
 | |
| #include <sys/xattr.h>
 | |
| 
 | |
| NSString *appName = @"Telegram.app";
 | |
| NSString *appDir = nil;
 | |
| NSString *workDir = nil;
 | |
| 
 | |
| #ifdef _DEBUG
 | |
| BOOL _debug = YES;
 | |
| #else
 | |
| BOOL _debug = NO;
 | |
| #endif
 | |
| 
 | |
| NSFileHandle *_logFile = nil;
 | |
| void openLog() {
 | |
| 	if (!_debug || _logFile) return;
 | |
| 	NSString *logDir = [workDir stringByAppendingString:@"DebugLogs"];
 | |
| 	if (![[NSFileManager defaultManager] createDirectoryAtPath:logDir withIntermediateDirectories:YES attributes:nil error:nil]) {
 | |
| 		return;
 | |
| 	}
 | |
| 
 | |
| 	NSDateFormatter *fmt = [[NSDateFormatter alloc] initWithDateFormat:@"DebugLogs/%Y%m%d_%H%M%S_upd.txt" allowNaturalLanguage:NO];
 | |
| 	NSString *logPath = [workDir stringByAppendingString:[fmt stringFromDate:[NSDate date]]];
 | |
| 	[[NSFileManager defaultManager] createFileAtPath:logPath contents:nil attributes:nil];
 | |
| 	_logFile = [NSFileHandle fileHandleForWritingAtPath:logPath];
 | |
| }
 | |
| 
 | |
| void closeLog() {
 | |
| 	if (!_logFile) return;
 | |
| 
 | |
| 	[_logFile closeFile];
 | |
| }
 | |
| 
 | |
| void writeLog(NSString *msg) {
 | |
| 	if (!_logFile) return;
 | |
| 
 | |
| 	[_logFile writeData:[[msg stringByAppendingString:@"\n"] dataUsingEncoding:NSUTF8StringEncoding]];
 | |
| 	[_logFile synchronizeFile];
 | |
| }
 | |
| 
 | |
| void RemoveQuarantineAttribute(NSString *path) {
 | |
| 	const char *kQuarantineAttribute = "com.apple.quarantine";
 | |
| 
 | |
| 	writeLog([@"Removing quarantine: " stringByAppendingString:path]);
 | |
| 	removexattr([path fileSystemRepresentation], kQuarantineAttribute, 0);
 | |
| }
 | |
| 
 | |
| void RemoveQuarantineFromBundle(NSString *path) {
 | |
| 	RemoveQuarantineAttribute(path);
 | |
| 	RemoveQuarantineAttribute([path stringByAppendingString:@"/Contents/MacOS/Telegram"]);
 | |
| 	RemoveQuarantineAttribute([path stringByAppendingString:@"/Contents/Helpers/crashpad_handler"]);
 | |
| 	RemoveQuarantineAttribute([path stringByAppendingString:@"/Contents/Frameworks/Updater"]);
 | |
| }
 | |
| 
 | |
| void delFolder() {
 | |
| 	writeLog([@"Fully clearing old path: " stringByAppendingString:[workDir stringByAppendingString:@"tupdates/ready"]]);
 | |
| 	if (![[NSFileManager defaultManager] removeItemAtPath:[workDir stringByAppendingString:@"tupdates/ready"] error:nil]) {
 | |
| 		writeLog(@"Failed to clear old path! :( New path was used?..");
 | |
| 	}
 | |
| 	writeLog([@"Fully clearing new path: " stringByAppendingString:[workDir stringByAppendingString:@"tupdates/temp"]]);
 | |
| 	if (![[NSFileManager defaultManager] removeItemAtPath:[workDir stringByAppendingString:@"tupdates/temp"] error:nil]) {
 | |
| 		writeLog(@"Error: failed to clear new path! :(");
 | |
| 	}
 | |
| 	rmdir([[workDir stringByAppendingString:@"tupdates"] fileSystemRepresentation]);
 | |
| }
 | |
| 
 | |
| int main(int argc, const char * argv[]) {
 | |
| 	NSString *path = [[NSBundle mainBundle] bundlePath];
 | |
| 	if (!path) {
 | |
| 		return -1;
 | |
| 	}
 | |
| 	NSRange range = [path rangeOfString:@".app/" options:NSBackwardsSearch];
 | |
| 	if (range.location == NSNotFound) {
 | |
| 		return -1;
 | |
| 	}
 | |
| 	path = [path substringToIndex:range.location > 0 ? range.location : 0];
 | |
| 
 | |
| 	range = [path rangeOfString:@"/" options:NSBackwardsSearch];
 | |
| 	NSString *appRealName = (range.location == NSNotFound) ? path : [path substringFromIndex:range.location + 1];
 | |
| 	appRealName = [[NSArray arrayWithObjects:appRealName, @".app", nil] componentsJoinedByString:@""];
 | |
| 	appDir = (range.location == NSNotFound) ? @"" : [path substringToIndex:range.location + 1];
 | |
| 	NSString *appDirFull = [appDir stringByAppendingString:appRealName];
 | |
| 
 | |
| 	openLog();
 | |
| 	pid_t procId = 0;
 | |
| 	BOOL update = YES, toSettings = NO, autoStart = NO, startInTray = NO, testMode = NO, freeType = NO, externalUpdater = NO;
 | |
| 	BOOL customWorkingDir = NO;
 | |
| 	NSString *key = nil;
 | |
| 	for (int i = 0; i < argc; ++i) {
 | |
| 		if ([@"-workpath" isEqualToString:[NSString stringWithUTF8String:argv[i]]]) {
 | |
| 			if (++i < argc) {
 | |
| 				workDir = [NSString stringWithUTF8String:argv[i]];
 | |
| 			}
 | |
| 		} else if ([@"-procid" isEqualToString:[NSString stringWithUTF8String:argv[i]]]) {
 | |
| 			if (++i < argc) {
 | |
| 				NSNumberFormatter *formatter = [[NSNumberFormatter alloc] init];
 | |
| 				[formatter setNumberStyle:NSNumberFormatterDecimalStyle];
 | |
| 				procId = [[formatter numberFromString:[NSString stringWithUTF8String:argv[i]]] intValue];
 | |
| 			}
 | |
| 		} else if ([@"-noupdate" isEqualToString:[NSString stringWithUTF8String:argv[i]]]) {
 | |
| 			update = NO;
 | |
| 		} else if ([@"-tosettings" isEqualToString:[NSString stringWithUTF8String:argv[i]]]) {
 | |
| 			toSettings = YES;
 | |
| 		} else if ([@"-autostart" isEqualToString:[NSString stringWithUTF8String:argv[i]]]) {
 | |
| 			autoStart = YES;
 | |
| 		} else if ([@"-debug" isEqualToString:[NSString stringWithUTF8String:argv[i]]]) {
 | |
| 			_debug = YES;
 | |
| 		} else if ([@"-startintray" isEqualToString:[NSString stringWithUTF8String:argv[i]]]) {
 | |
| 			startInTray = YES;
 | |
| 		} else if ([@"-testmode" isEqualToString:[NSString stringWithUTF8String:argv[i]]]) {
 | |
| 			testMode = YES;
 | |
| 		} else if ([@"-freetype" isEqualToString:[NSString stringWithUTF8String:argv[i]]]) {
 | |
| 			freeType = YES;
 | |
| 		} else if ([@"-externalupdater" isEqualToString:[NSString stringWithUTF8String:argv[i]]]) {
 | |
| 			externalUpdater = YES;
 | |
| 		} else if ([@"-workdir_custom" isEqualToString:[NSString stringWithUTF8String:argv[i]]]) {
 | |
| 			customWorkingDir = YES;
 | |
| 		} else if ([@"-key" isEqualToString:[NSString stringWithUTF8String:argv[i]]]) {
 | |
| 			if (++i < argc) key = [NSString stringWithUTF8String:argv[i]];
 | |
| 		}
 | |
| 	}
 | |
| 	if (!workDir) {
 | |
| 		workDir = appDir;
 | |
| 		customWorkingDir = NO;
 | |
| 	}
 | |
| 	openLog();
 | |
| 	NSMutableArray *argsArr = [[NSMutableArray alloc] initWithCapacity:argc];
 | |
| 	for (int i = 0; i < argc; ++i) {
 | |
| 		[argsArr addObject:[NSString stringWithUTF8String:argv[i]]];
 | |
| 	}
 | |
| 	writeLog([[NSArray arrayWithObjects:@"Arguments: '", [argsArr componentsJoinedByString:@"' '"], @"'..", nil] componentsJoinedByString:@""]);
 | |
| 	if (key) writeLog([@"Key: " stringByAppendingString:key]);
 | |
| 	if (toSettings) writeLog(@"To Settings!");
 | |
| 
 | |
| 	if (procId) {
 | |
| 		NSRunningApplication *app = [NSRunningApplication runningApplicationWithProcessIdentifier:procId];
 | |
| 		for (int i = 0; i < 5 && app != nil && ![app isTerminated]; ++i) {
 | |
| 			usleep(200000);
 | |
| 			app = [NSRunningApplication runningApplicationWithProcessIdentifier:procId];
 | |
| 		}
 | |
| 		if (app) [app forceTerminate];
 | |
| 		app = [NSRunningApplication runningApplicationWithProcessIdentifier:procId];
 | |
| 		for (int i = 0; i < 5 && app != nil && ![app isTerminated]; ++i) {
 | |
| 			usleep(200000);
 | |
| 			app = [NSRunningApplication runningApplicationWithProcessIdentifier:procId];
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	if (update) {
 | |
| 		NSFileManager *fileManager = [NSFileManager defaultManager];
 | |
| 		NSString *readyFilePath = [workDir stringByAppendingString:@"tupdates/temp/ready"];
 | |
| 		NSString *srcDir = [workDir stringByAppendingString:@"tupdates/temp/"], *srcEnum = [workDir stringByAppendingString:@"tupdates/temp"];
 | |
| 		if ([fileManager fileExistsAtPath:readyFilePath]) {
 | |
| 			writeLog([@"Ready file found! Using new path: " stringByAppendingString: srcEnum]);
 | |
| 		} else {
 | |
| 			srcDir = [workDir stringByAppendingString:@"tupdates/ready/"]; // old
 | |
| 			srcEnum = [workDir stringByAppendingString:@"tupdates/ready"];
 | |
| 			writeLog([@"Ready file not found! Using old path: " stringByAppendingString: srcEnum]);
 | |
| 		}
 | |
| 
 | |
| 		writeLog([@"Starting update files iteration, path: " stringByAppendingString: srcEnum]);
 | |
| 
 | |
| 		// Take the Updater (this currently running binary) from the place where it was placed by Telegram
 | |
| 		// and copy it to the folder with the new version of the app (ready),
 | |
| 		// so it won't be deleted when we will clear the "Telegram.app/Contents" folder.
 | |
| 		NSString *oldVersionUpdaterPath = [appDirFull stringByAppendingString: @"/Contents/Frameworks/Updater" ];
 | |
| 		NSString *newVersionUpdaterPath = [srcEnum stringByAppendingString:[[NSArray arrayWithObjects:@"/", appName, @"/Contents/Frameworks/Updater", nil] componentsJoinedByString:@""]];
 | |
| 		writeLog([[NSArray arrayWithObjects: @"Copying Updater from old path ", oldVersionUpdaterPath, @" to new path ", newVersionUpdaterPath, nil] componentsJoinedByString:@""]);
 | |
| 		if (![fileManager fileExistsAtPath:newVersionUpdaterPath]) {
 | |
| 			if (![fileManager copyItemAtPath:oldVersionUpdaterPath toPath:newVersionUpdaterPath error:nil]) {
 | |
| 				writeLog([[NSArray arrayWithObjects: @"Failed to copy file from ", oldVersionUpdaterPath, @" to ", newVersionUpdaterPath, nil] componentsJoinedByString:@""]);
 | |
| 				delFolder();
 | |
| 				return -1;
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 
 | |
| 		NSString *contentsPath = [appDirFull stringByAppendingString: @"/Contents"];
 | |
| 		writeLog([[NSArray arrayWithObjects: @"Clearing dir ", contentsPath, nil] componentsJoinedByString:@""]);
 | |
| 		if (![fileManager removeItemAtPath:contentsPath error:nil]) {
 | |
| 			writeLog([@"Failed to clear path for directory " stringByAppendingString:contentsPath]);
 | |
| 			delFolder();
 | |
| 			return -1;
 | |
| 		}
 | |
| 
 | |
| 		NSArray *keys = [NSArray arrayWithObject:NSURLIsDirectoryKey];
 | |
| 		NSDirectoryEnumerator *enumerator = [fileManager
 | |
| 											 enumeratorAtURL:[NSURL fileURLWithPath:srcEnum]
 | |
| 											 includingPropertiesForKeys:keys
 | |
| 											 options:0
 | |
| 											 errorHandler:^(NSURL *url, NSError *error) {
 | |
| 												 writeLog([[[@"Error in enumerating " stringByAppendingString:[url absoluteString]] stringByAppendingString: @" error is: "] stringByAppendingString: [error description]]);
 | |
| 												 return NO;
 | |
| 											 }];
 | |
| 		for (NSURL *url in enumerator) {
 | |
| 			NSString *srcPath = [url path];
 | |
| 			writeLog([@"Handling file " stringByAppendingString:srcPath]);
 | |
| 			NSRange r = [srcPath rangeOfString:srcDir];
 | |
| 			if (r.location != 0) {
 | |
| 				writeLog([@"Bad file found, no base path " stringByAppendingString:srcPath]);
 | |
| 				delFolder();
 | |
| 				break;
 | |
| 			}
 | |
| 			NSString *pathPart = [srcPath substringFromIndex:r.length];
 | |
| 			r = [pathPart rangeOfString:appName];
 | |
| 			if (r.location != 0) {
 | |
| 				writeLog([@"Skipping not app file " stringByAppendingString:srcPath]);
 | |
| 				continue;
 | |
| 			}
 | |
| 			NSString *dstPath = [appDirFull stringByAppendingString:[pathPart substringFromIndex:r.length]];
 | |
| 			NSError *error;
 | |
| 			NSNumber *isDirectory = nil;
 | |
| 			if (![url getResourceValue:&isDirectory forKey:NSURLIsDirectoryKey error:&error]) {
 | |
| 				writeLog([@"Failed to get IsDirectory for file " stringByAppendingString:[url path]]);
 | |
| 				delFolder();
 | |
| 				break;
 | |
| 			}
 | |
| 			if ([isDirectory boolValue]) {
 | |
| 				writeLog([[NSArray arrayWithObjects: @"Copying dir ", srcPath, @" to ", dstPath, nil] componentsJoinedByString:@""]);
 | |
| 				if (![fileManager createDirectoryAtPath:dstPath withIntermediateDirectories:YES attributes:nil error:nil]) {
 | |
| 					writeLog([@"Failed to force path for directory " stringByAppendingString:dstPath]);
 | |
| 					delFolder();
 | |
| 					break;
 | |
| 				}
 | |
| 			} else if ([srcPath isEqualToString:readyFilePath]) {
 | |
| 				writeLog([[NSArray arrayWithObjects: @"Skipping ready file ", srcPath, nil] componentsJoinedByString:@""]);
 | |
| 			} else if ([fileManager fileExistsAtPath:dstPath]) {
 | |
| 				if (![[NSData dataWithContentsOfFile:srcPath] writeToFile:dstPath atomically:YES]) {
 | |
| 					writeLog([@"Failed to edit file " stringByAppendingString:dstPath]);
 | |
| 					delFolder();
 | |
| 					break;
 | |
| 				}
 | |
| 			} else {
 | |
| 				if (![fileManager copyItemAtPath:srcPath toPath:dstPath error:nil]) {
 | |
| 					writeLog([@"Failed to copy file to " stringByAppendingString:dstPath]);
 | |
| 					delFolder();
 | |
| 					break;
 | |
| 				}
 | |
| 			}
 | |
| 		}
 | |
| 		delFolder();
 | |
| 	}
 | |
| 
 | |
| 	NSString *appPath = [[NSArray arrayWithObjects:appDir, appRealName, nil] componentsJoinedByString:@""];
 | |
| 
 | |
| 	RemoveQuarantineFromBundle(appPath);
 | |
| 
 | |
| 	NSMutableArray *args = [[NSMutableArray alloc] initWithObjects: @"-noupdate", nil];
 | |
| 	if (toSettings) [args addObject:@"-tosettings"];
 | |
| 	if (_debug) [args addObject:@"-debug"];
 | |
| 	if (startInTray) [args addObject:@"-startintray"];
 | |
| 	if (testMode) [args addObject:@"-testmode"];
 | |
| 	if (freeType) [args addObject:@"-freetype"];
 | |
| 	if (externalUpdater) [args addObject:@"-externalupdater"];
 | |
| 	if (autoStart) [args addObject:@"-autostart"];
 | |
| 	if (key) {
 | |
| 		[args addObject:@"-key"];
 | |
| 		[args addObject:key];
 | |
| 	}
 | |
| 	if (customWorkingDir) {
 | |
| 		[args addObject:@"-workdir"];
 | |
| 		[args addObject:workDir];
 | |
| 	}
 | |
| 	writeLog([[NSArray arrayWithObjects:@"Running application '", appPath, @"' with args '", [args componentsJoinedByString:@"' '"], @"'..", nil] componentsJoinedByString:@""]);
 | |
| 
 | |
| 	for (int i = 0; i < 5; ++i) {
 | |
| 		NSError *error = nil;
 | |
| 		NSRunningApplication *result = [[NSWorkspace sharedWorkspace]
 | |
| 					launchApplicationAtURL:[NSURL fileURLWithPath:appPath]
 | |
| 					options:NSWorkspaceLaunchDefault
 | |
| 					configuration:[NSDictionary
 | |
| 								   dictionaryWithObject:args
 | |
| 								   forKey:NSWorkspaceLaunchConfigurationArguments]
 | |
| 					error:&error];
 | |
| 		if (result) {
 | |
| 			closeLog();
 | |
| 			return 0;
 | |
| 		}
 | |
| 		writeLog([[NSString stringWithFormat:@"Could not run application, error %ld: ", (long)[error code]] stringByAppendingString: error ? [error localizedDescription] : @"(nil)"]);
 | |
| 		usleep(200000);
 | |
| 	}
 | |
| 	closeLog();
 | |
| 	return -1;
 | |
| }
 | |
| 
 | 
