blob: fe895670f1302054a40525fb19ac940d29fcb276 [file] [log] [blame]
#!/usr/bin/env python
#
# Copyright 2013 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.
"""Runs semi-automated update testing on a non-rooted device."""
import logging
import optparse
import os
import shutil
import sys
import time
from pylib import android_commands
from pylib.device import device_utils
def _SaveAppData(device, package_name, from_apk=None, data_dir=None):
def _BackupAppData(data_dir=None):
device.old_interface.Adb().SendCommand('backup %s' % package_name)
backup_file = os.path.join(os.getcwd(), 'backup.ab')
assert os.path.exists(backup_file), 'Backup failed.'
if data_dir:
if not os.path.isdir(data_dir):
os.makedirs(data_dir)
shutil.move(backup_file, data_dir)
backup_file = os.path.join(data_dir, 'backup.ab')
print 'Application data saved to %s' % backup_file
if from_apk:
logging.info('Installing %s...', from_apk)
# TODO(jbudorick) Switch to AdbWrapper.Install on the impl switch.
output = device.old_interface.Install(from_apk, reinstall=True)
if 'Success' not in output:
raise Exception('Unable to install %s. output: %s' % (from_apk, output))
raw_input('Set the application state. Once ready, press enter and '
'select "Backup my data" on the device.')
_BackupAppData(data_dir)
def _VerifyAppUpdate(device, to_apk, app_data, from_apk=None):
def _RestoreAppData():
assert os.path.exists(app_data), 'Backup file does not exist!'
device.old_interface.Adb().SendCommand('restore %s' % app_data)
# It seems restore command is not synchronous.
time.sleep(15)
if from_apk:
logging.info('Installing %s...', from_apk)
# TODO(jbudorick) Switch to AdbWrapper.Install on the impl switch.
output = device.old_interface.Install(from_apk, reinstall=True)
if 'Success' not in output:
raise Exception('Unable to install %s. output: %s' % (from_apk, output))
logging.info('Restoring the application data...')
raw_input('Press enter and select "Restore my data" on the device.')
_RestoreAppData()
logging.info('Verifying that %s cannot be installed side-by-side...',
to_apk)
# TODO(jbudorick) Switch to AdbWrapper.Install on the impl switch.
output = device.old_interface.Install(to_apk)
if 'INSTALL_FAILED_ALREADY_EXISTS' not in output:
if 'Success' in output:
raise Exception('Package name has changed! output: %s' % output)
else:
raise Exception(output)
logging.info('Verifying that %s can be overinstalled...', to_apk)
# TODO(jbudorick) Switch to AdbWrapper.Install on the impl switch.
output = device.old_interface.Install(to_apk, reinstall=True)
if 'Success' not in output:
raise Exception('Unable to install %s.\n output: %s' % (to_apk, output))
logging.info('Successfully updated to the new apk. Please verify that the '
'the application data is preserved.')
def main():
logger = logging.getLogger()
logger.setLevel(logging.DEBUG)
desc = (
'Performs semi-automated application update verification testing. '
'When given --save, it takes a snapshot of the application data '
'on the device. (A dialog on the device will prompt the user to grant '
'permission to backup the data.) Otherwise, it performs the update '
'testing as follows: '
'1. Installs the |from-apk| (optional). '
'2. Restores the previously stored snapshot of application data '
'given by |app-data| '
'(A dialog on the device will prompt the user to grant permission to '
'restore the data.) '
'3. Verifies that |to-apk| cannot be installed side-by-side. '
'4. Verifies that |to-apk| can replace |from-apk|.')
parser = optparse.OptionParser(description=desc)
parser.add_option('--package-name', help='Package name for the application.')
parser.add_option('--save', action='store_true',
help=('Save a snapshot of application data. '
'This will be saved as backup.db in the '
'current directory if |app-data| directory '
'is not specifid.'))
parser.add_option('--from-apk',
help=('APK to update from. This is optional if you already '
'have the app installed.'))
parser.add_option('--to-apk', help='APK to update to.')
parser.add_option('--app-data',
help=('Path to the application data to be restored or the '
'directory where the data should be saved.'))
(options, args) = parser.parse_args()
if args:
parser.print_help(sys.stderr)
parser.error('Unknown arguments: %s.' % args)
devices = android_commands.GetAttachedDevices()
if len(devices) != 1:
parser.error('Exactly 1 device must be attached.')
device = device_utils.DeviceUtils(devices[0])
if options.from_apk:
assert os.path.isfile(options.from_apk)
if options.save:
if not options.package_name:
parser.print_help(sys.stderr)
parser.error('Missing --package-name.')
_SaveAppData(device, options.package_name, from_apk=options.from_apk,
data_dir=options.app_data)
else:
if not options.to_apk or not options.app_data:
parser.print_help(sys.stderr)
parser.error('Missing --to-apk or --app-data.')
assert os.path.isfile(options.to_apk)
assert os.path.isfile(options.app_data)
_VerifyAppUpdate(device, options.to_apk, options.app_data,
from_apk=options.from_apk)
if __name__ == '__main__':
main()