Home | History | Annotate | Download | only in app_shim
      1 // Copyright (c) 2012 The Chromium Authors. All rights reserved.
      2 // Use of this source code is governed by a BSD-style license that can be
      3 // found in the LICENSE file.
      4 
      5 // On Mac, shortcuts can't have command-line arguments. Instead, produce small
      6 // app bundles which locate the Chromium framework and load it, passing the
      7 // appropriate data. This is the code for such an app bundle. It should be kept
      8 // minimal and do as little work as possible (with as much work done on
      9 // framework side as possible).
     10 
     11 #include <dlfcn.h>
     12 
     13 #import <Cocoa/Cocoa.h>
     14 
     15 #include "base/command_line.h"
     16 #include "base/files/file_path.h"
     17 #include "base/files/file_util.h"
     18 #include "base/logging.h"
     19 #include "base/mac/foundation_util.h"
     20 #include "base/mac/launch_services_util.h"
     21 #include "base/mac/scoped_nsautorelease_pool.h"
     22 #include "base/process/launch.h"
     23 #include "base/strings/string_number_conversions.h"
     24 #include "base/strings/sys_string_conversions.h"
     25 #include "chrome/common/chrome_constants.h"
     26 #include "chrome/common/chrome_switches.h"
     27 #import "chrome/common/mac/app_mode_chrome_locator.h"
     28 #include "chrome/common/mac/app_mode_common.h"
     29 
     30 namespace {
     31 
     32 typedef int (*StartFun)(const app_mode::ChromeAppModeInfo*);
     33 
     34 int LoadFrameworkAndStart(app_mode::ChromeAppModeInfo* info) {
     35   using base::SysNSStringToUTF8;
     36   using base::SysNSStringToUTF16;
     37   using base::mac::CFToNSCast;
     38   using base::mac::CFCastStrict;
     39   using base::mac::NSToCFCast;
     40 
     41   base::mac::ScopedNSAutoreleasePool scoped_pool;
     42 
     43   // Get the current main bundle, i.e., that of the app loader that's running.
     44   NSBundle* app_bundle = [NSBundle mainBundle];
     45   CHECK(app_bundle) << "couldn't get loader bundle";
     46 
     47   // ** 1: Get path to outer Chrome bundle.
     48   // Get the bundle ID of the browser that created this app bundle.
     49   NSString* cr_bundle_id = base::mac::ObjCCast<NSString>(
     50       [app_bundle objectForInfoDictionaryKey:app_mode::kBrowserBundleIDKey]);
     51   CHECK(cr_bundle_id) << "couldn't get browser bundle ID";
     52 
     53   // First check if Chrome exists at the last known location.
     54   base::FilePath cr_bundle_path;
     55   NSString* cr_bundle_path_ns =
     56       [CFToNSCast(CFCastStrict<CFStringRef>(CFPreferencesCopyAppValue(
     57           NSToCFCast(app_mode::kLastRunAppBundlePathPrefsKey),
     58           NSToCFCast(cr_bundle_id)))) autorelease];
     59   cr_bundle_path = base::mac::NSStringToFilePath(cr_bundle_path_ns);
     60   bool found_bundle =
     61       !cr_bundle_path.empty() && base::DirectoryExists(cr_bundle_path);
     62 
     63   if (!found_bundle) {
     64     // If no such bundle path exists, try to search by bundle ID.
     65     if (!app_mode::FindBundleById(cr_bundle_id, &cr_bundle_path)) {
     66       // TODO(jeremy): Display UI to allow user to manually locate the Chrome
     67       // bundle.
     68       LOG(FATAL) << "Failed to locate bundle by identifier";
     69     }
     70   }
     71 
     72   // ** 2: Read the running Chrome version.
     73   // The user_data_dir for shims actually contains the app_data_path.
     74   // I.e. <user_data_dir>/<profile_dir>/Web Applications/_crx_extensionid/
     75   base::FilePath app_data_dir = base::mac::NSStringToFilePath([app_bundle
     76       objectForInfoDictionaryKey:app_mode::kCrAppModeUserDataDirKey]);
     77   base::FilePath user_data_dir = app_data_dir.DirName().DirName().DirName();
     78   CHECK(!user_data_dir.empty());
     79 
     80   // If the version file does not exist, |cr_version_str| will be empty and
     81   // app_mode::GetChromeBundleInfo will default to the latest version.
     82   base::FilePath cr_version_str;
     83   base::ReadSymbolicLink(
     84       user_data_dir.Append(app_mode::kRunningChromeVersionSymlinkName),
     85       &cr_version_str);
     86 
     87   // If the version file does exist, it may have been left by a crashed Chrome
     88   // process. Ensure the process is still running.
     89   if (!cr_version_str.empty()) {
     90     NSArray* existing_chrome = [NSRunningApplication
     91         runningApplicationsWithBundleIdentifier:cr_bundle_id];
     92     if ([existing_chrome count] == 0)
     93       cr_version_str.clear();
     94   }
     95 
     96   // ** 3: Read information from the Chrome bundle.
     97   base::FilePath executable_path;
     98   base::FilePath version_path;
     99   base::FilePath framework_shlib_path;
    100   if (!app_mode::GetChromeBundleInfo(cr_bundle_path,
    101                                      cr_version_str.value(),
    102                                      &executable_path,
    103                                      &version_path,
    104                                      &framework_shlib_path)) {
    105     LOG(FATAL) << "Couldn't ready Chrome bundle info";
    106   }
    107   base::FilePath app_mode_bundle_path =
    108       base::mac::NSStringToFilePath([app_bundle bundlePath]);
    109 
    110   // ** 4: Fill in ChromeAppModeInfo.
    111   info->chrome_outer_bundle_path = cr_bundle_path;
    112   info->chrome_versioned_path = version_path;
    113   info->app_mode_bundle_path = app_mode_bundle_path;
    114 
    115   // Read information about the this app shortcut from the Info.plist.
    116   // Don't check for null-ness on optional items.
    117   NSDictionary* info_plist = [app_bundle infoDictionary];
    118   CHECK(info_plist) << "couldn't get loader Info.plist";
    119 
    120   info->app_mode_id = SysNSStringToUTF8(
    121       [info_plist objectForKey:app_mode::kCrAppModeShortcutIDKey]);
    122   CHECK(info->app_mode_id.size()) << "couldn't get app shortcut ID";
    123 
    124   info->app_mode_name = SysNSStringToUTF16(
    125       [info_plist objectForKey:app_mode::kCrAppModeShortcutNameKey]);
    126 
    127   info->app_mode_url = SysNSStringToUTF8(
    128       [info_plist objectForKey:app_mode::kCrAppModeShortcutURLKey]);
    129 
    130   info->user_data_dir = base::mac::NSStringToFilePath(
    131       [info_plist objectForKey:app_mode::kCrAppModeUserDataDirKey]);
    132 
    133   info->profile_dir = base::mac::NSStringToFilePath(
    134       [info_plist objectForKey:app_mode::kCrAppModeProfileDirKey]);
    135 
    136   // ** 5: Open the framework.
    137   StartFun ChromeAppModeStart = NULL;
    138   void* cr_dylib = dlopen(framework_shlib_path.value().c_str(), RTLD_LAZY);
    139   if (cr_dylib) {
    140     // Find the entry point.
    141     ChromeAppModeStart = (StartFun)dlsym(cr_dylib, "ChromeAppModeStart");
    142     if (!ChromeAppModeStart)
    143       LOG(ERROR) << "Couldn't get entry point: " << dlerror();
    144   } else {
    145     LOG(ERROR) << "Couldn't load framework: " << dlerror();
    146   }
    147 
    148   if (ChromeAppModeStart)
    149     return ChromeAppModeStart(info);
    150 
    151   LOG(ERROR) << "Loading Chrome failed, launching Chrome with command line";
    152   CommandLine command_line(executable_path);
    153   // The user_data_dir from the plist is actually the app data dir.
    154   command_line.AppendSwitchPath(
    155       switches::kUserDataDir,
    156       info->user_data_dir.DirName().DirName().DirName());
    157   if (CommandLine::ForCurrentProcess()->HasSwitch(
    158           app_mode::kLaunchedByChromeProcessId) ||
    159       info->app_mode_id == app_mode::kAppListModeId) {
    160     // Pass --app-shim-error to have Chrome rebuild this shim.
    161     // If Chrome has rebuilt this shim once already, then rebuilding doesn't fix
    162     // the problem, so don't try again.
    163     if (!CommandLine::ForCurrentProcess()->HasSwitch(
    164             app_mode::kLaunchedAfterRebuild)) {
    165       command_line.AppendSwitchPath(app_mode::kAppShimError,
    166                                     app_mode_bundle_path);
    167     }
    168   } else {
    169     // If the shim was launched directly (instead of by Chrome), first ask
    170     // Chrome to launch the app. Chrome will launch the shim again, the same
    171     // error will occur and be handled above. This approach allows the app to be
    172     // started without blocking on fixing the shim and guarantees that the
    173     // profile is loaded when Chrome receives --app-shim-error.
    174     command_line.AppendSwitchPath(switches::kProfileDirectory,
    175                                   info->profile_dir);
    176     command_line.AppendSwitchASCII(switches::kAppId, info->app_mode_id);
    177   }
    178   // Launch the executable directly since base::mac::OpenApplicationWithPath
    179   // uses LSOpenApplication which doesn't pass command line arguments if the
    180   // application is already running.
    181   if (!base::LaunchProcess(command_line, base::LaunchOptions(), NULL)) {
    182     LOG(ERROR) << "Could not launch Chrome: "
    183                << command_line.GetCommandLineString();
    184     return 1;
    185   }
    186 
    187   return 0;
    188 }
    189 
    190 } // namespace
    191 
    192 __attribute__((visibility("default")))
    193 int main(int argc, char** argv) {
    194   CommandLine::Init(argc, argv);
    195   app_mode::ChromeAppModeInfo info;
    196 
    197   // Hard coded info parameters.
    198   info.major_version = app_mode::kCurrentChromeAppModeInfoMajorVersion;
    199   info.minor_version = app_mode::kCurrentChromeAppModeInfoMinorVersion;
    200   info.argc = argc;
    201   info.argv = argv;
    202 
    203   // Exit instead of returning to avoid the the removal of |main()| from stack
    204   // backtraces under tail call optimization.
    205   exit(LoadFrameworkAndStart(&info));
    206 }
    207