# Copyright 2014 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. """ Exception classes raised by AdbWrapper and DeviceUtils. The class hierarchy for device exceptions is: base_error.BaseError +-- CommandFailedError | +-- AdbCommandFailedError | | +-- AdbShellCommandFailedError | +-- AdbVersionError | +-- FastbootCommandFailedError | +-- DeviceVersionError | +-- DeviceChargingError | +-- RootUserBuildError +-- CommandTimeoutError +-- DeviceUnreachableError +-- NoDevicesError +-- MultipleDevicesError +-- NoAdbError """ from devil import base_error from devil.utils import cmd_helper from devil.utils import parallelizer class CommandFailedError(base_error.BaseError): """Exception for command failures.""" def __init__(self, message, device_serial=None): device_leader = '(device: %s)' % device_serial if device_serial is not None and not message.startswith(device_leader): message = '%s %s' % (device_leader, message) self.device_serial = device_serial super(CommandFailedError, self).__init__(message) def __eq__(self, other): return (super(CommandFailedError, self).__eq__(other) and self.device_serial == other.device_serial) def __ne__(self, other): return not self == other class _BaseCommandFailedError(CommandFailedError): """Base Exception for adb and fastboot command failures.""" def __init__(self, args, output, status=None, device_serial=None, message=None): self.args = args self.output = output self.status = status if not message: adb_cmd = ' '.join(cmd_helper.SingleQuote(arg) for arg in self.args) segments = ['adb %s: failed ' % adb_cmd] if status: segments.append('with exit status %s ' % self.status) if output: segments.append('and output:\n') segments.extend('- %s\n' % line for line in output.splitlines()) else: segments.append('and no output.') message = ''.join(segments) super(_BaseCommandFailedError, self).__init__(message, device_serial) def __eq__(self, other): return (super(_BaseCommandFailedError, self).__eq__(other) and self.args == other.args and self.output == other.output and self.status == other.status) def __ne__(self, other): return not self == other def __reduce__(self): """Support pickling.""" result = [None, None, None, None, None] super_result = super(_BaseCommandFailedError, self).__reduce__() result[:len(super_result)] = super_result # Update the args used to reconstruct this exception. result[1] = (self.args, self.output, self.status, self.device_serial, self.message) return tuple(result) class AdbCommandFailedError(_BaseCommandFailedError): """Exception for adb command failures.""" def __init__(self, args, output, status=None, device_serial=None, message=None): super(AdbCommandFailedError, self).__init__( args, output, status=status, message=message, device_serial=device_serial) class FastbootCommandFailedError(_BaseCommandFailedError): """Exception for fastboot command failures.""" def __init__(self, args, output, status=None, device_serial=None, message=None): super(FastbootCommandFailedError, self).__init__( args, output, status=status, message=message, device_serial=device_serial) class DeviceVersionError(CommandFailedError): """Exception for device version failures.""" def __init__(self, message, device_serial=None): super(DeviceVersionError, self).__init__(message, device_serial) class AdbVersionError(CommandFailedError): """Exception for running a command on an incompatible version of adb.""" def __init__(self, args, desc=None, actual_version=None, min_version=None): adb_cmd = ' '.join(cmd_helper.SingleQuote(arg) for arg in args) desc = desc or 'not supported' if min_version: desc += ' prior to %s' % min_version if actual_version: desc += ' (actual: %s)' % actual_version super(AdbVersionError, self).__init__(message='adb %s: %s' % (adb_cmd, desc)) class AdbShellCommandFailedError(AdbCommandFailedError): """Exception for shell command failures run via adb.""" def __init__(self, command, output, status, device_serial=None): self.command = command segments = [ 'shell command run via adb failed on the device:\n', ' command: %s\n' % command ] segments.append(' exit status: %s\n' % status) if output: segments.append(' output:\n') if isinstance(output, basestring): output_lines = output.splitlines() else: output_lines = output segments.extend(' - %s\n' % line for line in output_lines) else: segments.append(" output: ''\n") message = ''.join(segments) super(AdbShellCommandFailedError, self).__init__( ['shell', command], output, status, device_serial, message) def __reduce__(self): """Support pickling.""" result = [None, None, None, None, None] super_result = super(AdbShellCommandFailedError, self).__reduce__() result[:len(super_result)] = super_result # Update the args used to reconstruct this exception. result[1] = (self.command, self.output, self.status, self.device_serial) return tuple(result) class CommandTimeoutError(base_error.BaseError): """Exception for command timeouts.""" def __init__(self, message, is_infra_error=False, output=None): super(CommandTimeoutError, self).__init__(message, is_infra_error) self.output = output class DeviceUnreachableError(base_error.BaseError): """Exception for device unreachable failures.""" pass class NoDevicesError(base_error.BaseError): """Exception for having no devices attached.""" def __init__(self, msg=None): super(NoDevicesError, self).__init__( msg or 'No devices attached.', is_infra_error=True) class MultipleDevicesError(base_error.BaseError): """Exception for having multiple attached devices without selecting one.""" def __init__(self, devices): parallel_devices = parallelizer.Parallelizer(devices) descriptions = parallel_devices.pMap(lambda d: d.build_description).pGet( None) msg = ('More than one device available. Use -d/--device to select a device ' 'by serial.\n\nAvailable devices:\n') for d, desc in zip(devices, descriptions): msg += ' %s (%s)\n' % (d, desc) super(MultipleDevicesError, self).__init__(msg, is_infra_error=True) class NoAdbError(base_error.BaseError): """Exception for being unable to find ADB.""" def __init__(self, msg=None): super(NoAdbError, self).__init__( msg or 'Unable to find adb.', is_infra_error=True) class DeviceChargingError(CommandFailedError): """Exception for device charging errors.""" def __init__(self, message, device_serial=None): super(DeviceChargingError, self).__init__(message, device_serial) class RootUserBuildError(CommandFailedError): """Exception for being unable to root a device with "user" build.""" def __init__(self, message=None, device_serial=None): super(RootUserBuildError, self).__init__( message or 'Unable to root device with user build.', device_serial)