blob: d7b787e8827c7e08f3fe5125ef6538923e4641c3 [file] [log] [blame]
# 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
"""
import six
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."""
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, six.string_types):
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."""
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."""
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)