| #!/usr/bin/env python | 
 |  | 
 | # Copyright (c) 2012, the Dart project authors.  Please see the AUTHORS file | 
 | # for details. All rights reserved. Use of this source code is governed by a | 
 | # BSD-style license that can be found in the LICENSE file. | 
 |  | 
 | """ | 
 | Find an Android device with a given ABI. | 
 |  | 
 | The name of the Android device is printed to stdout. | 
 |  | 
 | Optionally configure and launch an emulator if there's no existing device for a | 
 | given ABI. Will download and install Android SDK components as needed. | 
 | """ | 
 |  | 
 | import optparse | 
 | import os | 
 | import re | 
 | import sys | 
 | import traceback | 
 | import utils | 
 |  | 
 |  | 
 | DEBUG = False | 
 | VERBOSE = False | 
 |  | 
 |  | 
 | def BuildOptions(): | 
 |   result = optparse.OptionParser() | 
 |   result.add_option( | 
 |       "-a", "--abi", | 
 |       action="store", type="string", | 
 |       help="Desired ABI. armeabi-v7a or x86.") | 
 |   result.add_option( | 
 |       "-b", "--bootstrap", | 
 |       help='Bootstrap - create an emulator, installing SDK packages if needed.', | 
 |       default=False, action="store_true") | 
 |   result.add_option( | 
 |       "-d", "--debug", | 
 |       help='Turn on debugging diagnostics.', | 
 |       default=False, action="store_true") | 
 |   result.add_option( | 
 |       "-v", "--verbose", | 
 |       help='Verbose output.', | 
 |       default=False, action="store_true") | 
 |   return result | 
 |  | 
 |  | 
 | def ProcessOptions(options): | 
 |   global DEBUG | 
 |   DEBUG = options.debug | 
 |   global VERBOSE | 
 |   VERBOSE = options.verbose | 
 |   if options.abi is None: | 
 |     sys.stderr.write('--abi not specified.\n') | 
 |     return False | 
 |   return True | 
 |  | 
 |  | 
 | def ParseAndroidListSdkResult(text): | 
 |   """ | 
 |   Parse the output of an 'android list sdk' command. | 
 |  | 
 |   Return list of (id-num, id-key, type, description). | 
 |   """ | 
 |   header_regex = re.compile( | 
 |       r'Packages available for installation or update: \d+\n') | 
 |   packages = re.split(header_regex, text) | 
 |   if len(packages) != 2: | 
 |     raise utils.Error("Could not get a list of packages to install") | 
 |   entry_regex = re.compile( | 
 |       r'^id\: (\d+) or "([^"]*)"\n\s*Type\: ([^\n]*)\n\s*Desc\: (.*)') | 
 |   entries = [] | 
 |   for entry in packages[1].split('----------\n'): | 
 |     match = entry_regex.match(entry) | 
 |     if match == None: | 
 |       continue | 
 |     entries.append((int(match.group(1)), match.group(2), match.group(3), | 
 |         match.group(4))) | 
 |   return entries | 
 |  | 
 |  | 
 | def AndroidListSdk(): | 
 |   return ParseAndroidListSdkResult(utils.RunCommand( | 
 |       ["android", "list", "sdk", "-a", "-e"])) | 
 |  | 
 |  | 
 | def AndroidSdkFindPackage(packages, key): | 
 |   """ | 
 |   Args: | 
 |     packages: list of (id-num, id-key, type, description). | 
 |     key: (id-key, type, description-prefix). | 
 |   """ | 
 |   (key_id, key_type, key_description_prefix) = key | 
 |   for package in packages: | 
 |     (package_num, package_id, package_type, package_description) = package | 
 |     if (package_id == key_id and package_type == key_type | 
 |         and package_description.startswith(key_description_prefix)): | 
 |       return package | 
 |   return None | 
 |  | 
 |  | 
 | def EnsureSdkPackageInstalled(packages, key): | 
 |   """ | 
 |   Makes sure the package with a given key is installed. | 
 |  | 
 |   key is (id-key, type, description-prefix) | 
 |  | 
 |   Returns True if the package was not already installed. | 
 |   """ | 
 |   entry = AndroidSdkFindPackage(packages, key) | 
 |   if entry is None: | 
 |     raise utils.Error("Could not find a package for key %s" % key) | 
 |   packageId = entry[0] | 
 |   if VERBOSE: | 
 |     sys.stderr.write('Checking Android SDK package %s...\n' % str(entry)) | 
 |   out = utils.RunCommand( | 
 |       ["android", "update", "sdk", "-a", "-u", "--filter", str(packageId)]) | 
 |   return '\nInstalling Archives:\n' in out | 
 |  | 
 |  | 
 | def SdkPackagesForAbi(abi): | 
 |   packagesForAbi = { | 
 |     'armeabi-v7a': [ | 
 |       # The platform needed to install the armeabi ABI system image: | 
 |       ('android-15', 'Platform',  'Android SDK Platform 4.0.3'), | 
 |       # The armeabi-v7a ABI system image: | 
 |       ('sysimg-15', 'SystemImage',  'Android SDK Platform 4.0.3') | 
 |     ], | 
 |     'x86': [ | 
 |       # The platform needed to install the x86 ABI system image: | 
 |       ('android-15', 'Platform',  'Android SDK Platform 4.0.3'), | 
 |       # The x86 ABI system image: | 
 |       ('sysimg-15', 'SystemImage',  'Android SDK Platform 4.0.4') | 
 |     ] | 
 |   } | 
 |  | 
 |   if abi not in packagesForAbi: | 
 |     raise utils.Error('Unsupported abi %s' % abi) | 
 |   return packagesForAbi[abi] | 
 |  | 
 |  | 
 | def TargetForAbi(abi): | 
 |   for package in SdkPackagesForAbi(abi): | 
 |     if package[1] == 'Platform': | 
 |       return package[0] | 
 |  | 
 |  | 
 | def EnsureAndroidSdkPackagesInstalled(abi): | 
 |   """Return true if at least one package was not already installed.""" | 
 |   abiPackageList = SdkPackagesForAbi(abi) | 
 |   installedSomething = False | 
 |   packages = AndroidListSdk() | 
 |   for package in abiPackageList: | 
 |     installedSomething |= EnsureSdkPackageInstalled(packages, package) | 
 |   return installedSomething | 
 |  | 
 |  | 
 | def ParseAndroidListAvdResult(text): | 
 |   """ | 
 |   Parse the output of an 'android list avd' command. | 
 |   Return List of {Name: Path: Target: ABI: Skin: Sdcard:} | 
 |   """ | 
 |   text = text.split('Available Android Virtual Devices:\n')[-1] | 
 |   text = text.split( | 
 |       'The following Android Virtual Devices could not be loaded:\n')[0] | 
 |   result = [] | 
 |   line_re = re.compile(r'^\s*([^\:]+)\:\s*(.*)$') | 
 |   for chunk in text.split('\n---------\n'): | 
 |     entry = {} | 
 |     for line in chunk.split('\n'): | 
 |       line = line.strip() | 
 |       if len(line) == 0: | 
 |         continue | 
 |       match = line_re.match(line) | 
 |       if match is None: | 
 |         sys.stderr.write('Match fail %s\n' % str(line)) | 
 |         continue | 
 |         #raise utils.Error('Match failed') | 
 |       entry[match.group(1)] = match.group(2) | 
 |     if len(entry) > 0: | 
 |       result.append(entry) | 
 |   return result | 
 |  | 
 |  | 
 | def AndroidListAvd(): | 
 |   """Returns a list of available Android Virtual Devices.""" | 
 |   return ParseAndroidListAvdResult(utils.RunCommand(["android", "list", "avd"])) | 
 |  | 
 |  | 
 | def FindAvd(avds, key): | 
 |   for avd in avds: | 
 |     if avd['Name'] == key: | 
 |       return avd | 
 |   return None | 
 |  | 
 |  | 
 | def CreateAvd(avdName, abi): | 
 |   out = utils.RunCommand(["android", "create", "avd", "--name", avdName, | 
 |                           "--target", TargetForAbi(abi), '--abi', abi], | 
 |                          input="no\n") | 
 |   if out.find('Created AVD ') < 0: | 
 |     if VERBOSE: | 
 |         sys.stderr.write('Could not create AVD:\n%s\n' % out) | 
 |     raise utils.Error('Could not create AVD') | 
 |  | 
 |  | 
 | def AvdExists(avdName): | 
 |   avdList = AndroidListAvd() | 
 |   return FindAvd(avdList, avdName) is not None | 
 |  | 
 |  | 
 | def EnsureAvdExists(avdName, abi): | 
 |   if AvdExists(avdName): | 
 |     return | 
 |   if VERBOSE: | 
 |     sys.stderr.write('Checking SDK packages...\n') | 
 |   if EnsureAndroidSdkPackagesInstalled(abi): | 
 |     # Installing a new package could have made a previously invalid AVD valid | 
 |     if AvdExists(avdName): | 
 |         return | 
 |   CreateAvd(avdName, abi) | 
 |  | 
 |  | 
 | def StartEmulator(abi, avdName, pollFn): | 
 |   """ | 
 |   Start an emulator for a given abi and svdName. | 
 |  | 
 |   Echo the emulator's stderr and stdout output to our stderr. | 
 |  | 
 |   Call pollFn repeatedly until it returns False. Leave the emulator running | 
 |   when we return. | 
 |  | 
 |   Implementation note: Normally we would call the 'emulator' binary, which | 
 |   is a wrapper that launches the appropriate abi-specific emulator. But there | 
 |   is a bug that causes the emulator to exit immediately with a result code of | 
 |   -11 if run from a ssh shell or a No Machine shell. (And only if called from | 
 |   three levels of nested python scripts.) Calling the ABI-specific versions | 
 |   of the emulator directly works around this bug. | 
 |   """ | 
 |   emulatorName = {'x86' : 'emulator-x86', 'armeabi-v7a': 'emulator-arm'}[abi] | 
 |   command = [emulatorName, '-avd', avdName, '-no-boot-anim', '-no-window'] | 
 |   utils.RunCommand(command, pollFn=pollFn, killOnEarlyReturn=False, | 
 |           outStream=sys.stderr, errStream=sys.stderr) | 
 |  | 
 |  | 
 | def ParseAndroidDevices(text): | 
 |   """Return Dictionary [name] -> status""" | 
 |   text = text.split('List of devices attached')[-1] | 
 |   lines = [line.strip() for line in text.split('\n')] | 
 |   lines = [line for line in lines if len(line) > 0] | 
 |   devices = {} | 
 |   for line in lines: | 
 |     lineItems = line.split('\t') | 
 |     devices[lineItems[0]] = lineItems[1] | 
 |   return devices | 
 |  | 
 |  | 
 | def GetAndroidDevices(): | 
 |   return ParseAndroidDevices(utils.RunCommand(["adb", "devices"])) | 
 |  | 
 |  | 
 | def FilterOfflineDevices(devices): | 
 |   online = {} | 
 |   for device in devices.keys(): | 
 |     status = devices[device] | 
 |     if status != 'offline': | 
 |       online[device] = status | 
 |   return online | 
 |  | 
 |  | 
 | def GetOnlineAndroidDevices(): | 
 |   return FilterOfflineDevices(GetAndroidDevices()) | 
 |  | 
 |  | 
 | def GetAndroidDeviceProperty(device, property): | 
 |   return utils.RunCommand( | 
 |       ["adb", "-s", device, "shell", "getprop", property]).strip() | 
 |  | 
 |  | 
 | def GetAndroidDeviceAbis(device): | 
 |   abis = [] | 
 |   for property in ['ro.product.cpu.abi', 'ro.product.cpu.abi2']: | 
 |     out = GetAndroidDeviceProperty(device, property) | 
 |     if len(out) > 0: | 
 |       abis.append(out) | 
 |   return abis | 
 |  | 
 |  | 
 | def FindAndroidRunning(abi): | 
 |   for device in GetOnlineAndroidDevices().keys(): | 
 |     if abi in GetAndroidDeviceAbis(device): | 
 |       return device | 
 |   return None | 
 |  | 
 |  | 
 | def AddSdkToolsToPath(): | 
 |   script_dir = os.path.dirname(sys.argv[0]) | 
 |   dart_root = os.path.realpath(os.path.join(script_dir, '..', '..')) | 
 |   third_party_root = os.path.join(dart_root, 'third_party') | 
 |   android_tools = os.path.join(third_party_root, 'android_tools') | 
 |   android_sdk_root = os.path.join(android_tools, 'sdk') | 
 |   android_sdk_tools = os.path.join(android_sdk_root, 'tools') | 
 |   android_sdk_platform_tools = os.path.join(android_sdk_root, 'platform-tools') | 
 |   os.environ['PATH'] = ':'.join([ | 
 |       os.environ['PATH'], android_sdk_tools, android_sdk_platform_tools]) | 
 |   # Remove any environment variables that would affect our build. | 
 |   for i in ['ANDROID_NDK_ROOT', 'ANDROID_SDK_ROOT', 'ANDROID_TOOLCHAIN', | 
 |             'AR', 'BUILDTYPE', 'CC', 'CXX', 'GYP_DEFINES', | 
 |             'LD_LIBRARY_PATH', 'LINK', 'MAKEFLAGS', 'MAKELEVEL', | 
 |             'MAKEOVERRIDES', 'MFLAGS', 'NM']: | 
 |     if i in os.environ: | 
 |       del os.environ[i] | 
 |  | 
 |  | 
 | def FindAndroid(abi, bootstrap): | 
 |   if VERBOSE: | 
 |     sys.stderr.write('Looking for an Android device running abi %s...\n' % abi) | 
 |   AddSdkToolsToPath() | 
 |   device = FindAndroidRunning(abi) | 
 |   if not device: | 
 |     if bootstrap: | 
 |       if VERBOSE: | 
 |         sys.stderr.write("No emulator found, try to create one.\n") | 
 |       avdName = 'dart-build-%s' % abi | 
 |       EnsureAvdExists(avdName, abi) | 
 |  | 
 |       # It takes a while to start up an emulator. | 
 |       # Provide feedback while we wait. | 
 |       pollResult = [None] | 
 |       def pollFunction(): | 
 |         if VERBOSE: | 
 |           sys.stderr.write('.') | 
 |         pollResult[0] = FindAndroidRunning(abi) | 
 |         # Stop polling once we have our result. | 
 |         return pollResult[0] != None | 
 |       StartEmulator(abi, avdName, pollFunction) | 
 |       device = pollResult[0] | 
 |   return device | 
 |  | 
 |  | 
 | def Main(): | 
 |   # Parse options. | 
 |   parser = BuildOptions() | 
 |   (options, args) = parser.parse_args() | 
 |   if not ProcessOptions(options): | 
 |     parser.print_help() | 
 |     return 1 | 
 |  | 
 |   # If there are additional arguments, report error and exit. | 
 |   if args: | 
 |     parser.print_help() | 
 |     return 1 | 
 |  | 
 |   try: | 
 |     device = FindAndroid(options.abi, options.bootstrap) | 
 |     if device != None: | 
 |       sys.stdout.write("%s\n" % device) | 
 |       return 0 | 
 |     else: | 
 |       if VERBOSE: | 
 |         sys.stderr.write('Could not find device\n') | 
 |       return 2 | 
 |   except utils.Error as e: | 
 |     sys.stderr.write("error: %s\n" % e) | 
 |     if DEBUG: | 
 |       traceback.print_exc(file=sys.stderr) | 
 |     return -1 | 
 |  | 
 |  | 
 | if __name__ == '__main__': | 
 |   sys.exit(Main()) |