blob: 568e497496e3930b3710b0ca8d5e09a1bc2c4b7b [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.
"""
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)
message = ['adb %s: failed ' % adb_cmd]
if status:
message.append('with exit status %s ' % self.status)
if output:
message.append('and output:\n')
message.extend('- %s\n' % line for line in output.splitlines())
else:
message.append('and no output.')
message = ''.join(message)
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__()
for i in range(len(super_result)):
result[i] = super_result[i]
# 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 AdbShellCommandFailedError(AdbCommandFailedError):
"""Exception for shell command failures run via adb."""
def __init__(self, command, output, status, device_serial=None):
self.command = command
message = ['shell command run via adb failed on the device:\n',
' command: %s\n' % command]
message.append(' exit status: %s\n' % status)
if output:
message.append(' output:\n')
if isinstance(output, basestring):
output_lines = output.splitlines()
else:
output_lines = output
message.extend(' - %s\n' % line for line in output_lines)
else:
message.append(" output: ''\n")
message = ''.join(message)
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__()
for i in range(len(super_result)):
result[i] = super_result[i]
# 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."""
pass
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)