123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475 |
- // Copyright (c) 2017 GitHub, Inc.
- // Use of this source code is governed by the MIT license that can be
- // found in the LICENSE file.
- #import "shell/browser/ui/cocoa/electron_bundle_mover.h"
- #import <AppKit/AppKit.h>
- #import <Security/Security.h>
- #import <dlfcn.h>
- #import <sys/mount.h>
- #import <sys/param.h>
- #include <string>
- #include <utility>
- #include "gin/dictionary.h"
- #include "shell/browser/browser.h"
- #include "shell/common/gin_converters/callback_converter.h"
- #include "shell/common/gin_helper/error_thrower.h"
- namespace gin {
- template <>
- struct Converter<electron::BundlerMoverConflictType> {
- static v8::Local<v8::Value> ToV8(v8::Isolate* isolate,
- electron::BundlerMoverConflictType value) {
- switch (value) {
- case electron::BundlerMoverConflictType::kExists:
- return gin::StringToV8(isolate, "exists");
- case electron::BundlerMoverConflictType::kExistsAndRunning:
- return gin::StringToV8(isolate, "existsAndRunning");
- default:
- return gin::StringToV8(isolate, "");
- }
- }
- };
- } // namespace gin
- namespace {
- NSString* ContainingDiskImageDevice(NSString* bundlePath) {
- NSString* containingPath = [bundlePath stringByDeletingLastPathComponent];
- struct statfs fs;
- if (statfs([containingPath fileSystemRepresentation], &fs) ||
- (fs.f_flags & MNT_ROOTFS))
- return nil;
- NSString* device = [[NSFileManager defaultManager]
- stringWithFileSystemRepresentation:fs.f_mntfromname
- length:strlen(fs.f_mntfromname)];
- NSTask* hdiutil = [[NSTask alloc] init];
- [hdiutil setLaunchPath:@"/usr/bin/hdiutil"];
- [hdiutil setArguments:[NSArray arrayWithObjects:@"info", @"-plist", nil]];
- [hdiutil setStandardOutput:[NSPipe pipe]];
- [hdiutil launch];
- [hdiutil waitUntilExit];
- NSData* data =
- [[[hdiutil standardOutput] fileHandleForReading] readDataToEndOfFile];
- NSDictionary* info =
- [NSPropertyListSerialization propertyListWithData:data
- options:NSPropertyListImmutable
- format:nil
- error:nil];
- if (![info isKindOfClass:[NSDictionary class]])
- return nil;
- NSArray* images = (NSArray*)[info objectForKey:@"images"];
- if (![images isKindOfClass:[NSArray class]])
- return nil;
- for (NSDictionary* image in images) {
- if (![image isKindOfClass:[NSDictionary class]])
- return nil;
- id systemEntities = [image objectForKey:@"system-entities"];
- if (![systemEntities isKindOfClass:[NSArray class]])
- return nil;
- for (NSDictionary* systemEntity in systemEntities) {
- if (![systemEntity isKindOfClass:[NSDictionary class]])
- return nil;
- NSString* devEntry = [systemEntity objectForKey:@"dev-entry"];
- if (![devEntry isKindOfClass:[NSString class]])
- return nil;
- if ([devEntry isEqualToString:device])
- return device;
- }
- }
- return nil;
- }
- NSString* ResolvePath(NSString* path) {
- NSString* standardizedPath = [path stringByStandardizingPath];
- char resolved[PATH_MAX];
- if (realpath([standardizedPath UTF8String], resolved) == nullptr)
- return path;
- return @(resolved);
- }
- bool IsInApplicationsFolder(NSString* bundlePath) {
- // Check all the normal Application directories
- NSArray* applicationDirs = NSSearchPathForDirectoriesInDomains(
- NSApplicationDirectory, NSAllDomainsMask, true);
- NSString* resolvedBundlePath = ResolvePath(bundlePath);
- for (NSString* appDir in applicationDirs) {
- if ([resolvedBundlePath hasPrefix:appDir])
- return true;
- }
- // Also, handle the case that the user has some other Application directory
- // (perhaps on a separate data partition).
- if ([[resolvedBundlePath pathComponents] containsObject:@"Applications"])
- return true;
- return false;
- }
- bool AuthorizedInstall(NSString* srcPath, NSString* dstPath, bool* canceled) {
- if (canceled)
- *canceled = false;
- // Make sure that the destination path is an app bundle. We're essentially
- // running 'sudo rm -rf' so we really don't want to screw this up.
- if (![[dstPath pathExtension] isEqualToString:@"app"])
- return false;
- // Do some more checks
- if ([[dstPath stringByTrimmingCharactersInSet:[NSCharacterSet
- whitespaceCharacterSet]]
- length] == 0)
- return false;
- if ([[srcPath stringByTrimmingCharactersInSet:[NSCharacterSet
- whitespaceCharacterSet]]
- length] == 0)
- return false;
- int pid, status;
- AuthorizationRef myAuthorizationRef;
- // Get the authorization
- OSStatus err =
- AuthorizationCreate(nullptr, kAuthorizationEmptyEnvironment,
- kAuthorizationFlagDefaults, &myAuthorizationRef);
- if (err != errAuthorizationSuccess)
- return false;
- AuthorizationItem myItems = {kAuthorizationRightExecute, 0, nullptr, 0};
- AuthorizationRights myRights = {1, &myItems};
- AuthorizationFlags myFlags =
- (AuthorizationFlags)(kAuthorizationFlagInteractionAllowed |
- kAuthorizationFlagExtendRights |
- kAuthorizationFlagPreAuthorize);
- err = AuthorizationCopyRights(myAuthorizationRef, &myRights, nullptr, myFlags,
- nullptr);
- if (err != errAuthorizationSuccess) {
- if (err == errAuthorizationCanceled && canceled)
- *canceled = true;
- goto fail;
- }
- static OSStatus (*security_AuthorizationExecuteWithPrivileges)(
- AuthorizationRef authorization, const char* pathToTool,
- AuthorizationFlags options, char* const* arguments,
- FILE** communicationsPipe) = nullptr;
- if (!security_AuthorizationExecuteWithPrivileges) {
- // On 10.7, AuthorizationExecuteWithPrivileges is deprecated. We want to
- // still use it since there's no good alternative (without requiring code
- // signing). We'll look up the function through dyld and fail if it is no
- // longer accessible. If Apple removes the function entirely this will fail
- // gracefully. If they keep the function and throw some sort of exception,
- // this won't fail gracefully, but that's a risk we'll have to take for now.
- security_AuthorizationExecuteWithPrivileges = (OSStatus (*)(
- AuthorizationRef, const char*, AuthorizationFlags, char* const*,
- FILE**))dlsym(RTLD_DEFAULT, "AuthorizationExecuteWithPrivileges");
- }
- if (!security_AuthorizationExecuteWithPrivileges)
- goto fail;
- // Delete the destination
- {
- char rf[] = "-rf";
- char* args[] = {rf, (char*)[dstPath fileSystemRepresentation], nullptr};
- err = security_AuthorizationExecuteWithPrivileges(
- myAuthorizationRef, "/bin/rm", kAuthorizationFlagDefaults, args,
- nullptr);
- if (err != errAuthorizationSuccess)
- goto fail;
- // Wait until it's done
- pid = wait(&status);
- if (pid == -1 || !WIFEXITED(status))
- goto fail; // We don't care about exit status as the destination most
- // likely does not exist
- }
- // Copy
- {
- char pR[] = "-pR";
- char* args[] = {pR, (char*)[srcPath fileSystemRepresentation],
- (char*)[dstPath fileSystemRepresentation], nullptr};
- err = security_AuthorizationExecuteWithPrivileges(
- myAuthorizationRef, "/bin/cp", kAuthorizationFlagDefaults, args,
- nullptr);
- if (err != errAuthorizationSuccess)
- goto fail;
- // Wait until it's done
- pid = wait(&status);
- if (pid == -1 || !WIFEXITED(status) || WEXITSTATUS(status))
- goto fail;
- }
- AuthorizationFree(myAuthorizationRef, kAuthorizationFlagDefaults);
- return true;
- fail:
- AuthorizationFree(myAuthorizationRef, kAuthorizationFlagDefaults);
- return false;
- }
- bool CopyBundle(NSString* srcPath, NSString* dstPath) {
- NSFileManager* fileManager = [NSFileManager defaultManager];
- NSError* error = nil;
- return [fileManager copyItemAtPath:srcPath toPath:dstPath error:&error];
- }
- NSString* ShellQuotedString(NSString* string) {
- return [NSString
- stringWithFormat:@"'%@'",
- [string stringByReplacingOccurrencesOfString:@"'"
- withString:@"'\\''"]];
- }
- void Relaunch(NSString* destinationPath) {
- // The shell script waits until the original app process terminates.
- // This is done so that the relaunched app opens as the front-most app.
- int pid = [[NSProcessInfo processInfo] processIdentifier];
- // Command run just before running open /final/path
- NSString* preOpenCmd = @"";
- NSString* quotedDestinationPath = ShellQuotedString(destinationPath);
- // Before we launch the new app, clear xattr:com.apple.quarantine to avoid
- // duplicate "scary file from the internet" dialog.
- preOpenCmd = [NSString
- stringWithFormat:@"/usr/bin/xattr -d -r com.apple.quarantine %@",
- quotedDestinationPath];
- NSString* script =
- [NSString stringWithFormat:
- @"(while /bin/kill -0 %d >&/dev/null; do /bin/sleep 0.1; "
- @"done; %@; /usr/bin/open %@) &",
- pid, preOpenCmd, quotedDestinationPath];
- [NSTask
- launchedTaskWithLaunchPath:@"/bin/sh"
- arguments:[NSArray arrayWithObjects:@"-c", script, nil]];
- }
- bool Trash(NSString* path) {
- bool result = false;
- if (floor(NSAppKitVersionNumber) >= NSAppKitVersionNumber10_8) {
- result = [[NSFileManager defaultManager]
- trashItemAtURL:[NSURL fileURLWithPath:path]
- resultingItemURL:nil
- error:nil];
- }
- // As a last resort try trashing with AppleScript.
- // This allows us to trash the app in macOS Sierra even when the app is
- // running inside an app translocation image.
- if (!result) {
- auto* code = R"str(
- set theFile to POSIX file "%@"
- tell application "Finder"
- move theFile to trash
- end tell
- )str";
- NSAppleScript* appleScript = [[NSAppleScript alloc]
- initWithSource:[NSString stringWithFormat:@(code), path]];
- NSDictionary* errorDict = nil;
- NSAppleEventDescriptor* scriptResult =
- [appleScript executeAndReturnError:&errorDict];
- result = (scriptResult != nil);
- }
- return result;
- }
- bool DeleteOrTrash(NSString* path) {
- NSError* error;
- if ([[NSFileManager defaultManager] removeItemAtPath:path error:&error]) {
- return true;
- } else {
- return Trash(path);
- }
- }
- bool IsApplicationAtPathRunning(NSString* bundlePath) {
- bundlePath = [bundlePath stringByStandardizingPath];
- for (NSRunningApplication* runningApplication in
- [[NSWorkspace sharedWorkspace] runningApplications]) {
- NSString* runningAppBundlePath =
- [[[runningApplication bundleURL] path] stringByStandardizingPath];
- if ([runningAppBundlePath isEqualToString:bundlePath]) {
- return true;
- }
- }
- return false;
- }
- } // namespace
- namespace electron {
- bool ElectronBundleMover::ShouldContinueMove(gin_helper::ErrorThrower thrower,
- BundlerMoverConflictType type,
- gin::Arguments* args) {
- gin::Dictionary options(args->isolate());
- bool hasOptions = args->GetNext(&options);
- base::OnceCallback<v8::Local<v8::Value>(BundlerMoverConflictType)>
- conflict_cb;
- if (hasOptions && options.Get("conflictHandler", &conflict_cb)) {
- v8::Local<v8::Value> value = std::move(conflict_cb).Run(type);
- if (value->IsBoolean()) {
- if (!value.As<v8::Boolean>()->Value())
- return false;
- } else if (!value->IsUndefined()) {
- // we only want to throw an error if a user has returned a non-boolean
- // value; this allows for client-side error handling should something in
- // the handler throw
- thrower.ThrowError("Invalid conflict handler return type.");
- }
- }
- return true;
- }
- bool ElectronBundleMover::Move(gin_helper::ErrorThrower thrower,
- gin::Arguments* args) {
- // Path of the current bundle
- NSString* bundlePath = [[NSBundle mainBundle] bundlePath];
- // Skip if the application is already in the Applications folder
- if (IsInApplicationsFolder(bundlePath))
- return true;
- NSFileManager* fileManager = [NSFileManager defaultManager];
- NSString* diskImageDevice = ContainingDiskImageDevice(bundlePath);
- NSString* applicationsDirectory = [[NSSearchPathForDirectoriesInDomains(
- NSApplicationDirectory, NSLocalDomainMask, true) lastObject]
- stringByResolvingSymlinksInPath];
- NSString* bundleName = [bundlePath lastPathComponent];
- NSString* destinationPath =
- [applicationsDirectory stringByAppendingPathComponent:bundleName];
- // Check if we can write to the applications directory
- // and then make sure that if the app already exists we can overwrite it
- bool needAuthorization =
- ![fileManager isWritableFileAtPath:applicationsDirectory] ||
- ([fileManager fileExistsAtPath:destinationPath] &&
- ![fileManager isWritableFileAtPath:destinationPath]);
- // Activate app -- work-around for focus issues related to "scary file from
- // internet" OS dialog.
- if (![NSApp isActive]) {
- [NSApp activateIgnoringOtherApps:true];
- }
- // Move to applications folder
- if (needAuthorization) {
- bool authorizationCanceled;
- if (!AuthorizedInstall(bundlePath, destinationPath,
- &authorizationCanceled)) {
- if (authorizationCanceled) {
- // User rejected the authorization request
- thrower.ThrowError("User rejected the authorization request");
- return false;
- } else {
- thrower.ThrowError(
- "Failed to copy to applications directory even with authorization");
- return false;
- }
- }
- } else {
- // If a copy already exists in the Applications folder, put it in the Trash
- if ([fileManager fileExistsAtPath:destinationPath]) {
- // But first, make sure that it's not running
- if (IsApplicationAtPathRunning(destinationPath)) {
- // Check for callback handler and get user choice for open/quit
- if (!ShouldContinueMove(
- thrower, BundlerMoverConflictType::kExistsAndRunning, args))
- return false;
- // Unless explicitly denied, give running app focus and terminate self
- [[NSTask
- launchedTaskWithLaunchPath:@"/usr/bin/open"
- arguments:[NSArray
- arrayWithObject:destinationPath]]
- waitUntilExit];
- electron::Browser::Get()->Quit();
- return true;
- } else {
- // Check callback handler and get user choice for app trashing
- if (!ShouldContinueMove(thrower, BundlerMoverConflictType::kExists,
- args))
- return false;
- // Unless explicitly denied, attempt to trash old app
- if (!Trash([applicationsDirectory
- stringByAppendingPathComponent:bundleName])) {
- thrower.ThrowError("Failed to delete existing application");
- return false;
- }
- }
- }
- if (!CopyBundle(bundlePath, destinationPath)) {
- thrower.ThrowError(
- "Failed to copy current bundle to the applications folder");
- return false;
- }
- }
- // Trash the original app. It's okay if this fails.
- // NOTE: This final delete does not work if the source bundle is in a network
- // mounted volume.
- // Calling rm or file manager's delete method doesn't work either. It's
- // unlikely to happen but it'd be great if someone could fix this.
- if (diskImageDevice == nil && !DeleteOrTrash(bundlePath)) {
- // Could not delete original but we just don't care
- }
- // Relaunch.
- Relaunch(destinationPath);
- // Launched from within a disk image? -- unmount (if no files are open after 5
- // seconds, otherwise leave it mounted).
- if (diskImageDevice) {
- NSString* script = [NSString
- stringWithFormat:@"(/bin/sleep 5 && /usr/bin/hdiutil detach %@) &",
- ShellQuotedString(diskImageDevice)];
- [NSTask launchedTaskWithLaunchPath:@"/bin/sh"
- arguments:[NSArray arrayWithObjects:@"-c", script,
- nil]];
- }
- electron::Browser::Get()->Quit();
- return true;
- }
- bool ElectronBundleMover::IsCurrentAppInApplicationsFolder() {
- return IsInApplicationsFolder([[NSBundle mainBundle] bundlePath]);
- }
- } // namespace electron
|