2017-03-16 14:27:55 +00:00
|
|
|
##########################################################################
|
2016-07-18 13:50:21 +00:00
|
|
|
#
|
|
|
|
# pgAdmin 4 - PostgreSQL Tools
|
|
|
|
#
|
2017-01-04 13:33:32 +00:00
|
|
|
# Copyright (C) 2013 - 2017, The pgAdmin Development Team
|
2016-07-18 13:50:21 +00:00
|
|
|
# This software is released under the PostgreSQL Licence
|
|
|
|
#
|
2017-03-16 14:27:55 +00:00
|
|
|
##########################################################################
|
2016-07-18 13:50:21 +00:00
|
|
|
|
|
|
|
""" This file collect all modules/files present in tests directory and add
|
|
|
|
them to TestSuite. """
|
2016-09-14 15:26:12 +00:00
|
|
|
from __future__ import print_function
|
2017-03-01 13:20:06 +00:00
|
|
|
|
2016-07-18 13:50:21 +00:00
|
|
|
import argparse
|
2016-09-14 15:26:12 +00:00
|
|
|
import atexit
|
2016-07-18 13:50:21 +00:00
|
|
|
import logging
|
2017-03-01 13:20:06 +00:00
|
|
|
import os
|
|
|
|
import signal
|
|
|
|
import sys
|
2016-10-07 12:59:43 +00:00
|
|
|
import traceback
|
|
|
|
|
2017-03-01 13:20:06 +00:00
|
|
|
from selenium import webdriver
|
|
|
|
|
2016-10-07 12:59:43 +00:00
|
|
|
if sys.version_info < (2, 7):
|
|
|
|
import unittest2 as unittest
|
|
|
|
else:
|
|
|
|
import unittest
|
2016-07-18 13:50:21 +00:00
|
|
|
|
2016-09-22 11:58:38 +00:00
|
|
|
logger = logging.getLogger(__name__)
|
|
|
|
file_name = os.path.basename(__file__)
|
|
|
|
|
2016-07-18 13:50:21 +00:00
|
|
|
from testscenarios.scenarios import generate_scenarios
|
|
|
|
|
|
|
|
CURRENT_PATH = os.path.dirname(os.path.realpath(__file__))
|
|
|
|
|
|
|
|
# Set sys path to current directory so that we can import pgadmin package
|
|
|
|
root = os.path.dirname(CURRENT_PATH)
|
|
|
|
|
|
|
|
if sys.path[0] != root:
|
|
|
|
sys.path.insert(0, root)
|
2016-09-14 15:26:12 +00:00
|
|
|
os.chdir(root)
|
2016-07-18 13:50:21 +00:00
|
|
|
|
|
|
|
from pgadmin import create_app
|
|
|
|
import config
|
2016-10-07 12:59:43 +00:00
|
|
|
from regression import test_setup
|
2017-03-01 13:20:06 +00:00
|
|
|
from regression.feature_utils.app_starter import AppStarter
|
2016-09-14 15:26:12 +00:00
|
|
|
|
2016-09-22 11:58:38 +00:00
|
|
|
# Delete SQLite db file if exists
|
2016-09-14 15:26:12 +00:00
|
|
|
if os.path.isfile(config.TEST_SQLITE_PATH):
|
2016-09-22 11:58:38 +00:00
|
|
|
os.remove(config.TEST_SQLITE_PATH)
|
|
|
|
|
|
|
|
config.TESTING_MODE = True
|
2017-01-31 10:29:07 +00:00
|
|
|
|
|
|
|
# Disable upgrade checks - no need during testing, and it'll cause an error if there's
|
|
|
|
# no network connection when it runs.
|
|
|
|
config.UPGRADE_CHECK_ENABLED = False
|
|
|
|
|
2016-09-22 11:58:38 +00:00
|
|
|
pgadmin_credentials = test_setup.config_data
|
|
|
|
|
|
|
|
# Set environment variables for email and password
|
|
|
|
os.environ['PGADMIN_SETUP_EMAIL'] = ''
|
|
|
|
os.environ['PGADMIN_SETUP_PASSWORD'] = ''
|
|
|
|
if pgadmin_credentials:
|
|
|
|
if 'pgAdmin4_login_credentials' in pgadmin_credentials:
|
|
|
|
if all(item in pgadmin_credentials['pgAdmin4_login_credentials']
|
|
|
|
for item in ['login_username', 'login_password']):
|
|
|
|
pgadmin_credentials = pgadmin_credentials[
|
|
|
|
'pgAdmin4_login_credentials']
|
|
|
|
os.environ['PGADMIN_SETUP_EMAIL'] = pgadmin_credentials[
|
|
|
|
'login_username']
|
|
|
|
os.environ['PGADMIN_SETUP_PASSWORD'] = pgadmin_credentials[
|
|
|
|
'login_password']
|
|
|
|
|
|
|
|
# Execute the setup file
|
|
|
|
exec (open("setup.py").read())
|
2016-07-18 13:50:21 +00:00
|
|
|
|
|
|
|
# Get the config database schema version. We store this in pgadmin.model
|
|
|
|
# as it turns out that putting it in the config files isn't a great idea
|
|
|
|
from pgadmin.model import SCHEMA_VERSION
|
2016-09-14 15:26:12 +00:00
|
|
|
|
|
|
|
# Delay the import test_utils as it needs updated config.SQLITE_PATH
|
2017-03-23 11:59:31 +00:00
|
|
|
from regression.python_test_utils import test_utils
|
2016-07-18 13:50:21 +00:00
|
|
|
|
|
|
|
config.SETTINGS_SCHEMA_VERSION = SCHEMA_VERSION
|
|
|
|
|
|
|
|
# Override some other defaults
|
|
|
|
from logging import WARNING
|
2016-09-14 15:26:12 +00:00
|
|
|
|
2016-07-18 13:50:21 +00:00
|
|
|
config.CONSOLE_LOG_LEVEL = WARNING
|
|
|
|
|
|
|
|
# Create the app
|
|
|
|
app = create_app()
|
|
|
|
app.config['WTF_CSRF_ENABLED'] = False
|
2017-03-06 15:24:52 +00:00
|
|
|
app.PGADMIN_KEY = ''
|
2016-07-18 13:50:21 +00:00
|
|
|
test_client = app.test_client()
|
2017-03-08 09:21:28 +00:00
|
|
|
driver = None
|
|
|
|
app_starter = None
|
|
|
|
handle_cleanup = None
|
2016-07-18 13:50:21 +00:00
|
|
|
|
2016-09-22 11:58:38 +00:00
|
|
|
def get_suite(module_list, test_server, test_app_client):
|
2016-07-18 13:50:21 +00:00
|
|
|
"""
|
2016-09-22 11:58:38 +00:00
|
|
|
This function add the tests to test suite and return modified test suite
|
|
|
|
variable.
|
|
|
|
:param module_list: test module list
|
|
|
|
:type module_list: list
|
|
|
|
:param test_server: server details
|
|
|
|
:type test_server: dict
|
2016-07-18 13:50:21 +00:00
|
|
|
:param test_app_client: test client
|
|
|
|
:type test_app_client: pgadmin app object
|
|
|
|
:return pgadmin_suite: test suite with test cases
|
|
|
|
:rtype: TestSuite
|
|
|
|
"""
|
|
|
|
modules = []
|
|
|
|
pgadmin_suite = unittest.TestSuite()
|
|
|
|
|
|
|
|
# Get the each test module and add into list
|
2016-09-14 15:26:12 +00:00
|
|
|
for key, klass in module_list:
|
2016-07-18 13:50:21 +00:00
|
|
|
gen = klass
|
|
|
|
modules.append(gen)
|
|
|
|
|
|
|
|
# Set the test client to each module & generate the scenarios
|
|
|
|
for module in modules:
|
|
|
|
obj = module()
|
2016-09-14 15:26:12 +00:00
|
|
|
obj.setApp(app)
|
2016-07-18 13:50:21 +00:00
|
|
|
obj.setTestClient(test_app_client)
|
2016-09-22 11:58:38 +00:00
|
|
|
obj.setTestServer(test_server)
|
2017-03-01 13:20:06 +00:00
|
|
|
obj.setDriver(driver)
|
2016-07-18 13:50:21 +00:00
|
|
|
scenario = generate_scenarios(obj)
|
|
|
|
pgadmin_suite.addTests(scenario)
|
|
|
|
|
|
|
|
return pgadmin_suite
|
|
|
|
|
|
|
|
|
2016-09-22 11:58:38 +00:00
|
|
|
def get_test_modules(arguments):
|
|
|
|
"""
|
|
|
|
This function loads the all modules in the tests directory into testing
|
|
|
|
environment.
|
|
|
|
|
|
|
|
:param arguments: this is command line arguments for module name to
|
|
|
|
which test suite will run
|
|
|
|
:type arguments: str
|
|
|
|
:return module list: test module list
|
|
|
|
:rtype: list
|
|
|
|
"""
|
|
|
|
|
|
|
|
from pgadmin.utils.route import TestsGeneratorRegistry
|
|
|
|
|
2017-02-15 15:10:31 +00:00
|
|
|
exclude_pkgs = []
|
2017-03-08 09:21:28 +00:00
|
|
|
global driver, app_starter, handle_cleanup
|
2017-02-15 15:10:31 +00:00
|
|
|
|
|
|
|
if not config.SERVER_MODE:
|
|
|
|
exclude_pkgs.append("browser.tests")
|
|
|
|
if arguments['exclude'] is not None:
|
|
|
|
exclude_pkgs += arguments['exclude'].split(',')
|
|
|
|
|
2017-03-08 09:21:28 +00:00
|
|
|
if 'feature_tests' not in exclude_pkgs:
|
|
|
|
driver = webdriver.Chrome()
|
|
|
|
app_starter = AppStarter(driver, config)
|
|
|
|
app_starter.start_app()
|
|
|
|
|
|
|
|
handle_cleanup = test_utils.get_cleanup_handler(test_client, app_starter)
|
|
|
|
# Register cleanup function to cleanup on exit
|
|
|
|
atexit.register(handle_cleanup)
|
|
|
|
|
2016-09-22 11:58:38 +00:00
|
|
|
# Load the test modules which are in given package(i.e. in arguments.pkg)
|
|
|
|
if arguments['pkg'] is None or arguments['pkg'] == "all":
|
2017-02-15 15:10:31 +00:00
|
|
|
TestsGeneratorRegistry.load_generators('pgadmin', exclude_pkgs)
|
2016-09-22 11:58:38 +00:00
|
|
|
else:
|
2017-02-15 15:10:31 +00:00
|
|
|
TestsGeneratorRegistry.load_generators('pgadmin.%s' %
|
|
|
|
arguments['pkg'],
|
|
|
|
exclude_pkgs)
|
2016-09-22 11:58:38 +00:00
|
|
|
|
|
|
|
# Sort module list so that test suite executes the test cases sequentially
|
|
|
|
module_list = TestsGeneratorRegistry.registry.items()
|
|
|
|
module_list = sorted(module_list, key=lambda module_tuple: module_tuple[0])
|
|
|
|
|
|
|
|
return module_list
|
|
|
|
|
|
|
|
|
2016-07-18 13:50:21 +00:00
|
|
|
def add_arguments():
|
|
|
|
"""
|
|
|
|
This function parse the command line arguments(project's package name
|
|
|
|
e.g. browser) & add into parser
|
|
|
|
|
|
|
|
:return args: command line argument for pgadmin's package name
|
|
|
|
:rtype: argparse namespace
|
|
|
|
"""
|
|
|
|
|
|
|
|
parser = argparse.ArgumentParser(description='Test suite for pgAdmin4')
|
|
|
|
parser.add_argument('--pkg', help='Executes the test cases of particular'
|
2017-02-15 15:10:31 +00:00
|
|
|
' package and subpackages')
|
|
|
|
parser.add_argument('--exclude', help='Skips execution of the test '
|
|
|
|
'cases of particular package and sub-packages')
|
2016-07-18 13:50:21 +00:00
|
|
|
arg = parser.parse_args()
|
|
|
|
|
|
|
|
return arg
|
|
|
|
|
|
|
|
|
2016-09-14 15:26:12 +00:00
|
|
|
def sig_handler(signo, frame):
|
2017-03-08 09:21:28 +00:00
|
|
|
if handle_cleanup:
|
|
|
|
handle_cleanup()
|
2016-09-14 15:26:12 +00:00
|
|
|
|
|
|
|
|
2016-09-22 11:58:38 +00:00
|
|
|
def get_tests_result(test_suite):
|
2016-09-19 15:49:06 +00:00
|
|
|
"""This function returns the total ran and total failed test cases count"""
|
2016-09-22 11:58:38 +00:00
|
|
|
try:
|
|
|
|
total_ran = test_suite.testsRun
|
|
|
|
failed_cases_result = []
|
|
|
|
skipped_cases_result = []
|
|
|
|
if total_ran:
|
|
|
|
if test_suite.failures:
|
|
|
|
for failed_case in test_suite.failures:
|
|
|
|
class_name = str(
|
|
|
|
failed_case[0]).split('.')[-1].split()[0].strip(')')
|
2016-09-19 15:49:06 +00:00
|
|
|
failed_cases_result.append(class_name)
|
2016-09-22 11:58:38 +00:00
|
|
|
if test_suite.errors:
|
|
|
|
for error_case in test_suite.errors:
|
|
|
|
class_name = str(
|
|
|
|
error_case[0]).split('.')[-1].split()[0].strip(')')
|
|
|
|
if class_name not in failed_cases_result:
|
|
|
|
failed_cases_result.append(class_name)
|
2016-09-26 13:02:59 +00:00
|
|
|
if test_suite.skipped:
|
|
|
|
for skip_test in test_suite.skipped:
|
|
|
|
class_name = str(
|
|
|
|
skip_test[0]).split('.')[-1].split()[0].strip(')')
|
|
|
|
if class_name not in failed_cases_result:
|
|
|
|
skipped_cases_result.append(class_name)
|
2016-09-22 11:58:38 +00:00
|
|
|
return total_ran, failed_cases_result, skipped_cases_result
|
2016-10-07 12:59:43 +00:00
|
|
|
except Exception:
|
|
|
|
traceback.print_exc(file=sys.stderr)
|
2016-09-19 15:49:06 +00:00
|
|
|
|
|
|
|
|
2016-07-18 13:50:21 +00:00
|
|
|
class StreamToLogger(object):
|
|
|
|
def __init__(self, logger, log_level=logging.INFO):
|
|
|
|
self.terminal = sys.stderr
|
|
|
|
self.logger = logger
|
|
|
|
self.log_level = log_level
|
|
|
|
self.linebuf = ''
|
|
|
|
|
|
|
|
def write(self, buf):
|
|
|
|
"""
|
|
|
|
This function writes the log in the logger file as well as on console
|
|
|
|
|
|
|
|
:param buf: log message
|
|
|
|
:type buf: str
|
|
|
|
:return: None
|
|
|
|
"""
|
|
|
|
|
|
|
|
self.terminal.write(buf)
|
|
|
|
for line in buf.rstrip().splitlines():
|
|
|
|
self.logger.log(self.log_level, line.rstrip())
|
|
|
|
|
|
|
|
def flush(self):
|
|
|
|
pass
|
|
|
|
|
|
|
|
|
|
|
|
if __name__ == '__main__':
|
2017-02-08 16:52:03 +00:00
|
|
|
# Failure detected?
|
|
|
|
failure = False
|
|
|
|
|
2016-09-19 15:49:06 +00:00
|
|
|
test_result = dict()
|
2016-09-14 15:26:12 +00:00
|
|
|
# Set signal handler for cleanup
|
2017-01-08 13:18:04 +00:00
|
|
|
signal_list = dir(signal)
|
|
|
|
required_signal_list = ['SIGTERM', 'SIGABRT', 'SIGQUIT', 'SIGINT']
|
|
|
|
# Get the OS wise supported signals
|
|
|
|
supported_signal_list = [sig for sig in required_signal_list if
|
|
|
|
sig in signal_list]
|
|
|
|
for sig in supported_signal_list:
|
|
|
|
signal.signal(getattr(signal, sig), sig_handler)
|
2016-09-14 15:26:12 +00:00
|
|
|
|
2016-07-18 13:50:21 +00:00
|
|
|
# Set basic logging configuration for log file
|
|
|
|
logging.basicConfig(level=logging.DEBUG,
|
2016-07-27 14:33:36 +00:00
|
|
|
format='%(asctime)s:%(levelname)s:%(name)s:%(message)s'
|
|
|
|
,
|
2016-07-18 13:50:21 +00:00
|
|
|
filename=CURRENT_PATH + "/" + "regression.log",
|
|
|
|
filemode='w'
|
|
|
|
)
|
|
|
|
|
|
|
|
# Create logger to write log in the logger file as well as on console
|
|
|
|
stderr_logger = logging.getLogger('STDERR')
|
|
|
|
sys.stderr = StreamToLogger(stderr_logger, logging.ERROR)
|
2016-07-27 14:33:36 +00:00
|
|
|
args = vars(add_arguments())
|
2016-09-22 11:58:38 +00:00
|
|
|
# Get test module list
|
|
|
|
test_module_list = get_test_modules(args)
|
|
|
|
# Login the test client
|
|
|
|
test_utils.login_tester_account(test_client)
|
2016-07-18 13:50:21 +00:00
|
|
|
|
2016-09-14 15:26:12 +00:00
|
|
|
servers_info = test_utils.get_config_data()
|
2016-10-07 12:59:43 +00:00
|
|
|
node_name = "all"
|
|
|
|
if args['pkg'] is not None:
|
|
|
|
node_name = args['pkg'].split('.')[-1]
|
2016-09-14 15:26:12 +00:00
|
|
|
try:
|
|
|
|
for server in servers_info:
|
|
|
|
print("\n=============Running the test cases for '%s'============="
|
|
|
|
% server['name'], file=sys.stderr)
|
2016-09-22 11:58:38 +00:00
|
|
|
# Create test server
|
2017-02-15 15:10:31 +00:00
|
|
|
test_utils.create_parent_server_node(server)
|
2016-09-19 15:49:06 +00:00
|
|
|
|
2016-09-22 11:58:38 +00:00
|
|
|
suite = get_suite(test_module_list, server, test_client)
|
2016-09-14 15:26:12 +00:00
|
|
|
tests = unittest.TextTestRunner(stream=sys.stderr,
|
|
|
|
descriptions=True,
|
|
|
|
verbosity=2).run(suite)
|
2016-09-19 15:49:06 +00:00
|
|
|
|
2016-09-22 11:58:38 +00:00
|
|
|
ran_tests, failed_cases, skipped_cases = \
|
|
|
|
get_tests_result(tests)
|
|
|
|
test_result[server['name']] = [ran_tests, failed_cases,
|
|
|
|
skipped_cases]
|
2017-02-08 16:52:03 +00:00
|
|
|
|
|
|
|
if len(failed_cases) > 0:
|
|
|
|
failure = True
|
|
|
|
|
2016-09-22 11:58:38 +00:00
|
|
|
# Delete test server
|
2017-02-15 15:10:31 +00:00
|
|
|
test_utils.delete_test_server(test_client)
|
2016-09-14 15:26:12 +00:00
|
|
|
except SystemExit:
|
2017-03-08 09:21:28 +00:00
|
|
|
if handle_cleanup:
|
|
|
|
handle_cleanup()
|
2016-07-27 14:33:36 +00:00
|
|
|
|
2016-10-07 12:59:43 +00:00
|
|
|
print("\n==============================================================="
|
|
|
|
"=======", file=sys.stderr)
|
2016-09-26 13:16:04 +00:00
|
|
|
print("Test Result Summary", file=sys.stderr)
|
2016-10-07 12:59:43 +00:00
|
|
|
print(
|
|
|
|
"==================================================================="
|
|
|
|
"===\n", file=sys.stderr)
|
2016-09-19 15:49:06 +00:00
|
|
|
for server_res in test_result:
|
2016-09-26 13:16:04 +00:00
|
|
|
failed_cases = "\n\t\t".join(test_result[server_res][1])
|
|
|
|
skipped_cases = "\n\t\t".join(test_result[server_res][2])
|
2016-09-19 15:49:06 +00:00
|
|
|
total_failed = len(test_result[server_res][1])
|
2016-09-26 13:02:59 +00:00
|
|
|
total_skipped = len(test_result[server_res][2])
|
|
|
|
total_passed_cases = int(
|
|
|
|
test_result[server_res][0]) - total_failed - total_skipped
|
|
|
|
|
|
|
|
print(
|
2016-09-26 13:16:04 +00:00
|
|
|
"%s:\n\n\t%s test%s passed\n\t%s test%s failed%s%s"
|
|
|
|
"\n\t%s test%s skipped%s%s\n" %
|
2016-09-26 13:02:59 +00:00
|
|
|
(server_res, total_passed_cases,
|
|
|
|
(total_passed_cases != 1 and "s" or ""),
|
|
|
|
total_failed, (total_failed != 1 and "s" or ""),
|
2016-09-26 13:16:04 +00:00
|
|
|
(total_failed != 0 and ":\n\t\t" or ""), failed_cases,
|
2016-09-26 13:02:59 +00:00
|
|
|
total_skipped, (total_skipped != 1 and "s" or ""),
|
2016-09-26 13:16:04 +00:00
|
|
|
(total_skipped != 0 and ":\n\t\t" or ""), skipped_cases),
|
2016-09-26 13:02:59 +00:00
|
|
|
file=sys.stderr)
|
2016-09-22 11:58:38 +00:00
|
|
|
|
2016-10-07 12:59:43 +00:00
|
|
|
print(
|
|
|
|
"==================================================================="
|
|
|
|
"===\n", file=sys.stderr)
|
2016-09-19 15:49:06 +00:00
|
|
|
|
2016-09-26 13:16:04 +00:00
|
|
|
print("Please check output in file: %s/regression.log\n" % CURRENT_PATH)
|
2017-02-08 16:52:03 +00:00
|
|
|
|
|
|
|
if failure == True:
|
|
|
|
sys.exit(1)
|
|
|
|
else:
|
|
|
|
sys.exit(0)
|