2018-08-27 04:30:56 -05:00
|
|
|
##########################################################################
|
|
|
|
#
|
|
|
|
# pgAdmin 4 - PostgreSQL Tools
|
|
|
|
#
|
2022-01-04 02:24:25 -06:00
|
|
|
# Copyright (C) 2013 - 2022, The pgAdmin Development Team
|
2018-08-27 04:30:56 -05:00
|
|
|
# This software is released under the PostgreSQL Licence
|
|
|
|
#
|
|
|
|
##########################################################################
|
|
|
|
|
|
|
|
import sys
|
|
|
|
import random
|
|
|
|
|
|
|
|
from selenium.webdriver import ActionChains
|
|
|
|
from selenium.webdriver.common.keys import Keys
|
2021-11-10 00:20:20 -06:00
|
|
|
from selenium.webdriver.common.by import By
|
2018-08-27 04:30:56 -05:00
|
|
|
from regression.python_test_utils import test_utils
|
|
|
|
from regression.feature_utils.base_feature_test import BaseFeatureTest
|
2019-08-22 04:20:51 -05:00
|
|
|
from regression.feature_utils.locators import QueryToolLocators
|
2018-08-27 04:30:56 -05:00
|
|
|
|
|
|
|
|
|
|
|
class QueryToolAutoCompleteFeatureTest(BaseFeatureTest):
|
|
|
|
"""
|
|
|
|
This feature test will test the query tool auto complete feature.
|
|
|
|
"""
|
2018-09-11 06:56:14 -05:00
|
|
|
first_schema_name = ""
|
|
|
|
second_schema_name = ""
|
2018-08-27 04:30:56 -05:00
|
|
|
first_table_name = ""
|
|
|
|
second_table_name = ""
|
|
|
|
|
|
|
|
scenarios = [
|
|
|
|
("Query tool auto complete feature test", dict())
|
|
|
|
]
|
|
|
|
|
|
|
|
def before(self):
|
|
|
|
self.page.wait_for_spinner_to_disappear()
|
|
|
|
|
|
|
|
self.page.add_server(self.server)
|
2018-09-11 06:56:14 -05:00
|
|
|
|
|
|
|
self.first_schema_name = "test_schema" + \
|
2019-12-03 08:05:48 -06:00
|
|
|
str(random.randint(1000, 2000))
|
2018-09-11 06:56:14 -05:00
|
|
|
test_utils.create_schema(self.server, self.test_db,
|
|
|
|
self.first_schema_name)
|
|
|
|
|
|
|
|
self.second_schema_name = "comp_schema" + \
|
2019-12-03 08:05:48 -06:00
|
|
|
str(random.randint(2000, 3000))
|
2018-09-11 06:56:14 -05:00
|
|
|
test_utils.create_schema(self.server, self.test_db,
|
|
|
|
self.second_schema_name)
|
|
|
|
|
2018-08-27 04:30:56 -05:00
|
|
|
self.first_table_name = "auto_comp_" + \
|
2019-12-03 08:05:48 -06:00
|
|
|
str(random.randint(1000, 2000))
|
2018-08-27 04:30:56 -05:00
|
|
|
test_utils.create_table(self.server, self.test_db,
|
|
|
|
self.first_table_name)
|
|
|
|
|
|
|
|
self.second_table_name = "auto_comp_" + \
|
2019-12-03 08:05:48 -06:00
|
|
|
str(random.randint(2000, 3000))
|
2018-08-27 04:30:56 -05:00
|
|
|
test_utils.create_table(self.server, self.test_db,
|
|
|
|
self.second_table_name)
|
|
|
|
|
2021-11-10 00:20:20 -06:00
|
|
|
self.page.expand_database_node("Server", self.server['name'],
|
|
|
|
self.server['db_password'],
|
|
|
|
self.test_db)
|
2019-08-22 04:20:51 -05:00
|
|
|
|
2018-08-27 04:30:56 -05:00
|
|
|
self.page.open_query_tool()
|
|
|
|
self.page.wait_for_spinner_to_disappear()
|
|
|
|
|
|
|
|
def runTest(self):
|
|
|
|
# Test case for keywords
|
2020-08-06 02:12:19 -05:00
|
|
|
select_keyword = "SELECT * FROM public."
|
2018-08-27 04:30:56 -05:00
|
|
|
print("\nAuto complete ALTER keyword... ", file=sys.stderr, end="")
|
|
|
|
self._auto_complete("A", "ALTER")
|
|
|
|
print("OK.", file=sys.stderr)
|
2019-08-22 04:20:51 -05:00
|
|
|
self.page.clear_query_tool()
|
2018-08-27 04:30:56 -05:00
|
|
|
|
|
|
|
print("Auto complete BEGIN keyword... ", file=sys.stderr, end="")
|
|
|
|
self._auto_complete("BE", "BEGIN")
|
|
|
|
print("OK.", file=sys.stderr)
|
2019-08-22 04:20:51 -05:00
|
|
|
self.page.clear_query_tool()
|
2018-08-27 04:30:56 -05:00
|
|
|
|
|
|
|
print("Auto complete CASCADED keyword... ", file=sys.stderr, end="")
|
|
|
|
self._auto_complete("CAS", "CASCADED")
|
|
|
|
print("OK.", file=sys.stderr)
|
2019-08-22 04:20:51 -05:00
|
|
|
self.page.clear_query_tool()
|
2018-08-27 04:30:56 -05:00
|
|
|
|
|
|
|
print("Auto complete SELECT keyword... ", file=sys.stderr, end="")
|
|
|
|
self._auto_complete("SE", "SELECT")
|
|
|
|
print("OK.", file=sys.stderr)
|
2019-08-22 04:20:51 -05:00
|
|
|
self.page.clear_query_tool()
|
2018-08-27 04:30:56 -05:00
|
|
|
|
|
|
|
print("Auto complete pg_backend_pid() function ... ",
|
|
|
|
file=sys.stderr, end="")
|
2020-10-05 04:38:09 -05:00
|
|
|
self._auto_complete("SELECT pg_catalog.pg_", "pg_backend_pid()")
|
2018-08-27 04:30:56 -05:00
|
|
|
print("OK.", file=sys.stderr)
|
2019-08-22 04:20:51 -05:00
|
|
|
self.page.clear_query_tool()
|
2018-08-27 04:30:56 -05:00
|
|
|
|
|
|
|
print("Auto complete current_query() function ... ",
|
|
|
|
file=sys.stderr, end="")
|
2020-10-05 04:38:09 -05:00
|
|
|
self._auto_complete("SELECT pg_catalog.current_", "current_query()")
|
2018-08-27 04:30:56 -05:00
|
|
|
print("OK.", file=sys.stderr)
|
2019-08-22 04:20:51 -05:00
|
|
|
self.page.clear_query_tool()
|
2018-08-27 04:30:56 -05:00
|
|
|
|
|
|
|
print("Auto complete function with argument ... ",
|
|
|
|
file=sys.stderr, end="")
|
2020-10-05 04:39:39 -05:00
|
|
|
self._auto_complete("SELECT pg_catalog.pg_st",
|
|
|
|
"pg_stat_file(filename)")
|
2018-08-27 04:30:56 -05:00
|
|
|
print("OK.", file=sys.stderr)
|
2019-08-22 04:20:51 -05:00
|
|
|
self.page.clear_query_tool()
|
2018-08-27 04:30:56 -05:00
|
|
|
|
2018-09-11 06:56:14 -05:00
|
|
|
print("Auto complete schema other than default start with test_ ... ",
|
|
|
|
file=sys.stderr, end="")
|
|
|
|
self._auto_complete("SELECT * FROM te", self.first_schema_name)
|
|
|
|
print("OK.", file=sys.stderr)
|
2019-08-22 04:20:51 -05:00
|
|
|
self.page.clear_query_tool()
|
2018-09-11 06:56:14 -05:00
|
|
|
|
|
|
|
print("Auto complete schema other than default starts with comp_ ... ",
|
|
|
|
file=sys.stderr, end="")
|
|
|
|
self._auto_complete("SELECT * FROM co", self.second_schema_name)
|
|
|
|
print("OK.", file=sys.stderr)
|
2019-08-22 04:20:51 -05:00
|
|
|
self.page.clear_query_tool()
|
2018-09-11 06:56:14 -05:00
|
|
|
|
2018-08-27 04:30:56 -05:00
|
|
|
print("Auto complete first table in public schema ... ",
|
|
|
|
file=sys.stderr, end="")
|
2020-08-06 02:12:19 -05:00
|
|
|
self._auto_complete(select_keyword, self.first_table_name)
|
2018-08-27 04:30:56 -05:00
|
|
|
print("OK.", file=sys.stderr)
|
2019-08-22 04:20:51 -05:00
|
|
|
self.page.clear_query_tool()
|
2018-08-27 04:30:56 -05:00
|
|
|
|
|
|
|
print("Auto complete second table in public schema ... ",
|
|
|
|
file=sys.stderr, end="")
|
2020-08-06 02:12:19 -05:00
|
|
|
self._auto_complete(select_keyword, self.second_table_name)
|
2018-08-27 04:30:56 -05:00
|
|
|
print("OK.", file=sys.stderr)
|
2019-08-22 04:20:51 -05:00
|
|
|
self.page.clear_query_tool()
|
2018-08-27 04:30:56 -05:00
|
|
|
|
|
|
|
print("Auto complete JOIN second table with after schema name ... ",
|
|
|
|
file=sys.stderr, end="")
|
2020-08-06 02:12:19 -05:00
|
|
|
query = select_keyword + self.first_table_name + \
|
|
|
|
" JOIN public."
|
2018-08-27 04:30:56 -05:00
|
|
|
self._auto_complete(query, self.second_table_name)
|
|
|
|
print("OK.", file=sys.stderr)
|
2019-08-22 04:20:51 -05:00
|
|
|
self.page.clear_query_tool()
|
2018-08-27 04:30:56 -05:00
|
|
|
|
|
|
|
print("Auto complete JOIN ON some columns ... ",
|
|
|
|
file=sys.stderr, end="")
|
2020-08-06 02:12:19 -05:00
|
|
|
query = select_keyword + self.first_table_name + \
|
|
|
|
" JOIN public." + self.second_table_name + " ON " + \
|
|
|
|
self.second_table_name + "."
|
2018-08-27 04:30:56 -05:00
|
|
|
expected_string = "some_column = " + self.first_table_name + \
|
|
|
|
".some_column"
|
|
|
|
self._auto_complete(query, expected_string)
|
|
|
|
print("OK.", file=sys.stderr)
|
2019-08-22 04:20:51 -05:00
|
|
|
self.page.clear_query_tool()
|
2018-08-27 04:30:56 -05:00
|
|
|
|
2019-08-22 04:20:51 -05:00
|
|
|
print("Auto complete JOIN ON some columns using table alias ... ",
|
2018-08-27 04:30:56 -05:00
|
|
|
file=sys.stderr, end="")
|
2020-08-06 02:12:19 -05:00
|
|
|
query = select_keyword + self.first_table_name + \
|
|
|
|
" t1 JOIN public." + self.second_table_name + " t2 ON t2."
|
2018-08-27 04:30:56 -05:00
|
|
|
self._auto_complete(query, "some_column = t1.some_column")
|
|
|
|
print("OK.", file=sys.stderr)
|
2019-08-22 04:20:51 -05:00
|
|
|
self.page.clear_query_tool()
|
2018-08-27 04:30:56 -05:00
|
|
|
|
|
|
|
def after(self):
|
|
|
|
self.page.remove_server(self.server)
|
2019-08-22 04:20:51 -05:00
|
|
|
test_utils.delete_table(self.server, self.test_db,
|
|
|
|
self.first_table_name)
|
|
|
|
test_utils.delete_table(self.server, self.test_db,
|
|
|
|
self.second_table_name)
|
2018-08-27 04:30:56 -05:00
|
|
|
|
|
|
|
def _auto_complete(self, word, expected_string):
|
|
|
|
self.page.fill_codemirror_area_with(word)
|
2019-08-22 04:20:51 -05:00
|
|
|
hint_displayed = False
|
|
|
|
retry = 3
|
|
|
|
while retry > 0:
|
|
|
|
ActionChains(self.page.driver).key_down(
|
|
|
|
Keys.CONTROL).send_keys(Keys.SPACE).key_up(
|
|
|
|
Keys.CONTROL).perform()
|
|
|
|
if self.page.check_if_element_exist_by_xpath(
|
2021-11-10 00:20:20 -06:00
|
|
|
QueryToolLocators.code_mirror_hint_box_xpath, 15):
|
2019-08-22 04:20:51 -05:00
|
|
|
hint_displayed = True
|
|
|
|
break
|
|
|
|
else:
|
|
|
|
retry -= 1
|
|
|
|
if hint_displayed:
|
|
|
|
# if IntelliSense is present then verify this
|
2019-03-21 07:04:37 -05:00
|
|
|
self.page.find_by_xpath(
|
2019-08-22 04:20:51 -05:00
|
|
|
QueryToolLocators.code_mirror_hint_item_xpath.format(
|
|
|
|
expected_string))
|
2019-03-21 07:04:37 -05:00
|
|
|
else:
|
|
|
|
# if no IntelliSense is present it means there is only one option
|
|
|
|
# so check if required string is present in codeMirror
|
2021-11-10 00:20:20 -06:00
|
|
|
code_mirror = self.driver.find_elements(
|
|
|
|
By.XPATH, QueryToolLocators.code_mirror_data_xpath)
|
2019-08-22 04:20:51 -05:00
|
|
|
for data in code_mirror:
|
|
|
|
code_mirror_text = data.text
|
|
|
|
print("Single entry..........")
|
|
|
|
if expected_string not in code_mirror_text:
|
|
|
|
print("single entry exception.........")
|
2020-08-07 02:07:00 -05:00
|
|
|
raise RuntimeError("Required String %s is not "
|
|
|
|
"present" % expected_string)
|