2018-08-10 13:09:19 +03:00
|
|
|
import requests
|
|
|
|
import pytest
|
|
|
|
import re
|
2018-04-28 11:02:39 +02:00
|
|
|
from _pytest.runner import runtestprotocol
|
2018-11-29 20:16:26 +02:00
|
|
|
from http.client import RemoteDisconnected
|
2018-11-07 13:26:32 +01:00
|
|
|
from support.device_stats_db import DeviceStatsDB
|
2018-04-28 11:02:39 +02:00
|
|
|
from support.test_rerun import should_rerun_test
|
2018-11-07 13:26:32 +01:00
|
|
|
from tests import test_suite_data, appium_container
|
2017-08-31 16:39:41 +03:00
|
|
|
from datetime import datetime
|
|
|
|
from os import environ
|
2017-09-26 13:50:34 +03:00
|
|
|
from io import BytesIO
|
2019-01-30 14:02:24 +02:00
|
|
|
from sauceclient import SauceClient, SauceException
|
2018-09-17 11:50:01 +03:00
|
|
|
from support.api.network_api import NetworkApi
|
2018-04-26 15:34:15 +03:00
|
|
|
from support.github_report import GithubHtmlReport
|
|
|
|
from support.testrail_report import TestrailReport
|
2018-09-17 11:50:01 +03:00
|
|
|
from tests.users import transaction_senders
|
2017-08-31 16:39:41 +03:00
|
|
|
|
|
|
|
sauce_username = environ.get('SAUCE_USERNAME')
|
|
|
|
sauce_access_key = environ.get('SAUCE_ACCESS_KEY')
|
2018-01-26 13:07:09 +02:00
|
|
|
github_token = environ.get('GIT_HUB_TOKEN')
|
|
|
|
|
|
|
|
sauce = SauceClient(sauce_username, sauce_access_key)
|
2018-10-26 15:32:05 +02:00
|
|
|
github_report = GithubHtmlReport()
|
|
|
|
testrail_report = TestrailReport()
|
2017-08-31 16:39:41 +03:00
|
|
|
|
|
|
|
|
2017-08-28 13:02:20 +03:00
|
|
|
def pytest_addoption(parser):
|
2017-08-31 16:39:41 +03:00
|
|
|
parser.addoption("--build",
|
|
|
|
action="store",
|
2018-04-25 20:02:03 +03:00
|
|
|
default=datetime.now().strftime('%Y-%m-%d-%H-%M'),
|
2017-08-28 13:02:20 +03:00
|
|
|
help="Specify build name")
|
2017-08-31 16:39:41 +03:00
|
|
|
parser.addoption('--apk',
|
|
|
|
action='store',
|
2018-04-30 12:58:20 +03:00
|
|
|
default=None,
|
2017-11-13 12:49:45 +02:00
|
|
|
help='Url or local path to apk')
|
2017-10-05 22:41:17 +03:00
|
|
|
parser.addoption('--env',
|
|
|
|
action='store',
|
|
|
|
default='sauce',
|
2018-06-07 17:13:37 +02:00
|
|
|
help='Specify environment: local/sauce/api')
|
2018-11-07 13:26:32 +01:00
|
|
|
parser.addoption('--platform_version',
|
|
|
|
action='store',
|
2018-12-19 12:50:39 +02:00
|
|
|
default='8.0',
|
2018-11-07 13:26:32 +01:00
|
|
|
help='Android device platform version')
|
2019-05-23 12:04:41 +03:00
|
|
|
parser.addoption('--log_steps',
|
2017-11-13 12:49:45 +02:00
|
|
|
action='store',
|
|
|
|
default=False,
|
|
|
|
help='Display each test step in terminal as plain text: True/False')
|
2018-01-26 13:07:09 +02:00
|
|
|
parser.addoption('--pr_number',
|
|
|
|
action='store',
|
|
|
|
default=None,
|
|
|
|
help='Pull Request number')
|
2018-05-23 16:02:45 +03:00
|
|
|
parser.addoption('--testrail_report',
|
2018-04-26 15:34:15 +03:00
|
|
|
action='store',
|
|
|
|
default=False,
|
2018-05-23 16:02:45 +03:00
|
|
|
help='boolean; For creating testrail report per run')
|
2018-08-07 12:08:54 +03:00
|
|
|
parser.addoption('--network',
|
|
|
|
action='store',
|
|
|
|
default='ropsten',
|
|
|
|
help='string; ropsten or rinkeby')
|
2018-04-28 11:02:39 +02:00
|
|
|
parser.addoption('--rerun_count',
|
|
|
|
action='store',
|
|
|
|
default=0,
|
|
|
|
help='How many times tests should be re-run if failed')
|
2018-12-19 12:50:39 +02:00
|
|
|
parser.addoption("--run_testrail_ids",
|
|
|
|
action="store",
|
|
|
|
metavar="NAME",
|
|
|
|
default=None,
|
|
|
|
help="only run tests matching the environment NAME.")
|
2018-11-30 13:42:48 +02:00
|
|
|
|
2018-12-19 12:50:39 +02:00
|
|
|
# chat bot
|
2018-11-30 13:42:48 +02:00
|
|
|
|
2018-03-31 23:05:11 +03:00
|
|
|
parser.addoption('--messages_number',
|
|
|
|
action='store',
|
|
|
|
default=20,
|
|
|
|
help='Messages number')
|
2018-09-04 00:18:18 +03:00
|
|
|
parser.addoption('--public_keys',
|
|
|
|
action='store',
|
2018-09-28 15:31:02 +03:00
|
|
|
default='',
|
2018-09-04 00:18:18 +03:00
|
|
|
help='List of public keys for one-to-one chats')
|
|
|
|
parser.addoption('--running_time',
|
|
|
|
action='store',
|
2018-09-14 16:35:37 +03:00
|
|
|
default=600,
|
2018-09-04 00:18:18 +03:00
|
|
|
help='Running time in seconds')
|
2019-06-03 15:02:39 +03:00
|
|
|
parser.addoption('--chat_name',
|
|
|
|
action='store',
|
|
|
|
default='test_chat',
|
|
|
|
help='Public chat name')
|
|
|
|
parser.addoption('--device_number',
|
|
|
|
action='store',
|
|
|
|
default=2,
|
|
|
|
help='Public chat name')
|
2018-09-04 00:18:18 +03:00
|
|
|
|
2018-11-07 13:26:32 +01:00
|
|
|
# running tests using appium docker instance
|
|
|
|
|
|
|
|
parser.addoption('--docker',
|
|
|
|
action='store',
|
|
|
|
default=False,
|
|
|
|
help='Are you using the appium docker container to run the tests?')
|
|
|
|
parser.addoption('--docker_shared_volume',
|
|
|
|
action='store',
|
|
|
|
default=None,
|
|
|
|
help='Path to a directory with .apk that will be shared with docker instance. Test reports will be also saved there')
|
|
|
|
parser.addoption('--device_ip',
|
|
|
|
action='store',
|
|
|
|
default=None,
|
|
|
|
help='Android device IP address used for battery tests')
|
|
|
|
parser.addoption('--bugreport',
|
|
|
|
action='store',
|
|
|
|
default=False,
|
|
|
|
help='Should generate bugreport for each test?')
|
|
|
|
parser.addoption('--stats_db_host',
|
|
|
|
action='store',
|
|
|
|
default=None,
|
|
|
|
help='Host address for device stats database')
|
|
|
|
parser.addoption('--stats_db_port',
|
|
|
|
action='store',
|
|
|
|
default=8086,
|
|
|
|
help='Port for device stats db')
|
|
|
|
parser.addoption('--stats_db_username',
|
|
|
|
action='store',
|
|
|
|
default=None,
|
|
|
|
help='Username for device stats db')
|
|
|
|
parser.addoption('--stats_db_password',
|
|
|
|
action='store',
|
|
|
|
default=None,
|
|
|
|
help='Password for device stats db')
|
|
|
|
parser.addoption('--stats_db_database',
|
|
|
|
action='store',
|
|
|
|
default='example9',
|
|
|
|
help='Database name for device stats db')
|
|
|
|
|
2018-04-28 11:02:39 +02:00
|
|
|
|
2017-08-31 16:39:41 +03:00
|
|
|
def is_master(config):
|
|
|
|
return not hasattr(config, 'slaveinput')
|
|
|
|
|
|
|
|
|
|
|
|
def is_uploaded():
|
2018-01-26 13:07:09 +02:00
|
|
|
stored_files = sauce.storage.get_stored_files()
|
2017-08-31 16:39:41 +03:00
|
|
|
for i in range(len(stored_files['files'])):
|
2018-02-27 00:38:56 +01:00
|
|
|
if stored_files['files'][i]['name'] == test_suite_data.apk_name:
|
2017-08-31 16:39:41 +03:00
|
|
|
return True
|
|
|
|
|
|
|
|
|
|
|
|
def pytest_configure(config):
|
2018-12-19 12:50:39 +02:00
|
|
|
config.addinivalue_line("markers", "testrail_id(name): empty")
|
|
|
|
|
2019-05-23 12:04:41 +03:00
|
|
|
if config.getoption('log_steps'):
|
2017-11-13 12:49:45 +02:00
|
|
|
import logging
|
|
|
|
logging.basicConfig(level=logging.INFO)
|
2018-06-07 17:13:37 +02:00
|
|
|
if config.getoption('env') != 'api':
|
|
|
|
test_suite_data.apk_name = ([i for i in [i for i in config.getoption('apk').split('/')
|
|
|
|
if '.apk' in i]])[0]
|
|
|
|
if is_master(config):
|
2018-11-24 17:18:31 +01:00
|
|
|
pr_number = config.getoption('pr_number')
|
2018-06-07 17:13:37 +02:00
|
|
|
if config.getoption('testrail_report'):
|
2018-08-07 19:38:12 +03:00
|
|
|
if pr_number:
|
|
|
|
run_number = len(testrail_report.get_runs(pr_number)) + 1
|
|
|
|
run_name = 'PR-%s run #%s' % (pr_number, run_number)
|
|
|
|
else:
|
|
|
|
run_name = test_suite_data.apk_name
|
|
|
|
testrail_report.add_run(run_name)
|
2018-11-24 17:18:31 +01:00
|
|
|
if pr_number:
|
|
|
|
from github import Github
|
|
|
|
repo = Github(github_token).get_user('status-im').get_repo('status-react')
|
|
|
|
pull = repo.get_pull(int(pr_number))
|
|
|
|
pull.get_commits()[0].create_status(state='pending', context='Mobile e2e tests',
|
2019-03-12 10:32:50 +02:00
|
|
|
description='e2e tests are running')
|
2018-06-07 17:13:37 +02:00
|
|
|
if config.getoption('env') == 'sauce':
|
|
|
|
if not is_uploaded():
|
|
|
|
if 'http' in config.getoption('apk'):
|
|
|
|
response = requests.get(config.getoption('apk'), stream=True)
|
|
|
|
response.raise_for_status()
|
|
|
|
file = BytesIO(response.content)
|
|
|
|
del response
|
|
|
|
requests.post('http://saucelabs.com/rest/v1/storage/'
|
|
|
|
+ sauce_username + '/' + test_suite_data.apk_name + '?overwrite=true',
|
|
|
|
auth=(sauce_username, sauce_access_key),
|
|
|
|
data=file,
|
|
|
|
headers={'Content-Type': 'application/octet-stream'})
|
|
|
|
else:
|
|
|
|
sauce.storage.upload_file(config.getoption('apk'))
|
2018-01-26 13:07:09 +02:00
|
|
|
|
|
|
|
|
|
|
|
def pytest_unconfigure(config):
|
2018-04-26 15:34:15 +03:00
|
|
|
if is_master(config):
|
2018-10-26 15:32:05 +02:00
|
|
|
if config.getoption('testrail_report'):
|
|
|
|
testrail_report.add_results()
|
2018-04-26 15:34:15 +03:00
|
|
|
if config.getoption('pr_number'):
|
|
|
|
from github import Github
|
|
|
|
repo = Github(github_token).get_user('status-im').get_repo('status-react')
|
|
|
|
pull = repo.get_pull(int(config.getoption('pr_number')))
|
2018-11-24 17:18:31 +01:00
|
|
|
comment = pull.create_issue_comment(github_report.build_html_report(testrail_report.run_id))
|
|
|
|
if not testrail_report.is_run_successful():
|
|
|
|
pull.get_commits()[0].create_status(state='failure', context='Mobile e2e tests',
|
|
|
|
description='Failure - e2e tests are failed',
|
|
|
|
target_url=comment.html_url)
|
|
|
|
else:
|
|
|
|
pull.get_commits()[0].create_status(state='success', context='Mobile e2e tests',
|
|
|
|
description='Success - e2e tests are passed',
|
|
|
|
target_url=comment.html_url)
|
2018-01-26 13:07:09 +02:00
|
|
|
|
|
|
|
|
2018-11-28 10:57:18 +01:00
|
|
|
def should_save_device_stats(config):
|
|
|
|
db_args = [config.getoption(option) for option in
|
|
|
|
('stats_db_host', 'stats_db_port', 'stats_db_username', 'stats_db_password', 'stats_db_database')]
|
|
|
|
return all(db_args)
|
|
|
|
|
|
|
|
|
2018-01-26 13:07:09 +02:00
|
|
|
@pytest.mark.hookwrapper
|
|
|
|
def pytest_runtest_makereport(item, call):
|
|
|
|
outcome = yield
|
|
|
|
report = outcome.get_result()
|
2018-02-27 00:38:56 +01:00
|
|
|
if report.when == 'call':
|
2018-10-15 10:06:21 +03:00
|
|
|
is_sauce_env = item.config.getoption('env') == 'sauce'
|
2018-04-28 11:02:39 +02:00
|
|
|
current_test = test_suite_data.current_test
|
2018-02-27 00:38:56 +01:00
|
|
|
if report.failed:
|
2018-08-10 13:09:19 +03:00
|
|
|
error = report.longreprtext
|
2018-08-29 17:04:12 +03:00
|
|
|
exception = re.findall('E.*Message:|E.*Error:|E.*Failed:', error)
|
2018-08-10 13:09:19 +03:00
|
|
|
if exception:
|
2018-08-15 15:51:52 +03:00
|
|
|
error = error.replace(re.findall('E.*Message:|E.*Error:|E.*Failed:', report.longreprtext)[0], '')
|
2018-08-10 13:09:19 +03:00
|
|
|
current_test.testruns[-1].error = error
|
2018-02-27 00:38:56 +01:00
|
|
|
if is_sauce_env:
|
2018-04-28 11:02:39 +02:00
|
|
|
update_sauce_jobs(current_test.name, current_test.testruns[-1].jobs, report.passed)
|
2018-11-07 13:26:32 +01:00
|
|
|
if pytest.config.getoption('docker'):
|
|
|
|
device_stats = appium_container.get_device_stats()
|
|
|
|
if pytest.config.getoption('bugreport'):
|
|
|
|
appium_container.generate_bugreport(item.name)
|
|
|
|
|
|
|
|
build_name = pytest.config.getoption('apk')
|
|
|
|
# Find type of tests that are run on the device
|
|
|
|
if 'battery_consumption' in item.keywords._markers:
|
|
|
|
test_group = 'battery_consumption'
|
|
|
|
else:
|
|
|
|
test_group = None
|
|
|
|
|
2018-11-28 10:57:18 +01:00
|
|
|
if should_save_device_stats(item.config):
|
|
|
|
device_stats_db = DeviceStatsDB(
|
|
|
|
item.config.getoption('stats_db_host'),
|
|
|
|
item.config.getoption('stats_db_port'),
|
|
|
|
item.config.getoption('stats_db_username'),
|
|
|
|
item.config.getoption('stats_db_password'),
|
|
|
|
item.config.getoption('stats_db_database'),
|
|
|
|
)
|
|
|
|
device_stats_db.save_stats(build_name, item.name, test_group, not report.failed, device_stats)
|
2018-02-27 00:38:56 +01:00
|
|
|
|
|
|
|
|
|
|
|
def update_sauce_jobs(test_name, job_ids, passed):
|
2018-08-15 15:51:52 +03:00
|
|
|
for job_id in job_ids.keys():
|
2018-11-29 20:16:26 +02:00
|
|
|
try:
|
|
|
|
sauce.jobs.update_job(job_id, name=test_name, passed=passed)
|
2019-01-30 14:02:24 +02:00
|
|
|
except (RemoteDisconnected, SauceException):
|
2018-11-29 20:16:26 +02:00
|
|
|
pass
|
2017-11-13 12:49:45 +02:00
|
|
|
|
|
|
|
|
2018-12-19 12:50:39 +02:00
|
|
|
def get_testrail_case_id(item):
|
|
|
|
testrail_id = item.get_closest_marker('testrail_id')
|
2018-10-15 10:06:21 +03:00
|
|
|
if testrail_id:
|
|
|
|
return testrail_id.args[0]
|
2018-04-26 15:34:15 +03:00
|
|
|
|
|
|
|
|
2017-11-13 12:49:45 +02:00
|
|
|
def pytest_runtest_setup(item):
|
2019-06-03 15:02:39 +03:00
|
|
|
try:
|
|
|
|
testrail_id = [mark.args[0] for mark in item.iter_markers(name='testrail_id')][0]
|
|
|
|
except IndexError:
|
|
|
|
pass
|
2018-12-19 12:50:39 +02:00
|
|
|
run_testrail_ids = item.config.getoption("run_testrail_ids")
|
|
|
|
if run_testrail_ids:
|
|
|
|
if str(testrail_id) not in run_testrail_ids:
|
|
|
|
pytest.skip("test requires testrail case id %s" % testrail_id)
|
2018-04-28 11:02:39 +02:00
|
|
|
test_suite_data.set_current_test(item.name, testrail_case_id=get_testrail_case_id(item))
|
|
|
|
test_suite_data.current_test.create_new_testrun()
|
|
|
|
|
|
|
|
|
|
|
|
def pytest_runtest_protocol(item, nextitem):
|
2018-10-15 18:38:42 +03:00
|
|
|
rerun_count = int(item.config.getoption('rerun_count'))
|
2018-10-15 10:06:21 +03:00
|
|
|
for i in range(rerun_count):
|
2018-04-28 11:02:39 +02:00
|
|
|
reports = runtestprotocol(item, nextitem=nextitem)
|
|
|
|
for report in reports:
|
|
|
|
if report.failed and should_rerun_test(report.longreprtext):
|
|
|
|
break # rerun
|
|
|
|
else:
|
|
|
|
return True # no need to rerun
|
2018-03-31 23:05:11 +03:00
|
|
|
|
|
|
|
|
2018-10-10 17:24:36 +03:00
|
|
|
@pytest.fixture(scope="session", autouse=False)
|
2018-09-17 11:50:01 +03:00
|
|
|
def faucet_for_senders():
|
|
|
|
network_api = NetworkApi()
|
|
|
|
for user in transaction_senders.values():
|
|
|
|
network_api.faucet(address=user['address'])
|
|
|
|
|
|
|
|
|
2018-03-31 23:05:11 +03:00
|
|
|
@pytest.fixture
|
2018-10-15 10:06:21 +03:00
|
|
|
def messages_number(request):
|
|
|
|
return int(request.config.getoption('messages_number'))
|
2018-03-31 23:05:11 +03:00
|
|
|
|
|
|
|
|
|
|
|
@pytest.fixture
|
2018-10-15 10:06:21 +03:00
|
|
|
def message_wait_time(request):
|
|
|
|
return int(request.config.getoption('message_wait_time'))
|
2018-03-31 23:05:11 +03:00
|
|
|
|
|
|
|
|
|
|
|
@pytest.fixture
|
2018-10-15 10:06:21 +03:00
|
|
|
def participants_number(request):
|
|
|
|
return int(request.config.getoption('participants_number'))
|
2018-06-05 17:12:15 +02:00
|
|
|
|
|
|
|
|
|
|
|
@pytest.fixture
|
2018-10-15 10:06:21 +03:00
|
|
|
def chat_name(request):
|
|
|
|
return request.config.getoption('chat_name')
|
2018-06-05 17:12:15 +02:00
|
|
|
|
|
|
|
|
|
|
|
@pytest.fixture
|
2018-10-15 10:06:21 +03:00
|
|
|
def user_public_key(request):
|
|
|
|
return request.config.getoption('user_public_key')
|