#!/usr/bin/env python # Copyright 2015 The Chromium Authors. All rights reserved. # Use of this source code is governed by a BSD-style license that can be # found in the LICENSE file. """Launches a daemon to monitor android device temperatures & status. This script will repeatedly poll the given devices for their temperatures and status every 60 seconds and dump the stats to file on the host. """ import argparse import collections import json import logging import logging.handlers import os import re import socket import sys import time if __name__ == '__main__': sys.path.append( os.path.abspath(os.path.join(os.path.dirname(__file__), '..', '..', '..'))) from devil import devil_env from devil.android import battery_utils from devil.android import device_blacklist from devil.android import device_errors from devil.android import device_utils # Various names of sensors used to measure cpu temp CPU_TEMP_SENSORS = [ # most nexus devices 'tsens_tz_sensor0', # android one 'mtktscpu', # nexus 9 'CPU-therm', ] DEVICE_FILE_VERSION = 1 DEVICE_FILE = os.path.join( os.path.expanduser('~'), '.android', '%s__android_device_status.json' % socket.gethostname().split('.')[0]) MEM_INFO_REGEX = re.compile(r'.*?\:\s*(\d+)\s*kB') # ex: 'MemTotal: 185735 kB' def get_device_status_unsafe(device): """Polls the given device for various info. Returns: A dict of the following format: { 'battery': { 'level': 100, 'temperature': 123 }, 'build': { 'build.id': 'ABC12D', 'product.device': 'chickenofthesea' }, 'mem': { 'avail': 1000000, 'total': 1234567, }, 'processes': 123, 'state': 'good', 'temp': { 'some_sensor': 30 }, 'uptime': 1234.56, } """ status = collections.defaultdict(dict) # Battery battery = battery_utils.BatteryUtils(device) battery_info = battery.GetBatteryInfo() try: level = int(battery_info.get('level')) except (KeyError, TypeError, ValueError): level = None if level and level >= 0 and level <= 100: status['battery']['level'] = level try: temperature = int(battery_info.get('temperature')) except (KeyError, TypeError, ValueError): temperature = None if temperature: status['battery']['temperature'] = temperature # Build status['build']['build.id'] = device.build_id status['build']['product.device'] = device.build_product # Memory mem_info = '' try: mem_info = device.ReadFile('/proc/meminfo') except device_errors.AdbShellCommandFailedError: logging.exception('Unable to read /proc/meminfo') for line in mem_info.splitlines(): match = MEM_INFO_REGEX.match(line) if match: try: value = int(match.group(1)) except ValueError: continue key = line.split(':')[0].strip() if 'MemTotal' == key: status['mem']['total'] = value elif 'MemFree' == key: status['mem']['free'] = value # Process try: # TODO(catapult:#3215): Migrate to device.GetPids() lines = device.RunShellCommand(['ps'], check_return=True) status['processes'] = len(lines) - 1 # Ignore the header row. except device_errors.AdbShellCommandFailedError: logging.exception('Unable to count process list.') # CPU Temps # Find a thermal sensor that matches one in CPU_TEMP_SENSORS and read its # temperature. files = [] try: files = device.RunShellCommand( 'grep -lE "%s" /sys/class/thermal/thermal_zone*/type' % '|'.join( CPU_TEMP_SENSORS), shell=True, check_return=True) except device_errors.AdbShellCommandFailedError: logging.exception('Unable to list thermal sensors.') for f in files: try: sensor_name = device.ReadFile(f).strip() temp = float(device.ReadFile(f[:-4] + 'temp').strip()) # s/type^/temp status['temp'][sensor_name] = temp except (device_errors.AdbShellCommandFailedError, ValueError): logging.exception('Unable to read thermal sensor %s', f) # Uptime try: uptimes = device.ReadFile('/proc/uptime').split() status['uptime'] = float(uptimes[0]) # Take the first field (actual uptime) except (device_errors.AdbShellCommandFailedError, ValueError): logging.exception('Unable to read /proc/uptime') status['state'] = 'available' return status def get_device_status(device): try: status = get_device_status_unsafe(device) except device_errors.DeviceUnreachableError: status = {'state': 'offline'} return status def get_all_status(blacklist): status_dict = { 'version': DEVICE_FILE_VERSION, 'devices': {}, } healthy_devices = device_utils.DeviceUtils.HealthyDevices(blacklist) parallel_devices = device_utils.DeviceUtils.parallel(healthy_devices) results = parallel_devices.pMap(get_device_status).pGet(None) status_dict['devices'] = { device.serial: result for device, result in zip(healthy_devices, results) } if blacklist: for device, reason in blacklist.Read().iteritems(): status_dict['devices'][device] = { 'state': reason.get('reason', 'blacklisted')} status_dict['timestamp'] = time.time() return status_dict def main(argv): """Launches the device monitor. Polls the devices for their battery and cpu temperatures and scans the blacklist file every 60 seconds and dumps the data to DEVICE_FILE. """ parser = argparse.ArgumentParser( description='Launches the device monitor.') parser.add_argument('--adb-path', help='Path to adb binary.') parser.add_argument('--blacklist-file', help='Path to device blacklist file.') args = parser.parse_args(argv) logger = logging.getLogger() logger.setLevel(logging.DEBUG) handler = logging.handlers.RotatingFileHandler( '/tmp/device_monitor.log', maxBytes=10 * 1024 * 1024, backupCount=5) fmt = logging.Formatter('%(asctime)s %(levelname)s %(message)s', datefmt='%y%m%d %H:%M:%S') handler.setFormatter(fmt) logger.addHandler(handler) devil_dynamic_config = devil_env.EmptyConfig() if args.adb_path: devil_dynamic_config['dependencies'].update( devil_env.LocalConfigItem( 'adb', devil_env.GetPlatform(), args.adb_path)) devil_env.config.Initialize(configs=[devil_dynamic_config]) blacklist = (device_blacklist.Blacklist(args.blacklist_file) if args.blacklist_file else None) logging.info('Device monitor running with pid %d, adb: %s, blacklist: %s', os.getpid(), args.adb_path, args.blacklist_file) while True: start = time.time() status_dict = get_all_status(blacklist) with open(DEVICE_FILE, 'wb') as f: json.dump(status_dict, f, indent=2, sort_keys=True) logging.info('Got status of all devices in %.2fs.', time.time() - start) time.sleep(60) if __name__ == '__main__': sys.exit(main(sys.argv[1:]))