2017-03-16 14:27:55 +00:00
|
|
|
##########################################################################
|
|
|
|
#
|
|
|
|
# pgAdmin 4 - PostgreSQL Tools
|
|
|
|
#
|
|
|
|
# Copyright (C) 2013 - 2017, The pgAdmin Development Team
|
|
|
|
# This software is released under the PostgreSQL Licence
|
|
|
|
#
|
|
|
|
##########################################################################
|
|
|
|
|
2017-02-22 12:41:28 +00:00
|
|
|
import time
|
2017-05-26 14:46:58 +00:00
|
|
|
import math
|
2017-02-22 12:41:28 +00:00
|
|
|
|
2017-06-06 11:54:20 +00:00
|
|
|
from selenium.common.exceptions import NoSuchElementException, \
|
|
|
|
WebDriverException, StaleElementReferenceException
|
2017-02-22 12:41:28 +00:00
|
|
|
from selenium.webdriver import ActionChains
|
|
|
|
from selenium.webdriver.common.keys import Keys
|
2017-03-01 13:20:06 +00:00
|
|
|
from selenium.webdriver.support import expected_conditions as EC
|
|
|
|
from selenium.webdriver.support.ui import WebDriverWait
|
|
|
|
from selenium.webdriver.common.by import By
|
2017-02-22 12:41:28 +00:00
|
|
|
|
|
|
|
|
|
|
|
class PgadminPage:
|
|
|
|
"""
|
|
|
|
Helper class for interacting with the page, given a selenium driver
|
|
|
|
"""
|
2017-03-01 13:20:06 +00:00
|
|
|
|
2017-02-22 12:41:28 +00:00
|
|
|
def __init__(self, driver, app_config):
|
|
|
|
self.driver = driver
|
|
|
|
self.app_config = app_config
|
2017-06-27 14:55:57 +00:00
|
|
|
self.timeout = 30
|
2017-05-27 18:17:06 +00:00
|
|
|
self.app_start_timeout = 60
|
2017-02-22 12:41:28 +00:00
|
|
|
|
2017-03-01 13:20:06 +00:00
|
|
|
def reset_layout(self):
|
|
|
|
self.click_element(self.find_by_partial_link_text("File"))
|
|
|
|
self.find_by_partial_link_text("Reset Layout").click()
|
2017-06-13 08:50:41 +00:00
|
|
|
self.click_modal('OK')
|
2017-03-09 15:34:51 +00:00
|
|
|
self.wait_for_reloading_indicator_to_disappear()
|
2017-03-01 13:20:06 +00:00
|
|
|
|
2017-06-13 08:50:41 +00:00
|
|
|
def click_modal(self, button_text):
|
2017-03-09 15:34:51 +00:00
|
|
|
time.sleep(0.5)
|
2017-04-10 13:07:48 +00:00
|
|
|
# Find active alertify dialog in case of multiple alertify dialog & click on that dialog
|
2017-06-13 08:50:41 +00:00
|
|
|
modal_button = self.find_by_xpath(
|
|
|
|
"//div[contains(@class, 'alertify') and not(contains(@class, 'ajs-hidden'))]//button[.='%s']"
|
|
|
|
% button_text)
|
|
|
|
self.click_element(modal_button)
|
2017-02-22 12:41:28 +00:00
|
|
|
|
2017-03-01 13:20:06 +00:00
|
|
|
def add_server(self, server_config):
|
2017-02-22 12:41:28 +00:00
|
|
|
self.find_by_xpath("//*[@class='aciTreeText' and contains(.,'Servers')]").click()
|
|
|
|
self.driver.find_element_by_link_text("Object").click()
|
|
|
|
ActionChains(self.driver) \
|
|
|
|
.move_to_element(self.driver.find_element_by_link_text("Create")) \
|
|
|
|
.perform()
|
|
|
|
self.find_by_partial_link_text("Server...").click()
|
|
|
|
|
|
|
|
self.fill_input_by_field_name("name", server_config['name'])
|
|
|
|
self.find_by_partial_link_text("Connection").click()
|
|
|
|
self.fill_input_by_field_name("host", server_config['host'])
|
|
|
|
self.fill_input_by_field_name("port", server_config['port'])
|
|
|
|
self.fill_input_by_field_name("username", server_config['username'])
|
|
|
|
self.fill_input_by_field_name("password", server_config['db_password'])
|
|
|
|
self.find_by_xpath("//button[contains(.,'Save')]").click()
|
|
|
|
|
|
|
|
self.find_by_xpath("//*[@id='tree']//*[.='" + server_config['name'] + "']")
|
|
|
|
|
2017-06-23 13:07:31 +00:00
|
|
|
def open_query_tool(self):
|
|
|
|
self.driver.find_element_by_link_text("Tools").click()
|
|
|
|
tools_menu = self.driver.find_element_by_id('mnu_tools')
|
|
|
|
|
|
|
|
# Query Tool is first li
|
|
|
|
query_tool = tools_menu.find_element_by_tag_name('li')
|
|
|
|
|
|
|
|
self.enable_menu_item(query_tool, 10)
|
|
|
|
|
|
|
|
self.find_by_partial_link_text("Query Tool").click()
|
|
|
|
self.click_tab('Query -')
|
|
|
|
|
|
|
|
def enable_menu_item(self, menu_item, wait_time):
|
|
|
|
start_time = time.time()
|
|
|
|
# wait until menu becomes enabled.
|
|
|
|
while time.time() - start_time < wait_time: # wait_time seconds
|
|
|
|
# if menu is disabled then it will have
|
|
|
|
# two classes 'menu-item disabled'.
|
|
|
|
# And if menu is enabled the it will have
|
|
|
|
# only one class 'menu-item'.
|
|
|
|
|
|
|
|
if 'menu-item' == str(menu_item.get_attribute('class')):
|
|
|
|
break
|
2017-06-27 13:03:04 +00:00
|
|
|
time.sleep(0.1)
|
2017-06-23 13:07:31 +00:00
|
|
|
else:
|
|
|
|
assert False, "'Tools -> Query Tool' menu did not enable."
|
|
|
|
|
2017-04-18 12:28:45 +00:00
|
|
|
def close_query_tool(self):
|
|
|
|
self.driver.switch_to.default_content()
|
|
|
|
tab = self.find_by_xpath("//*[contains(@class,'wcPanelTab') and contains(.,'" + "Query" + "')]")
|
|
|
|
ActionChains(self.driver).context_click(tab).perform()
|
|
|
|
self.find_by_xpath("//li[contains(@class, 'context-menu-item')]/span[contains(text(), 'Remove Panel')]").click()
|
|
|
|
self.driver.switch_to.frame(self.driver.find_elements_by_tag_name("iframe")[0])
|
|
|
|
time.sleep(.5)
|
|
|
|
self.click_element(self.find_by_xpath('//button[contains(@class, "ajs-button") and contains(.,"Yes")]'))
|
|
|
|
self.driver.switch_to.default_content()
|
|
|
|
|
2017-06-06 11:54:20 +00:00
|
|
|
def close_data_grid(self):
|
|
|
|
self.driver.switch_to_default_content()
|
|
|
|
xpath = "//*[@id='dockerContainer']/div/div[3]/div/div[2]/div[1]"
|
|
|
|
self.click_element(self.find_by_xpath(xpath))
|
|
|
|
|
2017-02-22 12:41:28 +00:00
|
|
|
def remove_server(self, server_config):
|
2017-04-18 12:28:45 +00:00
|
|
|
self.driver.switch_to.default_content()
|
2017-06-13 08:50:41 +00:00
|
|
|
server_to_remove = self.find_by_xpath("//*[@id='tree']//*[.='" + server_config['name'] + "' and @class='aciTreeItem']")
|
|
|
|
self.click_element(server_to_remove)
|
|
|
|
object_menu_item = self.find_by_partial_link_text("Object")
|
|
|
|
self.click_element(object_menu_item)
|
|
|
|
delete_menu_item = self.find_by_partial_link_text("Delete/Drop")
|
|
|
|
self.click_element(delete_menu_item)
|
|
|
|
self.click_modal('OK')
|
2017-02-22 12:41:28 +00:00
|
|
|
|
2017-03-23 11:47:21 +00:00
|
|
|
def select_tree_item(self, tree_item_text):
|
|
|
|
self.find_by_xpath("//*[@id='tree']//*[.='" + tree_item_text + "' and @class='aciTreeItem']").click()
|
|
|
|
|
2017-02-22 12:41:28 +00:00
|
|
|
def toggle_open_tree_item(self, tree_item_text):
|
|
|
|
self.find_by_xpath("//*[@id='tree']//*[.='" + tree_item_text + "']/../*[@class='aciTreeButton']").click()
|
|
|
|
|
2017-06-08 12:31:36 +00:00
|
|
|
def toggle_open_server(self, tree_item_text):
|
|
|
|
def check_for_password_dialog_or_tree_open(driver):
|
|
|
|
try:
|
|
|
|
dialog = driver.find_element_by_id("frmPassword")
|
|
|
|
except WebDriverException:
|
|
|
|
dialog = None
|
|
|
|
|
|
|
|
try:
|
|
|
|
database_node = driver.find_element_by_xpath("//*[@id='tree']//*[.='Databases']/../*[@class='aciTreeButton']")
|
|
|
|
except WebDriverException:
|
|
|
|
database_node = None
|
|
|
|
|
|
|
|
return dialog is not None or database_node is not None
|
|
|
|
|
|
|
|
self.toggle_open_tree_item(tree_item_text)
|
|
|
|
self._wait_for("Waiting for password dialog or tree to open", check_for_password_dialog_or_tree_open)
|
|
|
|
|
|
|
|
try:
|
|
|
|
self.driver.find_element_by_id("frmPassword")
|
|
|
|
# Enter password here if needed
|
2017-06-14 08:59:11 +00:00
|
|
|
self.click_modal('OK')
|
2017-06-08 12:31:36 +00:00
|
|
|
except WebDriverException:
|
|
|
|
return
|
|
|
|
|
2017-02-22 12:41:28 +00:00
|
|
|
def find_by_xpath(self, xpath):
|
2017-03-08 09:49:48 +00:00
|
|
|
return self.wait_for_element(lambda driver: driver.find_element_by_xpath(xpath))
|
2017-02-22 12:41:28 +00:00
|
|
|
|
|
|
|
def find_by_id(self, element_id):
|
2017-03-08 09:49:48 +00:00
|
|
|
return self.wait_for_element(lambda driver: driver.find_element_by_id(element_id))
|
2017-02-22 12:41:28 +00:00
|
|
|
|
2017-06-06 11:54:20 +00:00
|
|
|
def find_by_css_selector(self, css_selector):
|
|
|
|
return self.wait_for_element(lambda driver: driver.find_element_by_css_selector(css_selector))
|
|
|
|
|
2017-02-22 12:41:28 +00:00
|
|
|
def find_by_partial_link_text(self, link_text):
|
2017-03-01 13:20:06 +00:00
|
|
|
return self._wait_for(
|
2017-03-23 11:47:21 +00:00
|
|
|
'link with text "{0}"'.format(link_text),
|
2017-03-01 13:20:06 +00:00
|
|
|
EC.element_to_be_clickable((By.PARTIAL_LINK_TEXT, link_text))
|
|
|
|
)
|
|
|
|
|
|
|
|
def click_element(self, element):
|
2017-06-13 08:50:41 +00:00
|
|
|
# driver must be here to adhere to the method contract in selenium.webdriver.support.wait.WebDriverWait.until()
|
2017-03-01 13:20:06 +00:00
|
|
|
def click_succeeded(driver):
|
|
|
|
try:
|
|
|
|
element.click()
|
|
|
|
return True
|
|
|
|
except WebDriverException:
|
|
|
|
return False
|
|
|
|
|
|
|
|
return self._wait_for("clicking the element not to throw an exception", click_succeeded)
|
2017-02-22 12:41:28 +00:00
|
|
|
|
|
|
|
def fill_input_by_field_name(self, field_name, field_content):
|
|
|
|
field = self.find_by_xpath("//input[@name='" + field_name + "']")
|
2017-03-01 13:20:06 +00:00
|
|
|
backspaces = [Keys.BACKSPACE] * len(field.get_attribute('value'))
|
2017-02-22 12:41:28 +00:00
|
|
|
|
|
|
|
field.click()
|
|
|
|
field.send_keys(backspaces)
|
|
|
|
field.send_keys(str(field_content))
|
|
|
|
self.wait_for_input_field_content(field_name, field_content)
|
|
|
|
|
|
|
|
def fill_codemirror_area_with(self, field_content):
|
2017-05-26 14:46:58 +00:00
|
|
|
# For long text, if we try to execute send_keys and perform back to back, then the actions are
|
|
|
|
# not executed properly as the driver can send only 50 to 60 characters. To avoid this, sleep
|
|
|
|
# on the basis of content length.
|
2017-06-07 13:26:42 +00:00
|
|
|
def find_codemirror(driver):
|
|
|
|
try:
|
|
|
|
driver.switch_to.default_content()
|
|
|
|
driver.switch_to_frame(driver.find_element_by_tag_name("iframe"))
|
|
|
|
element = driver.find_element_by_xpath(
|
|
|
|
"//pre[contains(@class,'CodeMirror-line')]/../../../*[contains(@class,'CodeMirror-code')]")
|
|
|
|
if element.is_displayed() and element.is_enabled():
|
|
|
|
return element
|
|
|
|
except (NoSuchElementException, WebDriverException):
|
|
|
|
return False
|
|
|
|
|
|
|
|
WebDriverWait(self.driver, timeout=self.timeout, poll_frequency=0.01).\
|
|
|
|
until(find_codemirror, "Timed out waiting for codemirror to appear").\
|
|
|
|
click()
|
|
|
|
time.sleep(1)
|
|
|
|
|
2017-05-26 14:46:58 +00:00
|
|
|
action = ActionChains(self.driver)
|
|
|
|
action.send_keys(field_content)
|
2017-06-07 13:26:42 +00:00
|
|
|
action.perform()
|
2017-05-26 14:46:58 +00:00
|
|
|
sleep_time = math.ceil(len(field_content) / 50)
|
|
|
|
time.sleep(sleep_time)
|
2017-02-22 12:41:28 +00:00
|
|
|
|
|
|
|
def click_tab(self, tab_name):
|
2017-06-13 08:50:41 +00:00
|
|
|
tab = self.find_by_xpath("//*[contains(@class,'wcTabTop')]//*[contains(@class,'wcPanelTab') "
|
|
|
|
"and contains(.,'" + tab_name + "')]")
|
|
|
|
self.click_element(tab)
|
2017-02-22 12:41:28 +00:00
|
|
|
|
|
|
|
def wait_for_input_field_content(self, field_name, content):
|
2017-03-01 13:20:06 +00:00
|
|
|
def input_field_has_content(driver):
|
|
|
|
element = driver.find_element_by_xpath(
|
2017-02-22 12:41:28 +00:00
|
|
|
"//input[@name='" + field_name + "']")
|
|
|
|
|
|
|
|
return str(content) == element.get_attribute('value')
|
|
|
|
|
|
|
|
return self._wait_for("field to contain '" + str(content) + "'", input_field_has_content)
|
|
|
|
|
|
|
|
def wait_for_element(self, find_method_with_args):
|
2017-03-01 13:20:06 +00:00
|
|
|
def element_if_it_exists(driver):
|
2017-02-22 12:41:28 +00:00
|
|
|
try:
|
2017-03-01 13:20:06 +00:00
|
|
|
element = find_method_with_args(driver)
|
|
|
|
if element.is_displayed() and element.is_enabled():
|
2017-02-22 12:41:28 +00:00
|
|
|
return element
|
|
|
|
except NoSuchElementException:
|
|
|
|
return False
|
|
|
|
|
|
|
|
return self._wait_for("element to exist", element_if_it_exists)
|
|
|
|
|
2017-03-09 15:34:51 +00:00
|
|
|
def wait_for_reloading_indicator_to_disappear(self):
|
|
|
|
def reloading_indicator_has_disappeared(driver):
|
|
|
|
try:
|
|
|
|
driver.find_element_by_id("reloading-indicator")
|
|
|
|
return False
|
|
|
|
except NoSuchElementException:
|
|
|
|
return True
|
|
|
|
|
|
|
|
self._wait_for("reloading indicator to disappear", reloading_indicator_has_disappeared)
|
|
|
|
|
2017-02-22 12:41:28 +00:00
|
|
|
def wait_for_spinner_to_disappear(self):
|
2017-03-01 13:20:06 +00:00
|
|
|
def spinner_has_disappeared(driver):
|
2017-02-22 12:41:28 +00:00
|
|
|
try:
|
2017-03-01 13:20:06 +00:00
|
|
|
driver.find_element_by_id("pg-spinner")
|
2017-02-22 12:41:28 +00:00
|
|
|
return False
|
|
|
|
except NoSuchElementException:
|
|
|
|
return True
|
|
|
|
|
|
|
|
self._wait_for("spinner to disappear", spinner_has_disappeared)
|
|
|
|
|
2017-06-27 13:03:04 +00:00
|
|
|
def wait_for_query_tool_loading_indicator_to_disappear(self):
|
|
|
|
def spinner_has_disappeared(driver):
|
|
|
|
try:
|
|
|
|
driver.find_element_by_xpath(
|
|
|
|
"//*[@id='fetching_data' and @class='hide']"
|
|
|
|
)
|
|
|
|
return False
|
|
|
|
except NoSuchElementException:
|
|
|
|
# wait for loading indicator disappear animation to complete.
|
|
|
|
time.sleep(0.5)
|
|
|
|
return True
|
|
|
|
|
|
|
|
self._wait_for("spinner to disappear", spinner_has_disappeared)
|
|
|
|
|
2017-02-22 12:41:28 +00:00
|
|
|
def wait_for_app(self):
|
2017-03-01 13:20:06 +00:00
|
|
|
def page_shows_app(driver):
|
|
|
|
if driver.title == self.app_config.APP_NAME:
|
2017-02-22 12:41:28 +00:00
|
|
|
return True
|
|
|
|
else:
|
2017-03-01 13:20:06 +00:00
|
|
|
driver.refresh()
|
2017-02-22 12:41:28 +00:00
|
|
|
return False
|
|
|
|
|
2017-05-27 18:17:06 +00:00
|
|
|
self._wait_for("app to start", page_shows_app, self.app_start_timeout)
|
2017-02-22 12:41:28 +00:00
|
|
|
|
2017-05-27 18:17:06 +00:00
|
|
|
def _wait_for(self, waiting_for_message, condition_met_function, timeout = None):
|
|
|
|
if timeout is None:
|
|
|
|
timeout = self.timeout
|
|
|
|
return WebDriverWait(self.driver, timeout, 0.01).until(condition_met_function,
|
2017-03-01 13:20:06 +00:00
|
|
|
"Timed out waiting for " + waiting_for_message)
|