2017-03-27 11:45:21 +01:00
|
|
|
#!/usr/bin/env python3
|
|
|
|
|
2017-03-24 15:49:07 +00:00
|
|
|
import json
|
|
|
|
import logging
|
|
|
|
import os
|
|
|
|
import subprocess
|
|
|
|
|
|
|
|
from src import CONFIG_FILE, ROOT
|
|
|
|
from src import log
|
|
|
|
|
|
|
|
log.init()
|
|
|
|
logger = logging.getLogger('app')
|
|
|
|
|
|
|
|
|
2017-03-27 11:45:21 +01:00
|
|
|
def get_or_raise(env: str) -> str:
|
2017-03-24 15:49:07 +00:00
|
|
|
"""
|
|
|
|
Check if needed environment variables are given.
|
|
|
|
|
|
|
|
:param env: key
|
|
|
|
:return: value
|
|
|
|
"""
|
|
|
|
env_value = os.getenv(env)
|
|
|
|
if not env_value:
|
|
|
|
raise RuntimeError('The environment variable {0:s} is missing.'
|
|
|
|
'Please check docker image or Dockerfile!'.format(env))
|
|
|
|
return env_value
|
|
|
|
|
|
|
|
|
2017-05-24 10:52:48 +01:00
|
|
|
def convert_str_to_bool(str: str) -> bool:
|
2017-03-24 15:49:07 +00:00
|
|
|
"""
|
|
|
|
Convert string to boolean.
|
|
|
|
|
|
|
|
:param str: given string
|
|
|
|
:return: converted string
|
|
|
|
"""
|
|
|
|
try:
|
|
|
|
return str.lower() in ('yes', 'true', 't', '1')
|
|
|
|
except AttributeError as err:
|
|
|
|
logger.error(err)
|
|
|
|
|
|
|
|
|
|
|
|
ANDROID_HOME = get_or_raise('ANDROID_HOME')
|
|
|
|
ANDROID_VERSION = get_or_raise('ANDROID_VERSION')
|
|
|
|
API_LEVEL = get_or_raise('API_LEVEL')
|
|
|
|
PROCESSOR = get_or_raise('PROCESSOR')
|
|
|
|
SYS_IMG = get_or_raise('SYS_IMG')
|
2017-04-26 14:16:26 +01:00
|
|
|
IMG_TYPE = get_or_raise('IMG_TYPE')
|
2017-03-24 15:49:07 +00:00
|
|
|
|
|
|
|
logger.info('Android version: {version} \n'
|
|
|
|
'API level: {level} \n'
|
|
|
|
'Processor: {processor} \n'
|
2017-04-26 14:16:26 +01:00
|
|
|
'System image: {img} \n'
|
|
|
|
'Image type: {img_type}'.format(version=ANDROID_VERSION, level=API_LEVEL, processor=PROCESSOR,
|
|
|
|
img=SYS_IMG, img_type=IMG_TYPE))
|
2017-03-24 15:49:07 +00:00
|
|
|
|
|
|
|
|
2017-03-27 11:45:21 +01:00
|
|
|
def prepare_avd(device: str, avd_name: str):
|
2017-03-24 15:49:07 +00:00
|
|
|
"""
|
|
|
|
Create and run android virtual device.
|
|
|
|
|
|
|
|
:param device: Device name
|
|
|
|
:param avd_name: Name of android virtual device / emulator
|
|
|
|
"""
|
2017-04-26 14:16:26 +01:00
|
|
|
cmd = 'echo no | android create avd -f -n {name} -t android-{api} -b {img_type}{sys_img}'.format(
|
|
|
|
name=avd_name, api=API_LEVEL, img_type='google_apis/' if IMG_TYPE == 'google_apis' else '',
|
|
|
|
sys_img=SYS_IMG)
|
2017-03-24 15:49:07 +00:00
|
|
|
|
|
|
|
# Link emulator skins
|
|
|
|
skin_rsc_path = os.path.join(ROOT, 'devices', 'skins')
|
|
|
|
logger.info('Skin ressource path: {rsc}'.format(rsc=skin_rsc_path))
|
|
|
|
skin_dst_path = os.path.join(ANDROID_HOME, 'platforms', 'android-{api}'.format(api=API_LEVEL), 'skins')
|
|
|
|
logger.info('Skin destination path: {dst}'.format(dst=skin_dst_path))
|
|
|
|
for s in os.listdir(skin_rsc_path):
|
|
|
|
os.symlink(os.path.join(skin_rsc_path, s), os.path.join(skin_dst_path, s))
|
|
|
|
|
|
|
|
# Hardware and its skin
|
|
|
|
device_name_bash = device.replace(' ', '\ ')
|
|
|
|
skin_name = device.replace(' ', '_').lower()
|
|
|
|
logger.info('Device name in bash: {db}, Skin name: {skin}'.format(db=device_name_bash, skin=skin_name))
|
|
|
|
|
|
|
|
# For custom hardware profile
|
|
|
|
profile_dst_path = os.path.join(ROOT, '.android', 'devices.xml')
|
|
|
|
if 'samsung' in device.lower():
|
|
|
|
# profile file name = skin name
|
|
|
|
profile_src_path = os.path.join(ROOT, 'devices', 'profiles', '{profile}.xml'.format(profile=skin_name))
|
|
|
|
logger.info('Hardware profile resource path: {rsc}'.format(rsc=profile_src_path))
|
|
|
|
logger.info('Hardware profile destination path: {dst}'.format(dst=profile_dst_path))
|
|
|
|
os.symlink(profile_src_path, profile_dst_path)
|
|
|
|
|
|
|
|
# Append command
|
|
|
|
cmd += ' -d {device} -s {skin}'.format(device=device_name_bash, skin=skin_name)
|
|
|
|
logger.info('AVD creation command: {cmd}'.format(cmd=cmd))
|
2017-04-06 15:29:11 +01:00
|
|
|
subprocess.check_call(cmd, shell=True)
|
2017-03-24 15:49:07 +00:00
|
|
|
|
|
|
|
|
2017-03-27 11:45:21 +01:00
|
|
|
def appium_run(avd_name: str):
|
2017-03-24 15:49:07 +00:00
|
|
|
"""
|
|
|
|
Run appium server.
|
|
|
|
|
|
|
|
:param avd_name: Name of android virtual device / emulator
|
|
|
|
"""
|
|
|
|
cmd = 'appium'
|
2017-05-17 08:58:29 +01:00
|
|
|
local_ip = os.popen('ifconfig eth0 | grep \'inet addr:\' | cut -d: -f2 | awk \'{ print $1}\'').read().strip()
|
2017-03-24 15:49:07 +00:00
|
|
|
|
2017-05-24 10:52:48 +01:00
|
|
|
grid_connect = convert_str_to_bool(str(os.getenv('CONNECT_TO_GRID', False)))
|
2017-03-24 15:49:07 +00:00
|
|
|
logger.info('Connect to selenium grid? {connect}'.format(connect=grid_connect))
|
|
|
|
if grid_connect:
|
|
|
|
try:
|
2017-05-24 15:00:29 +01:00
|
|
|
mobile_web_test = convert_str_to_bool(str(os.getenv('MOBILE_WEB_TEST', False)))
|
2017-05-24 13:50:34 +01:00
|
|
|
default_web_browser = os.getenv('BROWSER')
|
2017-05-17 08:58:29 +01:00
|
|
|
appium_host = os.getenv('APPIUM_HOST', local_ip)
|
2017-03-24 15:49:07 +00:00
|
|
|
appium_port = int(os.getenv('APPIUM_PORT', 4723))
|
|
|
|
selenium_host = os.getenv('SELENIUM_HOST', '172.17.0.1')
|
|
|
|
selenium_port = int(os.getenv('SELENIUM_PORT', 4444))
|
2017-05-24 13:50:34 +01:00
|
|
|
browser_name = default_web_browser if mobile_web_test else 'android'
|
2017-05-23 14:47:41 +01:00
|
|
|
create_node_config(avd_name, browser_name, appium_host, appium_port, selenium_host, selenium_port)
|
2017-03-24 15:49:07 +00:00
|
|
|
cmd += ' --nodeconfig {file}'.format(file=CONFIG_FILE)
|
|
|
|
except ValueError as v_err:
|
|
|
|
logger.error(v_err)
|
2017-05-17 08:58:29 +01:00
|
|
|
title = 'Appium Server'
|
|
|
|
subprocess.check_call('xterm -T "{title}" -n "{title}" -e \"{cmd}\"'.format(title=title, cmd=cmd), shell=True)
|
2017-03-24 15:49:07 +00:00
|
|
|
|
|
|
|
|
2017-05-23 14:47:41 +01:00
|
|
|
def create_node_config(avd_name: str, browser_name: str, appium_host: str, appium_port: int, selenium_host: str, selenium_port: int):
|
2017-03-24 15:49:07 +00:00
|
|
|
"""
|
|
|
|
Create custom node config file in json format to be able to connect with selenium server.
|
|
|
|
|
|
|
|
:param avd_name: Name of android virtual device / emulator
|
|
|
|
:param appium_host: Host where appium server is running
|
|
|
|
:param appium_port: Port number where where appium server is running
|
|
|
|
:param selenium_host: Host where selenium server is running
|
|
|
|
:param selenium_port: Port number where selenium server is running
|
|
|
|
"""
|
|
|
|
config = {
|
|
|
|
'capabilities': [
|
|
|
|
{
|
|
|
|
'platform': 'Android',
|
|
|
|
'platformName': 'Android',
|
|
|
|
'version': ANDROID_VERSION,
|
2017-05-23 14:47:41 +01:00
|
|
|
'browserName': browser_name,
|
|
|
|
'deviceName': avd_name,
|
2017-03-24 15:49:07 +00:00
|
|
|
'maxInstances': 1,
|
|
|
|
}
|
|
|
|
],
|
|
|
|
'configuration': {
|
|
|
|
'cleanUpCycle': 2000,
|
|
|
|
'timeout': 30000,
|
|
|
|
'proxy': 'org.openqa.grid.selenium.proxy.DefaultRemoteProxy',
|
|
|
|
'url': 'http://{host}:{port}/wd/hub'.format(host=appium_host, port=appium_port),
|
|
|
|
'host': appium_host,
|
|
|
|
'port': appium_port,
|
|
|
|
'maxSession': 6,
|
|
|
|
'register': True,
|
|
|
|
'registerCycle': 5000,
|
|
|
|
'hubHost': selenium_host,
|
2017-05-25 15:48:59 +01:00
|
|
|
'hubPort': selenium_port,
|
|
|
|
'unregisterIfStillDownAfter': 120000
|
2017-03-24 15:49:07 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
logger.info('Appium node config: {config}'.format(config=config))
|
|
|
|
with open(CONFIG_FILE, 'w') as cf:
|
|
|
|
cf.write(json.dumps(config))
|
|
|
|
|
|
|
|
|
|
|
|
def run():
|
|
|
|
"""Run app."""
|
|
|
|
device = os.getenv('DEVICE', 'Nexus 5')
|
|
|
|
logger.info('Device: {device}'.format(device=device))
|
|
|
|
|
|
|
|
avd_name = '{device}_{version}'.format(device=device.replace(' ', '_').lower(), version=ANDROID_VERSION)
|
|
|
|
logger.info('AVD name: {avd}'.format(avd=avd_name))
|
|
|
|
|
2017-04-06 15:29:11 +01:00
|
|
|
logger.info('Preparing emulator...')
|
2017-03-24 15:49:07 +00:00
|
|
|
prepare_avd(device, avd_name)
|
2017-04-06 15:29:11 +01:00
|
|
|
logger.info('Run emulator...')
|
|
|
|
cmd = 'emulator -avd {name}'.format(name=avd_name)
|
|
|
|
subprocess.Popen(cmd.split())
|
|
|
|
|
2017-05-24 10:52:48 +01:00
|
|
|
appium = convert_str_to_bool(str(os.getenv('APPIUM', False)))
|
2017-03-24 15:49:07 +00:00
|
|
|
if appium:
|
2017-04-06 15:29:11 +01:00
|
|
|
logger.info('Run appium server...')
|
2017-03-24 15:49:07 +00:00
|
|
|
appium_run(avd_name)
|
|
|
|
|
|
|
|
if __name__ == '__main__':
|
|
|
|
run()
|