
Аудитория
QA-инженеры, тестировщики мобильных приложений, автоматизаторы.
Проблема
Во время тестирования приложений под Android (не только, но далее речь пойдет только про данную платформу), приходится устанавливать множество сборок тестируемого продукта / продуктов. Этот процесс отнимает время и силы, которые эффективнее потратить на поиск багов.
В настоящей статье мы рассмотрим существующее решение, напишем свое на Python и сравним их.
Готовое решение
Пожалуй, самое популярное на данный момент решение этой проблемы предоставляет сервис Crashlytics, включающий установщик Beta.
Рассмотрим типичный процесс инсталляции приложения при помощи Crashlytics Beta:
- Находим иконку Beta (1) → делаем тап (2) = запускается приложение.
- Находим нужный проект (3) → делаем тап (4) = открывается экран со списком сборок.
- Находим нужную сборку (5) → делаем тап «Download» (6) = загружается установочный файл на устройство.
- Отображается экран с предложение установить приложение → тап «Установить» (7) = отображается экран установки.
- Находим установленное приложение (8) → делаем тап (9) = приложение запускается; готово для тестирования.
Итак, для инсталляции и запуска одной сборки приложения при помощи Crashlytics Beta необходимо совершить, в общей сложности, минимум девять действий. Будем ориентироваться на эти показатели и постараемся создать установщик, требующий для решения аналогичных задач, меньшее количество действий.
Кастомное решение
В качестве языка программирования выберем Python, т. к. он подходит для нашей задачи и является весьма популярным, в том числе и среди QA-инженеров.
Для взаимодействия с Android будем использовать adb, входящий в стандартный пакет Android SDK.
Для скачивания файлов — Wget.
В нашем случае сборки осуществляются в TeamCity.
Теперь перейдем к написанию кода.
Первым делом импортируем в проект модуль subprocess, он необходим для выполнения команд wget и adb.
import subprocess
Добавим необходимые настройки для Wget.
settings = {'user': '—user=логин_аккаунта_teamcity', 'password': '—password=пароль_аккаунта_teamcity', 'way': 'путь_куда_будут_скачиваться_сборки'}
Устанавливать приложения будем по номеру сборки, поэтому научим скрипт спрашивать этот параметр.
number = input('Укажите № сборки: ')
Допустим, что нам необходимо устанавливать сразу две сборки: тестовую и боевую. Будем скачивать их из TeamCity. Для этого узнаем полный путь до файлов, открыв страницу сервиса и отыскав сборки в артефактах. URL до сборок будет выглядеть приблизительно так:
https://teamcity.mysite.com/repository/app/номер_сборки/тип_сборки/myapp-номер_сборки-тип_сборки.apk
В адресе вместо номера сборки вы можете увидеть id, например, /1234:id/. Здесь мы будем указывать не id, а номер сборки.
Напишем функцию для скачивания заданных сборок.
def download(type_b): url = 'http://teamcity.mysite.com/repository/app/{0}/{1}/myapp-{0}-{1}.apk'.format(number, type_b) # Если файла нет — скачиваем, если есть, но не изменился — ничего не делаем, # если есть и изменился — перезаписываем subprocess.check_output(['wget', '-N', '--cache=off', '--progress=bar', settings['user'], settings['password'], '-P', settings['way'], url])
Напишем функцию для установки и запуска приложений. Сначала удаляем ранее установленные сборки. Не забываем, что если хотя бы одного приложения нет на устройстве – скрипт завершится с ошибкой. Чтобы избежать этого, будем игнорировать ошибки.
В данном воображаемом примере два пакета:
- com.myapp.prod
- com.myapp.test
Стартовые активности:
- com.myapp.prod/com.myapp.StartActivity
- com.myapp.test/com.myapp.StartActivity
У вас имена пакетов и активностей будут другими.
def install(type_b): try: # Удаляем старую сборку, указав package name subprocess.check_output(['adb', 'uninstall', 'com.myapp.{0}'.format(type_b)]) except: # Если старой сборки нет — игнорируем ошибку. pass finally: # Устанавливаем новую сборку subprocess.check_output(['adb', 'install', '{0}/myapp-{1}-{2}.apk'.format(settings['way'], number, type_b)]) # Запускаем новую сборку, указав activity name subprocess.check_output(['adb', 'shell', 'am', 'start', 'com.myapp.{0}/com.myapp.StartActivity'.format(type_b)]) print('(^_^) Сборка ({0}-{1}) установлена и запущена.'.format(number, type_b))
Все необходимы функции написаны. Теперь их можно применить.
Дополнительно добавим обработчик случая, когда заданной сборки нет в TeamCity.
while True: try: # Скачиваем сборку, смотрящую на боевой сервер download('prod') # Скачиваем сборку, смотрящую на тестовый сервер download('test') except Exception: # В случае ошибки — снова запрашиваем номер сборки number = input('¯\_(ツ)_/¯ Такой сборки нет.\nУкажите другой №: ') else: print('Начинаю установку…') # Устанавливаем и запускаем боевую сборку install('prod') # Устанавливаем и запускаем тестовую сборку install('test') # В случае успеха — немного Гомера Симпсона для хорошего настроения print('Уху! (_8(|)\n') break
Скрипт готов. Сохраняем его, например, под именем installer.py
Добавляем алиас, например, alias inst='python ~/scripts/installer.py'
Проверка
Итак, для установки одной сборки при помощи Crashlytics Beta необходимо совершить 9 действий. Для сравнения измерим данный показатель у скрипта.
- Запускаем скрипт командой inst (1) = отображается предложение задать номер сборки.
- Задаем номер сборки (2) = удаляются старые сборки; скачиваются, устанавливаются и запускаются новые. Приложение готово для тестирования.
Результат
Beta (1 сборка) — 9 действий (не учитывая удаление старых сборок).
Свой скрипт (сколько угодно сборок) — 2 действия.
Дополнительным бонусом кастомного решения является то, что оно масштабируется (добавление установки нескольких продуктов на ряд устройств и т. д.), а также легко адаптируется под задачи автоматического тестирования.
